io.hyperfoil.tools.horreum.svc.user.KeycloakUserBackend Maven / Gradle / Ivy
package io.hyperfoil.tools.horreum.svc.user;
import static java.text.MessageFormat.format;
import static java.util.stream.Collectors.joining;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Stream;
import jakarta.enterprise.context.ApplicationScoped;
import jakarta.inject.Inject;
import jakarta.ws.rs.ClientErrorException;
import jakarta.ws.rs.NotFoundException;
import jakarta.ws.rs.core.Response;
import org.eclipse.microprofile.config.inject.ConfigProperty;
import org.jboss.logging.Logger;
import org.keycloak.admin.client.Keycloak;
import org.keycloak.admin.client.resource.ClientsResource;
import org.keycloak.admin.client.resource.RoleMappingResource;
import org.keycloak.admin.client.resource.RoleResource;
import org.keycloak.admin.client.resource.UsersResource;
import org.keycloak.representations.idm.ClientRepresentation;
import org.keycloak.representations.idm.CredentialRepresentation;
import org.keycloak.representations.idm.RoleRepresentation;
import org.keycloak.representations.idm.UserRepresentation;
import io.hyperfoil.tools.horreum.api.services.UserService;
import io.hyperfoil.tools.horreum.svc.Roles;
import io.hyperfoil.tools.horreum.svc.ServiceException;
import io.quarkus.arc.lookup.LookupIfProperty;
/**
* Implementation of {@link UserBackEnd} using an external Keycloak server.
* Relies on keycloak-admin-client to manage user information.
*/
@ApplicationScoped
@LookupIfProperty(name = "horreum.roles.provider", stringValue = "keycloak")
public class KeycloakUserBackend implements UserBackEnd {
private static final Logger LOG = Logger.getLogger(KeycloakUserBackend.class);
private static final String[] ROLE_TYPES = new String[] { "team", Roles.VIEWER, Roles.TESTER, Roles.UPLOADER,
Roles.MANAGER };
@ConfigProperty(name = "quarkus.keycloak.admin-client.realm", defaultValue = "horreum")
String realm;
// please make sure all calls to this object are in a try/catch block to avoid leaking information
@Inject
Keycloak keycloak;
private static UserService.UserData toUserInfo(UserRepresentation rep) {
return new UserService.UserData(rep.getId(), rep.getUsername(), rep.getFirstName(), rep.getLastName(), rep.getEmail());
}
private static String getTeamPrefix(String team) {
return team.substring(0, team.length() - 4);
}
private static boolean isTeam(String role) {
return role.endsWith("-team"); // definition of a "team role"
}
@Override
public List searchUsers(String query) {
try {
return keycloak.realm(realm).users().search(query, null, null).stream().map(KeycloakUserBackend::toUserInfo)
.toList();
} catch (Throwable t) {
throw ServiceException.serverError("Unable to search for users");
}
}
@Override
public List getRoles(String username) {
List representations = keycloak.realm(realm).users().get(findMatchingUserId(username)).roles()
.realmLevel().listAll();
// the realm level roles does not include the base roles, only the composites, so add them manually
Set roles = new HashSet<>(representations.stream().map(RoleRepresentation::getName).toList());
for (String type : ROLE_TYPES) {
Optional composite = roles.stream().filter(role -> role.endsWith(type)).findAny();
if (composite.isPresent()) {
roles.add(type);
roles.add(composite.get().substring(0, composite.get().length() - type.length() - 1) + "-team");
}
}
return new ArrayList<>(roles);
// the right way to do this would be something like this (avoided because it does call keycloak a bunch of times)
// return representations.stream().flatMap(this::getRoleAndComposites).toList();
}
private Stream getRoleAndComposites(RoleRepresentation representation) {
Set roles = new HashSet<>();
if (representation.isComposite()) {
keycloak.realm(realm).rolesById().getRealmRoleComposites(representation.getId()).stream()
.flatMap(this::getRoleAndComposites).forEach(roles::add);
}
roles.add(representation.getName());
return roles.stream();
}
@Override
public List info(List usernames) {
List users = new ArrayList<>();
for (String username : usernames) {
try {
keycloak.realm(realm).users().search(username).stream().filter(u -> username.equals(u.getUsername()))
.map(KeycloakUserBackend::toUserInfo).forEach(users::add);
} catch (Throwable t) {
LOG.warnv(t, "Failed to fetch info for user {0}", username);
throw ServiceException.serverError(format("Failed to fetch info for user {0}", username));
}
}
return users;
}
@Override
public void createUser(UserService.NewUser user) {
UserRepresentation rep = convertUserRepresentation(user); // do not blindly use the provided representation
try (Response response = keycloak.realm(realm).users().create(rep)) {
if (response.getStatusInfo().getFamily() != Response.Status.Family.SUCCESSFUL) {
LOG.warnv("Failed to create new user {0}: {1}", rep.getUsername(), response.getStatusInfo());
if (!keycloak.realm(realm).users().search(rep.getUsername(), true).isEmpty()) {
throw ServiceException.badRequest("User exists with same username");
} else if (!keycloak.realm(realm).users().searchByEmail(rep.getEmail(), true).isEmpty()) {
throw ServiceException.badRequest("User exists with same email");
} else {
throw ServiceException
.badRequest("Failed to create new user: " + response.getStatusInfo().getReasonPhrase());
}
}
} catch (ServiceException se) {
throw se; // thrown above, re-throw
} catch (Throwable t) {
throw ServiceException.serverError(format("Failed to create new user {0}", rep.getUsername()));
}
try { // assign the provided roles to the realm
UsersResource usersResource = keycloak.realm(realm).users();
String userId = findMatchingUserId(rep.getUsername());
if (user.team != null) {
String prefix = getTeamPrefix(user.team);
usersResource.get(userId).roles().realmLevel()
.add(user.roles.stream().map(r -> ensureRole(prefix + r)).toList());
}
// also add the "view-profile" role
ClientsResource clientsResource = keycloak.realm(realm).clients();
ClientRepresentation account = clientsResource.query("account").stream().filter(c -> "account".equals(c.getName()))
.findFirst().orElse(null);
if (account != null) {
RoleRepresentation viewProfile = clientsResource.get(account.getId()).roles().get("view-profile")
.toRepresentation();
if (viewProfile != null) {
usersResource.get(userId).roles().clientLevel(account.getClientId()).add(List.of(viewProfile));
}
}
} catch (ServiceException se) {
throw se; // thrown above, re-throw
} catch (Throwable t) {
LOG.warnv(t, "Unable to assign roles to new user {0}", rep.getUsername());
throw ServiceException.serverError(format("Unable to assign roles to new user {0}", rep.getUsername()));
}
}
@Override
public void removeUser(String username) {
try (Response response = keycloak.realm(realm).users().delete(findMatchingUserId(username))) {
if (response.getStatusInfo().getFamily() != Response.Status.Family.SUCCESSFUL) {
LOG.warnv("Got {0} response for removing user {0}", response.getStatusInfo(), username);
throw ServiceException.serverError(format("Unable to remove user {0}", username));
}
} catch (ServiceException se) {
throw se; // thrown above, re-throw
} catch (Throwable t) {
LOG.warnv(t, "Unable to remove user {0}", username);
throw ServiceException.serverError(format("Unable to remove user {0}", username));
}
}
private static UserRepresentation convertUserRepresentation(UserService.NewUser user) {
UserRepresentation rep = new UserRepresentation();
rep.setUsername(user.user.username);
rep.setEmail(user.user.email);
rep.setFirstName(user.user.firstName);
rep.setLastName(user.user.lastName);
rep.setEnabled(true);
CredentialRepresentation credentials = new CredentialRepresentation();
credentials.setType(CredentialRepresentation.PASSWORD);
credentials.setTemporary(true);
credentials.setValue(user.password);
rep.setCredentials(List.of(credentials));
return rep;
}
@Override
public List getTeams() { // get the "team roles" in the realm
try {
return keycloak.realm(realm).roles().list().stream().map(RoleRepresentation::getName)
.filter(KeycloakUserBackend::isTeam).toList();
} catch (Throwable t) {
throw ServiceException.serverError("Unable to get list of teams");
}
}
private String findMatchingUserId(String username) { // find the clientID of a single user
List matchingUsers = keycloak.realm(realm).users().search(username, true);
if (matchingUsers == null || matchingUsers.isEmpty()) {
LOG.warnv("Cannot find user with username {0}", username);
throw ServiceException.notFound(format("User {0} does not exist", username));
} else if (matchingUsers.size() > 1) {
LOG.warnv("Multiple matches for exact search for username {0}: {1}", username,
matchingUsers.stream().map(UserRepresentation::getId).collect(joining(" ")));
throw ServiceException.serverError(format("More than one user with username {0}", username));
}
return matchingUsers.get(0).getId();
}
@Override
public Map> teamMembers(String team) { // get a list of members of a team and their "UI roles"
String prefix = getTeamPrefix(team);
Map> userMap = new HashMap<>();
for (String role : ROLE_TYPES) {
try {
// the call below does not consider transitivity with composite roles
keycloak.realm(realm).roles().get(prefix + role).getUserMembers(0, Integer.MAX_VALUE)
.forEach(user -> userMap.computeIfAbsent(user.getUsername(), u -> new ArrayList<>()).add(role));
} catch (NotFoundException e) {
LOG.warnv("Cannot find role {0}{1} in Keycloak", prefix, role); // was there a failure when creating the team?
} catch (Throwable t) {
LOG.warnv("Error querying keycloak: {0}", t.getMessage());
throw ServiceException.serverError("Failed to retrieve role users from Keycloak");
}
}
return userMap;
}
@Override
public void updateTeamMembers(String team, Map> roles) { // update the team membership. the roles provided here are "UI roles"
String prefix = getTeamPrefix(team);
for (Map.Entry> entry : roles.entrySet()) {
List existingRoles;
RoleMappingResource rolesMappingResource;
try { // fetch the current roles for the user
String userId = findMatchingUserId(entry.getKey());
rolesMappingResource = keycloak.realm(realm).users().get(userId).roles();
existingRoles = rolesMappingResource.realmLevel().listAll().stream().map(RoleRepresentation::getName).toList();
} catch (Throwable t) {
LOG.warnv(t, "Failed to retrieve current roles of user {0} from Keycloak", entry.getKey());
throw ServiceException
.serverError(format("Failed to retrieve current roles of user {0} from Keycloak", entry.getKey()));
}
try { // add new roles that are not in the list of current roles and then remove the existing roles that are not on the new roles
List rolesToAdd = entry.getValue().stream()
.filter(uiRole -> !existingRoles.contains(prefix + uiRole)).map(uiRole -> ensureRole(prefix + uiRole))
.toList();
if (!rolesToAdd.isEmpty()) {
rolesMappingResource.realmLevel().add(rolesToAdd);
}
List rolesToRemove = existingRoles.stream()
.filter(r -> r.startsWith(prefix) && !entry.getValue().contains(r.substring(prefix.length())))
.map(this::ensureRole).toList();
if (!rolesToRemove.isEmpty()) {
rolesMappingResource.realmLevel().remove(rolesToRemove);
}
} catch (Throwable t) {
LOG.warnv(t, "Failed to modify roles of user {0}", entry.getKey());
throw ServiceException.serverError(format("Failed to modify roles of user {0}", entry.getKey()));
}
}
try { // remove all team roles to users not in the provided roles map
UsersResource usersResource = keycloak.realm(realm).users();
for (String type : ROLE_TYPES) {
RoleResource roleResource = keycloak.realm(realm).roles().get(prefix + type);
RoleRepresentation role = roleResource.toRepresentation();
for (UserRepresentation user : roleResource.getUserMembers(0, Integer.MAX_VALUE)) {
if (!roles.containsKey(user.getUsername())) {
usersResource.get(user.getId()).roles().realmLevel().remove(List.of(role));
}
}
}
} catch (NotFoundException e) {
throw ServiceException.serverError(format("The team {0} does not exist", team));
} catch (Throwable t) {
LOG.warnv(t, "Failed to remove all roles of team {0}", team);
throw ServiceException.serverError(format("Failed to remove all roles of team {0}", team));
}
}
private RoleRepresentation ensureRole(String roleName) {
try {
return keycloak.realm(realm).roles().get(roleName).toRepresentation();
} catch (NotFoundException e) {
keycloak.realm(realm).roles().create(new RoleRepresentation(roleName, null, false));
return keycloak.realm(realm).roles().get(roleName).toRepresentation();
} catch (Throwable t) {
throw ServiceException.serverError(format("Unable to fetch role {0}", roleName));
}
}
@Override
public List getAllTeams() {
try {
return keycloak.realm(realm).roles().list().stream().map(RoleRepresentation::getName)
.filter(KeycloakUserBackend::isTeam).toList();
} catch (Exception e) {
throw ServiceException
.serverError("Please check with the System Administrators that you have the correct permissions");
}
}
@Override
public void addTeam(String team) { // create the "team roles"
String prefix = getTeamPrefix(team); // perform validation of the team name
createRole(team, null);
for (String role : List.of(Roles.MANAGER, Roles.TESTER, Roles.VIEWER, Roles.UPLOADER)) {
createRole(prefix + role, Set.of(role, team));
}
}
private void createRole(String roleName, Set compositeRoles) {
RoleRepresentation role = new RoleRepresentation(roleName, null, false);
if (compositeRoles != null) {
role.setComposite(true);
RoleRepresentation.Composites composites = new RoleRepresentation.Composites();
composites.setRealm(compositeRoles);
role.setComposites(composites);
}
try {
keycloak.realm(realm).roles().create(role);
} catch (ClientErrorException e) {
if (e.getResponse().getStatus() == Response.Status.CONFLICT.getStatusCode()) {
LOG.warnv("Registration of role {0} failed because it already exists", roleName);
} else {
throw ServiceException.serverError(format("Unable to create role {0}", roleName));
}
} catch (Throwable t) {
throw ServiceException.serverError(format("Unable to create role {0}", roleName));
}
}
@Override
public void deleteTeam(String team) { // delete a team by deleting all the "team roles"
String prefix = getTeamPrefix(team);
for (String type : ROLE_TYPES) {
try {
keycloak.realm(realm).roles().deleteRole(prefix + type);
} catch (NotFoundException e) {
LOG.warnv("Role {0}{1} was not found when deleting it", prefix, type);
throw ServiceException.notFound(format("Team {0} not found", team));
} catch (Throwable t) {
LOG.warnv(t, "Unable to delete team {0}", team);
throw ServiceException.serverError(format("Unable to delete team {0}", team));
}
}
}
@Override
public List administrators() { // get the list of all the users with administrator role
try {
return keycloak.realm(realm).roles().get(Roles.ADMIN).getUserMembers(0, Integer.MAX_VALUE).stream()
.map(KeycloakUserBackend::toUserInfo).toList();
} catch (Throwable t) {
LOG.warnv(t, "Unable to list administrators");
throw ServiceException
.serverError("Please verify with the System Administrators that you have the correct permissions");
}
}
@Override
public void updateAdministrators(List newAdmins) { // update the list of administrator users
try {
UsersResource usersResource = keycloak.realm(realm).users();
RoleResource adminRoleResource = keycloak.realm(realm).roles().get(Roles.ADMIN);
RoleRepresentation adminRole = adminRoleResource.toRepresentation();
List oldAdmins = adminRoleResource.getUserMembers(0, Integer.MAX_VALUE);
for (UserRepresentation user : oldAdmins) { // remove admin role from `oldAdmins` not in `newAdmins`
if (!newAdmins.contains(user.getUsername())) {
try {
usersResource.get(user.getId()).roles().realmLevel().remove(List.of(adminRole));
LOG.infov("Removed administrator role from user {0}", user.getUsername());
} catch (Throwable t) {
LOG.warnv("Could not remove admin role from user {0} due to {1}", user.getUsername(), t.getMessage());
}
}
}
for (String username : newAdmins) { // add admin role for `newAdmins` not in `oldAdmins`
if (oldAdmins.stream().noneMatch(old -> username.equals(old.getUsername()))) {
try {
usersResource.get(findMatchingUserId(username)).roles().realmLevel().add(List.of(adminRole));
LOG.infov("Added administrator role to user {0}", username);
} catch (Throwable t) {
LOG.warnv("Could not add admin role to user {0} due to {1}", username, t.getMessage());
}
}
}
} catch (ServiceException se) {
throw se; // thrown above, re-throw
} catch (Throwable t) {
LOG.warnv(t, "Cannot fetch representation for admin role");
throw ServiceException.serverError("Cannot find admin role");
}
}
@Override
public void setPassword(String username, String password) {
try {
CredentialRepresentation credentials = new CredentialRepresentation();
credentials.setType(CredentialRepresentation.PASSWORD);
credentials.setValue(password);
keycloak.realm(realm).users().get(findMatchingUserId(username)).resetPassword(credentials);
} catch (Throwable t) {
LOG.warnv(t, "Failed to retrieve current representation of user {0} from Keycloak", username);
throw ServiceException
.serverError(format("Failed to retrieve current representation of user {0} from Keycloak", username));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy