All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.hyperfoil.tools.horreum.svc.user.DatabaseUserBackend Maven / Gradle / Ivy

package io.hyperfoil.tools.horreum.svc.user;

import static java.text.MessageFormat.format;
import static java.util.Collections.emptyMap;
import static java.util.stream.Collectors.toSet;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.function.Function;

import jakarta.enterprise.context.ApplicationScoped;
import jakarta.persistence.criteria.CriteriaBuilder;
import jakarta.persistence.criteria.CriteriaQuery;
import jakarta.transaction.Transactional;

import org.jboss.logging.Logger;

import io.hyperfoil.tools.horreum.api.services.UserService;
import io.hyperfoil.tools.horreum.entity.user.Team;
import io.hyperfoil.tools.horreum.entity.user.TeamMembership;
import io.hyperfoil.tools.horreum.entity.user.TeamRole;
import io.hyperfoil.tools.horreum.entity.user.UserInfo;
import io.hyperfoil.tools.horreum.entity.user.UserRole;
import io.hyperfoil.tools.horreum.server.WithRoles;
import io.hyperfoil.tools.horreum.svc.Roles;
import io.hyperfoil.tools.horreum.svc.ServiceException;
import io.quarkus.arc.lookup.LookupIfProperty;

/**
 * Implementation of {@link UserBackEnd} that uses Horreum database for storage.
 * 

* Relies on the {@link UserInfo} entity and it's mappings. */ @ApplicationScoped @LookupIfProperty(name = "horreum.roles.provider", stringValue = "database") public class DatabaseUserBackend implements UserBackEnd { private static final Logger LOG = Logger.getLogger(DatabaseUserBackend.class); private static UserService.UserData toUserInfo(UserInfo info) { return new UserService.UserData("", info.username, info.firstName, info.lastName, info.email); } private static String removeTeamSuffix(String team) { return team.substring(0, team.length() - 5); } @Transactional @Override public List getRoles(String username) { UserInfo user = UserInfo.findById(username); List roles = new ArrayList<>(); user.roles.stream().map(UserRole::toString).map(String::toLowerCase).forEach(roles::add); user.teams.stream().map(TeamMembership::asRole).forEach(roles::add); user.teams.stream().map(TeamMembership::asTeam).forEach(roles::add); user.teams.stream().map(TeamMembership::asUIRole).distinct().forEach(roles::add); return roles; } @Transactional @WithRoles(extras = Roles.HORREUM_SYSTEM) @Override public List searchUsers(String param) { String query = "lower(firstName) like ?1 or lower(lastName) like ?1 or lower(username) like ?1"; return UserInfo. stream(query, "%" + param.toLowerCase() + "%") .map(DatabaseUserBackend::toUserInfo) .toList(); } @Transactional @WithRoles(extras = Roles.HORREUM_SYSTEM) @Override public List info(List usernames) { return UserInfo. stream("username in ?1", usernames).map(DatabaseUserBackend::toUserInfo).toList(); } @Transactional @WithRoles(fromParams = NewUserParameterConverter.class, extras = Roles.HORREUM_SYSTEM) @Override public void createUser(UserService.NewUser user) { if (UserInfo.findByIdOptional(user.user.username).isPresent()) { throw ServiceException.badRequest("User exists with same username"); } if (UserInfo.count("email", user.user.email) > 0) { throw ServiceException.badRequest("User exists with same email"); } UserInfo userInfo = new UserInfo(user.user.username); userInfo.email = user.user.email; userInfo.firstName = user.user.firstName; userInfo.lastName = user.user.lastName; userInfo.setPassword(user.password); if (user.team != null) { // userInfo.defaultTeam = user.team; // Don't set default team to be consistent with keycloak backend String teamName = removeTeamSuffix(user.team); for (String role : user.roles) { if (Roles.VIEWER.equals(role)) { addTeamMembership(userInfo, teamName, TeamRole.TEAM_VIEWER); } else if (Roles.TESTER.equals(role)) { addTeamMembership(userInfo, teamName, TeamRole.TEAM_TESTER); } else if (Roles.UPLOADER.equals(role)) { addTeamMembership(userInfo, teamName, TeamRole.TEAM_UPLOADER); } else if (Roles.MANAGER.equals(role)) { addTeamMembership(userInfo, teamName, TeamRole.TEAM_MANAGER); } else { LOG.infov("Dropping role {0} for user {1} {2}", role, userInfo.firstName, userInfo.lastName); } } } userInfo.persist(); } private void addTeamMembership(UserInfo userInfo, String teamName, TeamRole role) { Optional storedTeam = Team.find("teamName", teamName).firstResultOptional(); userInfo.teams.add(new TeamMembership(userInfo, storedTeam.orElseGet(() -> Team.getEntityManager().merge(new Team(teamName))), role)); } @Transactional @WithRoles(fromParams = RemoveUserParameterConverter.class) @Override public void removeUser(String username) { if (!UserInfo.deleteById(username)) { throw ServiceException.notFound("User does not exist"); } } @Transactional @Override public List getTeams() { List teams = Team.listAll(); return teams.stream().map(t -> t.teamName + "-team").toList(); } @Transactional @WithRoles(extras = Roles.HORREUM_SYSTEM) @Override public Map> teamMembers(String team) { Team teamEntity = Team.find("teamName", removeTeamSuffix(team)).firstResult(); if (teamEntity == null) { return emptyMap(); } Map> userMap = new HashMap<>(); teamEntity.teams.forEach(membership -> userMap.computeIfAbsent(membership.user.username, s -> new ArrayList<>()) .add(membership.asUIRole())); return userMap; } @Transactional @WithRoles(fromParams = UpdateTeamMembersParameterConverter.class) @Override public void updateTeamMembers(String team, Map> roles) { Team teamEntity = Team.find("teamName", removeTeamSuffix(team)).firstResult(); if (teamEntity == null) { throw ServiceException.notFound(format("The team {0} does not exist", team)); } // need to remove from the "owning" side of the relationship roles.forEach((username, teamRoles) -> { Optional user = UserInfo.findByIdOptional(username); user.ifPresent(u -> { List removedMemberships = u.teams.stream() .filter(t -> t.team == teamEntity && !teamRoles.contains(t.asUIRole())).toList(); removedMemberships.forEach(TeamMembership::delete); removedMemberships.forEach(u.teams::remove); u.teams.addAll(teamRoles.stream().map( uiRole -> TeamMembership.getEntityManager().merge(new TeamMembership(user.get(), teamEntity, uiRole))) .collect(toSet())); }); }); } @Transactional @WithRoles(extras = Roles.HORREUM_SYSTEM) @Override public List getAllTeams() { List teams = Team.listAll(); return teams.stream().map(t -> t.teamName + "-team").toList(); } @Transactional @Override public void addTeam(String team) { Team.getEntityManager().merge(new Team(removeTeamSuffix(team))); } @Transactional @WithRoles(extras = Roles.HORREUM_SYSTEM) @Override public void deleteTeam(String team) { Team teamEntity = Team.find("teamName", removeTeamSuffix(team)).firstResult(); if (teamEntity == null) { throw ServiceException.notFound(format("The team {0} does not exist", team)); } // need to delete memberships with roles deleteTeamAndMemberships(teamEntity); } @Transactional @WithRoles(fromParams = DeleteTeamAndMembershipsParameterConverter.class) void deleteTeamAndMemberships(Team teamEntity) { try { // need to remove from the "owning" side of the relationship teamEntity.teams.stream().map(membership -> membership.user.username).forEach(username -> { Optional user = UserInfo.findByIdOptional(username); user.ifPresent(u -> u.teams.removeIf(membership -> teamEntity.equals(membership.team))); }); teamEntity.delete(); } catch (Throwable t) { LOG.warnv("Unable to delete team {0} due to {1}", teamEntity.teamName, t.getMessage()); throw ServiceException.serverError(format("Unable to delete team {0}", teamEntity.teamName)); } } @Transactional @WithRoles(extras = Roles.HORREUM_SYSTEM) @Override public List administrators() { return getAdministratorUsers().stream().map(DatabaseUserBackend::toUserInfo).toList(); } @Transactional @WithRoles(extras = Roles.HORREUM_SYSTEM) @Override public void updateAdministrators(List newAdmins) { getAdministratorUsers().forEach(u -> { if (!newAdmins.contains(u.username)) { u.roles.remove(UserRole.ADMIN); u.persist(); LOG.infov("Removed administrator role from user {0}", u.username); } }); newAdmins.forEach(username -> { Optional user = UserInfo.findByIdOptional(username); user.ifPresent(u -> { u.roles.add(UserRole.ADMIN); u.persist(); LOG.infov("Added administrator role to user {0}", username); }); }); } private List getAdministratorUsers() { CriteriaBuilder cb = UserInfo.getEntityManager().getCriteriaBuilder(); CriteriaQuery query = cb.createQuery(UserInfo.class); query.where(cb.isMember(UserRole.ADMIN, query.from(UserInfo.class).get("roles"))); return UserInfo.getEntityManager().createQuery(query).getResultList(); } @Transactional @WithRoles(fromParams = ResetPasswordParameterConverter.class) @Override public void setPassword(String username, String password) { UserInfo user = UserInfo.findById(username); if (user == null) { throw ServiceException.notFound(format("User {0} not found", username)); } user.setPassword(password); } /** * Extracts username from parameters of `createUser()` */ public static final class NewUserParameterConverter implements Function { @Override public String[] apply(Object[] objects) { return new String[] { ((UserService.NewUser) objects[0]).user.username }; } } /** * Extract usernames from parameters of `removeUser()` */ public static final class RemoveUserParameterConverter implements Function { @Override public String[] apply(Object[] objects) { return new String[] { (String) objects[0] }; } } /** * Extract usernames from parameters of `updateTeamMembers()` */ public static final class UpdateTeamMembersParameterConverter implements Function { @SuppressWarnings("unchecked") @Override public String[] apply(Object[] objects) { return ((Map>) objects[1]).keySet().toArray(String[]::new); } } /** * Extract usernames from parameters of `deleteTeamAndMemberships()` */ public static final class DeleteTeamAndMembershipsParameterConverter implements Function { @Override public String[] apply(Object[] objects) { return ((Team) objects[0]).teams.stream().map(membership -> membership.user.username).toArray(String[]::new); } } /** * Extract usernames from parameters of `resetPassword()` */ public static final class ResetPasswordParameterConverter implements Function { @Override public String[] apply(Object[] objects) { return new String[] { (String) objects[0] }; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy