spring security oauth2 的四种获取token 的模式及注意事项

authorizationCode授权码模式

https://www.cnblogs.com/hellxz/p/oauth2_oauthcode_pattern.html

password密码模式

https://www.cnblogs.com/hellxz/p/12041495.html

这里需要注意的是,认证端的endpoint必须配置authenticationManager,因为在源码里,不配置compositeGranter是没有password这种grantType的,导致会报错unsupport_grant_type的异常的。

认证配置类需要配置:

@Override
    public void configure(AuthorizationServerEndpointsConfigurer endpoints) throws Exception {
        endpoints.authenticationManager(authenticationManager);//密码模式必须添加authenticationManager
    }

AuthorizationServerEndpointsConfigurer类:

private List<TokenGranter> getDefaultTokenGranters() {
        ClientDetailsService clientDetails = clientDetailsService();
        AuthorizationServerTokenServices tokenServices = tokenServices();
        AuthorizationCodeServices authorizationCodeServices = authorizationCodeServices();
        OAuth2RequestFactory requestFactory = requestFactory();

        List<TokenGranter> tokenGranters = new ArrayList<TokenGranter>();
        tokenGranters.add(new AuthorizationCodeTokenGranter(tokenServices, authorizationCodeServices, clientDetails,
                requestFactory));
        tokenGranters.add(new RefreshTokenGranter(tokenServices, clientDetails, requestFactory));
        ImplicitTokenGranter implicit = new ImplicitTokenGranter(tokenServices, clientDetails, requestFactory);
        tokenGranters.add(implicit);
        tokenGranters.add(new ClientCredentialsTokenGranter(tokenServices, clientDetails, requestFactory));
        if (authenticationManager != null) {
            tokenGranters.add(new ResourceOwnerPasswordTokenGranter(authenticationManager, tokenServices,
                    clientDetails, requestFactory));
        }
        return tokenGranters;
    }

ClientCredentials凭证模式(客户端模式)

https://www.cnblogs.com/hellxz/p/12041588.html

这里需要注意,返回的token格式,虽然定义了 自定义格式的token(继承DefaultOAuth2AccessToken并为其定义序列化器,使得返回带上msg和code这种统一json格式)

在获取token的过程,我们会经过这一代码

TokenEndpoint类
OAuth2AccessToken token = getTokenGranter().grant(tokenRequest.getGrantType(), tokenRequest);

grant方法是AbstractTokenGranter中执行的,其中继承AbstractTokenGranter的有五种,分别是:

  • ResourceOwnerPasswordTokenGranter

  • RefreshTokenGranter

  • ClientCredentialsTokenGranter

  • ImplicitTokenGranter

  • AuthorizationCodeTokenGranter

其中 ClientCredentialsTokenGranter 是有自己的grant()方法,其余四种是没有的,这就使得我们拿到DefaultOAuth2AccessToken之后再在自定义的TokenEnhancer中转化成自定义格式的OAuth2AccessToken,会原封不动的按照序列化器返回出去,而ClientCredentialsTokenGranter 的grant方法,如下:

@Override
    public OAuth2AccessToken grant(String grantType, TokenRequest tokenRequest) {
        OAuth2AccessToken token = super.grant(grantType, tokenRequest); //这里拿到的还是自定义的OAuth2AccessToken
        if (token != null) {
            DefaultOAuth2AccessToken norefresh = new DefaultOAuth2AccessToken(token);
            // The spec says that client credentials should not be allowed to get a refresh token
            if (!allowRefresh) {
                norefresh.setRefreshToken(null);
            }
            token = norefresh;
        }
        return token;
    }

第一句代码其实返回就是自定义的OAuth2AccessToken,之后在下面的判断中,就变为DefaultOAuth2AccessToken ,所以返回就是DefaultOAuth2AccessToken ,没有code、msg的统一json格式的token。而且它是没有refresh_token。

代码部分

这时我们可以自定义ClientCredentialsTokenGranter ,并且配置上就可以了:
首先就是模拟ClientCredentialsTokenGranter ,构建出自定义的ClientCredentialsTokenGranter :注意返回的是MyOauth2AccessToken

public class CustomClientCredentialsTokenGranter extends AbstractTokenGranter {
    private static final String GRANT_TYPE = "client_credentials";
    private boolean allowRefresh = false;

    public CustomClientCredentialsTokenGranter(AuthorizationServerTokenServices tokenServices,
                                         ClientDetailsService clientDetailsService, OAuth2RequestFactory requestFactory) {
        this(tokenServices, clientDetailsService, requestFactory, GRANT_TYPE);
    }

    protected CustomClientCredentialsTokenGranter(AuthorizationServerTokenServices tokenServices,
                                            ClientDetailsService clientDetailsService, OAuth2RequestFactory requestFactory, String grantType) {
        super(tokenServices, clientDetailsService, requestFactory, grantType);
    }

    public void setAllowRefresh(boolean allowRefresh) {
        this.allowRefresh = allowRefresh;
    }

    @Override
    public OAuth2AccessToken grant(String grantType, TokenRequest tokenRequest) {
        OAuth2AccessToken token = super.grant(grantType, tokenRequest);
        if (token != null) {
            //注意类型,这个自定义的token是含有自定义的序列化器,有msg和code返回的
            MyOauth2AccessToken norefresh = new MyOauth2AccessToken(token);
            // The spec says that client credentials should not be allowed to get a refresh token
            if (!allowRefresh) {
                norefresh.setRefreshToken(null);
            }
            token = norefresh;
        }
        return token;
    }

}

然后在认证服务器配置类中:

    @Override
    public void configure(AuthorizationServerEndpointsConfigurer endpoints) throws Exception {
        //MUST:密码模式下需设置一个AuthenticationManager对象,获取 UserDetails信息
        //末确认点.userDetailsService(userDetailsService)

        /*
          使用 /pig4cloud/login 覆盖 原有的/oauth/token,注意这里是覆盖一旦配置 原有路径将失效
          endpoints.pathMapping("/oauth/token","/pig4cloud/login");
        */

        endpoints.allowedTokenEndpointRequestMethods(HttpMethod.GET, HttpMethod.POST);
        endpoints.approvalStore(jdbcApprovalStore());
     
        //tokenServices没有的话会自动创建一个默认的
        endpoints.tokenServices(customTokenService());
        //密码password模式必须加上
        endpoints.authenticationManager(authenticationManager);


        DefaultOAuth2RequestFactory factory = new DefaultOAuth2RequestFactory(jdbcClientDetailsService());
        CustomClientCredentialsTokenGranter granter = new CustomClientCredentialsTokenGranter(
                customTokenService(),jdbcClientDetailsService(),factory);

        granter.setAllowRefresh(true);

        endpoints.tokenGranter(granter);
    }

这时返回就是统一的json格式,以及返回有refresh_token

{
    "msg": "Success",
    "code": 1000,
    "data": {
        "access_token": "3d1f89d4-e410-4559-b50b-6afa4b63fe5f",
        "refresh_token": "8f68f072-8bc7-4f18-be97-16ae20e44926",
        "scope": "user_info",
        "customInfo": "extra thing额外的东西",
        "token_type": "bearer",
        "expires_in": 6943
    }
}

implicit隐秘模式

https://www.cnblogs.com/hellxz/p/oauth2_impilit_pattern.html

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

推荐阅读更多精彩内容