SpringBoot集成Shiro和JWT

173 阅读7分钟

携手创作,共同成长!这是我参与「掘金日新计划 · 8 月更文挑战」的第4天,点击查看活动详情

image.png

对之前创建的一个小的单体SpringBoot项目认证授权部分一个小结

Shiro相关信息

  • SecurityManager,可以理解成控制中心,所有请求最终基本上都通过它来代理转发,一般我们程序中不需要直接跟他打交道。
  • Subject ,请求主体。比如登录用户,比如一个被授权的app。在程序中任何地方都可以通过==SecurityUtils.getSubject()== 获取到当前的subject。subject中可以获取到Principal,这个是subject的标识,比如登陆用户的用户名或者id等,shiro不对值做限制。但是在登录和授权过程中,程序需要通过principal来识别唯一的用户。
  • Realm通俗一点理解就是realm可以访问安全相关数据,提供统一的数据封装来给上层做数据校验。shiro的建议是每种数据源定义一个realm,比如用户数据存在数据库可以使用JdbcRealm;存在属性配置文件可以使用PropertiesRealm。一般我们使用shiro都使用自定义的realm。
anon: 无需认证即可访问
authc: 需要认证才可访问
user: 点击“记住我”功能可访问
perms: 拥有权限才可以访问
role: 拥有某个角色权限才能访问

JWT

  • JSON Web Token(JWT)是一个开放标准(RFC 7519),它定义了一种紧凑且独立的方式,可以在各方之间作为JSON对象安全地传输信息。此信息可以通过数字签名进行验证和信任。JWT可以使用秘密(使用HMAC算法)或使用RSA或ECDSA的公钥/私钥对进行签名。 虽然JWT可以加密以在各方之间提供保密,但我们将专注于签名令牌。签名令牌可以验证其中包含的声明的完整性,而加密令牌则隐藏其他方的声明。当使用公钥/私钥对签署令牌时,签名还证明只有持有私钥的一方是签署私钥的一方。

JWT构成

第一部分我们称它为头部(header),第二部分我们称其为载荷(payload, 类似于飞机上承载的物品),第三部分是签证(signature)。

格式:
xxxxx.yyyyy.zzzzz

SpringBoot中集成Shiro和JWT

  1. pom文件中导入相关jar包依赖
<!-- 版本控制 -->
<properties>
    <shiro.version>1.4.0</shiro.version>
    <jwt.auth0.version>3.2.0</jwt.auth0.version>
</properties>
<!-- 依赖管理 -->
</dependencies>
    <!-- shiro -->
    <dependency>
        <groupId>org.apache.shiro</groupId>
        <artifactId>shiro-core</artifactId>
        <version>${shiro.version}</version>
    </dependency>
    <dependency>
        <groupId>org.apache.shiro</groupId>
        <artifactId>shiro-spring</artifactId>
        <version>${shiro.version}</version>
    </dependency>
    <!-- JWT -->
    <dependency>
        <groupId>com.auth0</groupId>
        <artifactId>java-jwt</artifactId>
        <version>${jwt.auth0.version}</version>
    </dependency>
</dependencies>
  1. JWT工具类,用于生成校验token
public class JWTUtil {

    /**
     * 过期时间 24 小时
     */
    private static final long EXPIRE_TIME = 24 * 60 * 60 * 1000;

    /**
     * 密钥,注意这里如果真实用到,应当设置的复杂点,相当于私钥的存在。如果被人拿到,相当于它可以自己制造token了。
     */
    private static final String SECRET = "NENE";

    /**
     * 生成 token, 24h后过期
     *
     * @param userName 用户名
     * @return 加密的token string
     * @throws UnsupportedEncodingException
     */
    public static String createToken(String userName) throws UnsupportedEncodingException {
        Date date = new Date(System.currentTimeMillis() + EXPIRE_TIME);
        Algorithm algorithm = Algorithm.HMAC256(SECRET);
        // 附带username信息
        return JWT.create()
                .withClaim("userName", userName)
                //到期时间
                .withExpiresAt(date)
                //创建一个新的JWT,并使用给定的算法进行标记
                .sign(algorithm);
    }

    /**
     * 校验 token 是否正确
     *
     * @param token    密钥
     * @param userName 用户名
     * @return 是否正确 boolean
     */
    public static boolean verify(String token, String userName) {
        try {
            Algorithm algorithm = Algorithm.HMAC256(SECRET);
            //在token中附带了username信息
            JWTVerifier verifier = JWT.require(algorithm)
                    .withClaim("userName", userName)
                    .build();
            //验证 token
            verifier.verify(token);
            return true;
        } catch (Exception exception) {
            return false;
        }
    }

    /**
     * 获得token中的信息,无需secret解密也能获得
     *
     * @param token the token
     * @return token中包含的用户名 userName
     */
    public static String getUsername(String token) {
        try {
            DecodedJWT jwt = JWT.decode(token);
            return jwt.getClaim("userName").asString();
        } catch (JWTDecodeException e) {
            return null;
        }
    }

}
  1. 实现AuthenticationToken类存放token信息
@Data
public class JWTToken implements AuthenticationToken {

    private String token;

    public JWTToken(String token) {
        this.token = token;
    }

    @Override
    public Object getPrincipal() {
        return token;
    }

    @Override
    public Object getCredentials() {
        return token;
    }

}
  1. 自定义JWT过滤器
@Slf4j
public class JWTFilter extends BasicHttpAuthenticationFilter {

    /**
     * 如果带有 token,则对 token 进行检查,否则直接通过
     * 如果在这里返回了false,请求会被直接拦截,用户看不到任何东西
     * 所以我们在这里返回true,Controller中可以通过 subject.isAuthenticated() 来判断用户是否登入
     * 如果有些资源只有登入用户才能访问,我们只需要在方法上面加上 @RequiresAuthentication 注解即可
     * 但是这样做有一个缺点,就是不能够对GET,POST等请求进行分别过滤鉴权(因为我们重写了官方的方法),但实际上对应用影响不大
     */
    @Override
    protected boolean isAccessAllowed(ServletRequest request, ServletResponse response, Object mappedValue) {
        // 判断请求的请求头是否带上 "token"
        if (isLoginAttempt(request, response)) {
            // 如果存在,则进入 executeLogin 方法执行登入,检查 token 是否正确
            try {
                executeLogin(request, response);
                return true;
            } catch (Exception e) {
                //token 错误
                responseError(response, e.getMessage());
            }
        }
        //如果请求头不存在 Token,则可能是执行登陆操作或者是游客状态访问,无需检查 token,直接返回 true
        return true;
    }

    /**
     * 判断用户是否想要登入。
     * 检测 header 里面是否包含 Token 字段
     */
    @Override
    protected boolean isLoginAttempt(ServletRequest request, ServletResponse response) {
        HttpServletRequest req = (HttpServletRequest) request;
        String token = req.getHeader("token");
        return token != null;
    }

    /**
     * 执行登陆操作
     */
    @Override
    protected boolean executeLogin(ServletRequest request, ServletResponse response) {
        HttpServletRequest req = (HttpServletRequest) request;
        String token = req.getHeader("token");
        JWTToken jwtToken = new JWTToken(token);
        // 提交给realm进行登入,如果错误他会抛出异常并被捕获
        getSubject(request, response).login(jwtToken);
        // 如果没有抛出异常则代表登入成功,返回true
        return true;
    }

    /**
     * 对跨域提供支持
     */
    @Override
    protected boolean preHandle(ServletRequest request, ServletResponse response) throws Exception {
        HttpServletRequest httpServletRequest = (HttpServletRequest) request;
        HttpServletResponse httpServletResponse = (HttpServletResponse) response;
        httpServletResponse.setHeader("Access-control-Allow-Origin", httpServletRequest.getHeader("Origin"));
        httpServletResponse.setHeader("Access-Control-Allow-Methods", "GET,POST,OPTIONS,PUT,DELETE");
        httpServletResponse.setHeader("Access-Control-Allow-Headers", httpServletRequest.getHeader("Access-Control-Request-Headers"));
        // 跨域时会首先发送一个option请求,这里我们给option请求直接返回正常状态
        if (httpServletRequest.getMethod().equals(RequestMethod.OPTIONS.name())) {
            httpServletResponse.setStatus(HttpStatus.OK.value());
            return false;
        }
        return super.preHandle(request, response);
    }

    /**
     * 将非法请求跳转到 /unauthorized/**
     */
    private void responseError(ServletResponse response, String message) {
        try {
            HttpServletResponse httpServletResponse = (HttpServletResponse) response;
            // 设置编码,否则中文字符在重定向时会变为空字符串
            message = URLEncoder.encode(message, "UTF-8");
            httpServletResponse.sendRedirect("/platform/sysUser/unauthorized/" + message);
        } catch (IOException e) {
            log.error(e.getMessage());
        }
    }

}
  1. 继承AuthorizingRealm,实现用户授权的验证和权限的验证
@Slf4j
public class MyShiroRealm extends AuthorizingRealm {

    @Autowired
    private SysMenuMapper sysMenuMapper;

    /**
     * 必须重写此方法,不然会报错
     */
    @Override
    public boolean supports(AuthenticationToken token) {
        return token instanceof JWTToken;
    }

    /**
     * Description: 只有检测用户权限调用此方法进行验证
     *
     * @param principals
     * @return org.apache.shiro.authz.AuthorizationInfo
     * @author: xyn
     * @date 2020/6/8 10:17
     */
    @Override
    protected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principals) {
        log.debug("===========权限认证方法===========");
        SimpleAuthorizationInfo authorizationInfo = new SimpleAuthorizationInfo();
        // 获取当前用户信息
        String token = principals.toString();
        UserInfo userInfo = getUserInfo(token);
        // 获取当前用户的角色信息
        List<Role> roles = userInfo.getRoles();
        List<String> roleIds = roles.stream().map(Role::getRoleId).collect(Collectors.toList());
        // 根据角色信息查询相应角色的菜单权限
        List<String> perms = sysMenuMapper.getPermsByRoleId(roleIds);
        authorizationInfo.addStringPermissions(perms);
        return authorizationInfo;
    }

    /**
     * Description: 默认调用此方法进行用户名验证
     *
     * @param authenticationToken
     * @return org.apache.shiro.authc.AuthenticationInfo
     * @author: xyn
     * @date 2020/6/8 10:17
     */
    @Override
    protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken authenticationToken) throws AuthenticationException {
        log.debug("===========身份认证方法===========");
        // 获取token信息
        String token = (String) authenticationToken.getCredentials();
        // 获取用户信息
        UserInfo userInfo = getUserInfo(token);
        if (ObjectUtil.isNull(userInfo)) {
            throw new AuthenticationException("用户名或者密码不正确!");
        }
        if (Constant.USER_STATUS_NO.equals(userInfo.getStatus())) {
            throw new AuthenticationException("该用户已被禁用,请联系管理员解锁!");
        }
        RedisUtil.setOpsForExpValue(Constant.JWT_TOKEN_INFO + token, JSONUtil.toJsonStr(userInfo), 30, TimeUnit.MINUTES);
        return new SimpleAuthenticationInfo(token, token, "MyRealm");
    }

    /**
     * Description: 获取用户信息
     *
     * @param token
     * @return com.xyn.exe.modules.platform.dto.UserInfo
     * @author: xyn
     * @date 2020/6/10 14:46
     */
    private UserInfo getUserInfo(String token) throws AuthenticationException {
        // 根据token获取用户名
        String username = JWTUtil.getUsername(token);
        if (StrUtil.isBlank(username) || !JWTUtil.verify(token, username)) {
            throw new AuthenticationException("token认证失败!");
        }
        // 根据token去redis中查找用户信息
        Object o = RedisUtil.getRedisValue(Constant.JWT_TOKEN_INFO + token);
        if (ObjectUtil.isNull(o)) {
            throw new AuthenticationException("用户登录已过期!");
        }
        JSONObject jsonObject = JSONUtil.parseObj(o);
        return jsonObject.toBean(UserInfo.class);
    }
}
  1. 配置ShiroConfig,将自定义的过滤器设置进去
@Configuration
public class ShiroConfig {

    /**
     * 通过过滤器检测请求头中是否包含token,通过token登录,realm验证
     */
    @Bean
    public ShiroFilterFactoryBean factory(SecurityManager securityManager) {
        ShiroFilterFactoryBean factoryBean = new ShiroFilterFactoryBean();
        // 添加过滤器
        Map<String, Filter> filterMap = new HashMap<>();
        //设置我们自定义的JWT过滤器
        filterMap.put("jwt", new JWTFilter());
        factoryBean.setFilters(filterMap);
        factoryBean.setSecurityManager(securityManager);
        Map<String, String> filterRuleMap = new HashMap<>();
        //访问/login和/unauthorized 不需要经过过滤器
        // 登录
        filterRuleMap.put("/platform/sysUser/login", "anon");
        filterRuleMap.put("/platform/sysUser/unauthorized/**", "anon");
        // 所有请求通过我们自己的JWT Filter
        filterRuleMap.put("/**", "jwt");
        factoryBean.setFilterChainDefinitionMap(filterRuleMap);
        return factoryBean;
    }

    /**
     * 注入 securityManager
     */
    @Bean
    public SecurityManager securityManager() {
        DefaultWebSecurityManager securityManager = new DefaultWebSecurityManager();
        // 设置自定义 realm.
        securityManager.setRealm(loginShiroRealm());
        /*
         * 关闭shiro自带的session,详情见文档
         * http://shiro.apache.org/session-management.html#SessionManagement-StatelessApplications%28Sessionless%29
         */
        DefaultSubjectDAO subjectDAO = new DefaultSubjectDAO();
        DefaultSessionStorageEvaluator defaultSessionStorageEvaluator = new DefaultSessionStorageEvaluator();
        defaultSessionStorageEvaluator.setSessionStorageEnabled(false);
        subjectDAO.setSessionStorageEvaluator(defaultSessionStorageEvaluator);
        securityManager.setSubjectDAO(subjectDAO);
        return securityManager;
    }

    /**
     * 开启Shiro的注解(如@RequiresRoles,@RequiresPermissions)
     * 配置以下两个bean(DefaultAdvisorAutoProxyCreator和AuthorizationAttributeSourceAdvisor)即可实现此功能
     *
     * @return
     */
    @Bean
    public DefaultAdvisorAutoProxyCreator advisorAutoProxyCreator() {
        DefaultAdvisorAutoProxyCreator advisorAutoProxyCreator = new DefaultAdvisorAutoProxyCreator();
        advisorAutoProxyCreator.setProxyTargetClass(true);
        return advisorAutoProxyCreator;
    }

    /**
     * 开启shiro aop注解支持. 使用代理方式; 所以需要开启代码支持;
     */
    @Bean
    public AuthorizationAttributeSourceAdvisor authorizationAttributeSourceAdvisor(SecurityManager securityManager) {
        AuthorizationAttributeSourceAdvisor advisor = new AuthorizationAttributeSourceAdvisor();
        advisor.setSecurityManager(securityManager);
        return advisor;
    }

    @Bean
    public MyShiroRealm loginShiroRealm() {
        return new MyShiroRealm();
    }

}
  1. 控制权限相关注解
@RequiresRoles("角色")
@RequiresPermissions("权限")

c28a6b0ce28b07820f8f117fcd842008.jpg