| 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@Configurationpublic 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;    }}
 |