From 8b57fd4b7477f28aa4822dfff3f3f9616e242e90 Mon Sep 17 00:00:00 2001 From: Stepan Ermakov Date: Tue, 20 Aug 2024 10:43:24 +0300 Subject: [PATCH] Signed-off-by: Stepan Ermakov ##Fixes issue In case when Keycloack configured, if you create a user with non-ascii names (first name, last name, e-mail address) then these names will be displayed using non-readable symbols in the list of users in the Administration console. ##Changes introduced with this PR In case when External SSO is configured we re-encode the names from ISO_8859_1 to UTF8 ##Are you the owner of the code you are sending in, or do you have permission of the owner? Yes --- .../sso/service/NegotiateAuthService.java | 8 +++-- .../engine/core/sso/service/SsoService.java | 31 +++++++++++++++++++ 2 files changed, 36 insertions(+), 3 deletions(-) diff --git a/backend/manager/modules/enginesso/src/main/java/org/ovirt/engine/core/sso/service/NegotiateAuthService.java b/backend/manager/modules/enginesso/src/main/java/org/ovirt/engine/core/sso/service/NegotiateAuthService.java index 232f139779c..bd14a03ccae 100644 --- a/backend/manager/modules/enginesso/src/main/java/org/ovirt/engine/core/sso/service/NegotiateAuthService.java +++ b/backend/manager/modules/enginesso/src/main/java/org/ovirt/engine/core/sso/service/NegotiateAuthService.java @@ -132,19 +132,21 @@ private AuthResult doAuth(HttpServletRequest req, HttpServletResponse rsp, Deque .mput( Authz.InvokeKeys.QUERY_FLAGS, Authz.QueryFlags.RESOLVE_GROUPS | Authz.QueryFlags.RESOLVE_GROUPS_RECURSIVE); - if (SsoService.getSsoContext(req) + boolean externalSso = SsoService.getSsoContext(req) .getSsoLocalConfig() - .getBoolean("ENGINE_SSO_ENABLE_EXTERNAL_SSO")) { + .getBoolean("ENGINE_SSO_ENABLE_EXTERNAL_SSO"); + if (externalSso) { input.put(Authz.InvokeKeys.HTTP_SERVLET_REQUEST, req); } ExtMap outputMap = profile.getAuthz().invoke(input); token = SsoService.getTokenFromHeader(req); + ExtMap principalRecord = outputMap.get(Authz.InvokeKeys.PRINCIPAL_RECORD); SsoSession ssoSession = SsoService.persistAuthInfoInContextWithToken(req, token, null, profile.getName(), authRecord, - outputMap.get(Authz.InvokeKeys.PRINCIPAL_RECORD)); + SsoService.fixExternalNames(principalRecord, externalSso)); log.info("User {}@{} with profile [{}] successfully logged in with scopes : {} ", SsoService.getUserId(outputMap.get(Authz.InvokeKeys.PRINCIPAL_RECORD)), profile.getAuthzName(), diff --git a/backend/manager/modules/enginesso/src/main/java/org/ovirt/engine/core/sso/service/SsoService.java b/backend/manager/modules/enginesso/src/main/java/org/ovirt/engine/core/sso/service/SsoService.java index b3f1e6b458d..38e9792078f 100644 --- a/backend/manager/modules/enginesso/src/main/java/org/ovirt/engine/core/sso/service/SsoService.java +++ b/backend/manager/modules/enginesso/src/main/java/org/ovirt/engine/core/sso/service/SsoService.java @@ -35,6 +35,7 @@ import org.apache.http.impl.client.CloseableHttpClient; import org.apache.http.message.BasicNameValuePair; import org.apache.http.util.EntityUtils; +import org.ovirt.engine.api.extensions.ExtKey; import org.ovirt.engine.api.extensions.ExtMap; import org.ovirt.engine.api.extensions.aaa.Authn; import org.ovirt.engine.api.extensions.aaa.Authz; @@ -925,4 +926,34 @@ private static Set processGroupMemberships( } return membershipIds; } + + /** + * Convert principal record names from ISO_8859_1 to UTF-8 in case when the "External SSO provider" configured + * Apache (httpd) encodes all names using ISO_8859_1 but ovirt-engine tries to work with the data using UTF-8. + * This causes names (like first name, last name, e-mail address) corruption if non-ascii characters are used in + * these names. This routine converts the names to avoid the corruption. + * @param principalRecord Principal Record content to update + * @param externalSso Flag that signals if the "External SSO provider" (Keycloak) configured for the system. + * If the flag is 'false' then no any changes performed. + * @return Updated Principal Record content with fixed names (first name, last name, e-mail address) + */ + public static ExtMap fixExternalNames(ExtMap principalRecord, boolean externalSso) { + if (externalSso && principalRecord != null) { + //If the principal record came from external system, this means it was passed via the mod_auth_openidc + //plugin of the Apache service. This plugin sends all claims using the ISO_8859_1 encoding. This corrupts + //non-ascii characters in names. We need to fix the names: + fixExternalName(principalRecord, Authz.PrincipalRecord.FIRST_NAME); + fixExternalName(principalRecord, Authz.PrincipalRecord.LAST_NAME); + fixExternalName(principalRecord, Authz.PrincipalRecord.EMAIL); + } + return principalRecord; + } + + private static void fixExternalName(ExtMap principalRecord, ExtKey key) { + String value = principalRecord.get(key); + if (value != null) { + String valueFixed = new String(value.getBytes(StandardCharsets.ISO_8859_1), StandardCharsets.UTF_8); + principalRecord.put(key, valueFixed); + } + } }