- 浏览: 15475 次
- 性别:
- 来自: 重庆
最新评论
-
leyou:
nano 写道请教:按照楼主做法,前端页面ajax没法取得lo ...
Cas自定义登录页面Ajax实现 -
nano:
请教:按照楼主做法,前端页面ajax没法取得loginTick ...
Cas自定义登录页面Ajax实现
本文是基于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
并且将该 action 声明在 cas-servlet.xml 中:
还需要定义 loginTicket 的生成页也就是当返回 loginTicketRequested 的 view:
位置/WEB-INF/view/jsp/default/ui
viewRedirectToRequestor.jsp
并且需要将该 jsp 声明在 default._views.properites 中:
接下来要做的就是将该action 的处理加入到 login-webflow.xml 请求流中:
调整 CAS Server端,使其适应 Iframe 方式登录,并使其支持回调。
打开 login-webflow.xml,找到 <action-state id="generateServiceTicket"> 的 Flow-Action 配置项:
再新增 loginResponse Action配置项:
再调整,当验证失败后,也需要判断是否是 iframe/ajax登录:
还需要配置 viewAjaxLoginView 的 state:
接着,再定义 ajaxLoginServiceTicketAction Bean 吧,直接在 cas-servlet.xml 声明该 bean:
再定义一下 view 的页面地址,修改 default_views.properties,添加:
再是 ajaxLogin.jsp 的代码,从 request attributes 中获取到 ST, Service 等参数信息:
然后客户端登录页面login.jsp:
Url中的service参数为登录成功后返回的页面,因为我在ajaxLogin.jsp页面并没有设置使用iframe时跳转,所以看到的结果是页面没跳转,但是设置该参数还是很有必要的。比如当使用spring security时,service=http://localhost/web/j_spring_cas_security_check , 当登录成功时就会通过该URL对用户进行授权认证。
至此,整个开发就算完成, 本人也是才开始学cas,该方法比较粗糙,但理解起来比较简单。
我也不能取到loginTicket,我的客户端采用了Spring Security 3,按照博主的方法,response.getWriter().print(request.getAttribute("loginTicket")+"&"+request.getAttribute("flowExecutionKey"));
这一步已经把loginTicket响应回去,但是客户端怎么获取不到呢?客户端接受到的响应为空,难道是因为SpringSecurity吗?
环境:
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
import javax.servlet.http.HttpServletRequest; import org.jasig.cas.util.UniqueTicketIdGenerator; import org.jasig.cas.web.support.WebUtils; import org.springframework.webflow.action.AbstractAction; import org.springframework.webflow.execution.Event; import org.springframework.webflow.execution.RequestContext; /** * Opens up the CAS web flow to allow external retrieval of a login ticket. * * @author cydiay */ public class ProvideLoginTicketAction extends AbstractAction{ private static final String PREFIX = "LT"; @Override protected Event doExecute(RequestContext context) throws Exception { final HttpServletRequest request = WebUtils.getHttpServletRequest(context); if (request.getParameter("get-lt") != null && request.getParameter("get-lt").equalsIgnoreCase("true")) { final String loginTicket = this.ticketIdGenerator.getNewTicketId(PREFIX); WebUtils.putLoginTicket(context, loginTicket); return result("loginTicketRequested"); } return result("continue"); } private UniqueTicketIdGenerator ticketIdGenerator; public void setTicketIdGenerator(final UniqueTicketIdGenerator generator) { this.ticketIdGenerator = generator; } }
并且将该 action 声明在 cas-servlet.xml 中:
<bean id="provideLoginTicketAction" class="org.jasig.cas.web.flow.ProvideLoginTicketAction" p:ticketIdGenerator-ref="loginTicketUniqueIdGenerator"/>
还需要定义 loginTicket 的生成页也就是当返回 loginTicketRequested 的 view:
位置/WEB-INF/view/jsp/default/ui
viewRedirectToRequestor.jsp
<%@ page contentType="text/html; charset=UTF-8"%> <% String ajax = request.getParameter("n"); //当执行Ajax自定义页面时执行以下操作 if(ajax!=null && ajax.length()>0){ response.getWriter().print(request.getAttribute("loginTicket")+"&"+request.getAttribute("flowExecutionKey")); } else { //正常cas执行 %> <script>window.location.href = "/cas/login";</script> <% } %>
并且需要将该 jsp 声明在 default._views.properites 中:
casRedirectToRequestorView.(class)=org.springframework.web.servlet.view.JstlView casRedirectToRequestorView.url=/WEB-INF/view/jsp/default/ui/viewRedirectToRequestor.jsp
接下来要做的就是将该action 的处理加入到 login-webflow.xml 请求流中:
<on-start> <evaluate expression="initialFlowSetupAction" /> </on-start> <action-state id="provideLoginTicket"> <evaluate expression="provideLoginTicketAction"/> <transition on="loginTicketRequested" to="viewRedirectToRequestor" /> <transition on="continue" to="ticketGrantingTicketExistsCheck" /> </action-state> <view-state id="viewRedirectToRequestor" view="casRedirectToRequestorView" model="credentials"> <binder> <binding property="username" /> <binding property="password" /> </binder> <on-entry> <set name="viewScope.commandName" value="'credentials'" /> </on-entry> <transition on="submit" bind="true" validate="true" to="realSubmit"> <set name="flowScope.credentials" value="credentials" /> <evaluate expression="authenticationViaFormAction.doBind(flowRequestContext, flowScope.credentials)" /> </transition> </view-state> <decision-state id="ticketGrantingTicketExistsCheck"> <if test="flowScope.ticketGrantingTicketId != null" then="hasServiceCheck" else="gatewayRequestCheck" /> </decision-state>
调整 CAS Server端,使其适应 Iframe 方式登录,并使其支持回调。
打开 login-webflow.xml,找到 <action-state id="generateServiceTicket"> 的 Flow-Action 配置项:
<!--当执行到该 action 的时候,表示已经登录成功,将生成 ST(Service Ticket)。--> <action-state id="generateServiceTicket"> <evaluate expression="generateServiceTicketAction" /> <!--当生成 ST 成功后,则进入登录成功页,新增 loginResponse Action 处理项,判断是否是 ajax/iframe 登录 --> <!-- <transition on="success" to="warn" /> --> <transition on="success" to="loginResponse" /> <!--<transition on="error" to="viewLoginForm" />--> <!-- 可能生成 service ticket 失败,同样,也是进入 loginResponse --> <transition on="error" to="loginResponse" /> <transition on="gateway" to="redirect" /> </action-state>
再新增 loginResponse Action配置项:
<action-state id="loginResponse"> <evaluate expression="ajaxLoginServiceTicketAction" /> <!--非ajax/iframe方式登录,采取原流程处理 --> <transition on="success" to="warn" /> <transition on="error" to="viewLoginForm" /> <!-- 反之,则进入 viewAjaxLoginView 页面 --> <transition on="local" to="viewAjaxLoginView" /> </action-state>
再调整,当验证失败后,也需要判断是否是 iframe/ajax登录:
<action-state id="realSubmit"> <evaluate expression="authenticationViaFormAction.submit(flowRequestContext, flowScope.credentials, messageContext)" /> <transition on="warn" to="warn" /> <transition on="success" to="sendTicketGrantingTicket" /> <!--将 to="viewLoginForm" 修改为 to="loginResponse" --> <transition on="error" to="loginResponse" /> </action-state>
还需要配置 viewAjaxLoginView 的 state:
<end-state id="viewAjaxLoginView" view="viewAjaxLoginView" />
接着,再定义 ajaxLoginServiceTicketAction Bean 吧,直接在 cas-servlet.xml 声明该 bean:
<bean id="ajaxLoginServiceTicketAction" class="com.unknow.cas.server.web.AjaxLoginServiceTicketAction"/>
import javax.servlet.http.HttpServletRequest; import org.apache.commons.lang.BooleanUtils; import org.apache.commons.lang.StringUtils; import org.jasig.cas.authentication.principal.Service; import org.jasig.cas.web.support.WebUtils; import org.springframework.webflow.action.AbstractAction; import org.springframework.webflow.execution.Event; import org.springframework.webflow.execution.RequestContext; public final class AjaxLoginServiceTicketAction extends AbstractAction { // The default call back function name. protected static final String J_CALLBACK = "feedBackUrlCallBack"; protected Event doExecute(final RequestContext context) { HttpServletRequest request = WebUtils.getHttpServletRequest(context); Event event = context.getCurrentEvent(); boolean isAjax = BooleanUtils.toBoolean(request.getParameter("isajax")); if (!isAjax){ // 非 ajax/iframe 方式登录,返回当前 event. return event; } boolean isLoginSuccess; // Login Successful. if ("success".equals(event.getId())){ //是否登录成功 final Service service = WebUtils.getService(context); final String serviceTicket = WebUtils.getServiceTicketFromRequestScope(context); if (service != null){ //设置登录成功之后 跳转的地址 request.setAttribute("service", service.getId()); } request.setAttribute("ticket", serviceTicket); isLoginSuccess = true; } else { // Login Fails.. isLoginSuccess = false; } boolean isFrame = BooleanUtils.toBoolean(request.getParameter("isframe")); String callback = request.getParameter("callback"); if(StringUtils.isEmpty(callback)){ // 如果未转入 callback 参数,则采用默认 callback 函数名 callback = J_CALLBACK; } if(isFrame){ // 如果采用了 iframe ,则 concat 其 parent 。 callback = "parent.".concat(callback); } request.setAttribute("isFrame", isFrame); request.setAttribute("callback", callback); request.setAttribute("isLogin", isLoginSuccess); return new Event(this, "local"); // 转入 ajaxLogin.jsp 页面 } }
再定义一下 view 的页面地址,修改 default_views.properties,添加:
viewAjaxLoginView.(class)=org.springframework.web.servlet.view.JstlView viewAjaxLoginView.url=/WEB-INF/view/jsp/custom/ui/ajaxLogin.jsp
再是 ajaxLogin.jsp 的代码,从 request attributes 中获取到 ST, Service 等参数信息:
<%@ page contentType="text/html; charset=UTF-8"%> <html> <head> <title>正在登录....</title> </head> <body> <script type="text/javascript"> <% Boolean isFrame = (Boolean)request.getAttribute("isFrame"); Boolean isLogin = (Boolean)request.getAttribute("isLogin"); // 登录成功 if(isLogin){ if(isFrame){%> //parent.location.replace('${service}?ticket=${ticket}') <%} else{%> location.replace('${service}?ticket=${ticket}') <%} } %> // 回调 ${callback}({'login':${isLogin ? '"success"': '"false"'}, 'msg': ${isLogin ? '""': '"用户名或密码错误!"'}}) </script> </body> </html>
然后客户端登录页面login.jsp:
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Cas登录</title> <script type="text/javascript" src="js/jquery.js"></script> <script type="text/javascript" src="js/jquery.form.js"></script> <script type="text/javascript"> $(document).ready(function(){ flushLoginTicket(); // 进入登录页,则获取login ticket,该函数在下面定义。 }); // 登录验证函数, 由 onsubmit 事件触发 var loginValidate = function(){ var msg; if ($.trim($('#J_Username').val()).length == 0 ){ msg = "用户名不能为空。"; } else if ($.trim($('#J_Password').val()).length == 0 ){ msg = "密码不能为空。"; } if (msg && msg.length > 0) { $('#J_ErrorMsg').fadeOut().text(msg).fadeIn(); return false; // Can't request the login ticket. } else if ($('#J_LoginTicket').val().length == 0){ $('#J_ErrorMsg').text('服务器正忙,请稍后再试..'); return false; } else { // 验证成功后,动态创建用于提交登录的 iframe $('body').append($('<iframe/>').attr({ style: "display:none;width:0;height:0", id: "ssoLoginFrame", name: "ssoLoginFrame", src: "javascript:false;" })); return true; } } // 登录处理回调函数,将由 iframe 中的页同自动回调 var feedBackUrlCallBack = function (result) { customLoginCallBack(result); deleteIFrame('#ssoLoginFrame');// 删除用完的iframe,但是一定不要在回调前删除,Firefox可能有问题的 }; // 自定义登录回调逻辑 var customLoginCallBack = function(result){ // 登录失败,显示错误信息 if (result.login == 'false'){ $('#J_ErrorMsg').fadeOut().text(result.msg).fadeIn(); // 重新刷新 login ticket flushLoginTicket(); }else{ //该处定义登录成功后需要执行的操作,比如刷新DIV等 //...... alert("登陆成功"); } } var deleteIFrame = function (iframeName) { var iframe = $(iframeName); if (iframe) { // 删除用完的iframe,避免页面刷新或前进、后退时,重复执行该iframe的请求 iframe.remove() } }; // 由于一个 login ticket 只允许使用一次, 当每次登录需要调用该函数刷新 lt var flushLoginTicket = function(){ var _services = 'service=' + encodeURIComponent('http://localhost/web/login.jsp'); var casUrl = 'http://localhost:8080/cas/login?'+_services+'&get-lt=true&n=' + new Date().getTime(); $.ajax({ type: "GET", url: casUrl, success: function(data){ var data = data.split('&'); $('#J_LoginTicket').val(data[0]); $("#J_Execution").val(data[1]); } }); } </script> </head> <body> <form action="http://localhost:8080/cas/login" method="post" onsubmit="return loginValidate();" target="ssoLoginFrame"> <ul> <li><span class="red" style="height:12px;" id="J_ErrorMsg"></span></li> <li> <em>用户名:</em> <input name="username" id="J_Username" type="text" style="width: 180px" /> </li> <li> <em>密 码:</em> <input name="password" type="password" id="J_Password" style="width: 180px" /> </li> <li class="mai"> <em> </em> <input type="checkbox" name="rememberMe" id="rememberMe" value="true"/> 自动登录 <a href="/retrieve">忘记密码?</a> </li> <li> <em> </em> <input type="hidden" name="isajax" value="true" /> <input type="hidden" name="isframe" value="true" /> <input type="text" name="callback" value="feedBackUrlCallBack" /> <input type="text" name="lt" value="1" id="J_LoginTicket"> <input type="text" name="execution" id="J_Execution" value="" /> <input type="hidden" name="_eventId" value="submit" /> <input name="" type="submit" value="登录" /> </li> </ul> </form> </body> </html>
Url中的service参数为登录成功后返回的页面,因为我在ajaxLogin.jsp页面并没有设置使用iframe时跳转,所以看到的结果是页面没跳转,但是设置该参数还是很有必要的。比如当使用spring security时,service=http://localhost/web/j_spring_cas_security_check , 当登录成功时就会通过该URL对用户进行授权认证。
至此,整个开发就算完成, 本人也是才开始学cas,该方法比较粗糙,但理解起来比较简单。
评论
2 楼
leyou
2013-10-11
nano 写道
请教:按照楼主做法,前端页面ajax没法取得loginTicket,初步怀疑login-webflow配置错误了,楼主能否把webflow内容贴出来看看
我也不能取到loginTicket,我的客户端采用了Spring Security 3,按照博主的方法,response.getWriter().print(request.getAttribute("loginTicket")+"&"+request.getAttribute("flowExecutionKey"));
这一步已经把loginTicket响应回去,但是客户端怎么获取不到呢?客户端接受到的响应为空,难道是因为SpringSecurity吗?
1 楼
nano
2013-02-28
请教:按照楼主做法,前端页面ajax没法取得loginTicket,初步怀疑login-webflow配置错误了,楼主能否把webflow内容贴出来看看
相关推荐
为了实现在不跳转到CAS服务器登录页面的情况下获取ticket,我们可以通过以下步骤进行: 1. 配置Spring Boot:首先,我们需要在Spring Boot应用中引入必要的依赖,如CasClient、Shiro和Pac4J的相关依赖。然后,在...
在这个前后端分离的项目中,Vue.js用于构建前端页面,与后端通过Ajax进行通信,实现动态交互和数据展示。 集成CAS(Central Authentication Service)是为了实现单点登录(Single Sign-On, SSO)。CAS是一个开源的...
4. **定制化模板**:允许管理员根据自身需求自定义 CAS 登录界面,如添加企业 logo 或修改默认布局。 5. **第三方库**:可能包含一些用于处理 AJAX 请求、表单验证或提供额外功能的 JavaScript 库。 **二、CAS ...
它们定义了颜色、字体、间距、背景和元素的对齐方式,使得CAS的登录页面和其他交互界面符合设计规范。 2. **JavaScript(JS)**:JS文件负责增加页面的动态功能,如表单验证、按钮交互、AJAX请求等。在CAS中,...
CAS(Central Authentication Service)是中央...这个"plugin.dj"自定义头像脚本的实现涉及到多方面的JavaScript技术,旨在为用户提供便捷且个性化的头像设置功能,同时确保与CAS系统集成顺畅,提供安全的认证体验。
- **实现自定义登录页面**:介绍如何创建一个符合项目设计风格的登录界面。 - **修改配置文件**:更新Spring Security配置以支持自定义登录页面。 - **登录页面中的参数配置**:设置登录表单所需的参数,如用户名和...
- **实现自定义登录页面:** 创建自定义的登录页面,替换Spring Security默认的登录界面。 - **修改配置文件:** 调整配置以使用自定义的登录页面。 - **登录页面中的参数配置:** 设置登录页面的请求参数,确保...
- **自定义拒绝页面**: 当用户尝试访问未授权的资源时显示自定义页面。 #### 三、保护 Web 篇 ##### 9. 图解过滤器 - **过滤器介绍**: 详细解释 Spring Security 中各种过滤器的作用和顺序。 - **过滤器流程**: ...
**20.2 使用Ajax实现Digest认证** 编写Ajax代码实现Digest认证。 **20.3 编程实现Digest客户端** 编写Java代码实现Digest认证的客户端。 **21. 通过LDAP获取用户信息** 集成LDAP服务器以获取用户信息。 **22. ...
AJAX(Asynchronous JavaScript and XML)技术是一种在无需重新加载整个页面的情况下,能够更新部分网页的技术。在Web开发中,AJAX常用于在客户端(浏览器)和服务器之间进行异步数据交换。文章介绍了使用AJAX技术从...
- 结合前两节的知识点,实现动态管理资源的同时提供自定义登录界面。 - **示例代码**: 整合动态资源管理和自定义登录页面的配置。 **9. 中文用户名** - 解决中文用户名在登录过程中可能出现的问题。 - **示例...
在SSO场景下,AjaxAnyWhere可能被用来无刷新地处理登录状态,比如在用户登录后自动更新页面内容,或者在用户访问受保护资源时,通过Ajax请求验证用户身份。 6. **SSODemo**:这个文件可能是一个示例项目,展示了...
Shiro与CAS单点登录整合,便于扩展多个应用模块。此外,自定义了UsernamePasswordToken和Realm,实现基于验证码和数据库用户密码的登录验证。 Spring的注解控制器用于控制器层,支持返回Velocity视图、Ajax JSON和...
页面展示和控制方式多样化,包括动态同步请求、Velocity模板生成页面、Ajax异步请求。前端使用jQuery库、jqzoom插件、jQuery-validator等技术进行交互处理。权限安全控制则由Apache Shiro框架负责,它支持基于角色和...
3. 单一登录(SSO)原理和实现:OAuth2、JWT或CAS协议的理解,以及如何在客户端实现SSO流程。 4. 客户端与服务器端通信:了解AJAX、Fetch API等异步请求技术,处理跨域问题。 5. 测试策略和自动化测试:编写测试用例...
无刷新设计,除了进入功能页面和新页面,其它情况下全部采用Ajax交互,优化体验和性能。 支持一件换肤,只需在properties里修改下主题名称即可快速切换整个UI的风格,不仅仅是色调和样式,布局也可改变。支持自定义...
JavaScript在这个过程中通常用于前端交互,比如处理重定向、显示登录界面、处理令牌等。例如,使用Ajax进行异步通信,获取和发送令牌,提升用户体验。 在提供的压缩包文件“sso-master”中,我们可以推测可能包含了...
- **安全框架**:熟悉Shiro、Spring Security和CAS,这些都是实现单点登录和权限控制的工具。 2. **后端技术**: - **框架整合**:包括SSH(Spring、Struts、Hibernate)和SSM(Spring、SpringMVC、MyBatis),...
21. **单点登录**:CAS,OAuth2等身份验证协议。 22. **NAT穿透**:STUN,TURN,ICE协议。 23. **邮件收发**:JavaMailSender,POP3,IMAP协议。 24. **应用服务器迁移**:不同应用服务器间的兼容性问题。 25. **...
Ajax和JSON请求 145 CookieCsrfTokenRepository 146 18.5 CSRF警告 147 18.5.1超时 148 18.5.2登录 148 18.5.3注销 149 18.5.4多部分(文件上传) 149 在Spring Security之前放置MultipartFilter 150 包含CSRF令牌 ...