123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275 |
- package org.jeecg.config.shiro;
- import lombok.extern.slf4j.Slf4j;
- import org.apache.commons.pool2.impl.GenericObjectPoolConfig;
- import org.apache.shiro.mgt.DefaultSessionStorageEvaluator;
- import org.apache.shiro.mgt.DefaultSubjectDAO;
- import org.apache.shiro.mgt.SecurityManager;
- import org.apache.shiro.spring.LifecycleBeanPostProcessor;
- import org.apache.shiro.spring.security.interceptor.AuthorizationAttributeSourceAdvisor;
- import org.apache.shiro.spring.web.ShiroFilterFactoryBean;
- import org.apache.shiro.web.mgt.DefaultWebSecurityManager;
- import org.crazycake.shiro.IRedisManager;
- import org.crazycake.shiro.RedisCacheManager;
- import org.crazycake.shiro.RedisClusterManager;
- import org.crazycake.shiro.RedisManager;
- import org.jeecg.common.constant.CommonConstant;
- import org.jeecg.common.util.oConvertUtils;
- import org.jeecg.config.shiro.filters.CustomShiroFilterFactoryBean;
- import org.jeecg.config.shiro.filters.JwtFilter;
- import org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator;
- import org.springframework.beans.factory.annotation.Autowired;
- import org.springframework.beans.factory.annotation.Value;
- import org.springframework.context.annotation.Bean;
- import org.springframework.context.annotation.Configuration;
- import org.springframework.context.annotation.DependsOn;
- import org.springframework.core.env.Environment;
- import org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory;
- import org.springframework.util.StringUtils;
- import redis.clients.jedis.HostAndPort;
- import redis.clients.jedis.JedisCluster;
- import javax.annotation.Resource;
- import javax.servlet.Filter;
- import java.util.*;
- /**
- * @author: Scott
- * @date: 2018/2/7
- * @description: shiro 配置类
- */
- @Slf4j
- @Configuration
- public class ShiroConfig {
- @Value("${jeecg.shiro.excludeUrls}")
- private String excludeUrls;
- @Resource
- LettuceConnectionFactory lettuceConnectionFactory;
- @Autowired
- private Environment env;
- /**
- * Filter Chain定义说明
- *
- * 1、一个URL可以配置多个Filter,使用逗号分隔
- * 2、当设置多个过滤器时,全部验证通过,才视为通过
- * 3、部分过滤器可指定参数,如perms,roles
- */
- @Bean("shiroFilter")
- public ShiroFilterFactoryBean shiroFilter(SecurityManager securityManager) {
- CustomShiroFilterFactoryBean shiroFilterFactoryBean = new CustomShiroFilterFactoryBean();
- shiroFilterFactoryBean.setSecurityManager(securityManager);
- // 拦截器
- Map<String, String> filterChainDefinitionMap = new LinkedHashMap<String, String>();
- if(oConvertUtils.isNotEmpty(excludeUrls)){
- String[] permissionUrl = excludeUrls.split(",");
- for(String url : permissionUrl){
- filterChainDefinitionMap.put(url,"anon");
- }
- }
- // 配置不会被拦截的链接 顺序判断
- filterChainDefinitionMap.put("/systemController/**","anon");
- filterChainDefinitionMap.put("/sys/common/upload/**","anon");
- filterChainDefinitionMap.put("/sys/cas/client/validateLogin", "anon"); //cas验证登录
- filterChainDefinitionMap.put("/sys/randomImage/**", "anon"); //登录验证码接口排除
- filterChainDefinitionMap.put("/sys/checkCaptcha", "anon"); //登录验证码接口排除
- filterChainDefinitionMap.put("/sys/login", "anon"); //登录接口排除
- filterChainDefinitionMap.put("/sys/mLogin", "anon"); //登录接口排除
- filterChainDefinitionMap.put("/sys/logout", "anon"); //登出接口排除
- filterChainDefinitionMap.put("/sys/thirdLogin/**", "anon"); //第三方登录
- filterChainDefinitionMap.put("/sys/getEncryptedString", "anon"); //获取加密串
- filterChainDefinitionMap.put("/sys/sms", "anon");//短信验证码
- filterChainDefinitionMap.put("/sys/phoneLogin", "anon");//手机登录
- filterChainDefinitionMap.put("/sys/user/checkOnlyUser", "anon");//校验用户是否存在
- filterChainDefinitionMap.put("/sys/user/register", "anon");//用户注册
- filterChainDefinitionMap.put("/sys/user/phoneVerification", "anon");//用户忘记密码验证手机号
- filterChainDefinitionMap.put("/sys/user/passwordChange", "anon");//用户更改密码
- filterChainDefinitionMap.put("/auth/2step-code", "anon");//登录验证码
- filterChainDefinitionMap.put("/sys/common/static/**", "anon");//图片预览 &下载文件不限制token
- filterChainDefinitionMap.put("/sys/common/pdf/**", "anon");//pdf预览
- filterChainDefinitionMap.put("/generic/**", "anon");//pdf预览需要文件
- filterChainDefinitionMap.put("/", "anon");
- filterChainDefinitionMap.put("/doc.html", "anon");
- filterChainDefinitionMap.put("/**/*.js", "anon");
- filterChainDefinitionMap.put("/**/*.css", "anon");
- filterChainDefinitionMap.put("/**/*.html", "anon");
- filterChainDefinitionMap.put("/**/*.svg", "anon");
- filterChainDefinitionMap.put("/**/*.pdf", "anon");
- filterChainDefinitionMap.put("/**/*.jpg", "anon");
- filterChainDefinitionMap.put("/**/*.png", "anon");
- filterChainDefinitionMap.put("/**/*.ico", "anon");
- filterChainDefinitionMap.put("/systemController/**","anon");
- filterChainDefinitionMap.put("/sys/common/upload/**","anon");
- filterChainDefinitionMap.put("/bussMapQueryController/**","anon"); //一张图:业务属性查询接口
- filterChainDefinitionMap.put("/hzz.hhhj.glfwx/rmGlfwx/queryByInfo", "anon"); //划界信息查询
- filterChainDefinitionMap.put("/hzz.hhhj.gsp/rmBulletinBoard/queryByRelId", "anon"); //公告牌查询
- filterChainDefinitionMap.put("/hzz.hhhj.jzd/rmBoundaryMarker/queryByRelId", "anon"); //界桩点查询
- filterChainDefinitionMap.put("/hzz.axgh.axghgnq/rmAxghgnqgeo/getGeojsonById", "anon"); //岸线功能区图形json查询
- filterChainDefinitionMap.put("/hzz.axgh.axghgnq/lskzx/getGeojsonById", "anon"); //临水控制线图形json查询
- filterChainDefinitionMap.put("/hzz.axgh.axghgnq/wykzx/getGeojsonById", "anon"); //外缘控制线图形json查询
- filterChainDefinitionMap.put("/hzz.shjsgc.dfgc/rmDfgc/getGeojsonById", "anon"); //堤防工程图形json查询
- filterChainDefinitionMap.put("/hzz.shjsgc.bzgc/rmBzgc/queryById", "anon"); //泵站工程查询
- filterChainDefinitionMap.put("/hzz.shjsgc.zbgc/rmZbgc/queryById", "anon"); //闸坝工程查询
- filterChainDefinitionMap.put("/hzz.shjsgc.dfgc/rmDfgc/queryById", "anon"); //堤防工程查询
- // update-begin--Author:sunjianlei Date:20190813 for:排除字体格式的后缀
- filterChainDefinitionMap.put("/**/*.ttf", "anon");
- filterChainDefinitionMap.put("/**/*.woff", "anon");
- filterChainDefinitionMap.put("/**/*.woff2", "anon");
- // update-begin--Author:sunjianlei Date:20190813 for:排除字体格式的后缀
- filterChainDefinitionMap.put("/druid/**", "anon");
- filterChainDefinitionMap.put("/swagger-ui.html", "anon");
- filterChainDefinitionMap.put("/swagger**/**", "anon");
- filterChainDefinitionMap.put("/webjars/**", "anon");
- filterChainDefinitionMap.put("/v2/**", "anon");
- filterChainDefinitionMap.put("/sys/annountCement/show/**", "anon");
- //积木报表排除
- filterChainDefinitionMap.put("/jmreport/**", "anon");
- filterChainDefinitionMap.put("/**/*.js.map", "anon");
- filterChainDefinitionMap.put("/**/*.css.map", "anon");
- //大屏模板例子
- filterChainDefinitionMap.put("/test/bigScreen/**", "anon");
- //websocket排除
- filterChainDefinitionMap.put("/websocket/**", "anon");//系统通知和公告
- filterChainDefinitionMap.put("/newsWebsocket/**", "anon");//CMS模块
- filterChainDefinitionMap.put("/vxeSocket/**", "anon");//JVxeTable无痕刷新示例
- //性能监控 TODO 存在安全漏洞泄露TOEKN(durid连接池也有)
- filterChainDefinitionMap.put("/actuator/**", "anon");
- // 添加自己的过滤器并且取名为jwt
- Map<String, Filter> filterMap = new HashMap<String, Filter>(1);
- //如果cloudServer为空 则说明是单体 需要加载跨域配置【微服务跨域切换】
- Object cloudServer = env.getProperty(CommonConstant.CLOUD_SERVER_KEY);
- filterMap.put("jwt", new JwtFilter(cloudServer==null));
- shiroFilterFactoryBean.setFilters(filterMap);
- // <!-- 过滤链定义,从上向下顺序执行,一般将/**放在最为下边
- filterChainDefinitionMap.put("/**", "jwt");
- // 未授权界面返回JSON
- shiroFilterFactoryBean.setUnauthorizedUrl("/sys/common/403");
- shiroFilterFactoryBean.setLoginUrl("/sys/common/403");
- shiroFilterFactoryBean.setFilterChainDefinitionMap(filterChainDefinitionMap);
- return shiroFilterFactoryBean;
- }
- @Bean("securityManager")
- public DefaultWebSecurityManager securityManager(ShiroRealm myRealm) {
- DefaultWebSecurityManager securityManager = new DefaultWebSecurityManager();
- securityManager.setRealm(myRealm);
- /*
- * 关闭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);
- //自定义缓存实现,使用redis
- securityManager.setCacheManager(redisCacheManager());
- return securityManager;
- }
- /**
- * 下面的代码是添加注解支持
- * @return
- */
- @Bean
- @DependsOn("lifecycleBeanPostProcessor")
- public DefaultAdvisorAutoProxyCreator defaultAdvisorAutoProxyCreator() {
- DefaultAdvisorAutoProxyCreator defaultAdvisorAutoProxyCreator = new DefaultAdvisorAutoProxyCreator();
- defaultAdvisorAutoProxyCreator.setProxyTargetClass(true);
- /**
- * 解决重复代理问题 github#994
- * 添加前缀判断 不匹配 任何Advisor
- */
- defaultAdvisorAutoProxyCreator.setUsePrefix(true);
- defaultAdvisorAutoProxyCreator.setAdvisorBeanNamePrefix("_no_advisor");
- return defaultAdvisorAutoProxyCreator;
- }
- @Bean
- public static LifecycleBeanPostProcessor lifecycleBeanPostProcessor() {
- return new LifecycleBeanPostProcessor();
- }
- @Bean
- public AuthorizationAttributeSourceAdvisor authorizationAttributeSourceAdvisor(DefaultWebSecurityManager securityManager) {
- AuthorizationAttributeSourceAdvisor advisor = new AuthorizationAttributeSourceAdvisor();
- advisor.setSecurityManager(securityManager);
- return advisor;
- }
- /**
- * cacheManager 缓存 redis实现
- * 使用的是shiro-redis开源插件
- *
- * @return
- */
- public RedisCacheManager redisCacheManager() {
- log.info("===============(1)创建缓存管理器RedisCacheManager");
- RedisCacheManager redisCacheManager = new RedisCacheManager();
- redisCacheManager.setRedisManager(redisManager());
- //redis中针对不同用户缓存(此处的id需要对应user实体中的id字段,用于唯一标识)
- redisCacheManager.setPrincipalIdFieldName("id");
- //用户权限信息缓存时间
- redisCacheManager.setExpire(200000);
- return redisCacheManager;
- }
- /**
- * 配置shiro redisManager
- * 使用的是shiro-redis开源插件
- *
- * @return
- */
- @Bean
- public IRedisManager redisManager() {
- log.info("===============(2)创建RedisManager,连接Redis..");
- IRedisManager manager;
- // redis 单机支持,在集群为空,或者集群无机器时候使用 add by jzyadmin@163.com
- if (lettuceConnectionFactory.getClusterConfiguration() == null || lettuceConnectionFactory.getClusterConfiguration().getClusterNodes().isEmpty()) {
- RedisManager redisManager = new RedisManager();
- redisManager.setHost(lettuceConnectionFactory.getHostName());
- redisManager.setPort(lettuceConnectionFactory.getPort());
- redisManager.setDatabase(lettuceConnectionFactory.getDatabase());
- redisManager.setTimeout(0);
- if (!StringUtils.isEmpty(lettuceConnectionFactory.getPassword())) {
- redisManager.setPassword(lettuceConnectionFactory.getPassword());
- }
- manager = redisManager;
- }else{
- // redis集群支持,优先使用集群配置
- RedisClusterManager redisManager = new RedisClusterManager();
- Set<HostAndPort> portSet = new HashSet<>();
- lettuceConnectionFactory.getClusterConfiguration().getClusterNodes().forEach(node -> portSet.add(new HostAndPort(node.getHost() , node.getPort())));
- //update-begin--Author:scott Date:20210531 for:修改集群模式下未设置redis密码的bug issues/I3QNIC
- if (oConvertUtils.isNotEmpty(lettuceConnectionFactory.getPassword())) {
- JedisCluster jedisCluster = new JedisCluster(portSet, 2000, 2000, 5,
- lettuceConnectionFactory.getPassword(), new GenericObjectPoolConfig());
- redisManager.setPassword(lettuceConnectionFactory.getPassword());
- redisManager.setJedisCluster(jedisCluster);
- } else {
- JedisCluster jedisCluster = new JedisCluster(portSet);
- redisManager.setJedisCluster(jedisCluster);
- }
- //update-end--Author:scott Date:20210531 for:修改集群模式下未设置redis密码的bug issues/I3QNIC
- manager = redisManager;
- }
- return manager;
- }
- }
|