OpenSAML 使用引导 II : Service Provider 的实现之AuthnRequest

前文OpenSAML 使用引导 I : 简介介绍了OpenSAML的基础概况, 本文将从Service Provider(SP)角度出发,讲解如何使用OpenSAML如申请身份鉴别请求(AuthnRequest),并从IDP出得到断言的引用标识——SAML Artifact

相关阅读SAML2.0入门指南,
源码地址:https://github.com/sunrongxin7666/OpenSAML-ref-project-demo-v3.git

1、从OpenSAML的角度再看身份鉴别过程

SAML

1. 用户尝试获取访问权限

用户对SP中的资源进行请求。这一步中,SP会做出判断是否要需要鉴别用户身份。一般而言,如果当前用户在SP上不存在已经被认证的session信息,就需要对用户提出身份鉴别请求。

2. 用户被重定向到IDP

如果需要对用户进行身份鉴别,SP将会创建AuthnRequest对象指明要用户要如何才能被鉴别。这个AuthnRequest将会被以URL参数的形式编码到HTTP请求中,然后通过浏览器重定向到IDP。

3. 用户身份别鉴别

IDP解码获得AuthnRequest并根据其中的要求来鉴别用户身份。

4. 已鉴别的用户被发送回SP

如果用户通过身份鉴别,IDP会将认证信息和一个标识联系起来,这个标识被称为SAML制品(SAML Artifact,简称制品)。这个制品也是以URL参数的形式加入到HTTP请求中,然后重定向发回SP。

5. SP请求认证信息

SP创建ArtifactResolve对象,并将Artifact包含在其中。这个ArtifactResolve对象通过SOAP请求发送到IDP。

6. IDP响应请求返回认证信息

IDP接受到ArtifactResolve并将Artifact抽离出来。IDP通过ArtifactResponse响应SOAP请求,ArtifactResponse中包含认证信息,以SAML断言格式包含在其中。

2. 第一步:用户拦截

用户拦截

这里讨论鉴别过程的第一步。实际上,用户拦截并不是认证过程的一部分,但是确实这一步却决定着鉴别过程是否会发生。

用户的交互以拦截非认证的用户去尝试获取SP资源时开始。对于Java Web应用而言,Servlet过滤器是一个很好的选择去做这样的拦截。过滤器检查是否当前用户已被认证,如果已被验证则允许访问,反之要启动身份验证流程。

public class AccessFilter implements Filter {
    private static Logger logger = LoggerFactory
.getLogger(AccessFilter.class);
    @Override
    public void init(FilterConfig filterConfig) throws ServletException{
        JavaCryptoValidationInitializer javaCryptoValidationInitializer = 
            new JavaCryptoValidationInitializer();
        try {       
            javaCryptoValidationInitializer.init();
        } catch (InitializationException e) {
            e.printStackTrace();
        } 
        try {
            InitializationService.initialize();
        } catch (InitializationException e) {
            new RuntimeException("Initialization failed");
    }

    public void doFilter(
        ServletRequest request,
        ServletResponse response, 
        FilterChain chain) throws IOException, ServletException {
        HttpServletRequest httpServletRequest =
            (HttpServletRequest) request;
        HttpServletResponse httpServletResponse =
            (HttpServletResponse) response;
        if (httpServletRequest.getSession().getAttribute(
        SPConstants.AUTHENTICATED_SESSION_ATTRIBUTE) != null) {
            chain.doFilter(request, response);
        } else {
        //将本来要访问的目标路径保存到Session
        setGotoURLOnSession(httpServletRequest);
        redirectUserForAuthentication(httpServletResponse);
        }
    }
}

如果用户已经通过身份鉴别,则session中会有AUTHENTICATED_SESSION_ATTRIBUTE,此时用户是已经被认证的,过滤器应该不对该操作做任何处理;反之,如果AUTHENTICATED_SESSION_ATTRIBUTE并不存在则意味着需要开启鉴别流程:保留当前的目标URL,然后重定向到IDP。

3. 第二步,鉴别请求

鉴别请求

这一部分才是SAML身份鉴别流程的开始。SP通过发送SAML AuthnRequest到IDP,来请求鉴别用户身份。这里将以最常见的的方法HTTP重定向为例来讲解。
AuthnRequest对象

1. 构建AuthnRequest对象

AuthnRequest authnRequest = OpenSAMLUtils
    .buildSAMLObject(AuthnRequest.class);

其中如下属性需要设置:

  • 请求时间:该对象创建的时间,以判断其时效性,

authnRequest.setIssueInstant(new DateTime());

  • 目标URL:AuthnRequest的目标地址,IDP地址,

authnRequest.setDestination(getIPDDestination());

  • 传输SAML断言所使用的绑定:也就是用何种协议来使用Artifact取回真正的认证信息,

authnRequest.setProtocolBinding(SAMLConstants.SAML2_ARTIFACT_BINDING_URI);

  • SP地址: 也就是SAML断言返回的地址

authnRequest
.setAssertionConsumerServiceURL(getAssertionConsumerEndpoint());

  • 请求的ID:为当前请求设置ID,一般为随机数,

authnRequest.setID(OpenSAMLUtils.generateSecureRandomId());
注意:获得安全随机数的方法:

RandomIdentifierGenerationStrategy secureRandomIdGenerator = new RandomIdentifierGenerationStrategy();
String id = secureRandomIdGenerator.generateIdentifier();
  • Issuer: 发行人信息,也就是SP的ID,一般是SP的URL
Issuer issuer = OpenSAMLUtils.buildSAMLObject(Issuer.class);
issuer.setValue(SPConstants.SP_ENTITY_ID);
authnRequest.setIssuer(issuer);
  • NameID:IDP对于用户身份的标识;NameID policy是SP关于NameID是如何被创建的说明;Format指明SP需要返回什么类型的标识(SAML Artifact);属性AllowCreate指明IDP是否被允许当发现用户不存在时创建用户账号。
NameIDPolicy nameIDPolicy =
OpenSAMLUtils.buildSAMLObject(NameIDPolicy.class);
nameIDPolicy.setFormat(NameIDType.TRANSIENT);
nameIDPolicy.setAllowCreate(true);
authnRequest.setNameIDPolicy(nameIDPolicy);

NameID Formats:
在SAML中有多种NameID的格式存在,比如Kerberos,邮箱以及Windows域限定名称(Windows Domain Qualified Name),这里要特别说明如下两种:

  • 持久标识(Persistent Identifier):一个随机的ID标识被分配给用户,以避免暴露用户的真实账户。无论用户何时登入,都会返回相同的标识。SP可以将这个标识和本地的用户账号绑定;
  • 临时标识(Transient Identifier):临时标识是一个和用户账户没有关系的随机标识,不会被重复使用,用户每次登陆所返回的标识都是不一样的。
  • 请求认证上下文(requested Authentication Context):
    SP对于认证的要求,包含SP希望IDP如何验证用户,也就是IDP要依据什么来验证用户身份。
authnRequest.setRequestedAuthnContext(buildRequestedAuthnContext());

可以如此获得*RequestedAuthnContext *

RequestedAuthnContext requestedAuthnContext = OpenSAMLUtils
.buildSAMLObject(RequestedAuthnContext.class);
  • authnContextClassRef
    authnContextClassRef代表着鉴别方式的一个选项。比如一个网站同时支持口令认证和Kerberos两种方式,则口令认证和Kerberos就是两个authnContextClassRef选项。请求认证上下文中就可以同时包含这两个。
AuthnContextClassRef passwordAuthnContextClassRef 
    = OpenSAMLUtils.buildSAMLObject(AuthnContextClassRef.class);
passwordAuthnContextClassRef
    .setAuthnContextClassRef(AuthnContext.PASSWORD_AUTHN_CTX);
requestedAuthnContext.getAuthnContextClassRefs()
    .add(passwordAuthnContextClassRef);
requestedAuthnContext
    .setComparison(AuthnContextComparisonTypeEnumeration.MINIMUM);

同时请求认证上下文也可能有多个,如果是这样的情况他们就要安装优先级排列。

Comparison代表着如何IDP要如何依据所给出的鉴别方式选项处理鉴别结果,其取值包括:

  1. Minimum,最少策略,满足这个方式或者比它更安全方式就通过验证;
  2. Better,更优策略,需要满足比这个方式更为安全的方式才能通过验证;
  3. Exact,精准模式,必须满足当前方式才能通过验证;
  4. Maximum,最多策略,需要满足安全性最强的方式才能通过认证。

发送鉴别请求

AuthnReques URL

使用HTTP重定向绑定(HTTP Redirect Binding)将鉴别请求发送到Idp。 AuthnRequest以参数的形式附加在HTTP请求中,但是没有强制要求需要对其签名,不过为了信息的完整性强烈建议这样做,签名的结果作为一个独立的URL参数传输,以便于验证。同时建议使用HTTPS来保证数据传输的完整性和机密性。

为了帮助数字签名以及序列化参数以发送重定向消息,OpenSAML提供了HTTPRedirectDefalteEncoder,它将帮助我们来对于AuthnRequest进行序列化和签名,并把消息和用户一起重定向到Idp。

OpenSAML message encoders是对SAML消息传输的一种抽象封装,HTTPRedirectDefalteEncoder也是如此,它是对HTTP重定向绑定过程的抽象。

编码器(encoder)用来处理数据对象,也就是消息上下文(MessageContext),其中包含消息的信息内容和传输细节。

Message Context

在新版OpenSAML中,Message Context相关的类如下:

  • MessageContext:主类,主环境上下文;
  • SAMLPeerEntityContext:关于传输对端实体的信息,对于IDP就是SP,对于SP就是IDP;通常该对象不包括很多信息,但是会包含一个和多个子内容;
  • SAMLEndpointContext:端点信息;
  • SecurityParametersContext:关于签名和加密的信息;
  • SAMLMessageInfoContext:记录issue和ID等基本信息。

以上contexts都是主环境上下文的子内容创建的,以下使用实例:

MessageContext context = new MessageContext();
SAMLPeerEntityContext peerEntityContext = context.getSubcontext(SAMLPeerEntityContext.class, true);
SAMLEndpointContext endpointContext = peerEntityContext.getSubcontext(SAMLEndpointContext.class, true);

getSubcontext方法的最后一个参数表示如果存在该信息不存在是否创建,当然也可以使用setter方法来设置:

endpointContext.setEndpoint(getIPDEndpoint());

一般而言,SAMLEndpointContext
AuthnRequest所必须的,用以指明消息发送的目的地。SAMLEndpointContextSAMLPeerEntityContext的子内容。如何创建请见下一部分。

创建MessageContext

  1. 创建主环境上下文:
 MessageContext context = new MessageContext();
  1. 设置要发送的消息(这里就是AuthnRequest)到主环境上下文:
context.setMessage(authnRequest);
  1. 创建SAMLPeerEntityContext and SAMLEndpointContext
SAMLPeerEntityContext peerEntityContext = context.getSubcontext(SAMLPeerEntityContext.class, true);
SAMLEndpointContext endpointContext = peerEntityContext.getSubcontext(SAMLEndpointContext.class, true);
  1. 创建目的地端点并将其设置到SAMLEndpointContext
SingleSignOnService endpoint = OpenSAMLUtils.buildSAMLObject(SingleSignOnService.class);
endpoint.setBinding(SAMLConstants.SAML2_REDIRECT_BINDING_URI); 
endpoint.setLocation(getIPDSSODestination()); 
context.setPeerEntityEndpoint(endpoint);
endpointContext.setEndpoint(endpoint);
  1. SecurityParametersContext是可选项,但强烈建议创建它来签名参数信息
SignatureSigningParameters signatureSigningParameters = new SignatureSigningParameters();
signatureSigningParameters.setSigningCredential(SPCredentials.getCredential());
signatureSigningParameters.setSignatureAlgorithm(SignatureConstants.ALGO_ID_SIGNATURE_RSA_SHA256);
context.getSubcontext(SecurityParametersContext.class,true)
    .setSignatureSigningParameters(signatureSigningParameters);

SecurityParametersContext被设置后,HTTPRedirectDefalteEncoder会自动帮我们对AuthnRequest签名,并添加签名结果和签名算法到URL参数中。

  1. 创建HTTPRedirectDefalteEncoder,并将消息环境上下文赋予它,同时为其设置HTTPServletResponse
HTTPRedirectDeflateEncoder encoder = new HTTPRedirectDeflateEncoder();

encoder.setMessageContext(context);
encoder.setHttpServletResponse(httpServletResponse);
  1. encoder被初始化,然后调用encode发送消息。encode方法将会压缩消息(先使用RC1951-DEFLATE Compressed Data Format Specification version 作为默认方法压缩数据,在对压缩后的数据信息Base64编码),生成签名,添加结果到URL并从定向用户到Idp.
encoder.initialize();
encoder.encode();

以下是redirect URLAuthnRequest XML的实例:

redirect URL

AuthnRequest XML

接下来SP将使用SOAP协议将Artifact发给IDP换取断言信息(Assertion),由于篇幅有限,这部分内容将会在后面的文章中讲解,敬请期待,欢迎关注和指教。
最后给出源码地址 https://github.com/sunrongxin7666/OpenSAML-ref-project-demo-v3.git

更多关于SAML协议的是实现的内容,请参见本人编写的一系列教程文章,其介绍如何使用OpenSAML,欢迎阅读指正:

  1. OpenSAML 使用引导 I : 简介
  2. OpenSAML 使用引导 II : Service Provider 的实现之AuthnRequest
  3. OpenSAML 使用引导 III: Service Provider 的实现之Artifact与断言
  4. OpenSAMl 使用引导IV: 安全特性
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352

推荐阅读更多精彩内容