单点登录SSO解决方案:SpringSecurity+JWT,8年Java开发教你如何写简历

  • 头部:主要设置一些规范信息,签名部分的编码格式就在头部中声明。

  • 载荷:token中存放有效信息的部分,比如用户名,用户角色,过期时间等,但是不要放密码,会泄露!

  • 签名:将头部与载荷分别采用base64编码后,用“.”相连,再加入盐,最后使用头部声明的编码类型进行编码,就得到了签名。

JWT生成token的安全性分析

从JWT生成的token组成上来看,要想避免token被伪造,主要就得看签名部分了,而签名部分又有三部分组成,其中头部和载荷的base64编码,几乎是透明的,毫无安全性可言,那么最终守护token安全的重担就落在了加入的盐上面了!

试想:如果生成token所用的盐与解析token时加入的盐是一样的。岂不是类似于中国人民银行把人民币防伪技术公开了?大家可以用这个盐来解析token,就能用来伪造token。这时,我们就需要对盐采用非对称加密的方式进行加密,以达到生成token与校验token方所用的盐不一致的安全效果!

非对称加密RSA介绍

基本原理: 同时生成两把密钥:私钥和公钥,私钥隐秘保存,公钥可以下发给信任客户端

  • 私钥加密,持有私钥或公钥才可以解密

  • 公钥加密,持有私钥才可解密

优点: 安全,难以破解

缺点: 算法比较耗时,为了安全,可以接受

历史: 三位数学家Rivest、Shamir 和 Adleman 设计了一种算法,可以实现非对称加密。这种算法用他们三个人的名字缩写:RSA。

四、SpringSecurity整合JWT


1.认证思路分析

SpringSecurity主要是通过过滤器来实现功能的!我们要找到SpringSecurity实现认证和校验身份的过滤器!

回顾集中式认证流程

**用户认证:**使用UsernamePasswordAuthenticationFilter过滤器中attemptAuthentication方法实现认证功能,该过滤器父类中successfulAuthentication方法实现认证成功后的操作。

**身份校验:**使用BasicAuthenticationFilter过滤器中doFilterInternal方法验证是否登录,以决定能否进入后续过滤器。

分析分布式认证流程

用户认证:

由于分布式项目,多数是前后端分离的架构设计,我们要满足可以接受异步post的认证请求参数,需要修改UsernamePasswordAuthenticationFilter过滤器中attemptAuthentication方法,让其能够接收请求体。

另外,默认successfulAuthentication方法在认证通过后,是把用户信息直接放入session就完事了,现在我们需要修改这个方法,在认证通过后生成token并返回给用户。

身份校验:

原来BasicAuthenticationFilter过滤器中doFilterInternal方法校验用户是否登录,就是看session中是否有用户信息,我们要修改为,验证用户携带的token是否合法,并解析出用户信息,交给SpringSecurity,以便于后续的授权功能可以正常使用。

2.具体实现

为了演示单点登录的效果,我们设计如下项目结构

2.1父工程创建

因为本案例需要创建多个系统,所以我们使用maven聚合工程来实现,首先创建一个父工程,导入springboot的父依赖即可

org.springframework.boot

spring-boot-starter-parent

2.1.3.RELEASE

2.2公共工程创建

然后创建一个common工程,其他工程依赖此系统

导入JWT相关的依赖

io.jsonwebtoken

jjwt-api

0.10.7

io.jsonwebtoken

jjwt-impl

0.10.7

runtime

io.jsonwebtoken

jjwt-jackson

0.10.7

runtime

com.fasterxml.jackson.core

jackson-databind

2.9.9

org.springframework.boot

spring-boot-starter-logging

joda-time

joda-time

org.projectlombok

lombok

org.springframework.boot

spring-boot-starter-test

创建相关的工具类

Payload

/**

* @program: springboot-54-security-jwt-demo

* @description:

* @author: 波波烤鸭

*/

@Data

public class Payload {

private String id;

private T userInfo;

private Date expiration;

}

JsonUtils

package com.dpb.utils;

import com.fasterxml.jackson.core.JsonProcessingException;

import com.fasterxml.jackson.core.type.TypeReference;

import com.fasterxml.jackson.databind.ObjectMapper;

import org.slf4j.Logger;

import org.slf4j.LoggerFactory;

import java.io.IOException;

import java.util.List;

import java.util.Map;

/**

* @author: 波波烤鸭

**/

public class JsonUtils {

public static final ObjectMapper mapper = new ObjectMapper();

private static final Logger logger = LoggerFactory.getLogger(JsonUtils.class);

public static String toString(Object obj) {

if (obj == null) {

return null;

}

if (obj.getClass() == String.class) {

return (String) obj;

}

try {

return mapper.writeValueAsString(obj);

} catch (JsonProcessingException e) {

logger.error(“json序列化出错:” + obj, e);

return null;

}

}

public static  T toBean(String json, Class tClass) {

try {

return mapper.readValue(json, tClass);

} catch (IOException e) {

logger.error(“json解析出错:” + json, e);

return null;

}

}

public static  List toList(String json, Class eClass) {

try {

return mapper.readValue(json, mapper.getTypeFactory().constructCollectionType(List.class, eClass));

} catch (IOException e) {

logger.error(“json解析出错:” + json, e);

return null;

}

}

public static <K, V> Map<K, V> toMap(String json, Class kClass, Class vClass) {

try {

return mapper.readValue(json, mapper.getTypeFactory().constructMapType(Map.class, kClass, vClass));

} catch (IOException e) {

logger.error(“json解析出错:” + json, e);

return null;

}

}

public static  T nativeRead(String json, TypeReference type) {

try {

return mapper.readValue(json, type);

} catch (IOException e) {

logger.error(“json解析出错:” + json, e);

return null;

}

}

}

JwtUtils

package com.dpb.utils;

import com.dpb.domain.Payload;

import io.jsonwebtoken.Claims;

import io.jsonwebtoken.Jws;

import io.jsonwebtoken.Jwts;

import io.jsonwebtoken.SignatureAlgorithm;

import org.joda.time.DateTime;

import java.security.PrivateKey;

import java.security.PublicKey;

import java.util.Base64;

import java.util.UUID;

/**

* @author: 波波烤鸭

* 生成token以及校验token相关方法

*/

public class JwtUtils {

private static final String JWT_PAYLOAD_USER_KEY = “user”;

/**

* 私钥加密token

* @param userInfo   载荷中的数据

* @param privateKey 私钥

* @param expire     过期时间,单位分钟

* @return JWT

*/

public static String generateTokenExpireInMinutes(Object userInfo, PrivateKey privateKey, int expire) {

return Jwts.builder()

.claim(JWT_PAYLOAD_USER_KEY, JsonUtils.toString(userInfo))

.setId(createJTI())

.setExpiration(DateTime.now().plusMinutes(expire).toDate())

.signWith(privateKey, SignatureAlgorithm.RS256)

.compact();

}

/**

* 私钥加密token

* @param userInfo   载荷中的数据

* @param privateKey 私钥

* @param expire     过期时间,单位秒

* @return JWT

*/

public static String generateTokenExpireInSeconds(Object userInfo, PrivateKey privateKey, int expire) {

return Jwts.builder()

.claim(JWT_PAYLOAD_USER_KEY, JsonUtils.toString(userInfo))

.setId(createJTI())

.setExpiration(DateTime.now().plusSeconds(expire).toDate())

.signWith(privateKey, SignatureAlgorithm.RS256)

.compact();

}

/**

* 公钥解析token

* @param token     用户请求中的token

* @param publicKey 公钥

* @return Jws

*/

private static Jws parserToken(String token, PublicKey publicKey) {

return Jwts.parser().setSigningKey(publicKey).parseClaimsJws(token);

}

private static String createJTI() {

return new String(Base64.getEncoder().encode(UUID.randomUUID().toString().getBytes()));

}

/**

* 获取token中的用户信息

* @param token     用户请求中的令牌

* @param publicKey 公钥

* @return 用户信息

*/

public static  Payload getInfoFromToken(String token, PublicKey publicKey, Class userType) {

Jws claimsJws = parserToken(token, publicKey);

Claims body = claimsJws.getBody();

Payload claims = new Payload<>();

claims.setId(body.getId());

claims.setUserInfo(JsonUtils.toBean(body.get(JWT_PAYLOAD_USER_KEY).toString(), userType));

claims.setExpiration(body.getExpiration());

return claims;

}

/**

* 获取token中的载荷信息

* @param token     用户请求中的令牌

* @param publicKey 公钥

* @return 用户信息

*/

public static  Payload getInfoFromToken(String token, PublicKey publicKey) {

Jws claimsJws = parserToken(token, publicKey);

Claims body = claimsJws.getBody();

Payload claims = new Payload<>();

claims.setId(body.getId());

claims.setExpiration(body.getExpiration());

return claims;

}

}

RsaUtils

package com.dpb.utils;

import java.io.File;

import java.io.IOException;

import java.nio.file.Files;

import java.security.*;

import java.security.spec.InvalidKeySpecException;

import java.security.spec.PKCS8EncodedKeySpec;

import java.security.spec.X509EncodedKeySpec;

import java.util.Base64;

/**

* @author 波波烤鸭

*/

public class RsaUtils {

private static final int DEFAULT_KEY_SIZE = 2048;

/**

* 从文件中读取公钥

* @param filename 公钥保存路径,相对于classpath

* @return 公钥对象

* @throws Exception

*/

public static PublicKey getPublicKey(String filename) throws Exception {

byte[] bytes = readFile(filename);

return getPublicKey(bytes);

}

/**

* 从文件中读取密钥

* @param filename 私钥保存路径,相对于classpath

* @return 私钥对象

* @throws Exception

*/

public static PrivateKey getPrivateKey(String filename) throws Exception {

byte[] bytes = readFile(filename);

return getPrivateKey(bytes);

}

/**

* 获取公钥

* @param bytes 公钥的字节形式

* @return

* @throws Exception

*/

private static PublicKey getPublicKey(byte[] bytes) throws Exception {

bytes = Base64.getDecoder().decode(bytes);

X509EncodedKeySpec spec = new X509EncodedKeySpec(bytes);

KeyFactory factory = KeyFactory.getInstance(“RSA”);

return factory.generatePublic(spec);

}

/**

* 获取密钥

* @param bytes 私钥的字节形式

* @return

* @throws Exception

*/

private static PrivateKey getPrivateKey(byte[] bytes) throws NoSuchAlgorithmException, InvalidKeySpecException {

bytes = Base64.getDecoder().decode(bytes);

PKCS8EncodedKeySpec spec = new PKCS8EncodedKeySpec(bytes);

KeyFactory factory = KeyFactory.getInstance(“RSA”);

return factory.generatePrivate(spec);

}

/**

* 根据密文,生存rsa公钥和私钥,并写入指定文件

* @param publicKeyFilename  公钥文件路径

* @param privateKeyFilename 私钥文件路径

* @param secret             生成密钥的密文

*/

public static void generateKey(String publicKeyFilename, String privateKeyFilename, String secret, int keySize) throws Exception {

KeyPairGenerator keyPairGenerator = KeyPairGenerator.getInstance(“RSA”);

SecureRandom secureRandom = new SecureRandom(secret.getBytes());

keyPairGenerator.initialize(Math.max(keySize, DEFAULT_KEY_SIZE), secureRandom);

KeyPair keyPair = keyPairGenerator.genKeyPair();

// 获取公钥并写出

byte[] publicKeyBytes = keyPair.getPublic().getEncoded();

publicKeyBytes = Base64.getEncoder().encode(publicKeyBytes);

writeFile(publicKeyFilename, publicKeyBytes);

// 获取私钥并写出

byte[] privateKeyBytes = keyPair.getPrivate().getEncoded();

privateKeyBytes = Base64.getEncoder().encode(privateKeyBytes);

writeFile(privateKeyFilename, privateKeyBytes);

}

private static byte[] readFile(String fileName) throws Exception {

return Files.readAllBytes(new File(fileName).toPath());

}

private static void writeFile(String destPath, byte[] bytes) throws IOException {

File dest = new File(destPath);

if (!dest.exists()) {

dest.createNewFile();

}

Files.write(dest.toPath(), bytes);

}

}

在通用子模块中编写测试类生成rsa公钥和私钥

/**

* @program: springboot-54-security-jwt-demo

* @description:

* @author: 波波烤鸭

*/

public class JwtTest {

private String privateKey = “c:/tools/auth_key/id_key_rsa”;

private String publicKey = “c:/tools/auth_key/id_key_rsa.pub”;

@Test

public void test1() throws Exception{

RsaUtils.generateKey(publicKey,privateKey,“dpb”,1024);

}

}

2.3认证系统创建

接下来我们创建我们的认证服务。(搜索公众号Java知音,回复“2021”,送你一份Java面试题宝典)

导入相关的依赖

org.springframework.boot

spring-boot-starter-web

org.springframework.boot

spring-boot-starter-security

security-jwt-common

com.dpb

1.0-SNAPSHOT

mysql

mysql-connector-java

5.1.47

org.mybatis.spring.boot

mybatis-spring-boot-starter

2.1.0

com.alibaba

druid

1.1.10

org.springframework.boot

spring-boot-configuration-processor

true

创建配置文件

spring:

datasource:

driver-class-name: com.mysql.jdbc.Driver

url: jdbc:mysql://localhost:3306/srm

username: root

password: 123456

type: com.alibaba.druid.pool.DruidDataSource

mybatis:

type-aliases-package: com.dpb.domain

mapper-locations: classpath:mapper/*.xml

logging:

level:

com.dpb: debug

rsa:

key:

pubKeyFile: c:\tools\auth_key\id_key_rsa.pub

priKeyFile: c:\tools\auth_key\id_key_rsa

提供公钥私钥的配置类

package com.dpb.config;

import com.dpb.utils.RsaUtils;

import lombok.Data;

import org.springframework.boot.context.properties.ConfigurationProperties;

import org.springframework.context.annotation.Configuration;

import javax.annotation.PostConstruct;

import java.security.PrivateKey;

import java.security.PublicKey;

/**

* @program: springboot-54-security-jwt-demo

* @description:

* @author: 波波烤鸭

*/

@Data

@ConfigurationProperties(prefix = “rsa.key”)

public class RsaKeyProperties {

private String pubKeyFile;

private String priKeyFile;

private PublicKey publicKey;

private PrivateKey privateKey;

/**

* 系统启动的时候触发

* @throws Exception

*/

@PostConstruct

public void createRsaKey() throws Exception {

publicKey = RsaUtils.getPublicKey(pubKeyFile);

privateKey = RsaUtils.getPrivateKey(priKeyFile);

}

}

创建启动类

/**

* @program: springboot-54-security-jwt-demo

* @description: 启动类

* @author: 波波烤鸭

*/

@SpringBootApplication

@MapperScan(“com.dpb.mapper”)

@EnableConfigurationProperties(RsaKeyProperties.class)

public class App {

public static void main(String[] args) {

SpringApplication.run(App.class,args);

}

}

完成数据认证的逻辑

pojo

package com.dpb.domain;

import com.fasterxml.jackson.annotation.JsonIgnore;

import lombok.Data;

import org.springframework.security.core.GrantedAuthority;

/**

* @program: springboot-54-security-jwt-demo

* @description:

* @author: 波波烤鸭

*/

@Data

public class RolePojo implements GrantedAuthority {

private Integer id;

private String roleName;

private String roleDesc;

@JsonIgnore

@Override

public String getAuthority() {

return roleName;

}

}

package com.dpb.domain;

import com.fasterxml.jackson.annotation.JsonIgnore;

import lombok.Data;

import org.springframework.security.core.GrantedAuthority;

import org.springframework.security.core.authority.SimpleGrantedAuthority;

import org.springframework.security.core.userdetails.UserDetails;

import java.util.ArrayList;

import java.util.Collection;

import java.util.List;

/**

* @program: springboot-54-security-jwt-demo

* @description:

* @author: 波波烤鸭

*/

@Data

public class UserPojo implements UserDetails {

private Integer id;

private String username;

private String password;

private Integer status;

private List roles;

@JsonIgnore

@Override

public Collection<? extends GrantedAuthority> getAuthorities() {

List auth = new ArrayList<>();

auth.add(new SimpleGrantedAuthority(“ADMIN”));

return auth;

}

@Override

public String getPassword() {

return this.password;

}

@Override

public String getUsername() {

return this.username;

}

@JsonIgnore

@Override

public boolean isAccountNonExpired() {

return true;

}

@JsonIgnore

@Override

public boolean isAccountNonLocked() {

return true;

}

@JsonIgnore

@Override

public boolean isCredentialsNonExpired() {

return true;

}

@JsonIgnore

@Override

public boolean isEnabled() {

return true;

}

}

Mapper接口

public interface UserMapper {

public UserPojo queryByUserName(@Param(“userName”) String userName);

}

Mapper映射文件

<?xml version="1.0" encoding="UTF-8" ?>

select * from t_user where username = #{userName}

Service

public interface UserService extends UserDetailsService {

}

@Service

@Transactional

public class UserServiceImpl implements UserService {

@Autowired

private UserMapper mapper;

@Override

public UserDetails loadUserByUsername(String s) throws UsernameNotFoundException {

UserPojo user = mapper.queryByUserName(s);

return user;

}

}

自定义认证过滤器

package com.dpb.filter;

import com.dpb.config.RsaKeyProperties;

import com.dpb.domain.RolePojo;

import com.dpb.domain.UserPojo;

import com.dpb.utils.JwtUtils;

import com.fasterxml.jackson.databind.ObjectMapper;

import net.bytebuddy.agent.builder.AgentBuilder;

import org.springframework.security.authentication.AuthenticationManager;

import org.springframework.security.authentication.UsernamePasswordAuthenticationToken;

import org.springframework.security.core.Authentication;

import org.springframework.security.core.AuthenticationException;

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Java)
img

读者福利

秋招我借这份PDF的复习思路,收获美团,小米,京东等Java岗offer

更多笔记分享

秋招我借这份PDF的复习思路,收获美团,小米,京东等Java岗offer

FoundException {

UserPojo user = mapper.queryByUserName(s);

return user;

}

}

自定义认证过滤器

package com.dpb.filter;

import com.dpb.config.RsaKeyProperties;

import com.dpb.domain.RolePojo;

import com.dpb.domain.UserPojo;

import com.dpb.utils.JwtUtils;

import com.fasterxml.jackson.databind.ObjectMapper;

import net.bytebuddy.agent.builder.AgentBuilder;

import org.springframework.security.authentication.AuthenticationManager;

import org.springframework.security.authentication.UsernamePasswordAuthenticationToken;

import org.springframework.security.core.Authentication;

import org.springframework.security.core.AuthenticationException;

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
[外链图片转存中…(img-MzgA5tSb-1712014702190)]
[外链图片转存中…(img-2pIlDbYJ-1712014702190)]
[外链图片转存中…(img-4J5EdyQ5-1712014702191)]
[外链图片转存中…(img-dI4pLR2o-1712014702191)]
[外链图片转存中…(img-h8EV5kXb-1712014702191)]
[外链图片转存中…(img-x16EMfLM-1712014702191)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Java)
[外链图片转存中…(img-RWVE53Wa-1712014702192)]

读者福利

[外链图片转存中…(img-WmjyZMKe-1712014702192)]

更多笔记分享

[外链图片转存中…(img-3bBb3Cfn-1712014702192)]

  • 24
    点赞
  • 29
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值