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

org.jasig.cas.services.web.ServiceThemeResolver Maven / Gradle / Ivy

package org.jasig.cas.services.web;

import org.apache.commons.lang3.StringUtils;
import org.jasig.cas.authentication.principal.Service;
import org.jasig.cas.services.RegisteredService;
import org.jasig.cas.services.ServicesManager;
import org.jasig.cas.web.support.WebUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.support.ResourceBundleMessageSource;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.theme.AbstractThemeResolver;
import org.springframework.webflow.execution.RequestContext;
import org.springframework.webflow.execution.RequestContextHolder;

import javax.annotation.Resource;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.HashMap;
import java.util.Locale;
import java.util.Map;
import java.util.ResourceBundle;
import java.util.regex.Pattern;

/**
 * ThemeResolver to determine the theme for CAS based on the service provided.
 * The theme resolver will extract the service parameter from the Request object
 * and attempt to match the URL provided to a Service Id. If the service is
 * found, the theme associated with it will be used. If not, these is associated
 * with the service or the service was not found, a default theme will be used.
 *
 * @author Scott Battaglia
 * @since 3.0.0
 */
@Component("serviceThemeResolver")
public final class ServiceThemeResolver extends AbstractThemeResolver {

    private static final Logger LOGGER = LoggerFactory.getLogger(ServiceThemeResolver.class);

    /** The ServiceRegistry to look up the service. */
    @Autowired
    @Qualifier("servicesManager")
    private ServicesManager servicesManager;

    private Map overrides = new HashMap<>();

    @Override
    public String resolveThemeName(final HttpServletRequest request) {
        if (this.servicesManager == null) {
            return getDefaultThemeName();
        }
        // retrieve the user agent string from the request
        final String userAgent = request.getHeader("User-Agent");

        if (StringUtils.isBlank(userAgent)) {
            return getDefaultThemeName();
        }

        for (final Map.Entry entry : this.overrides.entrySet()) {
            if (entry.getKey().matcher(userAgent).matches()) {
                request.setAttribute("isMobile", "true");
                request.setAttribute("browserType", entry.getValue());
                break;
            }
        }

        final RequestContext context = RequestContextHolder.getRequestContext();
        final Service service = WebUtils.getService(context);
        if (service != null) {
            final RegisteredService rService = this.servicesManager.findServiceBy(service);
            if (rService != null && rService.getAccessStrategy().isServiceAccessAllowed()
                    && StringUtils.isNotBlank(rService.getTheme())) {
                LOGGER.debug("Service [{}] is configured to use a custom theme [{}]", rService, rService.getTheme());
                final CasThemeResourceBundleMessageSource messageSource = new CasThemeResourceBundleMessageSource();
                messageSource.setBasename(rService.getTheme());
                if (messageSource.doGetBundle(rService.getTheme(), request.getLocale()) != null) {
                    LOGGER.debug("Found custom theme [{}] for service [{}]", rService.getTheme(), rService);
                    return rService.getTheme();
                } else {
                    LOGGER.warn("Custom theme {} for service {} cannot be located. Falling back to default theme...",
                            rService.getTheme(), rService);
                }
            }
        }
        return getDefaultThemeName();
    }

    @Override
    public void setThemeName(final HttpServletRequest request, final HttpServletResponse response, final String themeName) {
        // nothing to do here
    }

    public void setServicesManager(final ServicesManager servicesManager) {
        this.servicesManager = servicesManager;
    }

    @Override
    @Autowired
    public void setDefaultThemeName(@Value("${cas.themeResolver.defaultThemeName:cas-theme-default}")
                                        final String defaultThemeName) {
        super.setDefaultThemeName(defaultThemeName);
    }

    /**
     * Sets the map of mobile browsers.  This sets a flag on the request called "isMobile" and also
     * provides the custom flag called browserType which can be mapped into the theme.
     * 

* Themes that understand isMobile should provide an alternative stylesheet. * * @param mobileOverrides the list of mobile browsers. */ @Resource(name="serviceThemeResolverSupportedBrowsers") public void setMobileBrowsers(final Map mobileOverrides) { // initialize the overrides variable to an empty map this.overrides = new HashMap<>(); for (final Map.Entry entry : mobileOverrides.entrySet()) { this.overrides.put(Pattern.compile(entry.getKey()), entry.getValue()); } } private static class CasThemeResourceBundleMessageSource extends ResourceBundleMessageSource { @Override protected ResourceBundle doGetBundle(final String basename, final Locale locale) { try { final ResourceBundle bundle = ResourceBundle.getBundle(basename, locale, getBundleClassLoader()); if (bundle != null && !bundle.keySet().isEmpty()) { return bundle; } } catch (final Exception e) { LOGGER.debug(e.getMessage(), e); } return null; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy