Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Show error message to OAuth2 user when a matching local account already exists #116

Merged
Merged
Show file tree
Hide file tree
Changes from 9 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.function.Consumer;

import org.georchestra.gateway.security.exceptions.DuplicatedEmailFoundException;
import org.georchestra.security.model.GeorchestraUser;

import lombok.NonNull;
Expand All @@ -37,7 +38,7 @@ public abstract class AbstractAccountsManager implements AccountManager {
protected final ReadWriteLock lock = new ReentrantReadWriteLock();

@Override
public GeorchestraUser getOrCreate(@NonNull GeorchestraUser mappedUser) {
public GeorchestraUser getOrCreate(@NonNull GeorchestraUser mappedUser) throws DuplicatedEmailFoundException {
return find(mappedUser).orElseGet(() -> createIfMissing(mappedUser));
}

Expand All @@ -57,7 +58,7 @@ protected Optional<GeorchestraUser> findInternal(GeorchestraUser mappedUser) {
return findByUsername(mappedUser.getUsername());
}

GeorchestraUser createIfMissing(GeorchestraUser mapped) {
GeorchestraUser createIfMissing(GeorchestraUser mapped) throws DuplicatedEmailFoundException {
lock.writeLock().lock();
try {
GeorchestraUser existing = findInternal(mapped).orElse(null);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,9 @@
import java.util.Optional;
import java.util.WeakHashMap;

import org.georchestra.ds.users.DuplicatedEmailException;
import org.georchestra.gateway.security.GeorchestraUserCustomizerExtension;
import org.georchestra.gateway.security.exceptions.DuplicatedEmailFoundException;
import org.georchestra.security.model.GeorchestraUser;
import org.springframework.core.Ordered;
import org.springframework.security.core.Authentication;
Expand Down Expand Up @@ -61,7 +63,8 @@ public class CreateAccountUserCustomizer implements GeorchestraUserCustomizerExt
* otherwise.
*/
@Override
public @NonNull GeorchestraUser apply(@NonNull Authentication auth, @NonNull GeorchestraUser mappedUser) {
public @NonNull GeorchestraUser apply(@NonNull Authentication auth, @NonNull GeorchestraUser mappedUser)
throws DuplicatedEmailFoundException {
final boolean isOauth2 = auth instanceof OAuth2AuthenticationToken;
final boolean isPreAuth = auth instanceof PreAuthenticatedAuthenticationToken;
if (isOauth2) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -36,8 +36,10 @@
import org.georchestra.ds.users.AccountFactory;
import org.georchestra.ds.users.DuplicatedEmailException;
import org.georchestra.ds.users.DuplicatedUidException;
import org.georchestra.gateway.accounts.admin.AbstractAccountsManager;;
import org.georchestra.gateway.accounts.admin.AbstractAccountsManager;
import org.georchestra.gateway.accounts.admin.AccountManager;
import org.georchestra.gateway.security.exceptions.DuplicatedEmailFoundException;
import org.georchestra.gateway.security.exceptions.DuplicatedUsernameFoundException;
import org.georchestra.security.api.UsersApi;
import org.georchestra.security.model.GeorchestraUser;
import org.springframework.beans.factory.annotation.Value;
Expand Down Expand Up @@ -89,12 +91,16 @@ private GeorchestraUser ensureRolesPrefixed(GeorchestraUser user) {
}

@Override
protected void createInternal(GeorchestraUser mapped) {
protected void createInternal(GeorchestraUser mapped) throws DuplicatedEmailFoundException {
Account newAccount = mapToAccountBrief(mapped);
try {
accountDao.insert(newAccount);
} catch (DataServiceException | DuplicatedUidException | DuplicatedEmailException accountError) {
} catch (DataServiceException accountError) {
throw new IllegalStateException(accountError);
} catch (DuplicatedEmailException accountError) {
throw new DuplicatedEmailFoundException(accountError.getMessage());
} catch (DuplicatedUidException accountError) {
throw new DuplicatedUsernameFoundException(accountError.getMessage());
}

ensureOrgExists(newAccount);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@
import javax.annotation.PostConstruct;

import org.georchestra.gateway.security.GeorchestraUserMapper;
import org.georchestra.gateway.security.exceptions.DuplicatedEmailFoundException;
import org.georchestra.gateway.security.ldap.LdapConfigProperties;
import org.georchestra.security.model.GeorchestraUser;
import org.springframework.beans.factory.annotation.Autowired;
Expand Down Expand Up @@ -87,7 +88,12 @@ void initialize() {
@GetMapping(path = "/whoami", produces = "application/json")
@ResponseBody
public Mono<Map<String, Object>> whoami(Authentication principal, ServerWebExchange exchange) {
GeorchestraUser user = Optional.ofNullable(principal).flatMap(userMapper::resolve).orElse(null);
GeorchestraUser user = null;
try {
user = Optional.ofNullable(principal).flatMap(userMapper::resolve).orElse(null);
} catch (DuplicatedEmailFoundException e) {
}

Map<String, Object> ret = new LinkedHashMap<>();
ret.put("GeorchestraUser", user);
if (principal == null) {
Expand All @@ -96,8 +102,6 @@ public Mono<Map<String, Object>> whoami(Authentication principal, ServerWebExcha
ret.put(principal.getClass().getCanonicalName(), principal);
}
return Mono.just(ret);
// return principal == null ? Mono.empty() :
// Mono.just(Map.of(principal.getClass().getCanonicalName(), principal));
}

@GetMapping(path = "/logout")
Expand All @@ -123,6 +127,8 @@ public String loginPage(@RequestParam Map<String, String> allRequestParams, Mode
mdl.addAttribute("passwordExpired", expired);
boolean invalidCredentials = "invalid_credentials".equals(allRequestParams.get("error"));
mdl.addAttribute("invalidCredentials", invalidCredentials);
boolean duplicateAccount = "duplicate_account".equals(allRequestParams.get("error"));
mdl.addAttribute("duplicateAccount", duplicateAccount);
return "login";
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,9 @@
import java.util.List;
import java.util.Optional;

import org.georchestra.ds.users.DuplicatedEmailException;
import org.georchestra.gateway.model.GeorchestraUsers;
import org.georchestra.gateway.security.exceptions.DuplicatedEmailFoundException;
import org.georchestra.security.model.GeorchestraUser;
import org.springframework.core.Ordered;
import org.springframework.security.core.Authentication;
Expand Down Expand Up @@ -77,15 +79,16 @@ public class GeorchestraUserMapper {
* {@link Optional#empty()} if no extension point implementation can
* handle the auth token.
*/
public Optional<GeorchestraUser> resolve(@NonNull Authentication authToken) {
public Optional<GeorchestraUser> resolve(@NonNull Authentication authToken) throws DuplicatedEmailFoundException {
return resolvers.stream()//
.map(resolver -> resolver.resolve(authToken))//
.filter(Optional::isPresent)//
.map(Optional::orElseThrow)//
.map(mapped -> customize(authToken, mapped)).findFirst();
}

private GeorchestraUser customize(@NonNull Authentication authToken, GeorchestraUser mapped) {
private GeorchestraUser customize(@NonNull Authentication authToken, GeorchestraUser mapped)
throws DuplicatedEmailFoundException {
GeorchestraUser customized = mapped;
for (GeorchestraUserCustomizerExtension customizer : customizers) {
customized = customizer.apply(authToken, customized);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,25 +18,36 @@
*/
package org.georchestra.gateway.security;

import org.apache.commons.lang3.tuple.Pair;
import org.georchestra.gateway.model.GeorchestraOrganizations;
import org.georchestra.gateway.model.GeorchestraTargetConfig;
import org.georchestra.gateway.model.GeorchestraUsers;
import org.georchestra.gateway.security.exceptions.DuplicatedEmailFoundException;
import org.georchestra.gateway.security.ldap.extended.ExtendedGeorchestraUser;
import org.georchestra.security.model.GeorchestraUser;
import org.georchestra.security.model.Organization;
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
import org.georchestra.security.model.Organization;
import org.springframework.http.HttpStatus;
import org.springframework.http.server.reactive.ServerHttpResponse;
import org.springframework.cloud.gateway.filter.GlobalFilter;
import org.springframework.cloud.gateway.filter.RouteToRequestUrlFilter;
import org.springframework.cloud.gateway.route.Route;
import org.springframework.core.Ordered;
import org.springframework.security.core.Authentication;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.security.web.server.DefaultServerRedirectStrategy;
import org.springframework.security.web.server.ServerRedirectStrategy;
import org.springframework.web.server.ServerWebExchange;
import org.springframework.web.server.WebSession;

import lombok.NonNull;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import reactor.core.publisher.Mono;

import java.net.URI;
import java.util.Optional;

/**
* A {@link GlobalFilter} that resolves the {@link GeorchestraUser} from the
* request's {@link Authentication} so it can be {@link GeorchestraUsers#resolve
Expand All @@ -56,6 +67,10 @@ public class ResolveGeorchestraUserGlobalFilter implements GlobalFilter, Ordered

private final @NonNull GeorchestraUserMapper resolver;

private ServerRedirectStrategy redirectStrategy = new DefaultServerRedirectStrategy();

private static String DUPLICATE_ACCOUNT = "duplicate_account";

/**
* @return a lower precedence than {@link RouteToRequestUrlFilter}'s, in order
* to make sure the matched {@link Route} has been set as a
Expand All @@ -72,14 +87,28 @@ public class ResolveGeorchestraUserGlobalFilter implements GlobalFilter, Ordered
* chain.
*/
public @Override Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {

return exchange.getPrincipal()//
Mono<Void> res = exchange.getPrincipal()//
.doOnNext(p -> log.debug("resolving user from {}", p.getClass().getName()))//
.filter(Authentication.class::isInstance)//
.map(Authentication.class::cast)//
.map(resolver::resolve)//
.map(auth -> {
try {
return Pair.of(resolver.resolve(auth), "");
} catch (DuplicatedEmailFoundException exp) {
Optional<GeorchestraUser> op = Optional.empty();
return Pair.of(op, "/login?error=" + DUPLICATE_ACCOUNT);
}
})//
.map(user -> {
GeorchestraUser usr = user.orElse(null);
if (!user.getRight().isEmpty()) {
SecurityContextHolder.getContext();
ServerHttpResponse response = exchange.getResponse();
response.setStatusCode(HttpStatus.FOUND);
response.getHeaders().setLocation(URI.create(user.getRight()));
return exchange;
}

GeorchestraUser usr = user.getLeft().orElse(null);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've tried to better understand Mono and ended up with something like (not fully working) :

.map(resolver::resolve)//
                .map(user -> {
                    GeorchestraUser usr = user.orElse(null);
                    GeorchestraUsers.store(exchange, usr);
                    if (usr != null && usr instanceof ExtendedGeorchestraUser) {
                        ExtendedGeorchestraUser eu = (ExtendedGeorchestraUser) usr;
                        Organization org = eu.getOrg();
                        GeorchestraOrganizations.store(exchange, org);
                    }
                    return exchange;
                })//
                .doOnError(DuplicatedEmailFoundException.class, exp -> {
                    ServerHttpResponse response = exchange.getResponse();
                    response.setStatusCode(HttpStatus.FOUND);
                    response.getHeaders().setLocation(URI.create("/login?error=" + DUPLICATE_ACCOUNT));
                    exchange.getSession().flatMap(WebSession::invalidate);
                })

I think we can do something way better. I couldn't succeed to handle status code in error.

Copy link
Contributor Author

@emmdurin emmdurin Apr 11, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes I think we can do something this way, but still have to find the correct syntax. Your proposal run additional code if an exception is thrown but doesn't cancel the exception which is thrown anyway at the end of the Mono chain. In fact it's like a try-finally but we need instead something like a try-catch.

GeorchestraUsers.store(exchange, usr);
if (usr != null && usr instanceof ExtendedGeorchestraUser) {
ExtendedGeorchestraUser eu = (ExtendedGeorchestraUser) usr;
Expand All @@ -89,7 +118,11 @@ public class ResolveGeorchestraUserGlobalFilter implements GlobalFilter, Ordered
return exchange;
})//
.defaultIfEmpty(exchange)//
.flatMap(chain::filter);
.flatMap(e -> e.getResponse().getStatusCode() != HttpStatus.FOUND ? chain.filter(e)
: Mono.fromRunnable(() -> exchange.getSession().flatMap(WebSession::invalidate)));

System.out.println(res);
return res;
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
package org.georchestra.gateway.security.exceptions;

public class DuplicatedEmailFoundException extends RuntimeException {
private String message;

public DuplicatedEmailFoundException(String message) {
super(message);
this.message = message;
}

public DuplicatedEmailFoundException() {
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
package org.georchestra.gateway.security.exceptions;

public class DuplicatedUsernameFoundException extends RuntimeException {
private String message;

public DuplicatedUsernameFoundException(String message) {
super(message);
this.message = message;
}

public DuplicatedUsernameFoundException() {
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -38,16 +38,16 @@

import reactor.core.publisher.Mono;

class PreauthAuthenticationManager implements ReactiveAuthenticationManager, ServerAuthenticationConverter {
public class PreauthAuthenticationManager implements ReactiveAuthenticationManager, ServerAuthenticationConverter {

static final String PREAUTH_HEADER_NAME = "sec-georchestra-preauthenticated";
public static final String PREAUTH_HEADER_NAME = "sec-georchestra-preauthenticated";

private static final String PREAUTH_USERNAME = "preauth-username";
private static final String PREAUTH_EMAIL = "preauth-email";
private static final String PREAUTH_FIRSTNAME = "preauth-firstname";
private static final String PREAUTH_LASTNAME = "preauth-lastname";
private static final String PREAUTH_ORG = "preauth-org";
private static final String PREAUTH_ROLES = "preauth-roles";
public static final String PREAUTH_USERNAME = "preauth-username";
public static final String PREAUTH_EMAIL = "preauth-email";
public static final String PREAUTH_FIRSTNAME = "preauth-firstname";
public static final String PREAUTH_LASTNAME = "preauth-lastname";
public static final String PREAUTH_ORG = "preauth-org";
public static final String PREAUTH_ROLES = "preauth-roles";

/**
* @return {@code Mono.empty()} if the pre-auth request headers are not
Expand Down
3 changes: 2 additions & 1 deletion gateway/src/main/resources/messages/login.properties
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,5 @@ forget_password = Forgot password ?
identity_provider_title = Log in with an identity provider
expired_password = Your password has been expired
expired_password_link = and should be changed
invalid_credentials = Invalid username or password
invalid_credentials = Invalid username or password
duplicate_account = An account already exists using this email address
3 changes: 2 additions & 1 deletion gateway/src/main/resources/messages/login_de.properties
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,5 @@ forget_password = Passwort vergessen?
identity_provider_title = Melden Sie sich bei einem Identitätsanbieter an
expired_password = Ihr Passwort ist abgelaufen
expired_password_link = und sollte geändert werden
invalid_credentials = Ungültiger Benutzername oder Passwort
invalid_credentials = Ungültiger Benutzername oder Passwort
duplicate_account = Es existiert bereits ein Konto mit dieser E-Mail-Adresse
3 changes: 2 additions & 1 deletion gateway/src/main/resources/messages/login_en.properties
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,5 @@ forget_password = Forgot password ?
identity_provider_title = Log in with an identity provider
expired_password = Your password has been expired
expired_password_link = and should be changed
invalid_credentials = Invalid username or password
invalid_credentials = Invalid username or password
duplicate_account = An account already exists using this email address
3 changes: 2 additions & 1 deletion gateway/src/main/resources/messages/login_es.properties
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,5 @@ forget_password = Contraseña olvidada ?
identity_provider_title = Iniciar sesión con un proveedor de identidad
expired_password = Su contraseña ha caducado
expired_password_link = y debería ser cambiado
invalid_credentials = Nombre de usuario o contraseña invalido
invalid_credentials = Nombre de usuario o contraseña invalido
duplicate_account = Ya existe una cuenta usando esta dirección de correo electrónico
3 changes: 2 additions & 1 deletion gateway/src/main/resources/messages/login_fr.properties
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,5 @@ forget_password = Mot de passe oublié ?
identity_provider_title = Se connecter depuis un fournisseur d'identité
expired_password = Votre mot de passe a expiré
expired_password_link = et doit être changé
invalid_credentials = Nom d'utilisateur ou mot de passe non valide
invalid_credentials = Nom d'utilisateur ou mot de passe non valide
duplicate_account = Il existe déjà un compte utilisant cette adresse e-mail
3 changes: 2 additions & 1 deletion gateway/src/main/resources/messages/login_nl.properties
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,5 @@ forget_password = Wachtwoord vergeten ?
identity_provider_title = Log in met een identiteitsprovider
expired_password = Uw wachtwoord is verlopen
expired_password_link = en moet worden veranderd
invalid_credentials = ongeldige gebruikersnaam of wachtwoord
invalid_credentials = ongeldige gebruikersnaam of wachtwoord
duplicate_account = Er bestaat al een account met dit e-mailadres
3 changes: 2 additions & 1 deletion gateway/src/main/resources/messages/login_ru.properties
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,5 @@ forget_password = Забыли пароль ?
identity_provider_title = Войдите в систему с помощью поставщика удостоверений
expired_password = Срок действия вашего пароля истек,
expired_password_link = и следует изменить
invalid_credentials = неправильное имя пользователя или пароль
invalid_credentials = неправильное имя пользователя или пароль
duplicate_account = Учетная запись уже существует с использованием этого адреса электронной почты
1 change: 1 addition & 0 deletions gateway/src/main/resources/templates/login.html
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ <h2 class="form-signin-heading"><span th:text="#{login_message_title}"></span></
<div style="text-align: center; font-size: 18px; color: #ff0033;" th:if="${passwordExpired}"> <span th:text="#{expired_password}" ></span>
<a href="/console/account/passwordRecovery" > <span th:text="#{expired_password_link}" ></span> </a>
</div>
<div style="text-align: center; font-size: 18px; color: #ff0033;" th:if="${duplicateAccount}"> <span th:text="#{duplicate_account}"></span> </div>
</div>
</form>
<div th:if="${oauth2LoginLinks.size() != 0}" class="container"><h2 class="form-signin-heading">Login with OAuth 2.0</h2>
Expand Down
Loading
Loading