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

org.omnifaces.eleos.services.BaseAuthenticationService Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2022, 2022 OmniFish and/or its affiliates. All rights reserved.
 * Copyright (c) 1997, 2018 Oracle and/or its affiliates. All rights reserved.
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License v. 2.0, which is available at
 * http://www.eclipse.org/legal/epl-2.0.
 *
 * This Source Code may also be made available under the following Secondary
 * Licenses when the conditions for such availability set forth in the
 * Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
 * version 2 with the GNU Classpath Exception, which is available at
 * https://www.gnu.org/software/classpath/license.html.
 *
 * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
 */

package org.omnifaces.eleos.services;


import static jakarta.security.auth.message.AuthStatus.SUCCESS;
import static java.lang.Boolean.TRUE;
import static org.omnifaces.eleos.config.helper.HttpServletConstants.IS_MANDATORY;
import static org.omnifaces.eleos.config.helper.HttpServletConstants.REGISTER_SESSION;

import java.io.IOException;
import java.util.Map;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;

import javax.security.auth.Subject;
import javax.security.auth.callback.Callback;
import javax.security.auth.callback.CallbackHandler;
import javax.security.auth.callback.UnsupportedCallbackException;

import org.omnifaces.eleos.config.helper.AuthMessagePolicy;
import org.omnifaces.eleos.config.helper.Caller;
import org.omnifaces.eleos.config.servlet.HttpMessageInfo;

import jakarta.security.auth.message.AuthException;
import jakarta.security.auth.message.MessageInfo;
import jakarta.security.auth.message.callback.PasswordValidationCallback;
import jakarta.security.auth.message.config.AuthConfig;
import jakarta.security.auth.message.config.AuthConfigFactory;
import jakarta.security.auth.message.config.AuthConfigFactory.RegistrationContext;
import jakarta.security.auth.message.config.AuthConfigProvider;
import jakarta.security.auth.message.config.ClientAuthConfig;
import jakarta.security.auth.message.config.ClientAuthContext;
import jakarta.security.auth.message.config.ServerAuthConfig;
import jakarta.security.auth.message.config.ServerAuthContext;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;

/**
 */
public class BaseAuthenticationService {

    protected static final AuthConfigFactory authConfigFactory = AuthConfigFactory.getFactory();

    private static final String MESSAGE_INFO = BaseAuthenticationService.class.getName() + ".message.info";

    private ReadWriteLock readWriteLock;
    private Lock readLock;
    private Lock writeLock;

    protected String messageLayer;
    protected String appContextId;
    protected Map map;
    protected CallbackHandler callbackHandler;
    protected AuthConfigRegistrationWrapper listenerWrapper;

    protected void init(String messageLayer, String appContextId, Map properties, CallbackHandler callbackHandler, RegistrationWrapperRemover removerDelegate) {
        this.messageLayer = messageLayer;
        this.appContextId = appContextId;
        this.map = properties;
        this.callbackHandler = callbackHandler;
        if (this.callbackHandler == null) {
            this.callbackHandler = getCallbackHandler();
        }

        this.readWriteLock = new ReentrantReadWriteLock(true);
        this.readLock = readWriteLock.readLock();
        this.writeLock = readWriteLock.writeLock();

        listenerWrapper = new AuthConfigRegistrationWrapper(this.messageLayer, this.appContextId, removerDelegate);
    }

    public void setRegistrationId(String registrationId) {
        listenerWrapper.setRegistrationId(registrationId);
    }

    public AuthConfigRegistrationWrapper getRegistrationWrapper() {
        return listenerWrapper;
    }

    public void setRegistrationWrapper(AuthConfigRegistrationWrapper wrapper) {
        this.listenerWrapper = wrapper;
    }

    public AuthConfigRegistrationWrapper.AuthConfigRegistrationListener getRegistrationListener() {
        return listenerWrapper.getListener();
    }

    public void disable() {
        listenerWrapper.disable();
    }

    public Object getProperty(String key) {
        return map == null ? null : map.get(key);
    }

    public String getAppContextID() {
        return appContextId;
    }

    public ClientAuthConfig getClientAuthConfig() throws AuthException {
        return (ClientAuthConfig) getAuthConfig(false);
    }

    public ServerAuthConfig getServerAuthConfig() throws AuthException {
        return (ServerAuthConfig) getAuthConfig(true);
    }

    public ClientAuthContext getClientAuthContext(MessageInfo info, Subject clientSubject) throws AuthException {
        ClientAuthConfig clientConfig = (ClientAuthConfig) getAuthConfig(false);
        if (clientConfig != null) {
            return clientConfig.getAuthContext(clientConfig.getAuthContextID(info), clientSubject, map);
        }

        return null;
    }

    public ServerAuthContext getServerAuthContext(MessageInfo info) throws AuthException {
        return getServerAuthContext(info, null);
    }

    public ServerAuthContext getServerAuthContext(MessageInfo info, Subject serviceSubject) throws AuthException {
        ServerAuthConfig serverAuthConfig = (ServerAuthConfig) getAuthConfig(true);
        if (serverAuthConfig != null) {
            return serverAuthConfig.getAuthContext(serverAuthConfig.getAuthContextID(info), serviceSubject, map);
        }

        return null;
    }

    protected AuthConfig getAuthConfig(AuthConfigProvider authConfigProvider, boolean isServer) throws AuthException {
        AuthConfig authConfig = null;

        if (authConfigProvider != null) {
            if (isServer) {
                authConfig = authConfigProvider.getServerAuthConfig(messageLayer, appContextId, callbackHandler);
            } else {
                authConfig = authConfigProvider.getClientAuthConfig(messageLayer, appContextId, callbackHandler);
            }
        }

        return authConfig;
    }

    protected AuthConfig getAuthConfig(boolean isServer) throws AuthException {
        ConfigData configData = null;
        AuthConfig authConfig = null;
        boolean disabled = false;
        AuthConfigProvider lastConfigProvider = null;

        try {
            readLock.lock();
            disabled = !listenerWrapper.isEnabled();
            if (!disabled) {
                configData = listenerWrapper.getConfigData();
                if (configData != null) {
                    authConfig = isServer ? configData.getServerConfig() : configData.getClientConfig();
                    lastConfigProvider = configData.getProvider();
                }
            }

        } finally {
            readLock.unlock();
            if (disabled || authConfig != null || (configData != null && lastConfigProvider == null)) {
                return authConfig;
            }
        }

        // d == null || (d != null && lastP != null && c == null)
        if (configData == null) {
            try {
                writeLock.lock();
                if (listenerWrapper.getConfigData() == null) {
                    AuthConfigProvider nextConfigProvider = authConfigFactory.getConfigProvider(messageLayer, appContextId, getRegistrationListener());

                    if (nextConfigProvider != null) {
                        listenerWrapper.setConfigData(new ConfigData(nextConfigProvider, getAuthConfig(nextConfigProvider, isServer)));
                    } else {
                        listenerWrapper.setConfigData(new ConfigData());
                    }
                }
                configData = listenerWrapper.getConfigData();
            } finally {
                writeLock.unlock();
            }
        }

        return isServer ? configData.getServerConfig() : configData.getClientConfig();
    }

    /**
     * Check if there is a provider register for a given layer and appContextID.
     */
    protected boolean hasExactMatchAuthProvider() {
        boolean exactMatch = false;

        AuthConfigProvider configProvider = authConfigFactory.getConfigProvider(messageLayer, appContextId, null);

        if (configProvider != null) {
            for (String registrationId : authConfigFactory.getRegistrationIDs(configProvider)) {
                RegistrationContext registrationContext = authConfigFactory.getRegistrationContext(registrationId);
                if (messageLayer.equals(registrationContext.getMessageLayer()) && appContextId.equals(registrationContext.getAppContext())) {
                    exactMatch = true;
                    break;
                }
            }
        }

        return exactMatch;
    }

    /**
     * Get the callback default handler
     */
    protected CallbackHandler getCallbackHandler() {
       return AuthMessagePolicy.getDefaultCallbackHandler();
    }

    public Caller login(String username, String password) {
        Subject subject = new Subject();

        try {
            PasswordValidationCallback passwordValidation =
                new PasswordValidationCallback(subject,
                    username,
                    password == null? null : password.toCharArray());

            callbackHandler.handle(new Callback[] { passwordValidation });

            if (passwordValidation.getResult()) {
                return Caller.fromSubject(subject);
            }

            return null;

        } catch (UnsupportedCallbackException | RuntimeException | IOException e) {
            throw new IllegalStateException(e);
        }
    }

    public Caller validateRequest(HttpServletRequest servletRequest, HttpServletResponse servletResponse, boolean calledFromAuthenticate, boolean isMandatory) throws IOException {
        Subject subject = new Subject();
        MessageInfo messageInfo = getMessageInfo(servletRequest, servletResponse);

        try {
            if (isMandatory || calledFromAuthenticate) {
                setMandatory(messageInfo);
            }

            if (!SUCCESS.equals(getServerAuthContext(messageInfo).validateRequest(messageInfo, subject, null))) {
                return null;
            }

            return Caller.fromSubject(subject);

        } catch (AuthException | RuntimeException e) {
            throw new IllegalStateException(e);
        }
    }

    public boolean mustRegisterSession(HttpServletRequest servletRequest, HttpServletResponse servletResponse) {
        return isRegisterSession(getMessageInfo(servletRequest, servletResponse));
    }

    public HttpServletRequest getWrappedRequestIfSet(HttpServletRequest servletRequest, HttpServletResponse servletResponse) {
        return (HttpServletRequest) getMessageInfo(servletRequest, servletResponse).getRequestMessage();
    }

    public HttpServletResponse getWrappedResponseIfSet(HttpServletRequest servletRequest, HttpServletResponse servletResponse) {
        return (HttpServletResponse) getMessageInfo(servletRequest, servletResponse).getResponseMessage();
    }

    public void secureResponse(HttpServletRequest servletRequest, HttpServletResponse servletResponse) {
        MessageInfo messageInfo = getMessageInfo(servletRequest, servletResponse);

        try {
            getServerAuthContext(messageInfo).secureResponse(messageInfo, null);
        } catch (AuthException e) {
            throw new IllegalStateException(e);
        }
    }

    public void clearSubject(HttpServletRequest servletRequest, HttpServletResponse servletResponse, Subject subject) {
        MessageInfo messageInfo = getMessageInfo(servletRequest, servletResponse);

        try {
            getServerAuthContext(messageInfo).cleanSubject(messageInfo, subject);
        } catch (AuthException e) {
            throw new IllegalStateException(e);
        }
    }




    // ### Private methods

    private boolean isRegisterSession(MessageInfo messageInfo) {
        return Boolean.valueOf((String) messageInfo.getMap().get(REGISTER_SESSION));
    }

    @SuppressWarnings("unchecked")
    private void setMandatory(MessageInfo messageInfo) {
        messageInfo.getMap().put(IS_MANDATORY, TRUE.toString());
    }

    private MessageInfo getMessageInfo(HttpServletRequest servletRequest, HttpServletResponse servletResponse) {
        MessageInfo messageInfo = (MessageInfo) servletRequest.getAttribute(MESSAGE_INFO);
        if (messageInfo == null) {
            messageInfo = new HttpMessageInfo(servletRequest, servletResponse);

            saveMessageInfo(servletRequest, messageInfo);
        }

        return messageInfo;
    }

    private void saveMessageInfo(HttpServletRequest servletRequest, MessageInfo messageInfo) {
        servletRequest.setAttribute(MESSAGE_INFO, messageInfo);
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy