Spring Security OAuth2.0系列文章:html
前面两篇文章详细讲解了如何基于spring boot + oath2.0搭建认证中心和资源中心,本篇文章将会讲解集成jwt以及将客户端信息和受权码信息保存到数据库。java
JSON Web Token(JWT)是一个开放的行业标准(RFC 7519),它定义了一种简介的、自包含的协议格式,用于在通讯双方传递json对象,传递的信息通过数字签名能够被验证和信任。JWT能够使用HMAC算法或使用RSA的公钥/私钥对来签名,防止被篡改。git
官网:https://jwt.io/web
标准: https://tools.ietf.org/html/rfc7519算法
JWT令牌的优势:spring
1)jwt基于json,很是方便解析。sql
2)能够在令牌中自定义丰富的内容,易扩展。数据库
3)经过非对称加密算法及数字签名技术,JWT防止篡改,安全性高。json
4)资源服务使用JWT可不依赖认证服务便可完成受权。安全
缺点:
1)JWT令牌较长,占存储空间比较大,这意味着会耗费必定的带宽资源
2)JWT签名和验签都要耗费处理器资源
JWT令牌由三部分组成,每部分中间使用点(.)分隔,好比:xxxxx.yyyyy.zzzzz
2.1 Header
头部包括令牌的类型(即JWT)及使用的哈希算法(如HMAC SHA256或RSA)一个例子以下:
下边是Header部分的内容
{ "alg": "HS256", "typ": "JWT" }
将上边的内容使用Base64Url编码,获得一个字符串就是JWT令牌的第一部分。
2.2 Payload
第二部分是负载,内容也是一个json对象,它是存放有效信息的地方,它能够存放jwt提供的现成字段,好比:iss(签发者),exp(过时时间戳), sub(面向的用户)等,也可自定义字段。此部分不建议存放敏感信息,由于此部分能够解码还原原始内容。最后将第二部分负载使用Base64Url编码,获得一个字符串就是JWT令牌的第二部分。
一个例子:
{ "sub": "1234567890", "name": "456", "admin": true }
2.3 Signature
第三部分是签名,此部分用于防止jwt内容被篡改。
这个部分使用base64url将前两部分进行编码,编码后使用点(.)链接组成字符串,最后使用header中声明签名算法进行签名。
一个例子:
HMACSHA256( base64UrlEncode(header) + "." + base64UrlEncode(payload), secret )
TokenConfig类的修改
@Configuration public class TokenConfig { private static final String SIGNING_KEY = "auth123"; @Bean public TokenStore tokenStore() { return new JwtTokenStore(accessTokenConverter()); } @Bean public JwtAccessTokenConverter accessTokenConverter(){ JwtAccessTokenConverter jwtAccessTokenConverter = new JwtAccessTokenConverter(); jwtAccessTokenConverter.setSigningKey(SIGNING_KEY);//对称秘钥,资源服务器使用该秘钥来验证 return jwtAccessTokenConverter; } }
AuthorizationServerTokenServices设置Token加强类
@Autowired private JwtAccessTokenConverter jwtAccessTokenConverter; @Bean public AuthorizationServerTokenServices tokenServices(){ DefaultTokenServices services = new DefaultTokenServices(); services.setClientDetailsService(clientDetailsService); services.setSupportRefreshToken(true); services.setTokenStore(tokenStore); services.setAccessTokenValiditySeconds(7200); services.setRefreshTokenValiditySeconds(259200); TokenEnhancerChain tokenEnhancerChain = new TokenEnhancerChain(); tokenEnhancerChain.setTokenEnhancers(Collections.singletonList(jwtAccessTokenConverter)); services.setTokenEnhancer(tokenEnhancerChain); return services; }
而后就能够测试了:
获得响应结果:
{ "access_token": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJhdWQiOlsicmVzMSJdLCJ1c2VyX25hbWUiOiJ6aGFuZ3NhbiIsInNjb3BlIjpbIlJPTEVfQURNSU4iLCJST0xFX1VTRVIiLCJST0xFX0FQSSJdLCJleHAiOjE2MTAzNzI5MzUsImF1dGhvcml0aWVzIjpbInAxIiwicDIiXSwianRpIjoiOWQzMzRmZGMtOTcwZC00YmJkLWI2MmMtZDU4MDZkNTgzM2YwIiwiY2xpZW50X2lkIjoiYzEifQ.gZraRNeX-o_jKiH7XQgg3TlUQBpxUcXa2-qR_Treu8U", "token_type": "bearer", "refresh_token": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJhdWQiOlsicmVzMSJdLCJ1c2VyX25hbWUiOiJ6aGFuZ3NhbiIsInNjb3BlIjpbIlJPTEVfQURNSU4iLCJST0xFX1VTRVIiLCJST0xFX0FQSSJdLCJhdGkiOiI5ZDMzNGZkYy05NzBkLTRiYmQtYjYyYy1kNTgwNmQ1ODMzZjAiLCJleHAiOjE2MTA2MjQ5MzUsImF1dGhvcml0aWVzIjpbInAxIiwicDIiXSwianRpIjoiN2U1NzE0NTgtNmU2Zi00YjlmLTkxODQtOWUzZmVmZmQ1YTNjIiwiY2xpZW50X2lkIjoiYzEifQ.wyiS-z-xhBPZSODXZHQVDJCQ6dcmeJjAwBPWe2GhT94", "expires_in": 7199, "scope": "ROLE_ADMIN ROLE_USER ROLE_API", "jti": "9d334fdc-970d-4bbd-b62c-d5806d5833f0" }
会发现accessToken长了不少,这是由于token是jwt字符串,分为三部分,第二部分payload携带了不少信息,打开jwt.io网站,将上面的accessToken贴上去,能够看到Base64解码后的信息:
第一步,将认证服务中的TokenConfig直接拷贝到资源服务中
第二步,修改ResouceServerConfig 类
@Autowired private TokenStore tokenStore; @Override public void configure(ResourceServerSecurityConfigurer resources) throws Exception { resources .resourceId(RESOURCE_ID) // .tokenServices(resourceServerTokenServices)//令牌服务 .tokenStore(tokenStore) .stateless(true); }
便可完成资源服务集成jwt的功能。
POST请求 http://127.0.0.1:30000/oauth/token?client_id=c1&client_secret=secret&grant_type=password&username=zhangsan&password=123 获取令牌,获取到accessToken以后携带token GET 请求 http://127.0.0.1:30001/r1 ,获得响应结果:
访问资源r1
便可证实成功。
认证服务客户端信息仍是保存在内存中,如今将其改造放到数据库中
DROP TABLE IF EXISTS `oauth_client_details`; CREATE TABLE `oauth_client_details` ( `client_id` varchar(255) NOT NULL COMMENT '客户端标识', `resource_ids` varchar(255) DEFAULT NULL COMMENT '接入资源列表', `client_secret` varchar(255) DEFAULT NULL COMMENT '客户端秘钥', `scope` varchar(255) DEFAULT NULL, `authorized_grant_types` varchar(255) DEFAULT NULL, `web_server_redirect_uri` varchar(255) DEFAULT NULL, `authorities` varchar(255) DEFAULT NULL, `access_token_validity` int(11) DEFAULT NULL, `refresh_token_validity` int(11) DEFAULT NULL, `additional_information` longtext, `create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, `archived` tinyint(4) DEFAULT NULL, `trusted` tinyint(4) DEFAULT NULL, `autoapprove` varchar(255) DEFAULT NULL, PRIMARY KEY (`client_id`) USING BTREE ) ENGINE=InnoDB DEFAULT CHARSET=utf8 ROW_FORMAT=DYNAMIC COMMENT='接入客户端信息'; /*Data for the table `oauth_client_details` */ insert into `oauth_client_details`(`client_id`,`resource_ids`,`client_secret`,`scope`,`authorized_grant_types`,`web_server_redirect_uri`,`authorities`,`access_token_validity`,`refresh_token_validity`,`additional_information`,`create_time`,`archived`,`trusted`,`autoapprove`) values ('c1','res1','$2a$10$X2xVwW.7cOEh2niPqHYAne9EnjRJFj7QI4TqfmnDou9fT/45sCFEm','ROLE_ADMIN,ROLE_USER,ROLE_API','client_credentials,password,authorization_code,implicit,refresh_token','https://www.baidu.com',NULL,7200,259200,NULL,'2021-01-11 09:09:53',0,0,'false'), ('c2','res2','$2a$10$X2xVwW.7cOEh2niPqHYAne9EnjRJFj7QI4TqfmnDou9fT/45sCFEm','ROLE_API','client_credentials,password,authorization_code,implicit,refresh_token','https://www.baidu.com',NULL,31536000,2592000,NULL,'2021-01-11 09:09:56',0,0,'false'); /*Table structure for table `oauth_code` */ DROP TABLE IF EXISTS `oauth_code`; CREATE TABLE `oauth_code` ( `create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP, `code` varchar(255) DEFAULT NULL, `authentication` blob, KEY `code_index` (`code`) USING BTREE ) ENGINE=InnoDB DEFAULT CHARSET=utf8 ROW_FORMAT=COMPACT;
上述SQL新建了两张表oauth_client_details
以及oauth_code
分别用于存储客户端信息以及受权码信息,因为使用了jwt token(jwt token自己就存储了数据),因此再也不保存数据库,两张表均为spring oauth2.0内置表,不须要写SQL,内置框架自动识别表。
对应上述两张表,分别修改Bean对象的建立为jdbc类型的:
@Bean public AuthorizationCodeServices authorizationCodeServices(DataSource dataSource){ return new JdbcAuthorizationCodeServices(dataSource); } @Bean public ClientDetailsService clientDetailsService(DataSource dataSource) { JdbcClientDetailsService clientDetailsService = new JdbcClientDetailsService(dataSource); clientDetailsService.setPasswordEncoder(passwordEncoder); return clientDetailsService; }
以后修改客户端配置对象:
@Override public void configure(ClientDetailsServiceConfigurer clients) throws Exception { clients.withClientDetails(clientDetailsService); // clients.inMemory() // .withClient("c1") // .secret(new BCryptPasswordEncoder().encode("secret"))//$2a$10$0uhIO.ADUFv7OQ/kuwsC1.o3JYvnevt5y3qX/ji0AUXs4KYGio3q6 // .resourceIds("r1") // .authorizedGrantTypes("authorization_code", "password", "client_credentials", "implicit", "refresh_token") // .scopes("all") // .autoApprove(false) // .redirectUris("https://www.baidu.com"); }
完成修改。
GET请求:http://127.0.0.1:30000/oauth/authorize?client_id=c1&response_type=code&scope=ROLE_API&redirect_uri=https://www.baidu.com 获取受权码后,观察表oauth_code,里面应当已经有了受权码数据。
源码地址:https://gitee.com/kdyzm/spring-security-oauth-study/tree/v4.0.0
个人博客地址:https://blog.kdyzm.cn/