Shiro 自定義 filter 匹配異常(好文)

[轉(zhuǎn)](http://www.hillfly.com/2017/179.html

最近忙著研究在 Springboot 上使用 Shiro 的問題滑进。剛好就遇到個(gè)詭異事驯镊,百度 Google 也沒找到啥有價(jià)值的信息够庙,幾番周折自己解決了,這里稍微記錄下践盼。

自定義 Filter

Shiro 支持自定義 Filter 大家都知道思瘟,也經(jīng)常用环础,這里我也用到了一個(gè)自定義 Filter,主要用于驗(yàn)證接口調(diào)用的 AccessToken 是否有效剩拢。

// AccessTokenFilter.java

public class AccessTokenFilter extends AccessControlFilter {
    @Override
    protected boolean isAccessAllowed(ServletRequest servletRequest,
                                      ServletResponse servletResponse,
                                      Object o) {
        if (isValidAccessToken(request)) {
            return true;
        }
        return false;
    }

    @Override
    protected boolean onAccessDenied(ServletRequest servletRequest, 
                                    ServletResponse servletResponse) throws Exception {
        throw new UnAuthorizedException("操作授權(quán)失斚叩谩!" + SysConstant.ACCESSTOKEN + "失效徐伐!");
    }
}
// ShiroConfiguration.java

@Bean
public AccessTokenFilter accessTokenFilter(){
    return new AccessTokenFilter();
}

 @Bean
public ShiroFilterFactoryBean shiroFilter(SecurityManager securityManager,
                                          IUrlFilterService urlFilterService) {
    ShiroFilterFactoryBean shiroFilterFactoryBean = new ShiroFilterFactoryBean();
    shiroFilterFactoryBean.setSecurityManager(securityManager);

    // 自定義過濾器
    Map<String, Filter> filterMap = shiroFilterFactoryBean.getFilters();
    filterMap.put("hasToken", accessTokenFilter());
    shiroFilterFactoryBean.setFilters(filterMap);

    // URL過濾
    Map<String, String> filterChainDefinitionMap = new LinkedHashMap<>();
    List<UrlFilter> urlFilterList = urlFilterService.selectAll();
    for (UrlFilter filter : urlFilterList) {
        filterChainDefinitionMap.put(filter.getFilterUrl(),
                filter.getFilterList());
    }

    shiroFilterFactoryBean.setFilterChainDefinitionMap(filterChainDefinitionMap);
    return shiroFilterFactoryBean;
}

ShiroFilter 中的 FilterChain 是從數(shù)據(jù)庫讀取的贯钩,如下:


image.png

我們想要達(dá)到的效果是,除了登陸和訪問 Druid 監(jiān)控頁面外办素,訪問其它地址一律要先驗(yàn)證 Token角雷,即走我們的自定義過濾器。
修改完畢后啟動(dòng)無異常性穿,我們訪問地址驗(yàn)證下勺三。

POST /api/login
{
  "hasError": true,
  "errors": {
    "httpStatus": 401,
    "errorCode": "4001",
    "errorMsg": "授權(quán)異常:操作授權(quán)失敗需曾!AccessToken失效吗坚!",
    "timestamp": "2017-06-10 11:08:03"
  }
}

funny,結(jié)果出乎意料呆万,居然登陸接口走了咱們的那個(gè)自定義 Filter商源??黑人問號(hào)臉桑嘶。炊汹。。

問題排查

FilterChain

首先檢查 Shiro FilterChain 加載的順序是否異常逃顶。

image.png

1讨便、集合容器使用 LinkedHashMap,保證的 FilterChain 的順序以政。
2霸褒、從數(shù)據(jù)庫讀取 Filter 時(shí)也是按 sort 排序的。
從調(diào)試結(jié)果來看盈蛮,加載順序和數(shù)據(jù)并沒有任何問題废菱,都是正確的。

排除了自身的數(shù)據(jù)問題抖誉,那就要往深處挖掘原因了殊轴,有了之前解決 Quartz 問題的經(jīng)歷,這次毫不猶豫就決定跟源碼跟蹤 Filter 注冊到匹配的過程袒炉。

Filter 注冊

要查明白為何匹配異常旁理,就要先弄清楚咱們的自定義 Filter 是如何注冊到 Shiro 的,顯然我磁,問題的關(guān)鍵在于 ShiroFilter 返回的 ShiroFilterFactoryBean 這個(gè)類中孽文,我們打開看看驻襟。很快,我們就鎖定了關(guān)鍵 method:

//ShiroFilterFactoryBean.java

protected AbstractShiroFilter createInstance() throws Exception {
    log.debug("Creating Shiro Filter instance.");
    SecurityManager securityManager = this.getSecurityManager();
    String msg;
    if(securityManager == null) {
        msg = "SecurityManager property must be set.";
        throw new BeanInitializationException(msg);
    } else if(!(securityManager instanceof WebSecurityManager)) {
        msg = "The security manager does not implement the WebSecurityManager interface.";
        throw new BeanInitializationException(msg);
    } else {
        FilterChainManager manager = this.createFilterChainManager();
        PathMatchingFilterChainResolver chainResolver = new PathMatchingFilterChainResolver();
        chainResolver.setFilterChainManager(manager);
        return new ShiroFilterFactoryBean.SpringShiroFilter((WebSecurityManager)securityManager, chainResolver);
    }
}

protected FilterChainManager createFilterChainManager() {
    DefaultFilterChainManager manager = new DefaultFilterChainManager();
    Map<String, Filter> defaultFilters = manager.getFilters();
    Iterator var3 = defaultFilters.values().iterator();

    while(var3.hasNext()) {
        Filter filter = (Filter)var3.next();
        this.applyGlobalPropertiesIfNecessary(filter);
    }

    Map<String, Filter> filters = this.getFilters();
    String name;
    Filter filter;
    if(!CollectionUtils.isEmpty(filters)) {
        for(Iterator var10 = filters.entrySet().iterator(); var10.hasNext(); manager.addFilter(name, filter, false)) {
            Entry<String, Filter> entry = (Entry)var10.next();
            name = (String)entry.getKey();
            filter = (Filter)entry.getValue();
            this.applyGlobalPropertiesIfNecessary(filter);
            if(filter instanceof Nameable) {
                ((Nameable)filter).setName(name);
            }
        }
    }

    Map<String, String> chains = this.getFilterChainDefinitionMap();
    if(!CollectionUtils.isEmpty(chains)) {
        Iterator var12 = chains.entrySet().iterator();

        while(var12.hasNext()) {
            Entry<String, String> entry = (Entry)var12.next();
            String url = (String)entry.getKey();
            String chainDefinition = (String)entry.getValue();
            manager.createChain(url, chainDefinition);
        }
    }

    return manager;
}
//DefaultFilterChainManager.java
public DefaultFilterChainManager() {
    this.addDefaultFilters(false);
}

//DefaultFilter.java
public enum DefaultFilter {
    anon(AnonymousFilter.class),
    authc(FormAuthenticationFilter.class),
    authcBasic(BasicHttpAuthenticationFilter.class),
    logout(LogoutFilter.class),
    noSessionCreation(NoSessionCreationFilter.class),
    perms(PermissionsAuthorizationFilter.class),
    port(PortFilter.class),
    rest(HttpMethodPermissionFilter.class),
    roles(RolesAuthorizationFilter.class),
    ssl(SslFilter.class),
    user(UserFilter.class);
}

看到這總算弄清楚 Shiro 加載 Filter 的順序:

  1. 加載 DefaultFilter 中的默認(rèn) Filter芋哭;
  2. 加載自定義 Filter沉衣;
  3. 加載 FFilterChainDefinitionMap;

弄清楚了這 Filter 的加載與注冊减牺,那這與我們要解決的問題有何關(guān)系呢豌习?首先我們懷疑這里獲取的 Filter 是異常的,調(diào)試打個(gè)斷點(diǎn)看看拔疚。


image.png

然而奇怪的是斑鸦,從調(diào)試結(jié)果來看,一切加載的 Filter 都如我們預(yù)想的那樣草雕,并無異常。

Filter Match

既然基本排除了 Filter 加載上出現(xiàn)問題的可能固以,那么就要來排查 Filter 匹配的問題了墩虹。
重點(diǎn)在于 AbstractShiroFilter 的 doFilterInternal(),這里是匹配的起點(diǎn)憨琳。

protected void doFilterInternal(ServletRequest servletRequest, ServletResponse servletResponse, final FilterChain chain) throws ServletException, IOException {
    Throwable t = null;
    try {
        final ServletRequest request = this.prepareServletRequest(servletRequest, servletResponse, chain);
        final ServletResponse response = this.prepareServletResponse(request, servletResponse, chain);
        Subject subject = this.createSubject(request, response);
        subject.execute(new Callable() {
            public Object call() throws Exception {
                AbstractShiroFilter.this.updateSessionLastAccessTime(request, response);
                AbstractShiroFilter.this.executeChain(request, response, chain);
                return null;
            }
        });
    } catch (ExecutionException var8) {
        t = var8.getCause();
    } catch (Throwable var9) {
        t = var9;
    }

    if(t != null) {
        if(t instanceof ServletException) {
            throw (ServletException)t;
        } else if(t instanceof IOException) {
            throw (IOException)t;
        } else {
            String msg = "Filtered request failed.";
            throw new ServletException(msg, t);
        }
    }
}

protected void executeChain(ServletRequest request, ServletResponse response, FilterChain origChain) throws IOException, ServletException {
    FilterChain chain = this.getExecutionChain(request, response, origChain);
    chain.doFilter(request, response);
}

跟蹤到最后诫钓,會(huì)進(jìn)入到一個(gè)關(guān)鍵方法:

//PathMatchingFilterChainResolver.java

public FilterChain getChain(ServletRequest request, ServletResponse response, FilterChain originalChain) {
    FilterChainManager filterChainManager = this.getFilterChainManager();
    if(!filterChainManager.hasChains()) {
        return null;
    } else {
        String requestURI = this.getPathWithinApplication(request);
        Iterator var6 = filterChainManager.getChainNames().iterator();
        String pathPattern;
        do {
            if(!var6.hasNext()) {
                return null;
            }
            pathPattern = (String)var6.next();
        } while(!this.pathMatches(pathPattern, requestURI));

        if(log.isTraceEnabled()) {
            log.trace("Matched path pattern [" + pathPattern + "] for requestURI [" + requestURI + "].  Utilizing corresponding filter chain...");
        }
        return filterChainManager.proxy(originalChain, pathPattern);
    }
}

顯然,這里就是進(jìn)行 URL 匹配的地方篙螟。難道是這里匹配出異常了菌湃?我們打個(gè)斷點(diǎn)在這里再訪問一下。然而怪異出現(xiàn)了遍略,沒有進(jìn)斷點(diǎn)惧所,直接返回了異常信息,根本沒有進(jìn)行匹配P餍印下愈!我們再對自定義 filter 斷點(diǎn)調(diào)試后發(fā)現(xiàn)了 Filter 調(diào)用鏈如下:

image.png

MMP 的,完全沒有不是按我們預(yù)想的那樣進(jìn)行調(diào)用蕾久。這 TM 居然是作為 Spring 的全局 Filter 被調(diào)用了势似。Shiro 的 Filter 優(yōu)先級居然失效了?我們都知道之前在 SpringMVC+Shiro 時(shí)僧著,都會(huì)把 Shiro 的 Filter 配置順序盡量放前履因,以達(dá)到優(yōu)先加載的目的。難道這里沒有走 Shiro 的匹配是因?yàn)檫@個(gè)嗎盹愚?栅迄?難道是因?yàn)?Springboot 先加載了我們自定義的 Filter,然后再加載了 ShiroFilter 嗎杯拐,然后這個(gè) Filter 優(yōu)先順序就出問題了霞篡?

我們將斷點(diǎn)打到 ApplicationFilterChain.java 的 internalDoFilter() 中進(jìn)行驗(yàn)證下:


image.png

J勒帷!果然袄时污淋!咱們的自定義 Filter 居然還在 ShiroFilter 之前,這就導(dǎo)致請求被我們自定義 Filter 先消費(fèi)掉了余掖。寸爆。ShiroFilter 成了擺設(shè)。
那么把咱們的 Bean 放到 ShiroFilter 后面會(huì)如何呢盐欺?

@Bean
public ShiroFilterFactoryBean shiroFilter(){}

@Bean
public AccessTokenFilter accessTokenFilter(){}

image.png

果然順序變了赁豆,那么問題解決了嗎?
——沒有冗美,問題依舊魔种,咱們的 Filter 還是跑了,返回了異常粉洼。

看來應(yīng)該不是這里的順序問題节预,我們回過頭來繼續(xù)看 ApplicationFilterChain.java 的 internalDoFilter(),系統(tǒng)會(huì)將注冊的 filters 逐一調(diào)用属韧,也就是說無論我們的順序如何安拟,F(xiàn)ilter 最終都是會(huì)被調(diào)用的。

問題解決

眼下我暫時(shí)有兩種辦法去解決這個(gè)問題:

  1. 修改 AccessTokenFilter宵喂,在 Filter 內(nèi)部加入 path match 方法對需要驗(yàn)證 token 的路徑進(jìn)行過濾糠赦。
  2. 將咱們的自定義 Filter 注冊到 Shiro,不注冊到 ApplicationFilterChain锅棕。

顯然方案一是不可取的拙泽,這樣修改范圍過大,得不償失了裸燎。那我們怎么去實(shí)現(xiàn)第二個(gè)方法呢奔滑?SpringBoot 提供了 FilterRegistrationBean 方便我們對 Filter 進(jìn)行管理。

@Bean
public FilterRegistrationBean registration(AccessTokenFilter filter) {
    FilterRegistrationBean registration = new FilterRegistrationBean(filter);
    registration.setEnabled(false);
    return registration;
}

將不需要注冊的 Filter 注入方法即可顺少。這時(shí)候再啟動(dòng)項(xiàng)目進(jìn)行測試朋其,就可以發(fā)現(xiàn) filters 已經(jīng)不存在咱們的自定義 Filter 了。

還有個(gè)辦法不需要使用到 FilterRegistrationBean脆炎,因?yàn)槲覀儗?AccessTokenFilter 注冊為了 Bean 交給 Spring 托管了梅猿,所以它會(huì)被自動(dòng)注冊到 FilterChain 中,那我們?nèi)绻话阉詾?Bean 就可以避免這個(gè)問題了秒裕。

/**
 * 不需要顯示注冊Bean了
@Bean
public AccessTokenFilter accessTokenFilter(){}
**/

@Bean
public ShiroFilterFactoryBean shiroFilter(SecurityManager securityManager,
                                          IUrlFilterService urlFilterService) {
    //省略
    filterMap.put("hasToken", new AccessTokenFilter());
    //省略
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末袱蚓,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子几蜻,更是在濱河造成了極大的恐慌喇潘,老刑警劉巖体斩,帶你破解...
    沈念sama閱讀 211,265評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異颖低,居然都是意外死亡絮吵,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,078評論 2 385
  • 文/潘曉璐 我一進(jìn)店門忱屑,熙熙樓的掌柜王于貴愁眉苦臉地迎上來蹬敲,“玉大人,你說我怎么就攤上這事莺戒“槲耍” “怎么了?”我有些...
    開封第一講書人閱讀 156,852評論 0 347
  • 文/不壞的土叔 我叫張陵从铲,是天一觀的道長瘪校。 經(jīng)常有香客問我,道長名段,這世上最難降的妖魔是什么渣淤? 我笑而不...
    開封第一講書人閱讀 56,408評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮吉嫩,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘嗅定。我一直安慰自己自娩,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,445評論 5 384
  • 文/花漫 我一把揭開白布渠退。 她就那樣靜靜地躺著忙迁,像睡著了一般。 火紅的嫁衣襯著肌膚如雪碎乃。 梳的紋絲不亂的頭發(fā)上姊扔,一...
    開封第一講書人閱讀 49,772評論 1 290
  • 那天,我揣著相機(jī)與錄音梅誓,去河邊找鬼恰梢。 笑死,一個(gè)胖子當(dāng)著我的面吹牛梗掰,可吹牛的內(nèi)容都是我干的嵌言。 我是一名探鬼主播,決...
    沈念sama閱讀 38,921評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼及穗,長吁一口氣:“原來是場噩夢啊……” “哼摧茴!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起埂陆,我...
    開封第一講書人閱讀 37,688評論 0 266
  • 序言:老撾萬榮一對情侶失蹤苛白,失蹤者是張志新(化名)和其女友劉穎娃豹,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體购裙,經(jīng)...
    沈念sama閱讀 44,130評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡懂版,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,467評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了缓窜。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片定续。...
    茶點(diǎn)故事閱讀 38,617評論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖禾锤,靈堂內(nèi)的尸體忽然破棺而出私股,到底是詐尸還是另有隱情,我是刑警寧澤恩掷,帶...
    沈念sama閱讀 34,276評論 4 329
  • 正文 年R本政府宣布倡鲸,位于F島的核電站,受9級特大地震影響黄娘,放射性物質(zhì)發(fā)生泄漏峭状。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,882評論 3 312
  • 文/蒙蒙 一逼争、第九天 我趴在偏房一處隱蔽的房頂上張望优床。 院中可真熱鬧,春花似錦誓焦、人聲如沸胆敞。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,740評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽移层。三九已至,卻和暖如春赫粥,著一層夾襖步出監(jiān)牢的瞬間观话,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,967評論 1 265
  • 我被黑心中介騙來泰國打工越平, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留频蛔,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,315評論 2 360
  • 正文 我出身青樓秦叛,卻偏偏與公主長得像帽驯,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子书闸,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,486評論 2 348

推薦閱讀更多精彩內(nèi)容