SpringBoot 2.x 開(kāi)發(fā)案例之 Shiro 整合 Redis

image

前言

前段時(shí)間做了一個(gè)圖床的小項(xiàng)目仁讨,安全框架使用的是Shiro贱纠。為了使用戶7x24小時(shí)訪問(wèn)乌奇,決定把項(xiàng)目由單機(jī)升級(jí)為集群部署架構(gòu)驶鹉。但是安全框架shiro只有單機(jī)存儲(chǔ)的SessionDao,盡管Shrio有基于Ehcache-rmi的組播/廣播實(shí)現(xiàn)絮识,然而集群的分布往往是跨網(wǎng)段的绿聘,甚至是跨地域的,所以尋求新的方案次舌。

架構(gòu)

image

方案

使用 redis 集中存儲(chǔ)熄攘,實(shí)現(xiàn)分布式集群共享用戶信息,這里我們采用第三方開(kāi)源插件crazycake來(lái)實(shí)現(xiàn)彼念,pom.xml 引入:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<dependency>
    <groupId>org.crazycake</groupId>
    <artifactId>shiro-redis</artifactId>
    <version>3.2.3</version>
</dependency>

配置 application.properties

# Redis
# 數(shù)據(jù)庫(kù)索引(默認(rèn)為0)
redis.database=0
# 服務(wù)器地址 變更為自己的
redis.host=127.0.0.1
# 服務(wù)器連接端口
redis.port=6379
# 服務(wù)器連接密碼挪圾,如果不設(shè)置密碼注釋掉即可
# redis.password=
# 連接超時(shí)時(shí)間(毫秒)
redis.timeout=30000

本來(lái)crazycake插件已經(jīng)實(shí)現(xiàn)了RedisManager,但是參數(shù)不可配逐沙,這里我們需要自己重寫(xiě)一下:

public class RedisManager extends WorkAloneRedisManager implements IRedisManager {

    private RedisProperties redis;

    private JedisPool jedisPool;

    public RedisManager(RedisProperties redis) {
        this.redis = redis;
    }

    private void init() {
        synchronized(this) {
            if (this.jedisPool == null) {
                this.jedisPool = new JedisPool(this.getJedisPoolConfig(), redis.getHost(), redis.getPort(),
                        redis.getTimeout(), redis.getPassword(), redis.getDatabase());
            }
        }
    }

    @Override
    protected Jedis getJedis() {
        if (this.jedisPool == null) {
            this.init();
        }
        return this.jedisPool.getResource();
    }
}

參數(shù)配置 RedisProperties

@Data
@ConfigurationProperties(prefix = "redis")
public class RedisProperties {

    private String host;
    private int port;
    private int timeout;
    private String password;
    private int database;
}

配置 ShiroConfig

/**
 * Shiro權(quán)限配置
 * 一定要配置 @Configuration 和 @EnableConfigurationProperties 注解
 */
@Configuration
@EnableConfigurationProperties({RedisProperties.class})
public class ShiroConfig {

    private RedisProperties redis;

    public ShiroConfig(RedisProperties redis) {
        this.redis = redis;
    }

    @Bean
    public UserRealm userRealm() {
        return new UserRealm();
    }

    @Bean
    public ShiroFilterFactoryBean shiroFilterFactoryBean (SecurityManager securityManager) {
        ShiroFilterFactoryBean shiroFilterFactoryBean = new ShiroFilterFactoryBean();
        shiroFilterFactoryBean.setSecurityManager(securityManager);
        shiroFilterFactoryBean.setLoginUrl("/index.html");
        shiroFilterFactoryBean.setUnauthorizedUrl("/403");
        // 攔截器
        Map<String, String> filterChainDefinitionMap = new LinkedHashMap<>();
        /**
         * 靜態(tài)文件
         */
        filterChainDefinitionMap.put("/file/**","anon");
        /**
         * 登錄注冊(cè)
         */
        filterChainDefinitionMap.put("/register.shtml","anon");
        filterChainDefinitionMap.put("/login.shtml","anon");
        /**
         * 管理后臺(tái)
         */
        filterChainDefinitionMap.put("/sys/**", "roles[admin]");
        filterChainDefinitionMap.put("/**", "authc");
        shiroFilterFactoryBean.setFilterChainDefinitionMap(filterChainDefinitionMap);
        return shiroFilterFactoryBean;
    }

    @Bean
    public SessionsSecurityManager securityManager() {
        DefaultWebSecurityManager securityManager = new DefaultWebSecurityManager();
        securityManager.setRealm(userRealm());
        securityManager.setCacheManager(cacheManager());
        securityManager.setSessionManager(sessionManager());
        return securityManager;
    }
    @Bean
    public DefaultWebSessionManager sessionManager() {
        DefaultWebSessionManager sessionManager = new DefaultWebSessionManager();
        sessionManager.setSessionIdUrlRewritingEnabled(false);
        sessionManager.setSessionDAO(redisSessionDAO());
        return sessionManager;
    }
    @Bean
    public ShiroDialect shiroDialect(){
        return new ShiroDialect();
    }

    /**
     * cacheManager 緩存 redis實(shí)現(xiàn)
     * @return
     */
    public RedisCacheManager cacheManager() {
        RedisCacheManager redisCacheManager = new RedisCacheManager();
        redisCacheManager.setRedisManager(redisManager());
        return redisCacheManager;
    }

    /**
     * 配置shiro redisManager
     * @return
     */
    public RedisManager redisManager() {
        RedisManager redisManager = new RedisManager(redis);
        return redisManager;
    }

    /**
     * RedisSessionDAO shiro sessionDao層的實(shí)現(xiàn)
     * 原理就是重寫(xiě) AbstractSessionDAO
     * 有興趣的小伙伴自行閱讀源碼
     */
    @Bean
    public RedisSessionDAO redisSessionDAO() {
        RedisSessionDAO redisSessionDAO = new RedisSessionDAO();
        redisSessionDAO.setRedisManager(redisManager());
        return redisSessionDAO;
    }
}

小結(jié)

是不是很爽哲思,以后重啟應(yīng)用再也不用擔(dān)心用戶投訴了?

源碼

https://gitee.com/52itstyle/SPTools

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末吩案,一起剝皮案震驚了整個(gè)濱河市也殖,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖忆嗜,帶你破解...
    沈念sama閱讀 206,311評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異崎岂,居然都是意外死亡捆毫,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,339評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門(mén)冲甘,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)绩卤,“玉大人,你說(shuō)我怎么就攤上這事江醇”舯铮” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 152,671評(píng)論 0 342
  • 文/不壞的土叔 我叫張陵陶夜,是天一觀的道長(zhǎng)凛驮。 經(jīng)常有香客問(wèn)我,道長(zhǎng)条辟,這世上最難降的妖魔是什么黔夭? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,252評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮羽嫡,結(jié)果婚禮上本姥,老公的妹妹穿的比我還像新娘。我一直安慰自己杭棵,他們只是感情好婚惫,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,253評(píng)論 5 371
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著魂爪,像睡著了一般先舷。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上甫窟,一...
    開(kāi)封第一講書(shū)人閱讀 49,031評(píng)論 1 285
  • 那天密浑,我揣著相機(jī)與錄音,去河邊找鬼粗井。 笑死尔破,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的浇衬。 我是一名探鬼主播懒构,決...
    沈念sama閱讀 38,340評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼耘擂!你這毒婦竟也來(lái)了胆剧?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 36,973評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎秩霍,沒(méi)想到半個(gè)月后篙悯,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,466評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡铃绒,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,937評(píng)論 2 323
  • 正文 我和宋清朗相戀三年鸽照,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片颠悬。...
    茶點(diǎn)故事閱讀 38,039評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡矮燎,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出赔癌,到底是詐尸還是另有隱情诞外,我是刑警寧澤,帶...
    沈念sama閱讀 33,701評(píng)論 4 323
  • 正文 年R本政府宣布灾票,位于F島的核電站峡谊,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏铝条。R本人自食惡果不足惜靖苇,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,254評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望班缰。 院中可真熱鬧贤壁,春花似錦、人聲如沸埠忘。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,259評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)莹妒。三九已至名船,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間旨怠,已是汗流浹背渠驼。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,485評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留鉴腻,地道東北人迷扇。 一個(gè)月前我還...
    沈念sama閱讀 45,497評(píng)論 2 354
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像爽哎,于是被迫代替她去往敵國(guó)和親蜓席。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,786評(píng)論 2 345