mirror of
				https://gitee.com/dromara/hutool.git
				synced 2025-10-25 10:19:23 +08:00 
			
		
		
		
	Merge branch 'v5-dev' of gitee.com:dromara/hutool into v5-dev
This commit is contained in:
		| @@ -15,6 +15,7 @@ | |||||||
| * 【db     】     MetaUtil增加getTableMeta重载(issue#2157@Github) | * 【db     】     MetaUtil增加getTableMeta重载(issue#2157@Github) | ||||||
| * 【http   】     增加HttpGlobalConfig.setDecodeUrl(issue#I4U8YQ@Gitee) | * 【http   】     增加HttpGlobalConfig.setDecodeUrl(issue#I4U8YQ@Gitee) | ||||||
| * 【core   】     增加Base58(pr#2162@Github) | * 【core   】     增加Base58(pr#2162@Github) | ||||||
|  | * 【core   】     增加AntPathMatcher(issue#I4T7K5@Gitee) | ||||||
|  |  | ||||||
| ### 🐞Bug修复 | ### 🐞Bug修复 | ||||||
| * 【cache  】     修复ReentrantCache.toString方法线程不安全问题(issue#2140@Github) | * 【cache  】     修复ReentrantCache.toString方法线程不安全问题(issue#2140@Github) | ||||||
|   | |||||||
							
								
								
									
										942
									
								
								hutool-core/src/main/java/cn/hutool/core/text/AntPathMatcher.java
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										942
									
								
								hutool-core/src/main/java/cn/hutool/core/text/AntPathMatcher.java
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,942 @@ | |||||||
|  | package cn.hutool.core.text; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | import cn.hutool.core.util.StrUtil; | ||||||
|  |  | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.Comparator; | ||||||
|  | import java.util.LinkedHashMap; | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.Map; | ||||||
|  | import java.util.concurrent.ConcurrentHashMap; | ||||||
|  | import java.util.regex.Matcher; | ||||||
|  | import java.util.regex.Pattern; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Ant风格的路径匹配器。<br> | ||||||
|  |  * 来自Spring-core和Ant | ||||||
|  |  * | ||||||
|  |  * <p>匹配URL的规则如下:<br> | ||||||
|  |  * <ul> | ||||||
|  |  * <li>{@code ?} 匹配单个字符</li> | ||||||
|  |  * <li>{@code *} 匹配0个或多个字符</li> | ||||||
|  |  * <li>{@code **} 0个或多个路径中的<em>目录节点</em></li> | ||||||
|  |  * <li>{@code {hutool:[a-z]+}} 匹配以"hutool"命名的正则 {@code [a-z]+}</li> | ||||||
|  |  * </ul> | ||||||
|  |  * | ||||||
|  |  * <p>例子: </p> | ||||||
|  |  * <ul> | ||||||
|  |  * <li>{@code com/t?st.jsp} — 匹配 {@code com/test.jsp} 或 {@code com/tast.jsp} 或 {@code com/txst.jsp}</li> | ||||||
|  |  * <li>{@code com/*.jsp} — 匹配{@code com}目录下全部 {@code .jsp}文件</li> | ||||||
|  |  * <li>{@code com/**/test.jsp} — 匹配{@code com}目录下全部 {@code test.jsp}文件</li> | ||||||
|  |  * <li>{@code cn/hutool/**/*.jsp} — 匹配{@code cn/hutool}路径下全部{@code .jsp} 文件</li> | ||||||
|  |  * <li>{@code org/**/servlet/bla.jsp} — 匹配{@code cn/hutool/servlet/bla.jsp} 或{@code cn/hutool/testing/servlet/bla.jsp} 或 {@code org/servlet/bla.jsp}</li> | ||||||
|  |  * <li>{@code com/{filename:\\w+}.jsp} 匹配 {@code com/test.jsp} 并将 {@code test} 关联到 {@code filename} 变量</li> | ||||||
|  |  * </ul> | ||||||
|  |  * | ||||||
|  |  * <p><strong>注意:</strong> 表达式和路径必须都为绝对路径或都为相对路径。 | ||||||
|  |  * | ||||||
|  |  * @author Alef Arendsen, Juergen Hoeller, Rob Harrop, Arjen Poutsma, Rossen Stoyanchev, Sam Brannen, Vladislav Kisel | ||||||
|  |  * @since 5.7.22 | ||||||
|  |  */ | ||||||
|  | public class AntPathMatcher { | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Default path separator: "/". | ||||||
|  | 	 */ | ||||||
|  | 	public static final String DEFAULT_PATH_SEPARATOR = StrUtil.SLASH; | ||||||
|  |  | ||||||
|  | 	private static final int CACHE_TURNOFF_THRESHOLD = 65536; | ||||||
|  |  | ||||||
|  | 	private static final Pattern VARIABLE_PATTERN = Pattern.compile("\\{[^/]+?}"); | ||||||
|  |  | ||||||
|  | 	private static final char[] WILDCARD_CHARS = {'*', '?', '{'}; | ||||||
|  |  | ||||||
|  | 	private String pathSeparator; | ||||||
|  |  | ||||||
|  | 	private PathSeparatorPatternCache pathSeparatorPatternCache; | ||||||
|  |  | ||||||
|  | 	private boolean caseSensitive = true; | ||||||
|  |  | ||||||
|  | 	private boolean trimTokens = false; | ||||||
|  |  | ||||||
|  | 	private volatile Boolean cachePatterns; | ||||||
|  |  | ||||||
|  | 	private final Map<String, String[]> tokenizedPatternCache = new ConcurrentHashMap<>(256); | ||||||
|  |  | ||||||
|  | 	private final Map<String, AntPathStringMatcher> stringMatcherCache = new ConcurrentHashMap<>(256); | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * 使用 {@link #DEFAULT_PATH_SEPARATOR} 作为分隔符构造 | ||||||
|  | 	 */ | ||||||
|  | 	public AntPathMatcher() { | ||||||
|  | 		this(DEFAULT_PATH_SEPARATOR); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * 使用自定义的分隔符构造 | ||||||
|  | 	 * | ||||||
|  | 	 * @param pathSeparator the path separator to use, must not be {@code null}. | ||||||
|  | 	 * @since 4.1 | ||||||
|  | 	 */ | ||||||
|  | 	public AntPathMatcher(String pathSeparator) { | ||||||
|  | 		if (null == pathSeparator) { | ||||||
|  | 			pathSeparator = DEFAULT_PATH_SEPARATOR; | ||||||
|  | 		} | ||||||
|  | 		setPathSeparator(pathSeparator); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * 设置路径分隔符 | ||||||
|  | 	 * | ||||||
|  | 	 * @param pathSeparator 分隔符,{@code null}表示使用默认分隔符{@link #DEFAULT_PATH_SEPARATOR} | ||||||
|  | 	 * @return this | ||||||
|  | 	 */ | ||||||
|  | 	public AntPathMatcher setPathSeparator(String pathSeparator) { | ||||||
|  | 		if (null == pathSeparator) { | ||||||
|  | 			pathSeparator = DEFAULT_PATH_SEPARATOR; | ||||||
|  | 		} | ||||||
|  | 		this.pathSeparator = pathSeparator; | ||||||
|  | 		this.pathSeparatorPatternCache = new PathSeparatorPatternCache(this.pathSeparator); | ||||||
|  | 		return this; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * 设置是否大小写敏感,默认为{@code true} | ||||||
|  | 	 * | ||||||
|  | 	 * @param caseSensitive 是否大小写敏感 | ||||||
|  | 	 * @return this | ||||||
|  | 	 */ | ||||||
|  | 	public AntPathMatcher setCaseSensitive(boolean caseSensitive) { | ||||||
|  | 		this.caseSensitive = caseSensitive; | ||||||
|  | 		return this; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * 设置是否去除路径节点两边的空白符,默认为{@code false} | ||||||
|  | 	 * | ||||||
|  | 	 * @param trimTokens 是否去除路径节点两边的空白符 | ||||||
|  | 	 * @return this | ||||||
|  | 	 */ | ||||||
|  | 	public AntPathMatcher setTrimTokens(boolean trimTokens) { | ||||||
|  | 		this.trimTokens = trimTokens; | ||||||
|  | 		return this; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Specify whether to cache parsed pattern metadata for patterns passed | ||||||
|  | 	 * into this matcher's {@link #match} method. A value of {@code true} | ||||||
|  | 	 * activates an unlimited pattern cache; a value of {@code false} turns | ||||||
|  | 	 * the pattern cache off completely. | ||||||
|  | 	 * <p>Default is for the cache to be on, but with the variant to automatically | ||||||
|  | 	 * turn it off when encountering too many patterns to cache at runtime | ||||||
|  | 	 * (the threshold is 65536), assuming that arbitrary permutations of patterns | ||||||
|  | 	 * are coming in, with little chance for encountering a recurring pattern. | ||||||
|  | 	 * | ||||||
|  | 	 * @param cachePatterns 是否缓存表达式 | ||||||
|  | 	 * @see #getStringMatcher(String) | ||||||
|  | 	 */ | ||||||
|  | 	public AntPathMatcher setCachePatterns(boolean cachePatterns) { | ||||||
|  | 		this.cachePatterns = cachePatterns; | ||||||
|  | 		return this; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * 判断给定路径是否是表达式 | ||||||
|  | 	 * | ||||||
|  | 	 * @param path 路径 | ||||||
|  | 	 * @return 是否为表达式 | ||||||
|  | 	 */ | ||||||
|  | 	public boolean isPattern(String path) { | ||||||
|  | 		if (path == null) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 		boolean uriVar = false; | ||||||
|  | 		final int length = path.length(); | ||||||
|  | 		char c; | ||||||
|  | 		for (int i = 0; i < length; i++) { | ||||||
|  | 			c = path.charAt(i); | ||||||
|  | 			// 含有通配符 | ||||||
|  | 			if (c == '*' || c == '?') { | ||||||
|  | 				return true; | ||||||
|  | 			} | ||||||
|  | 			if (c == CharPool.DELIM_START) { | ||||||
|  | 				uriVar = true; | ||||||
|  | 				continue; | ||||||
|  | 			} | ||||||
|  | 			if (c == CharPool.DELIM_END && uriVar) { | ||||||
|  | 				return true; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * 给定路径是否匹配表达式 | ||||||
|  | 	 * | ||||||
|  | 	 * @param pattern 表达式 | ||||||
|  | 	 * @param path 路径 | ||||||
|  | 	 * @return 是否匹配 | ||||||
|  | 	 */ | ||||||
|  | 	public boolean match(String pattern, String path) { | ||||||
|  | 		return doMatch(pattern, path, true, null); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * 前置部分匹配 | ||||||
|  | 	 * | ||||||
|  | 	 * @param pattern 表达式 | ||||||
|  | 	 * @param path 路径 | ||||||
|  | 	 * @return 是否匹配 | ||||||
|  | 	 */ | ||||||
|  | 	public boolean matchStart(String pattern, String path) { | ||||||
|  | 		return doMatch(pattern, path, false, null); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * 执行匹配,判断给定的{@code path}是否匹配{@code pattern} | ||||||
|  | 	 * | ||||||
|  | 	 * @param pattern              表达式 | ||||||
|  | 	 * @param path                 路径 | ||||||
|  | 	 * @param fullMatch            是否全匹配。{@code true} 表示全路径匹配,{@code false}表示只匹配开始 | ||||||
|  | 	 * @param uriTemplateVariables 变量映射 | ||||||
|  | 	 * @return {@code true} 表示提供的 {@code path} 匹配, {@code false} 表示不匹配 | ||||||
|  | 	 */ | ||||||
|  | 	protected boolean doMatch(String pattern, String path, boolean fullMatch, Map<String, String> uriTemplateVariables) { | ||||||
|  | 		if (path == null || path.startsWith(this.pathSeparator) != pattern.startsWith(this.pathSeparator)) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		final String[] pattDirs = tokenizePattern(pattern); | ||||||
|  | 		if (fullMatch && this.caseSensitive && false == isPotentialMatch(path, pattDirs)) { | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		final String[] pathDirs = tokenizePath(path); | ||||||
|  | 		int pattIdxStart = 0; | ||||||
|  | 		int pattIdxEnd = pattDirs.length - 1; | ||||||
|  | 		int pathIdxStart = 0; | ||||||
|  | 		int pathIdxEnd = pathDirs.length - 1; | ||||||
|  |  | ||||||
|  | 		// Match all elements up to the first ** | ||||||
|  | 		while (pattIdxStart <= pattIdxEnd && pathIdxStart <= pathIdxEnd) { | ||||||
|  | 			String pattDir = pattDirs[pattIdxStart]; | ||||||
|  | 			if ("**".equals(pattDir)) { | ||||||
|  | 				break; | ||||||
|  | 			} | ||||||
|  | 			if (notMatchStrings(pattDir, pathDirs[pathIdxStart], uriTemplateVariables)) { | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 			pattIdxStart++; | ||||||
|  | 			pathIdxStart++; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if (pathIdxStart > pathIdxEnd) { | ||||||
|  | 			// Path is exhausted, only match if rest of pattern is * or **'s | ||||||
|  | 			if (pattIdxStart > pattIdxEnd) { | ||||||
|  | 				return (pattern.endsWith(this.pathSeparator) == path.endsWith(this.pathSeparator)); | ||||||
|  | 			} | ||||||
|  | 			if (false == fullMatch) { | ||||||
|  | 				return true; | ||||||
|  | 			} | ||||||
|  | 			if (pattIdxStart == pattIdxEnd && pattDirs[pattIdxStart].equals("*") && path.endsWith(this.pathSeparator)) { | ||||||
|  | 				return true; | ||||||
|  | 			} | ||||||
|  | 			for (int i = pattIdxStart; i <= pattIdxEnd; i++) { | ||||||
|  | 				if (false == pattDirs[i].equals("**")) { | ||||||
|  | 					return false; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			return true; | ||||||
|  | 		} else if (pattIdxStart > pattIdxEnd) { | ||||||
|  | 			// String not exhausted, but pattern is. Failure. | ||||||
|  | 			return false; | ||||||
|  | 		} else if (false == fullMatch && "**".equals(pattDirs[pattIdxStart])) { | ||||||
|  | 			// Path start definitely matches due to "**" part in pattern. | ||||||
|  | 			return true; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// up to last '**' | ||||||
|  | 		while (pattIdxStart <= pattIdxEnd && pathIdxStart <= pathIdxEnd) { | ||||||
|  | 			String pattDir = pattDirs[pattIdxEnd]; | ||||||
|  | 			if (pattDir.equals("**")) { | ||||||
|  | 				break; | ||||||
|  | 			} | ||||||
|  | 			if (notMatchStrings(pattDir, pathDirs[pathIdxEnd], uriTemplateVariables)) { | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 			pattIdxEnd--; | ||||||
|  | 			pathIdxEnd--; | ||||||
|  | 		} | ||||||
|  | 		if (pathIdxStart > pathIdxEnd) { | ||||||
|  | 			// String is exhausted | ||||||
|  | 			for (int i = pattIdxStart; i <= pattIdxEnd; i++) { | ||||||
|  | 				if (false == pattDirs[i].equals("**")) { | ||||||
|  | 					return false; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			return true; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		while (pattIdxStart != pattIdxEnd && pathIdxStart <= pathIdxEnd) { | ||||||
|  | 			int patIdxTmp = -1; | ||||||
|  | 			for (int i = pattIdxStart + 1; i <= pattIdxEnd; i++) { | ||||||
|  | 				if (pattDirs[i].equals("**")) { | ||||||
|  | 					patIdxTmp = i; | ||||||
|  | 					break; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			if (patIdxTmp == pattIdxStart + 1) { | ||||||
|  | 				// '**/**' situation, so skip one | ||||||
|  | 				pattIdxStart++; | ||||||
|  | 				continue; | ||||||
|  | 			} | ||||||
|  | 			// Find the pattern between padIdxStart & padIdxTmp in str between | ||||||
|  | 			// strIdxStart & strIdxEnd | ||||||
|  | 			int patLength = (patIdxTmp - pattIdxStart - 1); | ||||||
|  | 			int strLength = (pathIdxEnd - pathIdxStart + 1); | ||||||
|  | 			int foundIdx = -1; | ||||||
|  |  | ||||||
|  | 			strLoop: | ||||||
|  | 			for (int i = 0; i <= strLength - patLength; i++) { | ||||||
|  | 				for (int j = 0; j < patLength; j++) { | ||||||
|  | 					String subPat = pattDirs[pattIdxStart + j + 1]; | ||||||
|  | 					String subStr = pathDirs[pathIdxStart + i + j]; | ||||||
|  | 					if (notMatchStrings(subPat, subStr, uriTemplateVariables)) { | ||||||
|  | 						continue strLoop; | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 				foundIdx = pathIdxStart + i; | ||||||
|  | 				break; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if (foundIdx == -1) { | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			pattIdxStart = patIdxTmp; | ||||||
|  | 			pathIdxStart = foundIdx + patLength; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		for (int i = pattIdxStart; i <= pattIdxEnd; i++) { | ||||||
|  | 			if (false == pattDirs[i].equals("**")) { | ||||||
|  | 				return false; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private boolean isPotentialMatch(String path, String[] pattDirs) { | ||||||
|  | 		if (!this.trimTokens) { | ||||||
|  | 			int pos = 0; | ||||||
|  | 			for (String pattDir : pattDirs) { | ||||||
|  | 				int skipped = skipSeparator(path, pos, this.pathSeparator); | ||||||
|  | 				pos += skipped; | ||||||
|  | 				skipped = skipSegment(path, pos, pattDir); | ||||||
|  | 				if (skipped < pattDir.length()) { | ||||||
|  | 					return (skipped > 0 || (pattDir.length() > 0 && isWildcardChar(pattDir.charAt(0)))); | ||||||
|  | 				} | ||||||
|  | 				pos += skipped; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return true; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private int skipSegment(String path, int pos, String prefix) { | ||||||
|  | 		int skipped = 0; | ||||||
|  | 		for (int i = 0; i < prefix.length(); i++) { | ||||||
|  | 			char c = prefix.charAt(i); | ||||||
|  | 			if (isWildcardChar(c)) { | ||||||
|  | 				return skipped; | ||||||
|  | 			} | ||||||
|  | 			int currPos = pos + skipped; | ||||||
|  | 			if (currPos >= path.length()) { | ||||||
|  | 				return 0; | ||||||
|  | 			} | ||||||
|  | 			if (c == path.charAt(currPos)) { | ||||||
|  | 				skipped++; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return skipped; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private int skipSeparator(String path, int pos, String separator) { | ||||||
|  | 		int skipped = 0; | ||||||
|  | 		while (path.startsWith(separator, pos + skipped)) { | ||||||
|  | 			skipped += separator.length(); | ||||||
|  | 		} | ||||||
|  | 		return skipped; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private boolean isWildcardChar(char c) { | ||||||
|  | 		for (char candidate : WILDCARD_CHARS) { | ||||||
|  | 			if (c == candidate) { | ||||||
|  | 				return true; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Tokenize the given path pattern into parts, based on this matcher's settings. | ||||||
|  | 	 * <p>Performs caching based on {@link #setCachePatterns}, delegating to | ||||||
|  | 	 * {@link #tokenizePath(String)} for the actual tokenization algorithm. | ||||||
|  | 	 * | ||||||
|  | 	 * @param pattern the pattern to tokenize | ||||||
|  | 	 * @return the tokenized pattern parts | ||||||
|  | 	 */ | ||||||
|  | 	protected String[] tokenizePattern(String pattern) { | ||||||
|  | 		String[] tokenized = null; | ||||||
|  | 		Boolean cachePatterns = this.cachePatterns; | ||||||
|  | 		if (cachePatterns == null || cachePatterns) { | ||||||
|  | 			tokenized = this.tokenizedPatternCache.get(pattern); | ||||||
|  | 		} | ||||||
|  | 		if (tokenized == null) { | ||||||
|  | 			tokenized = tokenizePath(pattern); | ||||||
|  | 			if (cachePatterns == null && this.tokenizedPatternCache.size() >= CACHE_TURNOFF_THRESHOLD) { | ||||||
|  | 				// Try to adapt to the runtime situation that we're encountering: | ||||||
|  | 				// There are obviously too many different patterns coming in here... | ||||||
|  | 				// So let's turn off the cache since the patterns are unlikely to be reoccurring. | ||||||
|  | 				deactivatePatternCache(); | ||||||
|  | 				return tokenized; | ||||||
|  | 			} | ||||||
|  | 			if (cachePatterns == null || cachePatterns) { | ||||||
|  | 				this.tokenizedPatternCache.put(pattern, tokenized); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return tokenized; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private void deactivatePatternCache() { | ||||||
|  | 		this.cachePatterns = false; | ||||||
|  | 		this.tokenizedPatternCache.clear(); | ||||||
|  | 		this.stringMatcherCache.clear(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Tokenize the given path into parts, based on this matcher's settings. | ||||||
|  | 	 * | ||||||
|  | 	 * @param path the path to tokenize | ||||||
|  | 	 * @return the tokenized path parts | ||||||
|  | 	 */ | ||||||
|  | 	protected String[] tokenizePath(String path) { | ||||||
|  | 		return StrSplitter.splitToArray(path, this.pathSeparator, 0, this.trimTokens, true); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Test whether or not a string matches against a pattern. | ||||||
|  | 	 * | ||||||
|  | 	 * @param pattern the pattern to match against (never {@code null}) | ||||||
|  | 	 * @param str     the String which must be matched against the pattern (never {@code null}) | ||||||
|  | 	 * @return {@code true} if the string matches against the pattern, or {@code false} otherwise | ||||||
|  | 	 */ | ||||||
|  | 	private boolean notMatchStrings(String pattern, String str, Map<String, String> uriTemplateVariables) { | ||||||
|  | 		return false == getStringMatcher(pattern).matchStrings(str, uriTemplateVariables); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Build or retrieve an {@link AntPathStringMatcher} for the given pattern. | ||||||
|  | 	 * <p>The default implementation checks this AntPathMatcher's internal cache | ||||||
|  | 	 * (see {@link #setCachePatterns}), creating a new AntPathStringMatcher instance | ||||||
|  | 	 * if no cached copy is found. | ||||||
|  | 	 * <p>When encountering too many patterns to cache at runtime (the threshold is 65536), | ||||||
|  | 	 * it turns the default cache off, assuming that arbitrary permutations of patterns | ||||||
|  | 	 * are coming in, with little chance for encountering a recurring pattern. | ||||||
|  | 	 * <p>This method may be overridden to implement a custom cache strategy. | ||||||
|  | 	 * | ||||||
|  | 	 * @param pattern the pattern to match against (never {@code null}) | ||||||
|  | 	 * @return a corresponding AntPathStringMatcher (never {@code null}) | ||||||
|  | 	 * @see #setCachePatterns | ||||||
|  | 	 */ | ||||||
|  | 	protected AntPathStringMatcher getStringMatcher(String pattern) { | ||||||
|  | 		AntPathStringMatcher matcher = null; | ||||||
|  | 		Boolean cachePatterns = this.cachePatterns; | ||||||
|  | 		if (cachePatterns == null || cachePatterns) { | ||||||
|  | 			matcher = this.stringMatcherCache.get(pattern); | ||||||
|  | 		} | ||||||
|  | 		if (matcher == null) { | ||||||
|  | 			matcher = new AntPathStringMatcher(pattern, this.caseSensitive); | ||||||
|  | 			if (cachePatterns == null && this.stringMatcherCache.size() >= CACHE_TURNOFF_THRESHOLD) { | ||||||
|  | 				// Try to adapt to the runtime situation that we're encountering: | ||||||
|  | 				// There are obviously too many different patterns coming in here... | ||||||
|  | 				// So let's turn off the cache since the patterns are unlikely to be reoccurring. | ||||||
|  | 				deactivatePatternCache(); | ||||||
|  | 				return matcher; | ||||||
|  | 			} | ||||||
|  | 			if (cachePatterns == null || cachePatterns) { | ||||||
|  | 				this.stringMatcherCache.put(pattern, matcher); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return matcher; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Given a pattern and a full path, determine the pattern-mapped part. <p>For example: <ul> | ||||||
|  | 	 * <li>'{@code /docs/cvs/commit.html}' and '{@code /docs/cvs/commit.html} → ''</li> | ||||||
|  | 	 * <li>'{@code /docs/*}' and '{@code /docs/cvs/commit} → '{@code cvs/commit}'</li> | ||||||
|  | 	 * <li>'{@code /docs/cvs/*.html}' and '{@code /docs/cvs/commit.html} → '{@code commit.html}'</li> | ||||||
|  | 	 * <li>'{@code /docs/**}' and '{@code /docs/cvs/commit} → '{@code cvs/commit}'</li> | ||||||
|  | 	 * <li>'{@code /docs/**\/*.html}' and '{@code /docs/cvs/commit.html} → '{@code cvs/commit.html}'</li> | ||||||
|  | 	 * <li>'{@code /*.html}' and '{@code /docs/cvs/commit.html} → '{@code docs/cvs/commit.html}'</li> | ||||||
|  | 	 * <li>'{@code *.html}' and '{@code /docs/cvs/commit.html} → '{@code /docs/cvs/commit.html}'</li> | ||||||
|  | 	 * <li>'{@code *}' and '{@code /docs/cvs/commit.html} → '{@code /docs/cvs/commit.html}'</li> </ul> | ||||||
|  | 	 * <p>Assumes that {@link #match} returns {@code true} for '{@code pattern}' and '{@code path}', but | ||||||
|  | 	 * does <strong>not</strong> enforce this. | ||||||
|  | 	 * | ||||||
|  | 	 * @param pattern 表达式 | ||||||
|  | 	 * @param path    路径 | ||||||
|  | 	 * @return 表达式匹配到的部分 | ||||||
|  | 	 */ | ||||||
|  | 	public String extractPathWithinPattern(String pattern, String path) { | ||||||
|  | 		String[] patternParts = tokenizePath(pattern); | ||||||
|  | 		String[] pathParts = tokenizePath(path); | ||||||
|  | 		StringBuilder builder = new StringBuilder(); | ||||||
|  | 		boolean pathStarted = false; | ||||||
|  |  | ||||||
|  | 		for (int segment = 0; segment < patternParts.length; segment++) { | ||||||
|  | 			String patternPart = patternParts[segment]; | ||||||
|  | 			if (patternPart.indexOf('*') > -1 || patternPart.indexOf('?') > -1) { | ||||||
|  | 				for (; segment < pathParts.length; segment++) { | ||||||
|  | 					if (pathStarted || (segment == 0 && !pattern.startsWith(this.pathSeparator))) { | ||||||
|  | 						builder.append(this.pathSeparator); | ||||||
|  | 					} | ||||||
|  | 					builder.append(pathParts[segment]); | ||||||
|  | 					pathStarted = true; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return builder.toString(); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	public Map<String, String> extractUriTemplateVariables(String pattern, String path) { | ||||||
|  | 		Map<String, String> variables = new LinkedHashMap<>(); | ||||||
|  | 		boolean result = doMatch(pattern, path, true, variables); | ||||||
|  | 		if (!result) { | ||||||
|  | 			throw new IllegalStateException("Pattern \"" + pattern + "\" is not a match for \"" + path + "\""); | ||||||
|  | 		} | ||||||
|  | 		return variables; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Combine two patterns into a new pattern. | ||||||
|  | 	 * <p>This implementation simply concatenates the two patterns, unless | ||||||
|  | 	 * the first pattern contains a file extension match (e.g., {@code *.html}). | ||||||
|  | 	 * In that case, the second pattern will be merged into the first. Otherwise, | ||||||
|  | 	 * an {@code IllegalArgumentException} will be thrown. | ||||||
|  | 	 * <p>Examples</p> | ||||||
|  | 	 * <table border="1" summary=""> | ||||||
|  | 	 * <tr><th>Pattern 1</th><th>Pattern 2</th><th>Result</th></tr> | ||||||
|  | 	 * <tr><td>{@code null}</td><td>{@code null}</td><td> </td></tr> | ||||||
|  | 	 * <tr><td>/hotels</td><td>{@code null}</td><td>/hotels</td></tr> | ||||||
|  | 	 * <tr><td>{@code null}</td><td>/hotels</td><td>/hotels</td></tr> | ||||||
|  | 	 * <tr><td>/hotels</td><td>/bookings</td><td>/hotels/bookings</td></tr> | ||||||
|  | 	 * <tr><td>/hotels</td><td>bookings</td><td>/hotels/bookings</td></tr> | ||||||
|  | 	 * <tr><td>/hotels/*</td><td>/bookings</td><td>/hotels/bookings</td></tr> | ||||||
|  | 	 * <tr><td>/hotels/**</td><td>/bookings</td><td>/hotels/**/bookings</td></tr> | ||||||
|  | 	 * <tr><td>/hotels</td><td>{hotel}</td><td>/hotels/{hotel}</td></tr> | ||||||
|  | 	 * <tr><td>/hotels/*</td><td>{hotel}</td><td>/hotels/{hotel}</td></tr> | ||||||
|  | 	 * <tr><td>/hotels/**</td><td>{hotel}</td><td>/hotels/**/{hotel}</td></tr> | ||||||
|  | 	 * <tr><td>/*.html</td><td>/hotels.html</td><td>/hotels.html</td></tr> | ||||||
|  | 	 * <tr><td>/*.html</td><td>/hotels</td><td>/hotels.html</td></tr> | ||||||
|  | 	 * <tr><td>/*.html</td><td>/*.txt</td><td>{@code IllegalArgumentException}</td></tr> | ||||||
|  | 	 * </table> | ||||||
|  | 	 * | ||||||
|  | 	 * @param pattern1 the first pattern | ||||||
|  | 	 * @param pattern2 the second pattern | ||||||
|  | 	 * @return the combination of the two patterns | ||||||
|  | 	 * @throws IllegalArgumentException if the two patterns cannot be combined | ||||||
|  | 	 */ | ||||||
|  | 	public String combine(String pattern1, String pattern2) { | ||||||
|  | 		if (StrUtil.isEmpty(pattern1) && StrUtil.isEmpty(pattern2)) { | ||||||
|  | 			return StrUtil.EMPTY; | ||||||
|  | 		} | ||||||
|  | 		if (StrUtil.isEmpty(pattern1)) { | ||||||
|  | 			return pattern2; | ||||||
|  | 		} | ||||||
|  | 		if (StrUtil.isEmpty(pattern2)) { | ||||||
|  | 			return pattern1; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		boolean pattern1ContainsUriVar = (pattern1.indexOf('{') != -1); | ||||||
|  | 		if (!pattern1.equals(pattern2) && !pattern1ContainsUriVar && match(pattern1, pattern2)) { | ||||||
|  | 			// /* + /hotel -> /hotel ; "/*.*" + "/*.html" -> /*.html | ||||||
|  | 			// However /user + /user -> /usr/user ; /{foo} + /bar -> /{foo}/bar | ||||||
|  | 			return pattern2; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// /hotels/* + /booking -> /hotels/booking | ||||||
|  | 		// /hotels/* + booking -> /hotels/booking | ||||||
|  | 		if (pattern1.endsWith(this.pathSeparatorPatternCache.getEndsOnWildCard())) { | ||||||
|  | 			return concat(pattern1.substring(0, pattern1.length() - 2), pattern2); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// /hotels/** + /booking -> /hotels/**/booking | ||||||
|  | 		// /hotels/** + booking -> /hotels/**/booking | ||||||
|  | 		if (pattern1.endsWith(this.pathSeparatorPatternCache.getEndsOnDoubleWildCard())) { | ||||||
|  | 			return concat(pattern1, pattern2); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		int starDotPos1 = pattern1.indexOf("*."); | ||||||
|  | 		if (pattern1ContainsUriVar || starDotPos1 == -1 || this.pathSeparator.equals(".")) { | ||||||
|  | 			// simply concatenate the two patterns | ||||||
|  | 			return concat(pattern1, pattern2); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		String ext1 = pattern1.substring(starDotPos1 + 1); | ||||||
|  | 		int dotPos2 = pattern2.indexOf('.'); | ||||||
|  | 		String file2 = (dotPos2 == -1 ? pattern2 : pattern2.substring(0, dotPos2)); | ||||||
|  | 		String ext2 = (dotPos2 == -1 ? "" : pattern2.substring(dotPos2)); | ||||||
|  | 		boolean ext1All = (ext1.equals(".*") || ext1.isEmpty()); | ||||||
|  | 		boolean ext2All = (ext2.equals(".*") || ext2.isEmpty()); | ||||||
|  | 		if (!ext1All && !ext2All) { | ||||||
|  | 			throw new IllegalArgumentException("Cannot combine patterns: " + pattern1 + " vs " + pattern2); | ||||||
|  | 		} | ||||||
|  | 		String ext = (ext1All ? ext2 : ext1); | ||||||
|  | 		return file2 + ext; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	private String concat(String path1, String path2) { | ||||||
|  | 		boolean path1EndsWithSeparator = path1.endsWith(this.pathSeparator); | ||||||
|  | 		boolean path2StartsWithSeparator = path2.startsWith(this.pathSeparator); | ||||||
|  |  | ||||||
|  | 		if (path1EndsWithSeparator && path2StartsWithSeparator) { | ||||||
|  | 			return path1 + path2.substring(1); | ||||||
|  | 		} else if (path1EndsWithSeparator || path2StartsWithSeparator) { | ||||||
|  | 			return path1 + path2; | ||||||
|  | 		} else { | ||||||
|  | 			return path1 + this.pathSeparator + path2; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Given a full path, returns a {@link Comparator} suitable for sorting patterns in order of | ||||||
|  | 	 * explicitness. | ||||||
|  | 	 * <p>This {@code Comparator} will {@linkplain List#sort(Comparator) sort} | ||||||
|  | 	 * a list so that more specific patterns (without URI templates or wild cards) come before | ||||||
|  | 	 * generic patterns. So given a list with the following patterns, the returned comparator | ||||||
|  | 	 * will sort this list so that the order will be as indicated. | ||||||
|  | 	 * <ol> | ||||||
|  | 	 * <li>{@code /hotels/new}</li> | ||||||
|  | 	 * <li>{@code /hotels/{hotel}}</li> | ||||||
|  | 	 * <li>{@code /hotels/*}</li> | ||||||
|  | 	 * </ol> | ||||||
|  | 	 * <p>The full path given as parameter is used to test for exact matches. So when the given path | ||||||
|  | 	 * is {@code /hotels/2}, the pattern {@code /hotels/2} will be sorted before {@code /hotels/1}. | ||||||
|  | 	 * | ||||||
|  | 	 * @param path the full path to use for comparison | ||||||
|  | 	 * @return a comparator capable of sorting patterns in order of explicitness | ||||||
|  | 	 */ | ||||||
|  | 	public Comparator<String> getPatternComparator(String path) { | ||||||
|  | 		return new AntPatternComparator(path); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * Tests whether or not a string matches against a pattern via a {@link Pattern}. | ||||||
|  | 	 * <p>The pattern may contain special characters: '*' means zero or more characters; '?' means one and | ||||||
|  | 	 * only one character; '{' and '}' indicate a URI template pattern. For example <tt>/users/{user}</tt>. | ||||||
|  | 	 */ | ||||||
|  | 	protected static class AntPathStringMatcher { | ||||||
|  |  | ||||||
|  | 		private static final Pattern GLOB_PATTERN = Pattern.compile("\\?|\\*|\\{((?:\\{[^/]+?}|[^/{}]|\\\\[{}])+?)}"); | ||||||
|  |  | ||||||
|  | 		private static final String DEFAULT_VARIABLE_PATTERN = "((?s).*)"; | ||||||
|  |  | ||||||
|  | 		private final String rawPattern; | ||||||
|  |  | ||||||
|  | 		private final boolean caseSensitive; | ||||||
|  |  | ||||||
|  | 		private final boolean exactMatch; | ||||||
|  |  | ||||||
|  | 		private final Pattern pattern; | ||||||
|  |  | ||||||
|  | 		private final List<String> variableNames = new ArrayList<>(); | ||||||
|  |  | ||||||
|  | 		public AntPathStringMatcher(String pattern, boolean caseSensitive) { | ||||||
|  | 			this.rawPattern = pattern; | ||||||
|  | 			this.caseSensitive = caseSensitive; | ||||||
|  | 			StringBuilder patternBuilder = new StringBuilder(); | ||||||
|  | 			Matcher matcher = GLOB_PATTERN.matcher(pattern); | ||||||
|  | 			int end = 0; | ||||||
|  | 			while (matcher.find()) { | ||||||
|  | 				patternBuilder.append(quote(pattern, end, matcher.start())); | ||||||
|  | 				String match = matcher.group(); | ||||||
|  | 				if ("?".equals(match)) { | ||||||
|  | 					patternBuilder.append('.'); | ||||||
|  | 				} else if ("*".equals(match)) { | ||||||
|  | 					patternBuilder.append(".*"); | ||||||
|  | 				} else if (match.startsWith("{") && match.endsWith("}")) { | ||||||
|  | 					int colonIdx = match.indexOf(':'); | ||||||
|  | 					if (colonIdx == -1) { | ||||||
|  | 						patternBuilder.append(DEFAULT_VARIABLE_PATTERN); | ||||||
|  | 						this.variableNames.add(matcher.group(1)); | ||||||
|  | 					} else { | ||||||
|  | 						String variablePattern = match.substring(colonIdx + 1, match.length() - 1); | ||||||
|  | 						patternBuilder.append('('); | ||||||
|  | 						patternBuilder.append(variablePattern); | ||||||
|  | 						patternBuilder.append(')'); | ||||||
|  | 						String variableName = match.substring(1, colonIdx); | ||||||
|  | 						this.variableNames.add(variableName); | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 				end = matcher.end(); | ||||||
|  | 			} | ||||||
|  | 			// No glob pattern was found, this is an exact String match | ||||||
|  | 			if (end == 0) { | ||||||
|  | 				this.exactMatch = true; | ||||||
|  | 				this.pattern = null; | ||||||
|  | 			} else { | ||||||
|  | 				this.exactMatch = false; | ||||||
|  | 				patternBuilder.append(quote(pattern, end, pattern.length())); | ||||||
|  | 				this.pattern = (this.caseSensitive ? Pattern.compile(patternBuilder.toString()) : | ||||||
|  | 						Pattern.compile(patternBuilder.toString(), Pattern.CASE_INSENSITIVE)); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		private String quote(String s, int start, int end) { | ||||||
|  | 			if (start == end) { | ||||||
|  | 				return ""; | ||||||
|  | 			} | ||||||
|  | 			return Pattern.quote(s.substring(start, end)); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		/** | ||||||
|  | 		 * Main entry point. | ||||||
|  | 		 * | ||||||
|  | 		 * @return {@code true} if the string matches against the pattern, or {@code false} otherwise. | ||||||
|  | 		 */ | ||||||
|  | 		public boolean matchStrings(String str, Map<String, String> uriTemplateVariables) { | ||||||
|  | 			if (this.exactMatch) { | ||||||
|  | 				return this.caseSensitive ? this.rawPattern.equals(str) : this.rawPattern.equalsIgnoreCase(str); | ||||||
|  | 			} else if (this.pattern != null) { | ||||||
|  | 				Matcher matcher = this.pattern.matcher(str); | ||||||
|  | 				if (matcher.matches()) { | ||||||
|  | 					if (uriTemplateVariables != null) { | ||||||
|  | 						if (this.variableNames.size() != matcher.groupCount()) { | ||||||
|  | 							throw new IllegalArgumentException("The number of capturing groups in the pattern segment " + | ||||||
|  | 									this.pattern + " does not match the number of URI template variables it defines, " + | ||||||
|  | 									"which can occur if capturing groups are used in a URI template regex. " + | ||||||
|  | 									"Use non-capturing groups instead."); | ||||||
|  | 						} | ||||||
|  | 						for (int i = 1; i <= matcher.groupCount(); i++) { | ||||||
|  | 							String name = this.variableNames.get(i - 1); | ||||||
|  | 							if (name.startsWith("*")) { | ||||||
|  | 								throw new IllegalArgumentException("Capturing patterns (" + name + ") are not " + | ||||||
|  | 										"supported by the AntPathMatcher. Use the PathPatternParser instead."); | ||||||
|  | 							} | ||||||
|  | 							String value = matcher.group(i); | ||||||
|  | 							uriTemplateVariables.put(name, value); | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 					return true; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * The default {@link Comparator} implementation returned by | ||||||
|  | 	 * {@link #getPatternComparator(String)}. | ||||||
|  | 	 * <p>In order, the most "generic" pattern is determined by the following: | ||||||
|  | 	 * <ul> | ||||||
|  | 	 * <li>if it's null or a capture all pattern (i.e. it is equal to "/**")</li> | ||||||
|  | 	 * <li>if the other pattern is an actual match</li> | ||||||
|  | 	 * <li>if it's a catch-all pattern (i.e. it ends with "**"</li> | ||||||
|  | 	 * <li>if it's got more "*" than the other pattern</li> | ||||||
|  | 	 * <li>if it's got more "{foo}" than the other pattern</li> | ||||||
|  | 	 * <li>if it's shorter than the other pattern</li> | ||||||
|  | 	 * </ul> | ||||||
|  | 	 */ | ||||||
|  | 	protected static class AntPatternComparator implements Comparator<String> { | ||||||
|  |  | ||||||
|  | 		private final String path; | ||||||
|  |  | ||||||
|  | 		public AntPatternComparator(String path) { | ||||||
|  | 			this.path = path; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		/** | ||||||
|  | 		 * Compare two patterns to determine which should match first, i.e. which | ||||||
|  | 		 * is the most specific regarding the current path. | ||||||
|  | 		 * | ||||||
|  | 		 * @param pattern1 表达式1 | ||||||
|  | 		 * @param pattern2 表达式2 | ||||||
|  | 		 * @return a negative integer, zero, or a positive integer as pattern1 is | ||||||
|  | 		 * more specific, equally specific, or less specific than pattern2. | ||||||
|  | 		 */ | ||||||
|  | 		@Override | ||||||
|  | 		public int compare(String pattern1, String pattern2) { | ||||||
|  | 			PatternInfo info1 = new PatternInfo(pattern1); | ||||||
|  | 			PatternInfo info2 = new PatternInfo(pattern2); | ||||||
|  |  | ||||||
|  | 			if (info1.isLeastSpecific() && info2.isLeastSpecific()) { | ||||||
|  | 				return 0; | ||||||
|  | 			} else if (info1.isLeastSpecific()) { | ||||||
|  | 				return 1; | ||||||
|  | 			} else if (info2.isLeastSpecific()) { | ||||||
|  | 				return -1; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			boolean pattern1EqualsPath = pattern1.equals(this.path); | ||||||
|  | 			boolean pattern2EqualsPath = pattern2.equals(this.path); | ||||||
|  | 			if (pattern1EqualsPath && pattern2EqualsPath) { | ||||||
|  | 				return 0; | ||||||
|  | 			} else if (pattern1EqualsPath) { | ||||||
|  | 				return -1; | ||||||
|  | 			} else if (pattern2EqualsPath) { | ||||||
|  | 				return 1; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if (info1.isPrefixPattern() && info2.isPrefixPattern()) { | ||||||
|  | 				return info2.getLength() - info1.getLength(); | ||||||
|  | 			} else if (info1.isPrefixPattern() && info2.getDoubleWildcards() == 0) { | ||||||
|  | 				return 1; | ||||||
|  | 			} else if (info2.isPrefixPattern() && info1.getDoubleWildcards() == 0) { | ||||||
|  | 				return -1; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if (info1.getTotalCount() != info2.getTotalCount()) { | ||||||
|  | 				return info1.getTotalCount() - info2.getTotalCount(); | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if (info1.getLength() != info2.getLength()) { | ||||||
|  | 				return info2.getLength() - info1.getLength(); | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if (info1.getSingleWildcards() < info2.getSingleWildcards()) { | ||||||
|  | 				return -1; | ||||||
|  | 			} else if (info2.getSingleWildcards() < info1.getSingleWildcards()) { | ||||||
|  | 				return 1; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			if (info1.getUriVars() < info2.getUriVars()) { | ||||||
|  | 				return -1; | ||||||
|  | 			} else if (info2.getUriVars() < info1.getUriVars()) { | ||||||
|  | 				return 1; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			return 0; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 		/** | ||||||
|  | 		 * Value class that holds information about the pattern, e.g. number of | ||||||
|  | 		 * occurrences of "*", "**", and "{" pattern elements. | ||||||
|  | 		 */ | ||||||
|  | 		private static class PatternInfo { | ||||||
|  |  | ||||||
|  | 			private final String pattern; | ||||||
|  | 			private int uriVars; | ||||||
|  | 			private int singleWildcards; | ||||||
|  | 			private int doubleWildcards; | ||||||
|  | 			private boolean catchAllPattern; | ||||||
|  | 			private boolean prefixPattern; | ||||||
|  | 			private Integer length; | ||||||
|  |  | ||||||
|  | 			public PatternInfo(String pattern) { | ||||||
|  | 				this.pattern = pattern; | ||||||
|  | 				if (this.pattern != null) { | ||||||
|  | 					initCounters(); | ||||||
|  | 					this.catchAllPattern = this.pattern.equals("/**"); | ||||||
|  | 					this.prefixPattern = !this.catchAllPattern && this.pattern.endsWith("/**"); | ||||||
|  | 				} | ||||||
|  | 				if (this.uriVars == 0) { | ||||||
|  | 					this.length = (this.pattern != null ? this.pattern.length() : 0); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			protected void initCounters() { | ||||||
|  | 				int pos = 0; | ||||||
|  | 				if (this.pattern != null) { | ||||||
|  | 					while (pos < this.pattern.length()) { | ||||||
|  | 						if (this.pattern.charAt(pos) == '{') { | ||||||
|  | 							this.uriVars++; | ||||||
|  | 							pos++; | ||||||
|  | 						} else if (this.pattern.charAt(pos) == '*') { | ||||||
|  | 							if (pos + 1 < this.pattern.length() && this.pattern.charAt(pos + 1) == '*') { | ||||||
|  | 								this.doubleWildcards++; | ||||||
|  | 								pos += 2; | ||||||
|  | 							} else if (pos > 0 && !this.pattern.substring(pos - 1).equals(".*")) { | ||||||
|  | 								this.singleWildcards++; | ||||||
|  | 								pos++; | ||||||
|  | 							} else { | ||||||
|  | 								pos++; | ||||||
|  | 							} | ||||||
|  | 						} else { | ||||||
|  | 							pos++; | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			public int getUriVars() { | ||||||
|  | 				return this.uriVars; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			public int getSingleWildcards() { | ||||||
|  | 				return this.singleWildcards; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			public int getDoubleWildcards() { | ||||||
|  | 				return this.doubleWildcards; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			public boolean isLeastSpecific() { | ||||||
|  | 				return (this.pattern == null || this.catchAllPattern); | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			public boolean isPrefixPattern() { | ||||||
|  | 				return this.prefixPattern; | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			public int getTotalCount() { | ||||||
|  | 				return this.uriVars + this.singleWildcards + (2 * this.doubleWildcards); | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			/** | ||||||
|  | 			 * Returns the length of the given pattern, where template variables are considered to be 1 long. | ||||||
|  | 			 * | ||||||
|  | 			 * @return 长度 | ||||||
|  | 			 */ | ||||||
|  | 			public int getLength() { | ||||||
|  | 				if (this.length == null) { | ||||||
|  | 					this.length = (this.pattern != null ? | ||||||
|  | 							VARIABLE_PATTERN.matcher(this.pattern).replaceAll("#").length() : 0); | ||||||
|  | 				} | ||||||
|  | 				return this.length; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * A simple cache for patterns that depend on the configured path separator. | ||||||
|  | 	 */ | ||||||
|  | 	private static class PathSeparatorPatternCache { | ||||||
|  |  | ||||||
|  | 		private final String endsOnWildCard; | ||||||
|  |  | ||||||
|  | 		private final String endsOnDoubleWildCard; | ||||||
|  |  | ||||||
|  | 		public PathSeparatorPatternCache(String pathSeparator) { | ||||||
|  | 			this.endsOnWildCard = pathSeparator + "*"; | ||||||
|  | 			this.endsOnDoubleWildCard = pathSeparator + "**"; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		public String getEndsOnWildCard() { | ||||||
|  | 			return this.endsOnWildCard; | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		public String getEndsOnDoubleWildCard() { | ||||||
|  | 			return this.endsOnDoubleWildCard; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										115
									
								
								hutool-core/src/test/java/cn/hutool/core/text/AntPathMatcherTest.java
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										115
									
								
								hutool-core/src/test/java/cn/hutool/core/text/AntPathMatcherTest.java
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,115 @@ | |||||||
|  | package cn.hutool.core.text; | ||||||
|  |  | ||||||
|  | import org.junit.Assert; | ||||||
|  | import org.junit.Test; | ||||||
|  |  | ||||||
|  | import java.util.HashMap; | ||||||
|  |  | ||||||
|  | public class AntPathMatcherTest { | ||||||
|  |  | ||||||
|  | 	@Test | ||||||
|  | 	public void matchesTest() { | ||||||
|  | 		AntPathMatcher antPathMatcher = new AntPathMatcher(); | ||||||
|  | 		boolean matched = antPathMatcher.match("/api/org/organization/{orgId}", "/api/org/organization/999"); | ||||||
|  | 		Assert.assertTrue(matched); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@Test | ||||||
|  | 	public void matchesTest2() { | ||||||
|  | 		AntPathMatcher antPathMatcher = new AntPathMatcher(); | ||||||
|  |  | ||||||
|  | 		String pattern = "/**/*.xml*"; | ||||||
|  | 		String path = "/WEB-INF/web.xml"; | ||||||
|  | 		boolean isMatched = antPathMatcher.match(pattern, path); | ||||||
|  | 		Assert.assertTrue(isMatched); | ||||||
|  |  | ||||||
|  | 		pattern = "org/codelabor/*/**/*Service"; | ||||||
|  | 		path = "org/codelabor/example/HelloWorldService"; | ||||||
|  | 		isMatched = antPathMatcher.match(pattern, path); | ||||||
|  | 		Assert.assertTrue(isMatched); | ||||||
|  |  | ||||||
|  | 		pattern = "org/codelabor/*/**/*Service?"; | ||||||
|  | 		path = "org/codelabor/example/HelloWorldServices"; | ||||||
|  | 		isMatched = antPathMatcher.match(pattern, path); | ||||||
|  | 		Assert.assertTrue(isMatched); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@Test | ||||||
|  | 	public void matchesTest3(){ | ||||||
|  | 		AntPathMatcher pathMatcher = new AntPathMatcher(); | ||||||
|  | 		pathMatcher.setCachePatterns(true); | ||||||
|  | 		pathMatcher.setCaseSensitive(true); | ||||||
|  | 		pathMatcher.setPathSeparator("/"); | ||||||
|  | 		pathMatcher.setTrimTokens(true); | ||||||
|  |  | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("a", "a")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("a*", "ab")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("a*/**/a", "ab/asdsa/a")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("a*/**/a", "ab/asdsa/asdasd/a")); | ||||||
|  |  | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("*", "a")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("*/*", "a/a")); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * AntPathMatcher默认路径分隔符为“/”,而在匹配文件路径时,需要注意Windows下路径分隔符为“\”,Linux下为“/”。靠谱写法如下两种方式: | ||||||
|  | 	 * AntPathMatcher matcher = new AntPathMatcher(File.separator); | ||||||
|  | 	 * AntPathMatcher matcher = new AntPathMatcher(System.getProperty("file.separator")); | ||||||
|  | 	 */ | ||||||
|  | 	@Test | ||||||
|  | 	public void matchesTest4() { | ||||||
|  | 		AntPathMatcher pathMatcher = new AntPathMatcher(); | ||||||
|  |  | ||||||
|  | 		// 精确匹配 | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("/test", "/test")); | ||||||
|  | 		Assert.assertFalse(pathMatcher.match("test", "/test")); | ||||||
|  |  | ||||||
|  | 		//测试通配符? | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("t?st", "test")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("te??", "test")); | ||||||
|  | 		Assert.assertFalse(pathMatcher.match("tes?", "tes")); | ||||||
|  | 		Assert.assertFalse(pathMatcher.match("tes?", "testt")); | ||||||
|  |  | ||||||
|  | 		//测试通配符* | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("*", "test")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("test*", "test")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("test/*", "test/Test")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("*.*", "test.")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("*.*", "test.test.test")); | ||||||
|  | 		Assert.assertFalse(pathMatcher.match("test*", "test/")); //注意这里是false 因为路径不能用*匹配 | ||||||
|  | 		Assert.assertFalse(pathMatcher.match("test*", "test/t")); //这同理 | ||||||
|  | 		Assert.assertFalse(pathMatcher.match("test*aaa", "testblaaab")); //这个是false 因为最后一个b无法匹配了 前面都是能匹配成功的 | ||||||
|  |  | ||||||
|  | 		//测试通配符** 匹配多级URL | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("/*/**", "/testing/testing")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("/**/*", "/testing/testing")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("/bla/**/bla", "/bla/testing/testing/bla/bla")); //这里也是true哦 | ||||||
|  | 		Assert.assertFalse(pathMatcher.match("/bla*bla/test", "/blaXXXbl/test")); | ||||||
|  |  | ||||||
|  | 		Assert.assertFalse(pathMatcher.match("/????", "/bala/bla")); | ||||||
|  | 		Assert.assertFalse(pathMatcher.match("/**/*bla", "/bla/bla/bla/bbb")); | ||||||
|  |  | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("/*bla*/**/bla/**", "/XXXblaXXXX/testing/testing/bla/testing/testing/")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("/*bla*/**/bla/*", "/XXXblaXXXX/testing/testing/bla/testing")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("/*bla*/**/bla/**", "/XXXblaXXXX/testing/testing/bla/testing/testing")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("/*bla*/**/bla/**", "/XXXblaXXXX/testing/testing/bla/testing/testing.jpg")); | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("/foo/bar/**", "/foo/bar")); | ||||||
|  |  | ||||||
|  | 		//这个需要特别注意:{}里面的相当于Spring MVC里接受一个参数一样,所以任何东西都会匹配的 | ||||||
|  | 		Assert.assertTrue(pathMatcher.match("/{bla}.*", "/testing.html")); | ||||||
|  | 		Assert.assertFalse(pathMatcher.match("/{bla}.htm", "/testing.html")); //这样就是false了 | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * 测试 URI 模板变量提取 | ||||||
|  | 	 */ | ||||||
|  | 	@Test | ||||||
|  | 	public void testExtractUriTemplateVariables() { | ||||||
|  | 		AntPathMatcher antPathMatcher = new AntPathMatcher(); | ||||||
|  | 		HashMap<String, String> map = (HashMap<String, String>) antPathMatcher.extractUriTemplateVariables("/api/org/organization/{orgId}", | ||||||
|  | 				"/api/org" + | ||||||
|  | 						"/organization" + | ||||||
|  | 						"/999"); | ||||||
|  | 		Assert.assertEquals(1, map.size()); | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @@ -1,8 +1,13 @@ | |||||||
| package cn.hutool.core.util; | package cn.hutool.core.util; | ||||||
|  |  | ||||||
|  | import cn.hutool.core.date.DateUtil; | ||||||
|  | import cn.hutool.core.date.TimeInterval; | ||||||
|  | import cn.hutool.core.lang.Console; | ||||||
| import cn.hutool.core.lang.test.bean.ExamInfoDict; | import cn.hutool.core.lang.test.bean.ExamInfoDict; | ||||||
| import cn.hutool.core.util.ClassUtilTest.TestSubClass; | import cn.hutool.core.util.ClassUtilTest.TestSubClass; | ||||||
|  | import lombok.Data; | ||||||
| import org.junit.Assert; | import org.junit.Assert; | ||||||
|  | import org.junit.Ignore; | ||||||
| import org.junit.Test; | import org.junit.Test; | ||||||
|  |  | ||||||
| import java.lang.reflect.Field; | import java.lang.reflect.Field; | ||||||
| @@ -91,33 +96,77 @@ public class ReflectUtilTest { | |||||||
|  |  | ||||||
| 	@Test | 	@Test | ||||||
| 	public void noneStaticInnerClassTest() { | 	public void noneStaticInnerClassTest() { | ||||||
| 		final TestAClass testAClass = ReflectUtil.newInstanceIfPossible(TestAClass.class); | 		final NoneStaticClass testAClass = ReflectUtil.newInstanceIfPossible(NoneStaticClass.class); | ||||||
| 		Assert.assertNotNull(testAClass); | 		Assert.assertNotNull(testAClass); | ||||||
| 		Assert.assertEquals(2, testAClass.getA()); | 		Assert.assertEquals(2, testAClass.getA()); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	@Data | ||||||
| 	static class TestClass { | 	static class TestClass { | ||||||
| 		private int a; | 		private int a; | ||||||
|  |  | ||||||
| 		public int getA() { |  | ||||||
| 			return a; |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		public void setA(int a) { |  | ||||||
| 			this.a = a; |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	@Data | ||||||
| 	@SuppressWarnings("InnerClassMayBeStatic") | 	@SuppressWarnings("InnerClassMayBeStatic") | ||||||
| 	class TestAClass { | 	class NoneStaticClass { | ||||||
| 		private int a = 2; | 		private int a = 2; | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 		public int getA() { | 	@Test | ||||||
| 			return a; | 	@Ignore | ||||||
|  | 	public void getMethodBenchTest(){ | ||||||
|  | 		// 预热 | ||||||
|  | 		getMethod(TestBenchClass.class, false, "getH"); | ||||||
|  |  | ||||||
|  | 		final TimeInterval timer = DateUtil.timer(); | ||||||
|  | 		timer.start(); | ||||||
|  | 		for (int i = 0; i < 100000000; i++) { | ||||||
|  | 			ReflectUtil.getMethod(TestBenchClass.class, false, "getH"); | ||||||
|  | 		} | ||||||
|  | 		Console.log(timer.interval()); | ||||||
|  |  | ||||||
|  | 		timer.restart(); | ||||||
|  | 		for (int i = 0; i < 100000000; i++) { | ||||||
|  | 			getMethod(TestBenchClass.class, false, "getH"); | ||||||
|  | 		} | ||||||
|  | 		Console.log(timer.interval()); | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@Data | ||||||
|  | 	static class TestBenchClass { | ||||||
|  | 		private int a; | ||||||
|  | 		private String b; | ||||||
|  | 		private String c; | ||||||
|  | 		private String d; | ||||||
|  | 		private String e; | ||||||
|  | 		private String f; | ||||||
|  | 		private String g; | ||||||
|  | 		private String h; | ||||||
|  | 		private String i; | ||||||
|  | 		private String j; | ||||||
|  | 		private String k; | ||||||
|  | 		private String l; | ||||||
|  | 		private String m; | ||||||
|  | 		private String n; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	public static Method getMethod(Class<?> clazz, boolean ignoreCase, String methodName, Class<?>... paramTypes) throws SecurityException { | ||||||
|  | 		if (null == clazz || StrUtil.isBlank(methodName)) { | ||||||
|  | 			return null; | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		public void setA(int a) { | 		Method res = null; | ||||||
| 			this.a = a; | 		final Method[] methods = ReflectUtil.getMethods(clazz); | ||||||
|  | 		if (ArrayUtil.isNotEmpty(methods)) { | ||||||
|  | 			for (Method method : methods) { | ||||||
|  | 				if (StrUtil.equals(methodName, method.getName(), ignoreCase) | ||||||
|  | 						&& ClassUtil.isAllAssignableFrom(method.getParameterTypes(), paramTypes) | ||||||
|  | 						&& (res == null | ||||||
|  | 						|| res.getReturnType().isAssignableFrom(method.getReturnType()))) { | ||||||
|  | 					res = method; | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
|  | 		return res; | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -103,6 +103,14 @@ | |||||||
| 			<version>${mongo.version}</version> | 			<version>${mongo.version}</version> | ||||||
| 			<optional>true</optional> | 			<optional>true</optional> | ||||||
| 		</dependency> | 		</dependency> | ||||||
|  | 		<!-- | ||||||
|  | 		<dependency> | ||||||
|  | 			<groupId>org.mongodb</groupId> | ||||||
|  | 			<artifactId>mongodb-driver-sync</artifactId> | ||||||
|  | 			<version>4.5.0</version> | ||||||
|  | 			<optional>true</optional> | ||||||
|  | 		</dependency> | ||||||
|  | 		--> | ||||||
| 		<!-- Redis Java客户端 --> | 		<!-- Redis Java客户端 --> | ||||||
| 		<dependency> | 		<dependency> | ||||||
| 			<groupId>redis.clients</groupId> | 			<groupId>redis.clients</groupId> | ||||||
| @@ -155,13 +163,13 @@ | |||||||
| 		<dependency> | 		<dependency> | ||||||
| 			<groupId>com.microsoft.sqlserver</groupId> | 			<groupId>com.microsoft.sqlserver</groupId> | ||||||
| 			<artifactId>mssql-jdbc</artifactId> | 			<artifactId>mssql-jdbc</artifactId> | ||||||
| 			<version>9.4.1.jre8</version> | 			<version>10.2.0.jre8</version> | ||||||
| 			<scope>test</scope> | 			<scope>test</scope> | ||||||
| 		</dependency> | 		</dependency> | ||||||
| 		<dependency> | 		<dependency> | ||||||
| 			<groupId>org.slf4j</groupId> | 			<groupId>org.slf4j</groupId> | ||||||
| 			<artifactId>slf4j-simple</artifactId> | 			<artifactId>slf4j-simple</artifactId> | ||||||
| 			<version>1.7.32</version> | 			<version>1.7.36</version> | ||||||
| 			<scope>test</scope> | 			<scope>test</scope> | ||||||
| 		</dependency> | 		</dependency> | ||||||
| 		<dependency> | 		<dependency> | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Looly
					Looly