`

单点登录

 
阅读更多
本文是基于CAS 之自定义登录页实践CAS 之 跨域 Ajax 登录实践而实现的,主要是针对最新的Cas实现自定义登录页的Ajax跨域实现. 

环境: 
        cas-server-3.5.1-release 
        cas地址:  http://localhost:8080/cas/ 
        client地址:  http://localhost/web
 

从CAS服务端生成lt及execution,在cas的 login flow 中加入 ProvideLoginTicketAction 的流,主要用于判断该请求是否是来获取 lt,在cas-server端声明获取 login ticket action 类: 

org.jasig.cas.web.flow.ProvideLoginTicketAction 
Java代码  收藏代码
  1. import javax.servlet.http.HttpServletRequest;  
  2.   
  3. import org.jasig.cas.util.UniqueTicketIdGenerator;  
  4. import org.jasig.cas.web.support.WebUtils;  
  5. import org.springframework.webflow.action.AbstractAction;  
  6. import org.springframework.webflow.execution.Event;  
  7. import org.springframework.webflow.execution.RequestContext;  
  8.   
  9. /** 
  10.  * Opens up the CAS web flow to allow external retrieval of a login ticket. 
  11.  *  
  12.  * @author cydiay 
  13.  */  
  14. public class ProvideLoginTicketAction extends AbstractAction{  
  15.   
  16.     private static final String PREFIX = "LT";  
  17.       
  18.     @Override  
  19.     protected Event doExecute(RequestContext context) throws Exception {  
  20.         final HttpServletRequest request = WebUtils.getHttpServletRequest(context);  
  21.         if (request.getParameter("get-lt") != null && request.getParameter("get-lt").equalsIgnoreCase("true")) {  
  22.             final String loginTicket = this.ticketIdGenerator.getNewTicketId(PREFIX);  
  23.             WebUtils.putLoginTicket(context, loginTicket);  
  24.             return result("loginTicketRequested");  
  25.         }  
  26.         return result("continue");  
  27.     }  
  28.       
  29.     private UniqueTicketIdGenerator ticketIdGenerator;  
  30.       
  31.     public void setTicketIdGenerator(final UniqueTicketIdGenerator generator) {  
  32.         this.ticketIdGenerator = generator;  
  33.     }  
  34. }  


并且将该 action 声明在 cas-servlet.xml 中: 

Xml代码  收藏代码
  1. <bean id="provideLoginTicketAction" class="org.jasig.cas.web.flow.ProvideLoginTicketAction"   
  2.         p:ticketIdGenerator-ref="loginTicketUniqueIdGenerator"/>  


还需要定义 loginTicket 的生成页也就是当返回 loginTicketRequested 的 view: 
位置/WEB-INF/view/jsp/default/ui 

viewRedirectToRequestor.jsp 

Html代码  收藏代码
  1. <%@ page contentType="text/html; charset=UTF-8"%>  
  2. <%  
  3.     String ajax = request.getParameter("n");  
  4.     //当执行Ajax自定义页面时执行以下操作  
  5.     if(ajax!=null && ajax.length()>0){  
  6.         response.getWriter().print(request.getAttribute("loginTicket")+"&"+request.getAttribute("flowExecutionKey"));  
  7.     } else {  
  8.       
  9.     //正常cas执行  
  10. %>         
  11.      <script>window.location.href = "/cas/login";</script>  
  12. <%         
  13.     }  
  14. %>  


并且需要将该 jsp 声明在 default._views.properites 中: 

Properties代码  收藏代码
  1. casRedirectToRequestorView.(class)=org.springframework.web.servlet.view.JstlView  
  2. casRedirectToRequestorView.url=/WEB-INF/view/jsp/default/ui/viewRedirectToRequestor.jsp  


接下来要做的就是将该action 的处理加入到 login-webflow.xml 请求流中: 

Xml代码  收藏代码
  1.   <on-start>  
  2.       <evaluate expression="initialFlowSetupAction" />  
  3.   </on-start>  
  4.   
  5.   <action-state id="provideLoginTicket">  
  6.       <evaluate expression="provideLoginTicketAction"/>  
  7.       <transition on="loginTicketRequested" to="viewRedirectToRequestor" />  
  8.       <transition on="continue" to="ticketGrantingTicketExistsCheck" />  
  9.   </action-state>  
  10.   
  11.  <view-state id="viewRedirectToRequestor" view="casRedirectToRequestorView" model="credentials">  
  12.       <binder>  
  13.           <binding property="username" />  
  14.           <binding property="password" />  
  15.       </binder>  
  16.       <on-entry>  
  17.           <set name="viewScope.commandName" value="'credentials'" />  
  18.       </on-entry>  
  19.       <transition on="submit" bind="true" validate="true" to="realSubmit">  
  20.           <set name="flowScope.credentials" value="credentials" />  
  21.           <evaluate expression="authenticationViaFormAction.doBind(flowRequestContext, flowScope.credentials)" />  
  22.       </transition>  
  23.   </view-state>  
  24.   
  25.   <decision-state id="ticketGrantingTicketExistsCheck">  
  26. <if test="flowScope.ticketGrantingTicketId != null" then="hasServiceCheck" else="gatewayRequestCheck" />  
  27.   </decision-state>  


调整 CAS Server端,使其适应 Iframe 方式登录,并使其支持回调。 
打开 login-webflow.xml,找到 <action-state id="generateServiceTicket"> 的 Flow-Action 配置项: 

Xml代码  收藏代码
  1. <!--当执行到该 action 的时候,表示已经登录成功,将生成 ST(Service Ticket)。-->    
  2. <action-state id="generateServiceTicket">  
  3.     <evaluate expression="generateServiceTicketAction" />  
  4.         <!--当生成 ST 成功后,则进入登录成功页,新增 loginResponse Action 处理项,判断是否是 ajax/iframe 登录 -->  
  5.         <!-- <transition on="success" to="warn" /> -->  
  6.         <transition on="success" to="loginResponse" />  
  7.         <!--<transition on="error" to="viewLoginForm" />-->  
  8.         <!-- 可能生成 service ticket 失败,同样,也是进入 loginResponse -->  
  9.         <transition on="error" to="loginResponse" />  
  10.         <transition on="gateway" to="redirect" />  
  11.     </action-state>  


再新增 loginResponse Action配置项: 

Xml代码  收藏代码
  1. <action-state id="loginResponse">  
  2.     <evaluate expression="ajaxLoginServiceTicketAction" />  
  3.     <!--非ajax/iframe方式登录,采取原流程处理 -->  
  4.     <transition on="success" to="warn" />  
  5.     <transition on="error" to="viewLoginForm" />  
  6.     <!-- 反之,则进入 viewAjaxLoginView 页面 -->  
  7.     <transition on="local" to="viewAjaxLoginView" />  
  8. </action-state>  


再调整,当验证失败后,也需要判断是否是 iframe/ajax登录: 

Xml代码  收藏代码
  1. <action-state id="realSubmit">  
  2.     <evaluate  
  3.         expression="authenticationViaFormAction.submit(flowRequestContext, flowScope.credentials, messageContext)" />  
  4.     <transition on="warn" to="warn" />  
  5.     <transition on="success" to="sendTicketGrantingTicket" />  
  6.     <!--将 to="viewLoginForm" 修改为 to="loginResponse" -->                 
  7.     <transition on="error" to="loginResponse" />  
  8. </action-state>  


还需要配置 viewAjaxLoginView 的 state:   

Xml代码  收藏代码
  1. <end-state id="viewAjaxLoginView" view="viewAjaxLoginView" />  


接着,再定义 ajaxLoginServiceTicketAction Bean 吧,直接在 cas-servlet.xml 声明该 bean: 

Xml代码  收藏代码
  1. <bean id="ajaxLoginServiceTicketAction" class="com.unknow.cas.server.web.AjaxLoginServiceTicketAction"/>  


Java代码  收藏代码
  1. import javax.servlet.http.HttpServletRequest;  
  2.   
  3. import org.apache.commons.lang.BooleanUtils;  
  4. import org.apache.commons.lang.StringUtils;  
  5. import org.jasig.cas.authentication.principal.Service;  
  6. import org.jasig.cas.web.support.WebUtils;  
  7. import org.springframework.webflow.action.AbstractAction;  
  8. import org.springframework.webflow.execution.Event;  
  9. import org.springframework.webflow.execution.RequestContext;  
  10.   
  11. public final class AjaxLoginServiceTicketAction extends AbstractAction {  
  12.       
  13.     // The default call back function name.  
  14.     protected static final String J_CALLBACK = "feedBackUrlCallBack";  
  15.   
  16.     protected Event doExecute(final RequestContext context) {  
  17.         HttpServletRequest request = WebUtils.getHttpServletRequest(context);  
  18.         Event event = context.getCurrentEvent();  
  19.         boolean isAjax = BooleanUtils.toBoolean(request.getParameter("isajax"));  
  20.           
  21.         if (!isAjax){  // 非 ajax/iframe 方式登录,返回当前 event.  
  22.             return event;  
  23.         }  
  24.         boolean isLoginSuccess;  
  25.         // Login Successful.  
  26.         if ("success".equals(event.getId())){ //是否登录成功  
  27.             final Service service = WebUtils.getService(context);  
  28.             final String serviceTicket = WebUtils.getServiceTicketFromRequestScope(context);  
  29.             if (service != null){  //设置登录成功之后 跳转的地址  
  30.                 request.setAttribute("service", service.getId());  
  31.             }  
  32.             request.setAttribute("ticket", serviceTicket);  
  33.             isLoginSuccess = true;  
  34.         } else { // Login Fails..  
  35.             isLoginSuccess = false;  
  36.         }  
  37.   
  38.         boolean isFrame = BooleanUtils.toBoolean(request.getParameter("isframe"));  
  39.         String callback = request.getParameter("callback");  
  40.         if(StringUtils.isEmpty(callback)){ // 如果未转入 callback 参数,则采用默认 callback 函数名  
  41.             callback = J_CALLBACK;  
  42.         }  
  43.         if(isFrame){ // 如果采用了 iframe ,则 concat 其 parent 。  
  44.             callback = "parent.".concat(callback);  
  45.         }  
  46.         request.setAttribute("isFrame", isFrame);  
  47.         request.setAttribute("callback", callback);  
  48.         request.setAttribute("isLogin", isLoginSuccess);  
  49.           
  50.         return new Event(this"local"); // 转入 ajaxLogin.jsp 页面  
  51.     }  
  52. }  


再定义一下 view 的页面地址,修改 default_views.properties,添加: 

Properties代码  收藏代码
  1. viewAjaxLoginView.(class)=org.springframework.web.servlet.view.JstlView  
  2. viewAjaxLoginView.url=/WEB-INF/view/jsp/custom/ui/ajaxLogin.jsp  


再是 ajaxLogin.jsp 的代码,从 request attributes 中获取到  ST, Service 等参数信息: 

Java代码  收藏代码
  1. <%@ page contentType="text/html; charset=UTF-8"%>  
  2. <html>  
  3.     <head>  
  4.         <title>正在登录....</title>  
  5.     </head>  
  6.     <body>  
  7.         <script type="text/javascript">  
  8.             <%  
  9.                 Boolean isFrame = (Boolean)request.getAttribute("isFrame");  
  10.                 Boolean isLogin = (Boolean)request.getAttribute("isLogin");  
  11.                 // 登录成功  
  12.                 if(isLogin){  
  13.                     if(isFrame){%>  
  14.                         //parent.location.replace('${service}?ticket=${ticket}')  
  15.                     <%} else{%>  
  16.                         location.replace('${service}?ticket=${ticket}')  
  17.                     <%}  
  18.                 }  
  19.             %>  
  20.             // 回调  
  21.             ${callback}({'login':${isLogin ? '"success"''"false"'}, 'msg': ${isLogin ? '""''"用户名或密码错误!"'}})  
  22.         </script>  
  23.     </body>  
  24. </html>  



然后客户端登录页面login.jsp: 

Html代码  收藏代码
  1. <%@ page language="java" contentType="text/html; charset=UTF-8"  
  2.     pageEncoding="UTF-8"%>  
  3. <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">  
  4. <html>  
  5. <head>  
  6. <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">  
  7. <title>Cas登录</title>  
  8. <script type="text/javascript" src="js/jquery.js"></script>  
  9. <script type="text/javascript" src="js/jquery.form.js"></script>  
  10. <script type="text/javascript">  
  11.     $(document).ready(function(){   
  12.         flushLoginTicket();  // 进入登录页,则获取login ticket,该函数在下面定义。  
  13.     });  
  14.   
  15.     // 登录验证函数, 由 onsubmit 事件触发  
  16.     var loginValidate = function(){  
  17.         var msg;  
  18.         if ($.trim($('#J_Username').val()).length == 0 ){  
  19.             msg = "用户名不能为空。";  
  20.         } else if ($.trim($('#J_Password').val()).length == 0 ){  
  21.             msg = "密码不能为空。";  
  22.         }  
  23.         if (msg && msg.length > 0) {  
  24.             $('#J_ErrorMsg').fadeOut().text(msg).fadeIn();  
  25.             return false;  
  26.             // Can't request the login ticket.  
  27.         } else if ($('#J_LoginTicket').val().length == 0){  
  28.             $('#J_ErrorMsg').text('服务器正忙,请稍后再试..');  
  29.             return false;  
  30.         } else {  
  31.             // 验证成功后,动态创建用于提交登录的 iframe  
  32.             $('body').append($('<iframe/>').attr({  
  33.                 style: "display:none;width:0;height:0",   
  34.                 id: "ssoLoginFrame",  
  35.                 name: "ssoLoginFrame",  
  36.                 src: "javascript:false;"  
  37.             }));  
  38.             return true;  
  39.         }  
  40.     }  
  41.       
  42.     // 登录处理回调函数,将由 iframe 中的页同自动回调  
  43.     var feedBackUrlCallBack = function (result) {  
  44.         customLoginCallBack(result);  
  45.         deleteIFrame('#ssoLoginFrame');// 删除用完的iframe,但是一定不要在回调前删除,Firefox可能有问题的  
  46.     };  
  47.       
  48.     // 自定义登录回调逻辑  
  49.     var customLoginCallBack = function(result){  
  50.         // 登录失败,显示错误信息  
  51.         if (result.login == 'false'){  
  52.             $('#J_ErrorMsg').fadeOut().text(result.msg).fadeIn();  
  53.             // 重新刷新 login ticket  
  54.             flushLoginTicket();  
  55.         }else{  
  56.             //该处定义登录成功后需要执行的操作,比如刷新DIV等  
  57.             //......  
  58.             alert("登陆成功");  
  59.         }  
  60.     }  
  61.   
  62.     var deleteIFrame = function (iframeName) {  
  63.         var iframe = $(iframeName);   
  64.         if (iframe) { // 删除用完的iframe,避免页面刷新或前进、后退时,重复执行该iframe的请求  
  65.             iframe.remove()  
  66.         }  
  67.     };  
  68.   
  69.     // 由于一个 login ticket 只允许使用一次, 当每次登录需要调用该函数刷新 lt  
  70.     var flushLoginTicket = function(){  
  71.         var _services = 'service=' + encodeURIComponent('http://localhost/web/login.jsp');  
  72.         var casUrl = 'http://localhost:8080/cas/login?'+_services+'&get-lt=true&n=' + new Date().getTime();  
  73.         $.ajax({  
  74.             type: "GET",  
  75.             url: casUrl,  
  76.             success: function(data){  
  77.                 var datadata = data.split('&');  
  78.                 $('#J_LoginTicket').val(data[0]);  
  79.                 $("#J_Execution").val(data[1]);  
  80.             }  
  81.         });  
  82.     }  
  83. </script>  
  84. </head>  
  85. <body>  
  86.     <form action="http://localhost:8080/cas/login" method="post" onsubmit="return loginValidate();" target="ssoLoginFrame">  
  87.     <ul>  
  88.         <li><span class="red" style="height:12px;" id="J_ErrorMsg"></span></li>  
  89.   
  90.         <li>  
  91.             <em>用户名:</em>  
  92.             <input name="username" id="J_Username" type="text" style="width: 180px" />  
  93.         </li>  
  94.         <li>  
  95.             <em>密 码:</em>  
  96.             <input name="password" type="password"  id="J_Password" style="width: 180px" />  
  97.         </li>  
  98.   
  99.         <li class="mai">  
  100.             <em>&nbsp;</em>  
  101.             <input type="checkbox" name="rememberMe" id="rememberMe" value="true"/>  
  102.             &nbsp;自动登录  
  103.             <a href="/retrieve">忘记密码?</a>  
  104.         </li>  
  105.         <li>  
  106.             <em>&nbsp;</em>  
  107.             <input type="hidden" name="isajax" value="true" />  
  108.             <input type="hidden" name="isframe" value="true" />  
  109.             <input type="text" name="callback" value="feedBackUrlCallBack" />  
  110.             <input type="text" name="lt" value="1" id="J_LoginTicket">  
  111.             <input type="text" name="execution" id="J_Execution" value="" />  
  112.             <input type="hidden" name="_eventId" value="submit" />  
  113.             <input name="" type="submit" value="登录" />  
  114.         </li>  
  115.     </ul>  
  116. </form>  
  117. </body>  
  118. </html>  


Url中的service参数为登录成功后返回的页面,因为我在ajaxLogin.jsp页面并没有设置使用iframe时跳转,所以看到的结果是页面没跳转,但是设置该参数还是很有必要的。比如当使用spring security时,service=http://localhost/web/j_spring_cas_security_check , 当登录成功时就会通过该URL对用户进行授权认证。 

至此,整个开发就算完成, 本人也是才开始学cas,该方法比较粗糙,但理解起来比较简单
分享到:
评论

相关推荐

    exchange邮件系统单点登录整合

    ### Exchange邮件系统单点登录整合知识点 #### 一、单点登录(Single Sign-On, SSO)概述 单点登录是一种认证机制,允许用户通过一次身份验证就能访问多个应用程序和服务,而无需重复登录。这种机制提高了用户体验...

    用友U8开发单点登录方案

    用友U8开发单点登录方案 本文主要介绍了用友U8开发单点登录方案的设计背景、应用场景、实现方式和技术细节。单点登录方案的设计目标是实现非U8系统与U8系统的集成,共享用户名和密码,实现单点登录。 设计背景 ...

    cas 单点登录 解决方案.

    cas 单点登录解决方案 cas 单点登录解决方案是目前比较流行的企业业务整合的解决方案之一。SSO 的定义是在多个应用系统中,用户只需要登录一次就可以访问所有相互信任的应用系统。随着企业的发展,业务系统的数量...

    JEECG 单点登录集成文档(kisso集成)

    JEECG单点登录集成文档(kisso集成)详细介绍了如何将Kisso单点登录解决方案集成到JEECG智能开发平台v3版本中。文档中提到了Kisso的基本概念和实现单点登录(SSO)的技术细节,具体包括服务端和客户端的集成步骤以及...

    JEECG 单点登录说明文档

    JEECG智能开发平台的单点登录(SSO)功能是通过集成Kisso实现的,Kisso是一个轻量级Java权限框架,它利用加密会话cookie机制来实现单点登录服务。单点登录是一种用户登录认证方法,允许用户在多个应用系统中,只通过...

    Jeecg配置单点登录 登录验证完整代码

    在本场景中,我们关注的是Jeecg如何配置单点登录(Single Sign-On,简称SSO)以及相关的登录验证代码。单点登录是一种网络应用架构中的安全机制,允许用户在一次登录后,就能访问多个相互关联的应用系统,而无需再次...

    金蝶EAS portal单点登录到SHR文档

    金蝶EAS portal单点登录到SHR文档 单点登录(Single Sign-On,SSO)是指用户只需要输入一次用户名和密码,即可访问多个相关的应用系统,而不需要再次输入登录信息。金蝶EAS portal单点登录到SHR文档提供了详细的...

    第8部分:单点登录服务接口规范.docx

    本规范规定了单点登录服务的逻辑,包括单点登录服务的架构、单点登录票据的生成和验证、单点登录会话的管理等,以确保单点登录服务的安全性和可靠性。 6. 接口定义 本规范规定了单点登录服务接口的定义,包括单点...

    C#.net实现单点登录

    单点登录(Single Sign-On,简称SSO)是一种网络用户身份验证机制,允许用户在一个系统或应用中登录后,无须再次输入凭证就能访问多个相互信任的系统或应用。在IT行业中,C#.NET框架提供了丰富的功能来实现跨域单点...

    华为防火墙实现单点登录方式.docx

    然后,终端设备加入AD域后,AD域控服务器会往终端设备上安装一个脚本,这个脚本是来源于单点登录服务程序,所以终端的上下线等操作会通过域传给AD域控服务器,并且还会通过脚本将信息传递给单点登录服务程序,单点...

    Android端实现单点登录的方法详解

    单点登录(Single Sign-On,简称SSO)是一种身份验证机制,允许用户在一次登录后访问多个相互关联的应用系统,而无需再次输入凭证。在Android端实现SSO,主要是为了确保用户在同一时间只能在一个设备上保持活跃的...

    漂亮的单点登录网页模版

    单点登录(Single Sign-On,简称SSO)是一种网络身份验证机制,允许用户在一个系统上登录后,无需再次提供凭证就能访问多个相互信任的系统。它简化了用户管理和提升了用户体验,因为用户不再需要记住多个密码或者在...

    用cas实现mantis单点登录和登出

    ### 使用 CAS 实现 Mantis 单点登录与登出 #### 概述 单点登录(Single Sign-On,简称 SSO)是一种常见的身份认证模式,它允许用户在多个应用程序和服务中仅通过一次登录就能访问所有相关系统而无需多次输入密码。...

    nc63、nc65单点登录方案

    ### nc63、nc65单点登录方案详解 #### 一、概述 单点登录(Single Sign-On,简称SSO)是一种用户只需要一次登录就能访问所有相互信任的应用系统的认证方式。这种机制不仅提升了用户体验,同时也提高了系统的安全性...

    K3单点登录二次开发指导文档

    【K3单点登录二次开发指导文档】是针对金蝶K3系统中单点登录功能进行集成应用的技术文档,旨在帮助具有相应开发经验的人员理解并实现与K/3门户、K/3、K/3HR等应用系统的无缝对接。单点登录(Single Sign-On,SSO)是...

    基于JWT实现SSO单点登录流程图解

    基于JWT实现SSO单点登录流程图解 基于JWT实现SSO单点登录流程图解是指使用JSON Web Token(JWT)来实现单点登录(SSO)的机制。在这种机制中,用户只需要登录一次,就可以访问多个应用服务器上的资源,而不需要再次...

    springCloud-master_单点登录_springCloud单点登录_SpringCloud系统_springclou

    在"springCloud-master_单点登录_springCloud单点登录_SpringCloud系统_springcloud eureka单点登录"这个项目中,我们将重点探讨如何在SpringCloud环境中实现单点登录(Single Sign-On,简称SSO)。 单点登录是一种...

    oracle单点登录demo示例

    Oracle单点登录(Single Sign-On, SSO)是一种身份验证机制,它允许用户在一次登录后访问多个相互关联的应用系统,而无需再次输入凭证。Oracle的SSO解决方案提供了企业级的安全性和便利性,大大提高了用户体验并降低...

    第三方单点登录Ecology方案

    在IT行业中,第三方单点登录(Third-party Single Sign-On,3SSO)是一种常见的身份验证解决方案,它允许用户通过一个中央认证系统访问多个相互独立的应用系统,而无需反复登录。"第三方单点登录Ecology方案"是针对...

Global site tag (gtag.js) - Google Analytics