CAS 实现站内单点登录及实现第三方 OAuth、OpenId 登录(四)

Stella981
• 阅读 540

一、OAuth 配置

  1. 配置 OAuth 提供商

    <bean id="weibo" class="com.buession.oauth.provider.impl.WeiboProvider">
        <property name="key" value="the_key_for_weibo" />
        <property name="secret" value="the_secret_for_weibo" />
    </bean>
    <bean id="qq" class="com.buession.oauth.provider.impl.QqProvider">
        <property name="key" value="the_key_for_qq" />
        <property name="secret" value="the_secret_for_qq" />
    </bean>
    ... ...
    

    提供商 OAuth 2.0 版本需继承类 org.scribe.up.provider.BaseOAuth20Provider
    提供商 OAuth 1.0 版本需继承类 org.scribe.up.provider.BaseOAuth20Provider
    所有的 OAuth Provider 必须定义在 WEB-INF/spring-configuration/applicationContext.xml 中

  2. OAuthConfiguration 配置(在 WEB-INF/spring-configuration/applicationContext.xml 添加)

                                                             ... ...              

二、OAuth Action 配置


    在  WEB-INF/cas-servlet.xml 添加:

<bean id="oAuthAction" class="com.buession.cas.web.flow.OAuthAction"
    p:configuration-ref="oAuthConfiguration" 
    p:centralAuthenticationService-ref="centralAuthenticationService" />

    com.buession.cas.web.flow.OAuthAction 与 org.jasig.cas.support.oauth.web.flow.OAuthAction 的唯一区别就是:前者把 org.jasig.cas.support.oauth.authentication.principal.OAuthCredentials 注册到了 spring FlowScope 中,以便后续方便使用。

final Credentials credentials = new OAuthCredentials(credential);
flowScope.put(Constants.OAUTH_CREDENTIALS, credentials);

    在 WEB-INF/login-webflow.xml 中添加:

<action-state id="oAuthAction">
    <evaluate expression="oAuthAction" />
    <transition on="success" to="sendTicketGrantingTicket" />
    <transition on="error" to="ticketGrantingTicketExistsCheck" />
</action-state>

作用是 OAuth 身份验证后进行拦截来自供应商的 OAuth 回调
同时,此时会向页面注册“请求用户授权Token URL”的变量:驱动提供商 type+"Url",即:$provider.getType()+"Url",默认即为 Provider 的类名+"Url",如:WeiboProviderUrl,这样在模板中即可通过:${WeiboProviderUrl} 形式的来生成“请求用户授权Token URL”的链接地址。

<a href="${QqProviderUrl}">QQ 登录</a>
<a href="${WeiboProviderUrl}">新浪微博登录</a>

三、OAuth 用户站内绑定

    在大部分情况下,我们都需要把通过 OAuth 拿到的用户信息,如ID、昵称、头像、key 等存储一份到本地数据库,以方便使用。
    这是需要修改:

<action-state id="oAuthAction">
    <evaluate expression="oAuthAction" />
    <transition on="success" to="oAuthBindCheckAction" />
    <transition on="error" to="ticketGrantingTicketExistsCheck" />
</action-state>

    在  WEB-INF/cas-servlet.xml 添加:

<!-- 验证 OAuth 用户是否已绑定 Action -->
<bean id="oAuthBindCheckAction" class="com.sso.web.flow.OAuthBindCheckAction"
    p:configuration-ref="oAuthConfiguration" 
    p:jdbcTemplate-ref="slaveJdbcTemplate" 
    p:sql="SELECT count(0) FROM `members_bind` WHERE `type` = ? AND `key` = ? LIMIT 1" />
<!-- OAuth 用户绑定 Action -->
<bean id="oAuthBindAction" class="com.sso.web.flow.OAuthBindAction"
    p:configuration-ref="oAuthConfiguration" 
    p:jdbcTemplate-ref="masterJdbcTemplate"
    p:transactionManager-ref="masterTransactionManager"
    p:passwordEncoder-ref="passwordEncoder" />

    在 WEB-INF/login-webflow.xml 中添加:

<action-state id="oAuthBindCheckAction">
    <evaluate expression="oAuthBindCheckAction" />
    <transition on="success" to="sendTicketGrantingTicket" />
    <transition on="error" to="oAuthBindAction" />
</action-state>

<action-state id="oAuthBindAction">
    <evaluate expression="oAuthBindAction" />
    <transition on="success" to="sendTicketGrantingTicket" />
    <transition on="error" to="ticketGrantingTicketExistsCheck" />
</action-state>

    OAuthBindCheckAction.java(验证 OAuth 用户是否绑定 Action)

public class OAuthBindCheckAction extends com.buession.cas.web.flow.OAuthBindJdbcCheckAction {

    @Override
    protected boolean valid(final OAuthProvider provider, final OAuthCredentials credentials) {
        if (credentials == null) {
            return false;
        }

        BaseOAuthProfile profile = (BaseOAuthProfile) credentials.getUserProfile();
        if (profile == null) {
            return false;
        }

        String providerName = provider.getType().replace("Provider", "").toLowerCase();
        try {
           return jdbcTemplate.queryForObject(sql, Integer.class, providerName,
                    profile.getId()) > 0;
        } catch (final IncorrectResultSizeDataAccessException e) {
        }

        return false;
    }

}

    OAuthBindAction.java(OAuth 用户绑定 Action)

public class OAuthBindAction extends com.buession.cas.web.flow.OAuthBindAction {

    @NotNull
    private DataSourceTransactionManager transactionManager;

    @NotNull
    private PasswordEncoder passwordEncoder;

    public DataSourceTransactionManager getTransactionManager() {
        return transactionManager;
    }

    public void setTransactionManager(DataSourceTransactionManager transactionManager) {
        this.transactionManager = transactionManager;
    }

    public PasswordEncoder getPasswordEncoder() {
        return passwordEncoder;
    }

    public void setPasswordEncoder(PasswordEncoder passwordEncoder) {
        this.passwordEncoder = passwordEncoder;
    }

    @Override
    protected boolean bind(final HttpServletRequest request, final OAuthProvider provider,
            final OAuthCredentials credentials) {
        if (credentials == null) {
            return false;
        }

        BaseOAuthProfile profile = (BaseOAuthProfile) credentials.getUserProfile();
        if (profile == null) {
            return false;
        }

        TransactionTemplate transactionTemplate = new TransactionTemplate(getTransactionManager());
        return transactionTemplate.execute(new TransactionCallback<Boolean>() {
            @Override
            public Boolean doInTransaction(TransactionStatus status) {
                BaseOAuthProfile profile = (BaseOAuthProfile) credentials.getUserProfile();
                if (profile == null) {
                    return false;
                }

                ... ...

                return false;
            }
        });
    }

}

    在 WEB-INF/deployerConfigContext.xml 中添加

<bean id="authenticationManager" class="org.jasig.cas.authentication.AuthenticationManagerImpl">
    <property name="authenticationMetaDataPopulators">
        <list>
            <!-- 如果想从 OAuth 获取用户信息,则需添加 -->
            <bean class="com.buession.cas.oauth.authentication.OAuthAuthenticationMetaDataPopulator" />
            <!-- 或者 -->
            <bean class="org.jasig.cas.support.oauth.authentication.OAuthAuthenticationMetaDataPopulator" />
        </list>
    </property>
    <property name="credentialsToPrincipalResolvers">
        <list>
            <!--  -->
            <bean class="org.jasig.cas.support.oauth.authentication.principal.OAuthCredentialsToPrincipalResolver"
                p:attributeRepository-ref="oAuthAttributeRepositoryDao" />
        </list>
    </property>
    <property name="authenticationHandlers">
        <list>
            <!-- 使其 CAS 支持 OAuth 认证 -->
            <bean class="org.jasig.cas.support.oauth.authentication.handler.support.OAuthAuthenticationHandler"
                p:configuration-ref="oAuthConfiguration" />
        </list>
    </property>
</bean>

<!-- OAuth 登录成功后,查询本地用户数据 -->
<bean id="oAuthAttributeRepositoryDao" class="com.sso.persondir.OAuthSingleRowJdbcPersonAttributeDao">
    <property name="jdbcTemplate" ref="slaveJdbcTemplate" />
    <property name="queryTemplate" value="SELECT `m`.* FROM `members` AS `m`, `members_bind` AS `mb` WHERE `m`.`id` = `mb`.`members_id` AND `mb`.`type` = ? AND `mb`.`app_userid` = ? LIMIT 1" />
    <property name="resultAttributeMapping">
        <map>
            <entry key="uid" value="id" />
            <entry key="username" value="username" />
            <entry key="email" value="email" />
            ... ...
        </map>
    </property>
</bean>

com.buession.cas.oauth.authentication.OAuthAuthenticationMetaDataPopulator 与 org.jasig.cas.support.oauth.authentication.OAuthAuthenticationMetaDataPopulator 的区别:前者能够将 OAuth 获取到的用户信息与本地查询到的用户信息合并,得到更加丰富完善的用户信息;否则就只能返回 OAuth 获取到的用户信息。

OAuthSingleRowJdbcPersonAttributeDao.java()

public class OAuthSingleRowJdbcPersonAttributeDao extends com.buession.cas.service.persondir.support.jdbc.OAuthSingleRowJdbcPersonAttributeDao {

    @Override
    protected List<Map<String, Object>> query(ProviderId providerId) {
        final ParameterizedRowMapper<Map<String, Object>> rowMapper = getRowMapper();
        List<Map<String, Object>> results = null;

        if (providerId != null) {
            results = jdbcTemplate.query(queryTemplate, rowMapper, providerId.getProviderName(),
                    providerId.getId());
        } else {
            results = jdbcTemplate.query(queryTemplate, rowMapper);
        }

        return results;
    }

    /**
     * @param uid
     * @return ProviderId
     */
    @Override
    protected ProviderId convertAttributesMap(String uid) {
        String[] temp = uid.split("#");
        return temp.length >= 2 ? new ProviderId(temp[0].replace("Profile", "").toLowerCase(), temp[1]) : null;
    }

}

使用 org.jasig.cas.support.oauth.authentication.OAuthAuthenticationMetaDataPopulator,所得结果,如图:

CAS 实现站内单点登录及实现第三方 OAuth、OpenId 登录(四)

使用 com.buession.cas.oauth.authentication.OAuthAuthenticationMetaDataPopulator,所得结果,如图:

CAS 实现站内单点登录及实现第三方 OAuth、OpenId 登录(四)

点赞
收藏
评论区
推荐文章
blmius blmius
2年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Wesley13 Wesley13
2年前
java将前端的json数组字符串转换为列表
记录下在前端通过ajax提交了一个json数组的字符串,在后端如何转换为列表。前端数据转化与请求varcontracts{id:'1',name:'yanggb合同1'},{id:'2',name:'yanggb合同2'},{id:'3',name:'yang
Jacquelyn38 Jacquelyn38
2年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Stella981 Stella981
2年前
KVM调整cpu和内存
一.修改kvm虚拟机的配置1、virsheditcentos7找到“memory”和“vcpu”标签,将<namecentos7</name<uuid2220a6d1a36a4fbb8523e078b3dfe795</uuid
Wesley13 Wesley13
2年前
PHP创建多级树型结构
<!lang:php<?php$areaarray(array('id'1,'pid'0,'name''中国'),array('id'5,'pid'0,'name''美国'),array('id'2,'pid'1,'name''吉林'),array('id'4,'pid'2,'n
Stella981 Stella981
2年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Wesley13 Wesley13
2年前
MYSQL查询用户下多个角色信息
<resultMapid"baseBeanUser"type"com...vo.system.TUserVoOut"<idcolumn"id"property"id"/<resultcolumn"name"property"name"/
Wesley13 Wesley13
2年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
Python进阶者 Python进阶者
3个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这