明確需求
在使用Shiro的時候,鑒權失敗一般都是返回一個錯誤頁或者登錄頁給前端,特別是后台系統,這種模式用的特別多。但是現在的項目越來越多的趨向於使用前后端分離的方式開發,這時候就需要響應Json數據給前端了,前端再根據狀態碼做相應的操作。那么Shiro框架能不能在鑒權失敗的時候直接返回Json數據呢?答案當然是可以。
其實Shiro的自定義過濾器功能特別強大,可以實現很多實用的功能,向前端返回Json數據自然不在話下。通常我們沒有去關注它是因為Shiro內置的一下過濾器功能已經比較全了,后台系統的權限控制基本上只需要使用Shiro內置的一些過濾器就能實現了,此處再次貼上這個圖。

相關文檔地址:http://shiro.apache.org/web.html#default-filters
我最近的一個項目是需要為手機APP提供功能接口,需要做用戶登錄,Session持久化以及Session共享,但不需要細粒度的權限控制。面對這個需求我第一個想到的就是集成Shiro了,Session的持久化及共享在Shiro系列第二篇已經講過了,那么這篇順便用一下Shiro中的自定義過濾器。因為不需要提供細粒度權限控制,只需要做登錄鑒權,而且鑒權失敗后需要向前端響應Json數據,那么使用自定義Filter再好不過了。
自定義Filter
還是以第一篇的Demo為例,項目地址在文章尾部有放上,本篇在之前的代碼上繼續添加功能。
首發地址:https://www.guitu18.com/post/2020/01/06/64.html
在實現自定義Filter之前,我們先看看這個類:org.apache.shiro.web.filter.AccessControlFilter,點開它的子類,發現子類全部都是org.apache.shiro.web.filter.authc和org.apache.shiro.web.filter.authz這兩個包下的,大多都繼承了AccessControlFilter這個類。這些子類的類名是不是很眼熟,看上面那張我貼了三遍的圖,大部分都在這里面呢。

看來AccessControlFilter這個類是跟Shiro權限過濾密切相關的,那么先看看它的體系結構:

它的頂級父類是javax.servlet.Filter,前面我們也說過,Shiro中所有的權限過濾都是基於Filter來實現的。自定義Filter同樣需要實現AccessControlFilter,這里我們添加一個登錄驗證過濾器,代碼如下:
public class AuthLoginFilter extends AccessControlFilter {
// 未登錄登陸返狀態回碼
private int code;
// 未登錄登陸返提示信息
private String message;
public AuthLoginFilter(int code, String message) {
this.code = code;
this.message = message;
}
@Override
protected boolean isAccessAllowed(ServletRequest servletRequest, ServletResponse servletResponse,
Object mappedValue) throws Exception {
Subject subject = SecurityUtils.getSubject();
// 這里配合APP需求我只需要做登錄檢測即可
if (subject != null && subject.isAuthenticated()) {
// TODO 登錄檢測通過,這里可以添加一些自定義操作
return Boolean.TRUE;
}
// 登錄檢測失敗返貨False后會進入下面的onAccessDenied()方法
return Boolean.FALSE;
}
@Override
protected boolean onAccessDenied(ServletRequest servletRequest,
ServletResponse servletResponse) throws Exception {
PrintWriter out = null;
try {
// 這里就很簡單了,向Response中寫入Json響應數據,需要聲明ContentType及編碼格式
servletResponse.setCharacterEncoding("UTF-8");
servletResponse.setContentType("application/json; charset=utf-8");
out = servletResponse.getWriter();
out.write(JSONObject.toJSONString(R.error(code, message)));
} catch (IOException e) {
e.printStackTrace();
} finally {
if (out != null) {
out.close();
}
}
return Boolean.FALSE;
}
}
自定義過濾器寫好了,現在需要把它交給Shiro管理:
@Bean
public ShiroFilterFactoryBean shiroFilterFactoryBean(SecurityManager securityManager) {
ShiroFilterFactoryBean shiroFilterFactoryBean = new ShiroFilterFactoryBean();
shiroFilterFactoryBean.setSecurityManager(securityManager);
// 添加登錄過濾器
Map<String, Filter> filters = new LinkedHashMap<>();
// 這里注釋的一行是我這次踩的一個小坑,我一開始按下面這么配置產生一個我意料之外的問題
// filters.put("authLogin", authLoginFilter());
// 正確的配置是需要我們自己new出來,不能將這個Filter交給Spring管理,后面會說明
filters.put("authLogin", new AuthLoginFilter(500, "未登錄或登錄超時"));
shiroFilterFactoryBean.setFilters(filters);
// 設置過濾規則
Map<String, String> filterMap = new LinkedHashMap<>();
filterMap.put("/api/login", "anon");
filterMap.put("/api/**", "authLogin");
shiroFilterFactoryBean.setFilterChainDefinitionMap(filterMap);
return shiroFilterFactoryBean;
}
如此Shiro添加自定義過濾器就完成了。自定義的Filter可以添加多個以實現不同的需求,你僅僅需要在filters中將過濾器起好名字put進去,並在filterChainMap中添加過濾器別名和路徑的映射就可以使用這個過濾器了。需要注意的一點就是過濾器是從前往后順序匹配的,所以要把范圍大的路徑放在后面put進去。
到這里自定義Filter功能已經實現了,后面是采坑排查記錄,不感興趣可以跳過。
問題排查
前半段介紹了如何使用Shiro的自定義Filter功能實現過濾,在Shiro配置代碼中我提了一句這次配置踩的一個小坑,如果我們將自定義的Filter交給Spring管理,會產生一些意料之外的問題。確實,通常在Spring項目中做配置時,我們都默認將Bean交由Spring管理,一般不會有什么問題,但是這次不一樣,先看代碼如下:
public ShiroFilterFactoryBean shiroFilterFactoryBean(SecurityManager securityManager) {
...
filters.put("authLogin", authLoginFilter());
...
filterMap.put("/api/login", "anon");
filterMap.put("/api/**", "authLogin");
...
}
@Bean
public AuthLoginFilter authLoginFilter() {
return new AuthLoginFilter(500, "未登錄或登錄超時");
}
這樣配置后造成的現象是:無論前面的過濾器是否放行,最終都會走到自定義的AuthLoginFilter過濾器。
比如上面的配置,我們訪問/api/login正常來講會被anon匹配到AnonymousFilter中,這里是什么都沒做直接放行的,但是放行后還會繼續走到AuthLoginFilter中,怎么會這樣,說好的按順序匹配呢,怎么不按套路出牌。
打斷點一路往上追溯,我們找到了ApplicationFilterChain這里,它是Tomcat所實現的一個Java Servlet API的規范。所有的請求都必須通過filters里的過濾器層層過濾后才會調用Servlet中的方法service()方法。這里包括Spring中的各種過濾器,全部都是注冊到這里來的。
前面的四個Filter都是Spring的,第五個是Shiro的ShiroFilterFactoryBean,它的內部也維護了一個filters,用來保存Shiro內置的一些過濾器和我們自定義的過濾器,Tomcat所維護的filters和Shiro維護的filters是一個父子層級的關系,Shiro中的ShiroFilterFactoryBean僅僅只是Tomcat里filters中的一員。點開看ShiroFilterFactoryBean查看,果然Shiro內置的一些過濾器全都按順序排着呢,我們自定義的AuthLoginFilter在最后一個。

但是,再看看Tomcat中的第六個過濾器,居然也是我們自定義的AuthLoginFilter,它同時出現在Tomcat和Shiro的filters中,這樣也就造成了前面提到的問題,Shiro在匹配到anon之后確實會將請求放行,但是在外層Tomcat的Filter中依舊被匹配上了,造成的現象好像是Shiro的Filter配置規則失效了,其實這個問題跟Shiro並沒有關系。
問題的根源找到了,想要解決這個問題必須找到這個自定義的Filter何時被添加到Tomcat的過濾器執行鏈中以及其原因。
追根溯源
關於這個問題我找到了ServletContextInitializerBeans這個類中,它在Spring啟動時就會初始化,在它的構造方法中做了很多初始化相關的操作。至於這一系列初始化流程就不得不提ServletContextInitializer相關知識點了,關於它的內容完全可以另開一片博客細說了。先看看ServletContextInitializerBeans的構造方法:
@SafeVarargs
public ServletContextInitializerBeans(ListableBeanFactory beanFactory,
Class<? extends ServletContextInitializer>... initializerTypes) {
this.initializers = new LinkedMultiValueMap<>();
this.initializerTypes = (initializerTypes.length != 0) ? Arrays.asList(initializerTypes)
: Collections.singletonList(ServletContextInitializer.class);
// 上面提到的Filter正是在這個方法開始一步步被添加到ApplicationFilterChain中的
addServletContextInitializerBeans(beanFactory);
addAdaptableBeans(beanFactory);
List<ServletContextInitializer> sortedInitializers = this.initializers.values().stream()
.flatMap((value) -> value.stream().sorted(AnnotationAwareOrderComparator.INSTANCE))
.collect(Collectors.toList());
this.sortedList = Collections.unmodifiableList(sortedInitializers);
logMappings(this.initializers);
}
上面提到的ApplicationFilterChain中的Filter正是在addServletContextInitializerBeans(beanFactory)這個方法開始一步步被添加到Filters中的,限於篇幅這里就看一下關鍵步驟。
private void addServletContextInitializerBeans(ListableBeanFactory beanFactory) {
for (Class<? extends ServletContextInitializer> initializerType : this.initializerTypes) {
for (Entry<String, ? extends ServletContextInitializer> initializerBean :
// 這里根據type獲取Bean列表並遍歷
getOrderedBeansOfType(beanFactory, initializerType)) {
// 此處開始添加對應的ServletContextInitializer
addServletContextInitializerBean(initializerBean.getKey(), initializerBean.getValue(), beanFactory);
}
}
}
addServletContextInitializerBeans(beanFactory)一路走下去會到達getOrderedBeansOfType()方法中,然后調用了beanFactory的getBeanNamesForType(),默認的實現在DefaultListableBeanFactory中,這里所貼前后刪減掉了無關代碼:
private String[] doGetBeanNamesForType(ResolvableType type, boolean includeNonSingletons, boolean allowEagerInit) {
List<String> result = new ArrayList<>();
// 檢查所有的Bean
for (String beanName : this.beanDefinitionNames) {
// 當這個Bean名稱沒有定義為其他bean的別名時,才進行匹配
if (!isAlias(beanName)) {
RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);
// 檢查Bean的完整性,檢測是否是抽象類,是否懶加載等等屬性
if (!mbd.isAbstract() && (allowEagerInit || (mbd.hasBeanClass() || !mbd.isLazyInit() ||
isAllowEagerClassLoading()) && !requiresEagerInitForType(mbd.getFactoryBeanName()))) {
// 匹配的Bean是否是FactoryBean,對於FactoryBean,需要匹配它創建的對象
boolean isFactoryBean = isFactoryBean(beanName, mbd);
BeanDefinitionHolder dbd = mbd.getDecoratedDefinition();
// 這里也是做完整性檢查
boolean matchFound = (allowEagerInit || !isFactoryBean || (dbd != null && !mbd.isLazyInit())
|| containsSingleton(beanName)) && (includeNonSingletons ||
(dbd != null ? mbd.isSingleton() : isSingleton(beanName))) && isTypeMatch(beanName, type);
if (!matchFound && isFactoryBean) {
// 對於FactoryBean,接下來嘗試匹配FactoryBean實例本身
beanName = FACTORY_BEAN_PREFIX + beanName;
matchFound = (includeNonSingletons || mbd.isSingleton()) && isTypeMatch(beanName, type);
}
if (matchFound) {
result.add(beanName);
}
}
}
}
return StringUtils.toStringArray(result);
}
到這里就是關鍵所在了,它會根據目標類型調用isTypeMatch(beanName, type)匹配每一個被Spring接管的Bean,isTypeMatch方法很長,這里就不貼了,有興趣的可以自行去看看,它位於AbstractBeanFactory中。這里匹配的type就是ServletContextInitializerBeans遍歷自構造方法中的initializerTypes列表。
從doGetBeanNamesForType出來后,再看這個方法:
private void addServletContextInitializerBean(String beanName,
ServletContextInitializer initializer, ListableBeanFactory beanFactory) {
if (initializer instanceof ServletRegistrationBean) {
Servlet source = ((ServletRegistrationBean<?>) initializer).getServlet();
addServletContextInitializerBean(Servlet.class, beanName, initializer,
beanFactory, source);
}
else if (initializer instanceof FilterRegistrationBean) {
Filter source = ((FilterRegistrationBean<?>) initializer).getFilter();
addServletContextInitializerBean(Filter.class, beanName, initializer,
beanFactory, source);
}
else if (initializer instanceof DelegatingFilterProxyRegistrationBean) {
String source = ((DelegatingFilterProxyRegistrationBean) initializer)
.getTargetBeanName();
addServletContextInitializerBean(Filter.class, beanName, initializer,
beanFactory, source);
}
else if (initializer instanceof ServletListenerRegistrationBean) {
EventListener source = ((ServletListenerRegistrationBean<?>) initializer)
.getListener();
addServletContextInitializerBean(EventListener.class, beanName, initializer,
beanFactory, source);
}
else {
addServletContextInitializerBean(ServletContextInitializer.class, beanName,
initializer, beanFactory, initializer);
}
}
前面兩個配置過Filter和Servlet的應該很熟悉,Spring中添加自定義Filter經常這么用,添加Servlet同理:
@Bean
public FilterRegistrationBean xssFilterRegistration() {
FilterRegistrationBean registration = new FilterRegistrationBean();
registration.setDispatcherTypes(DispatcherType.REQUEST);
registration.setFilter(new XxxFilter());
registration.addUrlPatterns("/*");
registration.setName("xxxFilter");
return registration;
}
這樣Spring就會將其添加到過濾器執行鏈中,當然這只是添加Filter的眾多方式之一。
解決方案
那么問題的根源找到了,被Spring接管的Bean中所有的Filter都會被添加到ApplicationFilterChain,那我不讓Spring接管我的AuthLoginFilter不就行了。如何做?配置的時候直接new出來,還記得前面的那兩行代碼嗎:
// 這里注釋的一行是我這次踩的一個小坑,我一開始按下面這么配置產生了一個我意料之外的問題
// filters.put("authLogin", authLoginFilter());
// 正確的配置是需要我們自己new出來,不能將這個Filter交給Spring管理
filters.put("authLogin", new AuthLoginFilter(500, "未登錄或登錄超時"));
OK,問題解決,就是這么簡單。但就是這么小小的一個問題,在不清楚問題產生的原因的情況下,根本想不到是Spring接管Filter造成的,了解了底層,才能更好的排查問題。
尾巴
Shiro中自定義Filter僅需要繼承AccessControlFilter類后實現參與過濾的兩個方法,再將其配置到ShiroFilterFactoryBean中即可。- 需要注意的點是,因為
Spring的初始化機制,我們自定義的Filter如果被Spring接管,那么會被Spring添加到ApplicationFilterChain中,導致這個自定義過濾器會被重復執行,也就是無論Shiro中的過濾器過濾結果如何,最后依舊會走到被添加到ApplicationFilterChain中的自定義過濾器。 - 解決這個問題的方法非常簡單,不讓
Spring接管我們的Filter,直接new出來配置到Shiro中即可。 - 碼海無涯,不進則退,日積跬步,以至千里。
Shiro系列博客項目源代碼地址:
Gitee:https://gitee.com/guitu18/ShiroDemo
GitHub:https://github.com/guitu18/ShiroDemo
