cas-client单点登录客户端拦截请求和忽略/排除不需要拦截的请求URL的问题

cas-client单点登录客户端拦截请求和忽略/排除不需要拦截的请求URL的问题

2017年06月22日 23:43:59 阅读数:6319 标签: cascas-client单点登录排除请求URL 更多

个人分类: cas

所属专栏: 颠覆javaEE开发

版权声明:eguid新博客地址:https://blog.eguid.cc;温馨提示:本博客所有原创文章均采用知识共享署名-相同方式共享 3.0 中国大陆许可协议进行许可。如有转载请注明出处和作者名! https://blog.csdn.net/eguid_1/article/details/73611781

前言:今天在网上无意间看到cas单点登录排除请求的问题,发现很多人在讨论如何通过改写AuthenticationFilter类来实现忽略/排除请求URL的功能;突发奇想搜了一下,还真蛮多人都是这么干的,原谅我是个耿直的boy,当时我笑的饭都喷出来了,只需要一个配置的问题,被你们搞的这么麻烦;虽然很想回复他们“你们这帮人用别人的东西都不看源码的吗?”,转念一想,这也要怪作者不给力,文档里压根没有提到这个配置,在这里用少量篇幅讲解如何配置排除不需要拦截的请求URL,后面用大量篇幅介绍我是如何从源码中得知这个配置的,希望对大家有用!做好自己!--eguid始终坚持原创的开源技术文章分享,博客园与本博客保持同步更新。欢迎大家加群一起交流:608423839

1、cas-client单点登录配置

http://blog.csdn.net/eguid_1/article/details/51278622,cas-client完整配置。

没有实现忽略/排除请求URL的cas-client登录验证过滤器

 

 
  1. <filter>

  2. <filter-name>casAuthenticationFilter</filter-name>

  3. <filter-class>org.jasig.cas.client.authentication.AuthenticationFilter</filter-class>

  4. <init-param>

  5. <param-name>casServerLoginUrl</param-name>

  6. <param-value>https://cas.eguid.cc/cas-server/</param-value>

  7. </init-param>

  8. <init-param>

  9. <param-name>serverName</param-name>

  10. <param-value>http://client.eguid.cc/</param-value>

  11. </init-param>

  12. </filter>

  13. <filter-mapping>

  14. <filter-name>casAuthenticationFilter</filter-name>

  15. <url-pattern>/*</url-pattern>

  16. </filter-mapping>


这个配置依然是可用的,当然我们要实现忽略/排除请求URL的功能,那么我们该怎么做呢?

 

2、忽略/排除多个请求URL

 
  1. <filter>

  2. <filter-name>casAuthenticationFilter</filter-name>

  3. <filter-class>org.jasig.cas.client.authentication.AuthenticationFilter</filter-class>

  4. <init-param>

  5. <param-name>casServerLoginUrl</param-name>

  6. <param-value>http://cas.eguid.cc/cas-server/</param-value>

  7. </init-param>

  8. <init-param>

  9. <param-name>serverName</param-name>

  10. <param-value>http://cilent.eguid.cc/</param-value>

  11. </init-param>

  12. <init-param>

  13. <description>不拦截的请求(做好自己!--eguid)</description>

  14. <param-name>ignorePattern</param-name>

  15. <param-value>/js/*|/img/*|/view/*|/css/*</param-value>

  16. </init-param>

  17. </filter>

  18. <filter-mapping>

  19. <filter-name>casAuthenticationFilter</filter-name>

  20. <url-pattern>/*</url-pattern>

  21. </filter-mapping>




如上所见,我们排除了四个请求URL(必须是正则表达式形式,下面会讲为什么要这么配置)

 

3、cas-client默认登录验证过滤器源码解析

看源码,一定要带着目的去看;我们的目的就是找AuthenticationFilter这个cas-client默认登录验证过滤器是否具有排除登录请求URL的功能。

(1)打开cas-client项目源码

打开github上的cas-client项目,可以把项目导到本地或者直接在github上找到org.jasig.cas.client.authentication.AuthenticationFilter.java这个类。

(2)登录验证过滤器AuthenticationFilter的doFilter

既然是个过滤器,就直接找到该类的doFilter方法

 

 
  1. public final void doFilter(final ServletRequest servletRequest, final ServletResponse servletResponse,

  2. final FilterChain filterChain) throws IOException, ServletException {

  3. <!--做好自己!eguid原创-->

  4. final HttpServletRequest request = (HttpServletRequest) servletRequest;

  5. final HttpServletResponse response = (HttpServletResponse) servletResponse;

  6.  
  7. if (isRequestUrlExcluded(request)) {

  8. logger.debug("Request is ignored.");

  9. filterChain.doFilter(request, response);

  10. return;

  11. }

  12.  
  13. final HttpSession session = request.getSession(false);

  14. final Assertion assertion = session != null ? (Assertion) session.getAttribute(CONST_CAS_ASSERTION) : null;

  15.  
  16. if (assertion != null) {

  17. filterChain.doFilter(request, response);

  18. return;

  19. }

  20.  
  21. final String serviceUrl = constructServiceUrl(request, response);

  22. final String ticket = retrieveTicketFromRequest(request);

  23. final boolean wasGatewayed = this.gateway && this.gatewayStorage.hasGatewayedAlready(request, serviceUrl);

  24.  
  25. if (CommonUtils.isNotBlank(ticket) || wasGatewayed) {

  26. filterChain.doFilter(request, response);

  27. return;

  28. }

  29.  
  30. final String modifiedServiceUrl;

  31.  
  32. logger.debug("no ticket and no assertion found");

  33. if (this.gateway) {

  34. logger.debug("setting gateway attribute in session");

  35. modifiedServiceUrl = this.gatewayStorage.storeGatewayInformation(request, serviceUrl);

  36. } else {

  37. modifiedServiceUrl = serviceUrl;

  38. }

  39.  
  40. logger.debug("Constructed service url: {}", modifiedServiceUrl);

  41.  
  42. final String urlToRedirectTo = CommonUtils.constructRedirectUrl(this.casServerLoginUrl,

  43. getProtocol().getServiceParameterName(), modifiedServiceUrl, this.renew, this.gateway);

  44.  
  45. logger.debug("redirecting to \"{}\"", urlToRedirectTo);

  46. this.authenticationRedirectStrategy.redirect(request, response, urlToRedirectTo);

  47. }

 

(3)isRequestUrlExcluded方法

 

第一眼就看到了上面代码红色标识处的isRequestUrlExcluded,这个意思很直白,判断是不是需要忽略/排除的请求URL。

继续接着找到isRequestUrlExcluded这个方法的实现代码:

 

 
  1. private boolean isRequestUrlExcluded(final HttpServletRequest request) {

  2. if (this.ignoreUrlPatternMatcherStrategyClass == null) {

  3. return false;

  4. }

  5. <!--做好自己!eguid原创-->

  6. final StringBuffer urlBuffer = request.getRequestURL();

  7. if (request.getQueryString() != null) {

  8. urlBuffer.append("?").append(request.getQueryString());

  9. }

  10. final String requestUri = urlBuffer.toString();

  11. return this.ignoreUrlPatternMatcherStrategyClass.matches(requestUri);

  12. }

看红色标识位置的名字,这里用到了UrlPatternMatcherStrategy这个类,意思很简单直白:‘请求url的匹配策略类’,暂时还不知道这里是正则匹配,往后看:

 

(4)请求URL的匹配策略类UrlPatternMatcherStrategy

 

private UrlPatternMatcherStrategy ignoreUrlPatternMatcherStrategyClass = null;

发现该类是在初始化方法中进行初始化的:

 

 

 
  1. protected void initInternal(final FilterConfig filterConfig) throws ServletException {

  2. if (!isIgnoreInitConfiguration()) {

  3. super.initInternal(filterConfig);

  4. setCasServerLoginUrl(getString(ConfigurationKeys.CAS_SERVER_LOGIN_URL));

  5. setRenew(getBoolean(ConfigurationKeys.RENEW));

  6. setGateway(getBoolean(ConfigurationKeys.GATEWAY));

  7. <!--做好自己!eguid原创-->

  8. final String ignorePattern = getString(ConfigurationKeys.IGNORE_PATTERN);

  9. final String ignoreUrlPatternType = getString(ConfigurationKeys.IGNORE_URL_PATTERN_TYPE);

  10.  
  11. if (ignorePattern != null) {

  12. final Class<? extends UrlPatternMatcherStrategy> ignoreUrlMatcherClass = PATTERN_MATCHER_TYPES.get(ignoreUrlPatternType);

  13. if (ignoreUrlMatcherClass != null) {

  14. this.ignoreUrlPatternMatcherStrategyClass = ReflectUtils.newInstance(ignoreUrlMatcherClass.getName());

  15. } else {

  16. try {

  17. logger.trace("Assuming {} is a qualified class name...", ignoreUrlPatternType);

  18. this.ignoreUrlPatternMatcherStrategyClass = ReflectUtils.newInstance(ignoreUrlPatternType);

  19. } catch (final IllegalArgumentException e) {

  20. logger.error("Could not instantiate class [{}]", ignoreUrlPatternType, e);

  21. }

  22. }

  23. if (this.ignoreUrlPatternMatcherStrategyClass != null) {

  24. this.ignoreUrlPatternMatcherStrategyClass.setPattern(ignorePattern);

  25. }

  26. }

  27.  
  28. final Class<? extends GatewayResolver> gatewayStorageClass = getClass(ConfigurationKeys.GATEWAY_STORAGE_CLASS);

  29.  
  30. if (gatewayStorageClass != null) {

  31. setGatewayStorage(ReflectUtils.newInstance(gatewayStorageClass));

  32. }

  33.  
  34. final Class<? extends AuthenticationRedirectStrategy> authenticationRedirectStrategyClass = getClass(ConfigurationKeys.AUTHENTICATION_REDIRECT_STRATEGY_CLASS);

  35.  
  36. if (authenticationRedirectStrategyClass != null) {

  37. this.authenticationRedirectStrategy = ReflectUtils.newInstance(authenticationRedirectStrategyClass);

  38. }

  39. }

  40. }

虽然使用了反射,但是依然不影响我们找到根本所在,找到ConfigurationKeys这个类里面的变量究竟是什么:

 

 

 
  1. ConfigurationKey<String> IGNORE_PATTERN = new ConfigurationKey<String>("ignorePattern", null);

  2. ConfigurationKey<String> IGNORE_URL_PATTERN_TYPE = new ConfigurationKey<String>("ignoreUrlPatternType", "REGEX");

字面上理解这两个常量定义了忽略模式以及忽略模式类型是‘正则’,当然我们还是不确定是不是正则,那么继续往下找

 

 

 final Class<? extends UrlPatternMatcherStrategy> ignoreUrlMatcherClass = PATTERN_MATCHER_TYPES.get(ignoreUrlPatternType);

 

我们已经通过ConfigurationKeys类知道ignoreUrlPatternType是个‘REGEX’字符串,那么

 

   PATTERN_MATCHER_TYPES.put("REGEX", RegexUrlPatternMatcherStrategy.class);

那么按照REGEX对应的值找到RegexUrlPatternMatcherStrategy这个类:

 

(5)确定RegexUrlPatternMatcherStrategy类用于处理正则验证匹配

 

 
  1. public final class RegexUrlPatternMatcherStrategy implements UrlPatternMatcherStrategy {

  2. <!--做好自己!eguid原创-->

  3. private Pattern pattern;

  4.  
  5. public RegexUrlPatternMatcherStrategy() {}

  6.  
  7. public RegexUrlPatternMatcherStrategy(final String pattern) {

  8. this.setPattern(pattern);

  9. }

  10.  
  11. public boolean matches(final String url) {

  12. return this.pattern.matcher(url).find();

  13. }

  14.  
  15. public void setPattern(final String pattern) {

  16. this.pattern = Pattern.compile(pattern);

  17. }

  18. }

该类中用到了Pattern来编译和匹配正则表达式

 

到这里我们终于可以确定可以用ignorePattern来忽略/排除我们不需要拦截的请求URL,当然必须满足正则表达式。

 

Python正确的学习路线,你一定不知道的薪资翻倍秘诀

如何从8K提至20K月薪,你要掌握学习那些技能

  • l714417743

    l714417743: cas4.0测试无效,还是自己尝试重写吧(07-31 15:56#5楼)

  • zhanfp

    zhanfp: 很佩服作者的研究精神,但是作者忽略了cas的版本。(06-27 16:03#4楼)

  • jiangjia2706

    jiangjia2706: 您好,能耽误宝贵时间帮我看下么,用的jar包是3.2.1版本,但是我要做到的是不拦截webservice接口,我是这样配置( /webservice/* ),这样不行,能帮忙看下么,谢谢(03-08 16:34#3楼)查看回复(1)

  • sinat_19171485

    饭野兔: cas-client 3.3.0版本没有这个功能(11-13 14:57#2楼)查看回复(1)

查看 9 条热评

SSO - CAS:不拦截指定url过滤

qq_36698956

 461

首先我们看一下我们客户端中web.xml中的拦截器配置 &amp;lt;filter&amp;gt; &amp;lt;filter-name&amp;gt;CASFilter&a...

单点登录CAS使用记(五):cas-client拦截静态资源以及无需登录的请求

qq_33732975

 668

一.问题在哪? 在配置cas-client中,有这么一段配置: filter> filter-name>CAS Filterfilter-name> ...

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值