`
jinnianshilongnian
  • 浏览: 21433507 次
  • 性别: Icon_minigender_1
博客专栏
5c8dac6a-21dc-3466-8abb-057664ab39c7
跟我学spring3
浏览量:2404875
D659df3e-4ad7-3b12-8b9a-1e94abd75ac3
Spring杂谈
浏览量:2997548
43989fe4-8b6b-3109-aaec-379d27dd4090
跟开涛学SpringMVC...
浏览量:5631359
1df97887-a9e1-3328-b6da-091f51f886a1
Servlet3.1规范翻...
浏览量:257535
4f347843-a078-36c1-977f-797c7fc123fc
springmvc杂谈
浏览量:1593116
22722232-95c1-34f2-b8e1-d059493d3d98
hibernate杂谈
浏览量:248952
45b32b6f-7468-3077-be40-00a5853c9a48
跟我学Shiro
浏览量:5847389
Group-logo
跟我学Nginx+Lua开...
浏览量:698077
5041f67a-12b2-30ba-814d-b55f466529d5
亿级流量网站架构核心技术
浏览量:780398
社区版块
存档分类
最新评论

具备白名单、黑名单的Filter基类扩展

 
阅读更多

 

 

之前有朋友问《servlet-mapping中url-pattern配置问题

 

Mybeautiful  写道
比如我定义了一个mapping, 
<servlet-mapping> 
<servlet-name>html_dispatcher</servlet-name> 
<url-pattern>*.html</url-pattern> 
</servlet-mapping> 

就是说所有的.html请求都会由html_dispatcher来处理,但现在我的问题是: 
我希望某个文件夹下面的.html不用这个html_dispatcher处理,而所有其它的.html请求不变。 

因为把所有其它的.html列出来实在太多了,多到我都不知道有多少。所以不能一一列出。 

很明显是在白名单里边排除一部分黑名单需求。

 

 

大家在开发Java Web时,在使用过滤器时是比较挠头的,比如

	<filter-mapping>
		<filter-name>TestFilter</filter-name>
		<url-pattern>/*</url-pattern>
	</filter-mapping>

 

url-pattern只能是如下三种:

1、前缀匹配   /user/*
2、后缀匹配  *.do
3、绝对匹配  /myservlet

而且如果我想过滤多个url必须
	<filter-mapping>
		<filter-name>TestFilter</filter-name>
		<url-pattern>/a</url-pattern>
	</filter-mapping>
	<filter-mapping>
		<filter-name>TestFilter</filter-name>
		<url-pattern>/b</url-pattern>
	</filter-mapping>

两个问题:
1、有的朋友可能需要过滤*.html 但要排除一部分(黑名单);
2、有的朋友可能需要过滤多个url,如/a  /b,对于每一个url必须写一个filter-mapping比较繁琐;

因此我写了一个非常简单的具备白名单和黑名单功能的基类Filter,过滤url使用Ant Path:
package com.sishuok.web.filter;

import java.io.IOException;
import java.util.ArrayList;
import java.util.List;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.util.AntPathMatcher;
import org.springframework.util.PathMatcher;
/**
 * Ant语法
 * 请参考 http://jinnianshilongnian.iteye.com/blog/1416322
 *  
 *  配置方式
	<filter>
		<filter-name>TestFilter</filter-name>
		<filter-class>com.sishuok.web.filter.TestFilter</filter-class>
		<!-- url分隔符可以是 换行 空格 分号  逗号  白名单和黑名单都是可选-->
		<init-param>
		    <param-name>blackListURL</param-name> <!-- 配置黑名单url 表示不走过滤器的url order:1 -->
		    <param-value>
		    	/aa
		    	/bb/**
		    	/cc/*
		    </param-value>
		</init-param>
		<init-param>
		    <param-name>whiteListURL</param-name> <!-- 配置白名单url 表示走过滤器的url order:2-->
		    <param-value>
		    	/dd;/ee,/ff /list
		    </param-value>
		</init-param>
	</filter>
	<filter-mapping>
	<filter-name>TestFilter</filter-name>
		<url-pattern>/*</url-pattern>
	</filter-mapping>
 * 
 * @author Zhang Kaitao
 *
 */
public abstract class BaseFilter implements Filter {
	
	private FilterConfig config = null;
			
	private final String[] NULL_STRING_ARRAY = new String[0];
	private final String URL_SPLIT_PATTERN = "[, ;\r\n]";//逗号  空格 分号  换行

	private final PathMatcher pathMatcher = new AntPathMatcher();

	private final Logger logger = LoggerFactory.getLogger("url.filter");

	/**
	 * 白名单
	 */
	private String[] whiteListURLs = null;

	/**
	 * 黑名单
	 */
	private String[] blackListURLs = null;

	@Override
	public final void init(FilterConfig config) throws ServletException {
		this.config = config;
		this.initConfig();
		this.init();
	}

	/**
	 * 子类覆盖
	 * 
	 * @throws ServletException
	 */
	public void init() throws ServletException {

	}

	/**
	 * 1、黑名单匹配 2、白名单匹配
	 */
	@Override
	public final void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) 
		throws IOException, ServletException {

		HttpServletRequest httpRequest = (HttpServletRequest) request;
		HttpServletResponse httpResponse = (HttpServletResponse) response;

		String currentURL = httpRequest.getServletPath();

		logger.debug("url filter : current url : [{}]", currentURL);

		if (isBlackURL(currentURL)) {
			chain.doFilter(request, response);
			return;
		}

		if (isWhiteURL(currentURL)) {
			doFilter(httpRequest, httpResponse, chain);
			return;
		}

		logger.debug("url filter : no url list matches : [{}]  break", currentURL);
		chain.doFilter(request, response);
		return;
	}

	private boolean isWhiteURL(String currentURL) {
		for (String whiteURL : whiteListURLs) {
			if (pathMatcher.match(whiteURL, currentURL)) {
				logger.debug("url filter : white url list matches : [{}] match [{}] continue", whiteURL, currentURL);
				return true;
			}
			logger.debug("url filter : white url list not matches : [{}] match [{}]", whiteURL, currentURL);
		}
		return false;
	}

	private boolean isBlackURL(String currentURL) {
		for (String blackURL : blackListURLs) {
			if (pathMatcher.match(blackURL, currentURL)) {
				logger.debug("url filter : black url list matches : [{}] match [{}] break", blackURL, currentURL);
				return true;
			}
			logger.debug("url filter : black url list not matches : [{}] match [{}]", blackURL, currentURL);
		}
		return false;
	}

	/**
	 * 子类覆盖
	 * 
	 * @param request
	 * @param response
	 * @param chain
	 * @throws IOException
	 * @throws ServletException
	 */
	public void doFilter(HttpServletRequest request, HttpServletResponse response, FilterChain chain)
			throws IOException, ServletException {
		chain.doFilter(request, response);
	}

	/**
	 * 子类覆盖
	 */
	@Override
	public void destroy() {

	}

	private void initConfig() {
		String whiteListURLStr = this.config.getInitParameter("whiteListURL");
		whiteListURLs = strToArray(whiteListURLStr);

		String blackListURLStr = this.config.getInitParameter("blackListURL");
		blackListURLs = strToArray(blackListURLStr);

	}

	private String[] strToArray(String urlStr) {
		if (urlStr == null) {
			return NULL_STRING_ARRAY;
		}
		String[] urlArray = urlStr.split(URL_SPLIT_PATTERN);

		List<String> urlList = new ArrayList<String>();

		for (String url : urlArray) {
			url = url.trim();
			if (url.length() == 0) {
				continue;
			}

			urlList.add(url);
		}

		return urlList.toArray(NULL_STRING_ARRAY);
	}

	public FilterConfig getConfig() {
		return config;
	}
}
 
具体使用如下:
package com.sishuok.web.filter;

import java.io.IOException;

import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

public class TestFilter extends BaseFilter {
	@Override
	public void init() throws ServletException {//初始化
		FilterConfig config = getConfig();
	}
	
	
	@Override
	public void doFilter(HttpServletRequest request, HttpServletResponse response, FilterChain chain)
		throws IOException, ServletException {
                
		System.out.println("==============");
		chain.doFilter(request, response);
		
	}
	
	@Override
	public void destroy() {//销毁
	}
	
}
 
xml配置如下:
	<filter>
		<filter-name>TestFilter</filter-name>
		<filter-class>com.sishuok.web.filter.TestFilter</filter-class>
		<!-- url分隔符可以是 换行 空格 分号  逗号  白名单和黑名单都是可选-->
		<init-param>
		    <param-name>blackListURL</param-name> <!-- 配置黑名单url 表示不走过滤器的url order:1 -->
		    <param-value>
		    	/aa
		    	/bb/**
		    	/cc/*
		    </param-value>
		</init-param>
		<init-param>
		    <param-name>whiteListURL</param-name> <!-- 配置白名单url 表示走过滤器的url order:2-->
		    <param-value>
		    	/dd;/ee,/ff /list
		    </param-value>
		</init-param>
	</filter>
	<filter-mapping>
		<filter-name>TestFilter</filter-name>
		<url-pattern>/*</url-pattern>
	</filter-mapping>
 
因为相对来说实现很简单,具体实现就不详细介绍了。
16
12
分享到:
评论
8 楼 jinnianshilongnian 2013-05-13  
varyall 写道
实在找不到地方了,发在这里你帮我解决下,谢谢:
问题在这里呢
http://www.iteye.com/problems/95034

那我帮你看看
7 楼 varyall 2013-05-13  
实在找不到地方了,发在这里你帮我解决下,谢谢:
问题在这里呢
http://www.iteye.com/problems/95034
6 楼 jinnianshilongnian 2012-09-20  
hubert_bubert 写道
对于子类来说,把子类覆盖的doFilter()方法改为abstract是否更好一些?

当时也这么考虑  如果你仅仅需要过滤功能没有必要再去实现 
5 楼 hubert_bubert 2012-09-20  
对于子类来说,把子类覆盖的doFilter()方法改为abstract是否更好一些?
4 楼 jinnianshilongnian 2012-08-31  
锅巴49 写道
不错,有碰到过这个问题

呵呵
3 楼 锅巴49 2012-08-31  
不错,有碰到过这个问题
2 楼 jinnianshilongnian 2012-08-27  
飞天奔月 写道
l 以”/’开头和以”/*”结尾的是用来做路径映射的。

l 以前缀”*.”开头的是用来做扩展映射的。

l “/” 是用来定义default servlet映射的。

l 剩下的都是用来定义详细映射的。
 

比如: /aa/bb/cc.action

1 楼 飞天奔月 2012-08-27  
l 以”/’开头和以”/*”结尾的是用来做路径映射的。

l 以前缀”*.”开头的是用来做扩展映射的。

l “/” 是用来定义default servlet映射的。

l 剩下的都是用来定义详细映射的。
 

比如: /aa/bb/cc.action

相关推荐

Global site tag (gtag.js) - Google Analytics