当前位置: 首页 > news >正文

CAS (2) —— Mac下配置CAS到Tomcat(客户端)

CAS (2) —— Mac下配置CAS到Tomcat(客户端)


tomcat版本: tomcat-8.0.29

jdk版本: jdk1.8.0_65

cas版本: cas4.1.2
cas-client-3.4.1

参考来源:

CAS实现单点登录(SSO)经典完整教程

CAS 4.0 配置开发手册

cas客户端应用实现

使用 CAS 在 Tomcat 中实现单点登录

Tomcat (1) —— Mac下配置Tomcat Https/SSL

【高可用HA】Apache (2) —— Mac下安装多个Apache Tomcat实例

目标架构

613455-20151214150233412-2079087955.png

下载

首先登陆jasig网站http://downloads.jasig.org/,下载相应的cas版本。

由于网站只提供源码包而不提供发布包,所以需要自己下载来编译。

cas会为不同的客户端消费者提供client包,这里我们选择java-client作为演示。

编译客户端所需要的jar

java-cas-client-cas-client-3.4.1 Richard$ mvn clean install -Dmaven.test.skip
    

然后下载供测试的客户端示例代码(jasig在github上为我们提供了一个简易的客户端demo)

$ git clone https://github.com/UniconLabs/cas-sample-java-webapp.git

配置

客户端

  • 设置

参照以下文章为Tomcat配置好Https

Tomcat (1) —— Mac下配置Tomcat Https/SSL

【高可用HA】Apache (2) —— Mac下安装多个Apache Tomcat实例

  • 编译部署

然后编译我们的示例项目

:cas-sample-java-webapp Richard$ mvn clean install -Dmaven.test.skip

pom.xml

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>iamlabs.unicon.net</groupId>
<artifactId>cas-sample-java-webapp</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>war</packaging>
<name>CAS Example Java Web App</name>
<description>A sample web application that exercises the CAS protocol features via the Java CAS Client.</description>
<build>
    <finalName>cas-sample-java-webapp</finalName>
    <plugins>
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-compiler-plugin</artifactId>
            <version>2.5.1</version>
            <configuration>
                <source>1.6</source>
                <target>1.6</target>
            </configuration>
        </plugin>
    </plugins>
</build>

<dependencies>

    <dependency>
        <groupId>commons-logging</groupId>
        <artifactId>commons-logging</artifactId>
        <version>1.1.1</version>
    </dependency>

    <dependency>
        <groupId>log4j</groupId>
        <artifactId>log4j</artifactId>
        <version>1.2.16</version>
    </dependency>
    <dependency>
        <groupId>org.opensaml</groupId>
        <artifactId>opensaml1</artifactId>
        <version>1.1</version>
    </dependency>
    
    <dependency>
        <groupId>javax.servlet</groupId>
        <artifactId>javax.servlet-api</artifactId>
        <version>3.1.0</version>
        <scope>provided</scope>
    </dependency>

    <dependency>
        <groupId>org.jasig.cas.client</groupId>
        <artifactId>cas-client-core</artifactId>
        <version>3.2.1</version>
        <exclusions>
            <exclusion>
                <groupId>javax.servlet</groupId>
                <artifactId>servlet-api</artifactId>
            </exclusion>
        </exclusions>
    </dependency>

    <dependency>
        <groupId>commons-codec</groupId>
        <artifactId>commons-codec</artifactId>
        <version>1.6</version>
    </dependency>

    <dependency>
        <groupId>org.apache.santuario</groupId>
        <artifactId>xmlsec</artifactId>
        <version>1.4.3</version>
    </dependency>
</dependencies>

将编译生成的war包部署到我们的Tomcat容器中:

613455-20151214153711974-1840488430.png

项目主要包括两个页面

index.jsp(供登陆)

<%@page contentType="text/html"%>
<%@page pageEncoding="UTF-8"%>
<%@ page import="java.util.Map" %>
<%@ page import="java.util.Iterator" %>
<%@ page import="java.util.List" %>
<%@ page import="org.jasig.cas.client.authentication.AttributePrincipal" %>

<!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 Example Java Web App</title>
</head>
<body>

<h1>CAS Example Java Web App</h1>
<p>A sample web application that exercises the CAS protocol features via the Java CAS Client.</p>
<hr>

<p><b>Authenticated User Id:</b> <a href="../cas1/logout.jsp" title="Click here to log out"><%= request.getRemoteUser() %></a></p>

<%
if (request.getUserPrincipal() != null) {
  AttributePrincipal principal = (AttributePrincipal) request.getUserPrincipal();
  
  /*
  final String password = principal.getPassword();
  if (password != null) {
    out.println("<p><b>User Credentials:</b> " + password + "</p>");
  }
  */
  
  final Map attributes = principal.getAttributes();
  
  if (attributes != null) {
    Iterator attributeNames = attributes.keySet().iterator();
    out.println("<b>Attributes:</b>");
    
    if (attributeNames.hasNext()) {
      out.println("<hr><table border='3pt' width='100%'>");
      out.println("<th colspan='2'>Attributes</th>");
      out.println("<tr><td><b>Key</b></td><td><b>Value</b></td></tr>");

      for (; attributeNames.hasNext();) {
        out.println("<tr><td>");
        String attributeName = (String) attributeNames.next();
        out.println(attributeName);
        out.println("</td><td>");
        final Object attributeValue = attributes.get(attributeName);

        if (attributeValue instanceof List) {
          final List values = (List) attributeValue;
          out.println("<strong>Multi-valued attribute: " + values.size() + "</strong>");
          out.println("<ul>");
          for (Object value: values) {
            out.println("<li>" + value + "</li>");
          }
          out.println("</ul>");
        } else {
          out.println(attributeValue);
        }
        out.println("</td></tr>");
      }
      out.println("</table>");
    } else {
      out.print("No attributes are supplied by the CAS server.</p>");
    }
  } else {
    out.println("<pre>The attribute map is empty. Review your CAS filter configurations.</pre>");
  }
} else {
    out.println("<pre>The user principal is empty from the request object. Review the wrapper filter configuration.</pre>");
}
%>

</body>
</html>

logout.jsp(注销)

<%@page contentType="text/html"%>
<%@page pageEncoding="UTF-8"%>
<%@ page import="java.util.Map" %>
<%@ page import="java.util.Iterator" %>
<%@ page import="org.jasig.cas.client.authentication.AttributePrincipal" %>

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"
    "http://www.w3.org/TR/html4/loose.dtd">

<%
    session.invalidate();
%>

<html>
<head>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
    <title>CAS Example Java Web App</title>
</head>
<body>
    <h1>CAS Example Java Web App</h1>
    <p>Application session is now invalidated. You may also issue a request to "/cas/logout" to destroy the CAS SSO Session as well.</p>
    <hr>
    
    <a href="../cas1/index.jsp">Back to Home</a>
</body>
</html>
  • 配置CAS Authentication Filter

web.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">
<filter>
    <filter-name>CAS Authentication Filter</filter-name>
    <filter-class>org.jasig.cas.client.authentication.AuthenticationFilter</filter-class>
    <!-- <filter-class>org.jasig.cas.client.authentication.Saml11AuthenticationFilter</filter-class> -->
    <!-- <filter-class>org.jasig.cas.client.authentication.AuthenticationFilter</filter-class> -->
    <init-param>
        <param-name>casServerLoginUrl</param-name>
        <param-value>https://sso.hoau.com:8433/cas/login</param-value>
    </init-param>
    <init-param>
        <param-name>serverName</param-name>
        <param-value>https://app1.hoau.com:8413</param-value>
    </init-param>
</filter>

<filter>
    <filter-name>CAS Validation Filter</filter-name>
    <filter-class>org.jasig.cas.client.validation.Cas20ProxyReceivingTicketValidationFilter</filter-class>
    <!-- <filter-class>org.jasig.cas.client.validation.Saml11TicketValidationFilter</filter-class> -->  
    <!-- <filter-class>org.jasig.cas.client.validation.Cas20ProxyReceivingTicketValidationFilter</filter-class> --> 
    <init-param>
        <param-name>casServerUrlPrefix</param-name>
        <param-value>https://sso.hoau.com:8433/cas</param-value>
    </init-param>
    <init-param>
        <param-name>serverName</param-name>
        <param-value>https://app1.hoau.com:8413</param-value>
    </init-param>
    <init-param>
        <param-name>redirectAfterValidation</param-name>
        <param-value>true</param-value>
    </init-param>
    <init-param>
        <param-name>useSession</param-name>
        <param-value>true</param-value>
    </init-param>
    <init-param>
        <param-name>acceptAnyProxy</param-name>
        <param-value>true</param-value>
    </init-param>
    <!--
    <init-param>
        <param-name>proxyReceptorUrl</param-name>
        <param-value>/cas1/proxyUrl</param-value>
    </init-param>
    <init-param>
        <param-name>proxyCallbackUrl</param-name>
        <param-value>https://app1.hoau.com:8413/cas1/index.jsp</param-value>
    </init-param>
    -->
</filter>

<filter>
    <filter-name>CAS HttpServletRequest Wrapper Filter</filter-name>
    <filter-class>org.jasig.cas.client.util.HttpServletRequestWrapperFilter</filter-class>
</filter>

<filter-mapping>
    <filter-name>CAS Validation Filter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

<filter-mapping>
    <filter-name>CAS Authentication Filter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

<filter-mapping>
    <filter-name>CAS HttpServletRequest Wrapper Filter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

<welcome-file-list>
    <welcome-file>
        index.jsp
</welcome-file>
</welcome-file-list>
</web-app>
* 以上web.xml配置中有四处需要注意的地方
  1. cas目标服务器登陆页面的配置

     <init-param>
         <param-name>casServerLoginUrl</param-name>
         <param-value>https://sso.hoau.com:8433/cas/login</param-value>
     </init-param>
  2. 本机服务器的地址(即当前node-a)

     <init-param>
         <param-name>serverName</param-name>
         <param-value>https://app1.hoau.com:8413</param-value>
     </init-param>
  3. "Cas20ProxyReceivingTicketValidationFilter"

    2.x和3.x以前也有其他的Ticket校验方式,这里官方推荐用Cas20Proxy

     <filter>
     <filter-name>CAS Validation Filter</filter-name>
     <filter-class>org.jasig.cas.client.validation.Cas20ProxyReceivingTicketValidationFilter</filter-class>
     <init-param>
         <param-name>casServerUrlPrefix</param-name>
         <param-value>https://sso.hoau.com:8433/cas</param-value>
     </init-param>
     <init-param>
         <param-name>serverName</param-name>
         <param-value>https://app1.hoau.com:8413</param-value>
     </init-param>
  4. 暂时将proxyReceptorUrl和proxyCallbackUrl注释掉,因为这里没有使用代理

     <!--
     <init-param>
         <param-name>proxyReceptorUrl</param-name>
         <param-value>/cas1/proxyUrl</param-value>
     </init-param>
     <init-param>
         <param-name>proxyCallbackUrl</param-name>
         <param-value>https://app1.hoau.com:8413/cas1/index.jsp</param-value>
     </init-param>
     -->

测试

尝试访问

https://app1.hoau.com:8413/cas1

613455-20151214153754896-220731740.png

* 抱歉此处配图为node-b的

并使用我们在数据库里面预埋的数据"test01/psw01"登陆

613455-20151214153808646-506023957.png

使用相同方式将cas2部署到node-b节点,并指向cas的SSO服务器

*扩展

我们清空浏览器的cache和cookie,按照下列步骤操作

  1. 访问"https://app1.hoau.com:8413/cas1"

    系统会将我们重定向到"https://sso.hoau.com:8433/cas/login"。

  2. 输入用户名密码"test01/psw01"

    登陆成功

  3. 访问"https://app2.hoau.com:8423/cas2"

    系统会自动登陆使用用户名密码"test01/psw01"。

问题来了

这就是达到了SSO的效果,但是为什么是这样,原理是什么

结束

相关文章:

  • Median of Two Sorted Arrays
  • 固态硬盘上安装Windows8(ghost)启动问题
  • 王家林每日大数据语录Spark篇0003
  • [hive小技巧]同一份数据多种处理
  • 【191】◀▶ Powershell 命令集 Cmdlets
  • Github访问慢解决办法
  • ActionContext 中的数据详解
  • SpringMVC格式总结
  • QT内使用OpenCV
  • FRC与tableview的联合使用
  • Sublime Text 3安装与使用
  • 解放全天下开发者
  • Cesium 获取当前视图范围
  • Gradle介绍
  • 扩展SpringMVC以支持绑定JSON格式的请求参数
  • [译]Python中的类属性与实例属性的区别
  • 【译】React性能工程(下) -- 深入研究React性能调试
  • Centos6.8 使用rpm安装mysql5.7
  • CSS选择器——伪元素选择器之处理父元素高度及外边距溢出
  • Koa2 之文件上传下载
  • MySQL-事务管理(基础)
  • PAT A1092
  • 理解在java “”i=i++;”所发生的事情
  • 聊聊flink的BlobWriter
  • 系统认识JavaScript正则表达式
  • 小程序 setData 学问多
  • 一起参Ember.js讨论、问答社区。
  • FaaS 的简单实践
  • ​​​​​​​sokit v1.3抓手机应用socket数据包: Socket是传输控制层协议,WebSocket是应用层协议。
  • ​LeetCode解法汇总2808. 使循环数组所有元素相等的最少秒数
  • ## 临床数据 两两比较 加显著性boxplot加显著性
  • ### Error querying database. Cause: com.mysql.jdbc.exceptions.jdbc4.CommunicationsException
  • #pragma 指令
  • (30)数组元素和与数字和的绝对差
  • (保姆级教程)Mysql中索引、触发器、存储过程、存储函数的概念、作用,以及如何使用索引、存储过程,代码操作演示
  • (非本人原创)我们工作到底是为了什么?​——HP大中华区总裁孙振耀退休感言(r4笔记第60天)...
  • (附源码)springboot电竞专题网站 毕业设计 641314
  • (三) diretfbrc详解
  • (十七)devops持续集成开发——使用jenkins流水线pipeline方式发布一个微服务项目
  • (一)搭建springboot+vue前后端分离项目--前端vue搭建
  • (一一四)第九章编程练习
  • (转)c++ std::pair 与 std::make
  • (转)Linq学习笔记
  • (转)Scala的“=”符号简介
  • **CI中自动类加载的用法总结
  • .equal()和==的区别 怎样判断字符串为空问题: Illegal invoke-super to void nio.file.AccessDeniedException
  • .NET/C# 反射的的性能数据,以及高性能开发建议(反射获取 Attribute 和反射调用方法)
  • .NET开发不可不知、不可不用的辅助类(三)(报表导出---终结版)
  • .NET框架设计—常被忽视的C#设计技巧
  • .NET中winform传递参数至Url并获得返回值或文件
  • .NET中的十进制浮点类型,徐汇区网站设计
  • @Autowired 与@Resource的区别
  • @synthesize和@dynamic分别有什么作用?
  • [AIGC] Java 和 Kotlin 的区别
  • [AIGC] MySQL存储引擎详解