99db1b6344631e0297d337fb7cc799ea086d489a.svn-base 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  1. package org.jeecg.config.shiro;
  2. import lombok.extern.slf4j.Slf4j;
  3. import org.apache.commons.pool2.impl.GenericObjectPoolConfig;
  4. import org.apache.shiro.mgt.DefaultSessionStorageEvaluator;
  5. import org.apache.shiro.mgt.DefaultSubjectDAO;
  6. import org.apache.shiro.mgt.SecurityManager;
  7. import org.apache.shiro.spring.LifecycleBeanPostProcessor;
  8. import org.apache.shiro.spring.security.interceptor.AuthorizationAttributeSourceAdvisor;
  9. import org.apache.shiro.spring.web.ShiroFilterFactoryBean;
  10. import org.apache.shiro.web.mgt.DefaultWebSecurityManager;
  11. import org.crazycake.shiro.IRedisManager;
  12. import org.crazycake.shiro.RedisCacheManager;
  13. import org.crazycake.shiro.RedisClusterManager;
  14. import org.crazycake.shiro.RedisManager;
  15. import org.jeecg.common.constant.CommonConstant;
  16. import org.jeecg.common.util.oConvertUtils;
  17. import org.jeecg.config.shiro.filters.CustomShiroFilterFactoryBean;
  18. import org.jeecg.config.shiro.filters.JwtFilter;
  19. import org.springframework.aop.framework.autoproxy.DefaultAdvisorAutoProxyCreator;
  20. import org.springframework.beans.factory.annotation.Autowired;
  21. import org.springframework.beans.factory.annotation.Value;
  22. import org.springframework.context.annotation.Bean;
  23. import org.springframework.context.annotation.Configuration;
  24. import org.springframework.context.annotation.DependsOn;
  25. import org.springframework.core.env.Environment;
  26. import org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory;
  27. import org.springframework.util.StringUtils;
  28. import redis.clients.jedis.HostAndPort;
  29. import redis.clients.jedis.JedisCluster;
  30. import javax.annotation.Resource;
  31. import javax.servlet.Filter;
  32. import java.util.*;
  33. /**
  34. * @author: Scott
  35. * @date: 2018/2/7
  36. * @description: shiro 配置类
  37. */
  38. @Slf4j
  39. @Configuration
  40. public class ShiroConfig {
  41. @Value("${jeecg.shiro.excludeUrls}")
  42. private String excludeUrls;
  43. @Resource
  44. LettuceConnectionFactory lettuceConnectionFactory;
  45. @Autowired
  46. private Environment env;
  47. /**
  48. * Filter Chain定义说明
  49. *
  50. * 1、一个URL可以配置多个Filter,使用逗号分隔
  51. * 2、当设置多个过滤器时,全部验证通过,才视为通过
  52. * 3、部分过滤器可指定参数,如perms,roles
  53. */
  54. @Bean("shiroFilter")
  55. public ShiroFilterFactoryBean shiroFilter(SecurityManager securityManager) {
  56. CustomShiroFilterFactoryBean shiroFilterFactoryBean = new CustomShiroFilterFactoryBean();
  57. shiroFilterFactoryBean.setSecurityManager(securityManager);
  58. // 拦截器
  59. Map<String, String> filterChainDefinitionMap = new LinkedHashMap<String, String>();
  60. if(oConvertUtils.isNotEmpty(excludeUrls)){
  61. String[] permissionUrl = excludeUrls.split(",");
  62. for(String url : permissionUrl){
  63. filterChainDefinitionMap.put(url,"anon");
  64. }
  65. }
  66. // 配置不会被拦截的链接 顺序判断
  67. filterChainDefinitionMap.put("/systemController/**","anon");
  68. filterChainDefinitionMap.put("/sys/common/upload/**","anon");
  69. filterChainDefinitionMap.put("/sys/cas/client/validateLogin", "anon"); //cas验证登录
  70. filterChainDefinitionMap.put("/sys/randomImage/**", "anon"); //登录验证码接口排除
  71. filterChainDefinitionMap.put("/sys/checkCaptcha", "anon"); //登录验证码接口排除
  72. filterChainDefinitionMap.put("/sys/login", "anon"); //登录接口排除
  73. filterChainDefinitionMap.put("/sys/mLogin", "anon"); //登录接口排除
  74. filterChainDefinitionMap.put("/sys/logout", "anon"); //登出接口排除
  75. filterChainDefinitionMap.put("/sys/thirdLogin/**", "anon"); //第三方登录
  76. filterChainDefinitionMap.put("/sys/getEncryptedString", "anon"); //获取加密串
  77. filterChainDefinitionMap.put("/sys/sms", "anon");//短信验证码
  78. filterChainDefinitionMap.put("/sys/phoneLogin", "anon");//手机登录
  79. filterChainDefinitionMap.put("/sys/user/checkOnlyUser", "anon");//校验用户是否存在
  80. filterChainDefinitionMap.put("/sys/user/register", "anon");//用户注册
  81. filterChainDefinitionMap.put("/sys/user/phoneVerification", "anon");//用户忘记密码验证手机号
  82. filterChainDefinitionMap.put("/sys/user/passwordChange", "anon");//用户更改密码
  83. filterChainDefinitionMap.put("/auth/2step-code", "anon");//登录验证码
  84. filterChainDefinitionMap.put("/sys/common/static/**", "anon");//图片预览 &下载文件不限制token
  85. filterChainDefinitionMap.put("/sys/common/pdf/**", "anon");//pdf预览
  86. filterChainDefinitionMap.put("/generic/**", "anon");//pdf预览需要文件
  87. filterChainDefinitionMap.put("/", "anon");
  88. filterChainDefinitionMap.put("/doc.html", "anon");
  89. filterChainDefinitionMap.put("/**/*.js", "anon");
  90. filterChainDefinitionMap.put("/**/*.css", "anon");
  91. filterChainDefinitionMap.put("/**/*.html", "anon");
  92. filterChainDefinitionMap.put("/**/*.svg", "anon");
  93. filterChainDefinitionMap.put("/**/*.pdf", "anon");
  94. filterChainDefinitionMap.put("/**/*.jpg", "anon");
  95. filterChainDefinitionMap.put("/**/*.png", "anon");
  96. filterChainDefinitionMap.put("/**/*.ico", "anon");
  97. filterChainDefinitionMap.put("/systemController/**","anon");
  98. filterChainDefinitionMap.put("/sys/common/upload/**","anon");
  99. filterChainDefinitionMap.put("/bussMapQueryController/**","anon"); //一张图:业务属性查询接口
  100. filterChainDefinitionMap.put("/hzz.hhhj.glfwx/rmGlfwx/queryByInfo", "anon"); //划界信息查询
  101. filterChainDefinitionMap.put("/hzz.hhhj.gsp/rmBulletinBoard/queryByRelId", "anon"); //公告牌查询
  102. filterChainDefinitionMap.put("/hzz.hhhj.jzd/rmBoundaryMarker/queryByRelId", "anon"); //界桩点查询
  103. filterChainDefinitionMap.put("/hzz.axgh.axghgnq/rmAxghgnqgeo/getGeojsonById", "anon"); //岸线功能区图形json查询
  104. filterChainDefinitionMap.put("/hzz.axgh.axghgnq/lskzx/getGeojsonById", "anon"); //临水控制线图形json查询
  105. filterChainDefinitionMap.put("/hzz.axgh.axghgnq/wykzx/getGeojsonById", "anon"); //外缘控制线图形json查询
  106. filterChainDefinitionMap.put("/hzz.shjsgc.dfgc/rmDfgc/getGeojsonById", "anon"); //堤防工程图形json查询
  107. filterChainDefinitionMap.put("/hzz.shjsgc.bzgc/rmBzgc/queryById", "anon"); //泵站工程查询
  108. filterChainDefinitionMap.put("/hzz.shjsgc.zbgc/rmZbgc/queryById", "anon"); //闸坝工程查询
  109. filterChainDefinitionMap.put("/hzz.shjsgc.dfgc/rmDfgc/queryById", "anon"); //堤防工程查询
  110. // update-begin--Author:sunjianlei Date:20190813 for:排除字体格式的后缀
  111. filterChainDefinitionMap.put("/**/*.ttf", "anon");
  112. filterChainDefinitionMap.put("/**/*.woff", "anon");
  113. filterChainDefinitionMap.put("/**/*.woff2", "anon");
  114. // update-begin--Author:sunjianlei Date:20190813 for:排除字体格式的后缀
  115. filterChainDefinitionMap.put("/druid/**", "anon");
  116. filterChainDefinitionMap.put("/swagger-ui.html", "anon");
  117. filterChainDefinitionMap.put("/swagger**/**", "anon");
  118. filterChainDefinitionMap.put("/webjars/**", "anon");
  119. filterChainDefinitionMap.put("/v2/**", "anon");
  120. filterChainDefinitionMap.put("/sys/annountCement/show/**", "anon");
  121. //积木报表排除
  122. filterChainDefinitionMap.put("/jmreport/**", "anon");
  123. filterChainDefinitionMap.put("/**/*.js.map", "anon");
  124. filterChainDefinitionMap.put("/**/*.css.map", "anon");
  125. //大屏模板例子
  126. filterChainDefinitionMap.put("/test/bigScreen/**", "anon");
  127. //websocket排除
  128. filterChainDefinitionMap.put("/websocket/**", "anon");//系统通知和公告
  129. filterChainDefinitionMap.put("/newsWebsocket/**", "anon");//CMS模块
  130. filterChainDefinitionMap.put("/vxeSocket/**", "anon");//JVxeTable无痕刷新示例
  131. //性能监控 TODO 存在安全漏洞泄露TOEKN(durid连接池也有)
  132. filterChainDefinitionMap.put("/actuator/**", "anon");
  133. // 添加自己的过滤器并且取名为jwt
  134. Map<String, Filter> filterMap = new HashMap<String, Filter>(1);
  135. //如果cloudServer为空 则说明是单体 需要加载跨域配置【微服务跨域切换】
  136. Object cloudServer = env.getProperty(CommonConstant.CLOUD_SERVER_KEY);
  137. filterMap.put("jwt", new JwtFilter(cloudServer==null));
  138. shiroFilterFactoryBean.setFilters(filterMap);
  139. // <!-- 过滤链定义,从上向下顺序执行,一般将/**放在最为下边
  140. filterChainDefinitionMap.put("/**", "jwt");
  141. // 未授权界面返回JSON
  142. shiroFilterFactoryBean.setUnauthorizedUrl("/sys/common/403");
  143. shiroFilterFactoryBean.setLoginUrl("/sys/common/403");
  144. shiroFilterFactoryBean.setFilterChainDefinitionMap(filterChainDefinitionMap);
  145. return shiroFilterFactoryBean;
  146. }
  147. @Bean("securityManager")
  148. public DefaultWebSecurityManager securityManager(ShiroRealm myRealm) {
  149. DefaultWebSecurityManager securityManager = new DefaultWebSecurityManager();
  150. securityManager.setRealm(myRealm);
  151. /*
  152. * 关闭shiro自带的session,详情见文档
  153. * http://shiro.apache.org/session-management.html#SessionManagement-
  154. * StatelessApplications%28Sessionless%29
  155. */
  156. DefaultSubjectDAO subjectDAO = new DefaultSubjectDAO();
  157. DefaultSessionStorageEvaluator defaultSessionStorageEvaluator = new DefaultSessionStorageEvaluator();
  158. defaultSessionStorageEvaluator.setSessionStorageEnabled(false);
  159. subjectDAO.setSessionStorageEvaluator(defaultSessionStorageEvaluator);
  160. securityManager.setSubjectDAO(subjectDAO);
  161. //自定义缓存实现,使用redis
  162. securityManager.setCacheManager(redisCacheManager());
  163. return securityManager;
  164. }
  165. /**
  166. * 下面的代码是添加注解支持
  167. * @return
  168. */
  169. @Bean
  170. @DependsOn("lifecycleBeanPostProcessor")
  171. public DefaultAdvisorAutoProxyCreator defaultAdvisorAutoProxyCreator() {
  172. DefaultAdvisorAutoProxyCreator defaultAdvisorAutoProxyCreator = new DefaultAdvisorAutoProxyCreator();
  173. defaultAdvisorAutoProxyCreator.setProxyTargetClass(true);
  174. /**
  175. * 解决重复代理问题 github#994
  176. * 添加前缀判断 不匹配 任何Advisor
  177. */
  178. defaultAdvisorAutoProxyCreator.setUsePrefix(true);
  179. defaultAdvisorAutoProxyCreator.setAdvisorBeanNamePrefix("_no_advisor");
  180. return defaultAdvisorAutoProxyCreator;
  181. }
  182. @Bean
  183. public static LifecycleBeanPostProcessor lifecycleBeanPostProcessor() {
  184. return new LifecycleBeanPostProcessor();
  185. }
  186. @Bean
  187. public AuthorizationAttributeSourceAdvisor authorizationAttributeSourceAdvisor(DefaultWebSecurityManager securityManager) {
  188. AuthorizationAttributeSourceAdvisor advisor = new AuthorizationAttributeSourceAdvisor();
  189. advisor.setSecurityManager(securityManager);
  190. return advisor;
  191. }
  192. /**
  193. * cacheManager 缓存 redis实现
  194. * 使用的是shiro-redis开源插件
  195. *
  196. * @return
  197. */
  198. public RedisCacheManager redisCacheManager() {
  199. log.info("===============(1)创建缓存管理器RedisCacheManager");
  200. RedisCacheManager redisCacheManager = new RedisCacheManager();
  201. redisCacheManager.setRedisManager(redisManager());
  202. //redis中针对不同用户缓存(此处的id需要对应user实体中的id字段,用于唯一标识)
  203. redisCacheManager.setPrincipalIdFieldName("id");
  204. //用户权限信息缓存时间
  205. redisCacheManager.setExpire(200000);
  206. return redisCacheManager;
  207. }
  208. /**
  209. * 配置shiro redisManager
  210. * 使用的是shiro-redis开源插件
  211. *
  212. * @return
  213. */
  214. @Bean
  215. public IRedisManager redisManager() {
  216. log.info("===============(2)创建RedisManager,连接Redis..");
  217. IRedisManager manager;
  218. // redis 单机支持,在集群为空,或者集群无机器时候使用 add by jzyadmin@163.com
  219. if (lettuceConnectionFactory.getClusterConfiguration() == null || lettuceConnectionFactory.getClusterConfiguration().getClusterNodes().isEmpty()) {
  220. RedisManager redisManager = new RedisManager();
  221. redisManager.setHost(lettuceConnectionFactory.getHostName());
  222. redisManager.setPort(lettuceConnectionFactory.getPort());
  223. redisManager.setDatabase(lettuceConnectionFactory.getDatabase());
  224. redisManager.setTimeout(0);
  225. if (!StringUtils.isEmpty(lettuceConnectionFactory.getPassword())) {
  226. redisManager.setPassword(lettuceConnectionFactory.getPassword());
  227. }
  228. manager = redisManager;
  229. }else{
  230. // redis集群支持,优先使用集群配置
  231. RedisClusterManager redisManager = new RedisClusterManager();
  232. Set<HostAndPort> portSet = new HashSet<>();
  233. lettuceConnectionFactory.getClusterConfiguration().getClusterNodes().forEach(node -> portSet.add(new HostAndPort(node.getHost() , node.getPort())));
  234. //update-begin--Author:scott Date:20210531 for:修改集群模式下未设置redis密码的bug issues/I3QNIC
  235. if (oConvertUtils.isNotEmpty(lettuceConnectionFactory.getPassword())) {
  236. JedisCluster jedisCluster = new JedisCluster(portSet, 2000, 2000, 5,
  237. lettuceConnectionFactory.getPassword(), new GenericObjectPoolConfig());
  238. redisManager.setPassword(lettuceConnectionFactory.getPassword());
  239. redisManager.setJedisCluster(jedisCluster);
  240. } else {
  241. JedisCluster jedisCluster = new JedisCluster(portSet);
  242. redisManager.setJedisCluster(jedisCluster);
  243. }
  244. //update-end--Author:scott Date:20210531 for:修改集群模式下未设置redis密码的bug issues/I3QNIC
  245. manager = redisManager;
  246. }
  247. return manager;
  248. }
  249. }