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

org.beangle.security.cas.web.CasEntryPoint Maven / Gradle / Ivy

There is a newer version: 4.0.7
Show newest version
/*
 * Beangle, Agile Java/Scala Development Scaffold and Toolkit
 *
 * Copyright (c) 2005-2012, Beangle Software.
 *
 * Beangle is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * Beangle is distributed in the hope that it will be useful.
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with Beangle.  If not, see .
 */
package org.beangle.security.cas.web;

import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.net.URLEncoder;

import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.beangle.commons.bean.Initializing;
import org.beangle.commons.lang.Assert;
import org.beangle.commons.lang.Strings;
import org.beangle.security.auth.AccountStatusException;
import org.beangle.security.cas.CasConfig;
import org.beangle.security.core.AuthenticationException;
import org.beangle.security.core.userdetail.UsernameNotFoundException;
import org.beangle.security.web.AuthenticationEntryPoint;

/**
 * Used by the ExceptionTranslationFilter to commence
 * authentication via the JA-SIG Central Authentication Service (CAS).
 * 

* The user's browser will be redirected to the JA-SIG CAS enterprise-wide login page. This page is * specified by the loginUrl property. Once login is complete, the CAS login page will * redirect to the page indicated by the service property. The service is * a HTTP URL belonging to the current application. The service URL is monitored by the * {@link CasPreauthFilter}, which will validate the CAS login was successful. * * @author chaostone * @version $Id: CasProcessingFilterEntryPoint.java $ */ public class CasEntryPoint implements AuthenticationEntryPoint, Initializing { private CasConfig config; /** 本地登录地址 */ private String localLogin; public CasEntryPoint() { super(); } public CasEntryPoint(CasConfig config) { super(); this.config = config; } public void init() throws Exception { Assert.notNull(this.config, "cas config must be specified"); } public void commence(final ServletRequest servletRequest, final ServletResponse servletResponse, final AuthenticationException ae) throws IOException, ServletException { final HttpServletRequest request = (HttpServletRequest) servletRequest; final HttpServletResponse response = (HttpServletResponse) servletResponse; if (null != ae && (ae instanceof UsernameNotFoundException || ae instanceof AccountStatusException)) { response.getWriter().append(String.valueOf(ae.getAuthentication().getPrincipal())) .append(ae.getMessage()); return; } if (null != localLogin) { // 防止在localLogin也不是公开资源的错误配置情况下,出现CasEntryPoint和CasServer之间的死循环 if (request.getServletPath().endsWith(localLogin)) { response.sendRedirect(request.getContextPath() + localLogin); } else { final String encodedServiceUrl = constructLocalLoginServiceUrl(request, response, null, CasConfig.getLocalServer(request), config.getArtifactName(), config.isEncode()); final String redirectUrl = constructRedirectUrl(config.getLoginUrl(), "service", encodedServiceUrl, config.isRenew(), false); response.sendRedirect(redirectUrl + "&isLoginService=11"); } } else { final String encodedServiceUrl = constructServiceUrl(request, response, null, CasConfig.getLocalServer(request), config.getArtifactName(), config.isEncode()); final String redirectUrl = constructRedirectUrl(config.getLoginUrl(), "service", encodedServiceUrl, config.isRenew(), false); response.sendRedirect(redirectUrl); } } /** * Construct local login Service Url * * @param request * @param response * @param service * @param serverName * @param artifactParameterName * @param encode */ public String constructLocalLoginServiceUrl(final HttpServletRequest request, final HttpServletResponse response, final String service, final String serverName, final String artifactParameterName, final boolean encode) { if (Strings.isNotBlank(service)) return encode ? response.encodeURL(service) : service; final StringBuilder buffer = new StringBuilder(); if (!serverName.startsWith("https://") && !serverName.startsWith("http://")) { buffer.append(request.isSecure() ? "https://" : "http://"); } buffer.append(serverName); buffer.append(request.getContextPath()); buffer.append(localLogin); final String returnValue = encode ? response.encodeURL(buffer.toString()) : buffer.toString(); return returnValue; } /** * Constructs a service url from the HttpServletRequest or from the given * serviceUrl. Prefers the serviceUrl provided if both a serviceUrl and a * serviceName. * * @param request the HttpServletRequest * @param response the HttpServletResponse * @param service the configured service url (this will be used if not null) * @param serverName the server name to use to constuct the service url if service param is empty * @param artifactParameterName the artifact parameter name to remove (i.e. ticket) * @param encode whether to encode the url or not (i.e. Jsession). * @return the service url to use. */ public static String constructServiceUrl(final HttpServletRequest request, final HttpServletResponse response, final String service, final String serverName, final String artifactParameterName, final boolean encode) { if (Strings.isNotBlank(service)) { return encode ? response.encodeURL(service) : service; } final StringBuilder buffer = new StringBuilder(); if (!serverName.startsWith("https://") && !serverName.startsWith("http://")) { buffer.append(request.isSecure() ? "https://" : "http://"); } buffer.append(serverName); buffer.append(request.getRequestURI()); if (Strings.isNotBlank(request.getQueryString())) { final int location = request.getQueryString().indexOf(artifactParameterName + "="); if (location == 0) { final String returnValue = encode ? response.encodeURL(buffer.toString()) : buffer.toString(); return returnValue; } buffer.append("?"); if (location == -1) { buffer.append(request.getQueryString()); } else if (location > 0) { final int actualLocation = request.getQueryString().indexOf("&" + artifactParameterName + "="); if (actualLocation == -1) buffer.append(request.getQueryString()); else if (actualLocation > 0) buffer.append(request.getQueryString().substring(0, actualLocation)); } } final String returnValue = encode ? response.encodeURL(buffer.toString()) : buffer.toString(); return returnValue; } /** * Constructs the URL to use to redirect to the CAS server. * * @param casServerLoginUrl the CAS Server login url. * @param serviceParameterName the name of the parameter that defines the service. * @param serviceUrl the actual service's url. * @param renew whether we should send renew or not. * @param gateway where we should send gateway or not. * @return the fully constructed redirect url. */ public String constructRedirectUrl(final String casServerLoginUrl, final String serviceParameterName, final String serviceUrl, final boolean renew, final boolean gateway) { try { return casServerLoginUrl + (casServerLoginUrl.indexOf("?") != -1 ? "&" : "?") + serviceParameterName + "=" + URLEncoder.encode(serviceUrl, "UTF-8") + (renew ? "&renew=true" : "") + (gateway ? "&gateway=true" : ""); } catch (final UnsupportedEncodingException e) { throw new RuntimeException(e); } } public CasConfig getConfig() { return this.config; } public void setConfig(CasConfig config) { this.config = config; } public String getLocalLogin() { return localLogin; } public void setLocalLogin(String localLogin) { this.localLogin = localLogin; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy