public class AntPathMatcher extends java.lang.Object implements PathMatcher
PathMatcher
implementation for Ant-style path patterns.
Part of this mapping code has been kindly borrowed from Apache Ant.
The mapping matches URLs using the following rules:
?
matches one character*
matches zero or more characters**
matches zero or more directories in a path{spring:[a-z]+}
matches the regexp [a-z]+
as a path variable named "spring"com/t?st.jsp
— matches com/test.jsp
but also
com/tast.jsp
or com/txst.jsp
com/*.jsp
— matches all .jsp
files in the
com
directorycom/**/test.jsp
— matches all test.jsp
files underneath the com
pathorg/springframework/**/*.jsp
— matches all
.jsp
files underneath the org/springframework
pathorg/**/servlet/bla.jsp
— matches
org/springframework/servlet/bla.jsp
but also
org/springframework/testing/servlet/bla.jsp
and org/servlet/bla.jsp
com/{filename:\\w+}.jsp
will match com/test.jsp
and assign the value test
to the filename
variableNote: a pattern and a path must both be absolute or must both be relative in order for the two to match. Therefore it is recommended that users of this implementation to sanitize patterns in order to prefix them with "/" as it makes sense in the context in which they're used.
Modifier and Type | Class and Description |
---|---|
protected static class |
AntPathMatcher.AntPathStringMatcher
Tests whether or not a string matches against a pattern via a
Pattern . |
protected static class |
AntPathMatcher.AntPatternComparator
The default
Comparator implementation returned by
getPatternComparator(String) . |
private static class |
AntPathMatcher.PathSeparatorPatternCache
A simple cache for patterns that depend on the configured path separator.
|
Modifier and Type | Field and Description |
---|---|
private static int |
CACHE_TURNOFF_THRESHOLD |
private java.lang.Boolean |
cachePatterns |
private boolean |
caseSensitive |
static java.lang.String |
DEFAULT_PATH_SEPARATOR
Default path separator: "/"
|
private java.lang.String |
pathSeparator |
private AntPathMatcher.PathSeparatorPatternCache |
pathSeparatorPatternCache |
(package private) java.util.Map<java.lang.String,AntPathMatcher.AntPathStringMatcher> |
stringMatcherCache |
private java.util.Map<java.lang.String,java.lang.String[]> |
tokenizedPatternCache |
private boolean |
trimTokens |
private static java.util.regex.Pattern |
VARIABLE_PATTERN |
private static char[] |
WILDCARD_CHARS |
Constructor and Description |
---|
AntPathMatcher()
Create a new instance with the
DEFAULT_PATH_SEPARATOR . |
AntPathMatcher(java.lang.String pathSeparator)
A convenient, alternative constructor to use with a custom path separator.
|
Modifier and Type | Method and Description |
---|---|
java.lang.String |
combine(java.lang.String pattern1,
java.lang.String pattern2)
Combine two patterns into a new pattern.
|
private java.lang.String |
concat(java.lang.String path1,
java.lang.String path2) |
private void |
deactivatePatternCache() |
protected boolean |
doMatch(java.lang.String pattern,
java.lang.String path,
boolean fullMatch,
java.util.Map<java.lang.String,java.lang.String> uriTemplateVariables)
Actually match the given
path against the given pattern . |
java.lang.String |
extractPathWithinPattern(java.lang.String pattern,
java.lang.String path)
Given a pattern and a full path, determine the pattern-mapped part.
|
java.util.Map<java.lang.String,java.lang.String> |
extractUriTemplateVariables(java.lang.String pattern,
java.lang.String path)
Given a pattern and a full path, extract the URI template variables.
|
java.util.Comparator<java.lang.String> |
getPatternComparator(java.lang.String path)
Given a full path, returns a
Comparator suitable for sorting patterns in order of
explicitness. |
protected AntPathMatcher.AntPathStringMatcher |
getStringMatcher(java.lang.String pattern)
Build or retrieve an
AntPathMatcher.AntPathStringMatcher for the given pattern. |
boolean |
isPattern(java.lang.String path)
Does the given
path represent a pattern that can be matched
by an implementation of this interface? |
private boolean |
isPotentialMatch(java.lang.String path,
java.lang.String[] pattDirs) |
private boolean |
isWildcardChar(char c) |
boolean |
match(java.lang.String pattern,
java.lang.String path)
Match the given
path against the given pattern ,
according to this PathMatcher's matching strategy. |
boolean |
matchStart(java.lang.String pattern,
java.lang.String path)
Match the given
path against the corresponding part of the given
pattern , according to this PathMatcher's matching strategy. |
private boolean |
matchStrings(java.lang.String pattern,
java.lang.String str,
java.util.Map<java.lang.String,java.lang.String> uriTemplateVariables)
Test whether or not a string matches against a pattern.
|
void |
setCachePatterns(boolean cachePatterns)
Specify whether to cache parsed pattern metadata for patterns passed
into this matcher's
match(java.lang.String, java.lang.String) method. |
void |
setCaseSensitive(boolean caseSensitive)
Specify whether to perform pattern matching in a case-sensitive fashion.
|
void |
setPathSeparator(java.lang.String pathSeparator)
Set the path separator to use for pattern parsing.
|
void |
setTrimTokens(boolean trimTokens)
Specify whether to trim tokenized paths and patterns.
|
private int |
skipSegment(java.lang.String path,
int pos,
java.lang.String prefix) |
private int |
skipSeparator(java.lang.String path,
int pos,
java.lang.String separator) |
protected java.lang.String[] |
tokenizePath(java.lang.String path)
Tokenize the given path String into parts, based on this matcher's settings.
|
protected java.lang.String[] |
tokenizePattern(java.lang.String pattern)
Tokenize the given path pattern into parts, based on this matcher's settings.
|
public static final java.lang.String DEFAULT_PATH_SEPARATOR
private static final int CACHE_TURNOFF_THRESHOLD
private static final java.util.regex.Pattern VARIABLE_PATTERN
private static final char[] WILDCARD_CHARS
private java.lang.String pathSeparator
private AntPathMatcher.PathSeparatorPatternCache pathSeparatorPatternCache
private boolean caseSensitive
private boolean trimTokens
@Nullable private volatile java.lang.Boolean cachePatterns
private final java.util.Map<java.lang.String,java.lang.String[]> tokenizedPatternCache
final java.util.Map<java.lang.String,AntPathMatcher.AntPathStringMatcher> stringMatcherCache
public AntPathMatcher()
DEFAULT_PATH_SEPARATOR
.public AntPathMatcher(java.lang.String pathSeparator)
pathSeparator
- the path separator to use, must not be null
.public void setPathSeparator(@Nullable java.lang.String pathSeparator)
Default is "/", as in Ant.
public void setCaseSensitive(boolean caseSensitive)
Default is true
. Switch this to false
for case-insensitive matching.
public void setTrimTokens(boolean trimTokens)
Default is false
.
public void setCachePatterns(boolean cachePatterns)
match(java.lang.String, java.lang.String)
method. A value of true
activates an unlimited pattern cache; a value of false
turns
the pattern cache off completely.
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.
getStringMatcher(String)
private void deactivatePatternCache()
public boolean isPattern(java.lang.String path)
PathMatcher
path
represent a pattern that can be matched
by an implementation of this interface?
If the return value is false
, then the PathMatcher.match(java.lang.String, java.lang.String)
method does not have to be used because direct equality comparisons
on the static path Strings will lead to the same result.
isPattern
in interface PathMatcher
path
- the path String to checktrue
if the given path
represents a patternpublic boolean match(java.lang.String pattern, java.lang.String path)
PathMatcher
path
against the given pattern
,
according to this PathMatcher's matching strategy.match
in interface PathMatcher
pattern
- the pattern to match againstpath
- the path String to testtrue
if the supplied path
matched,
false
if it didn'tpublic boolean matchStart(java.lang.String pattern, java.lang.String path)
PathMatcher
path
against the corresponding part of the given
pattern
, according to this PathMatcher's matching strategy.
Determines whether the pattern at least matches as far as the given base path goes, assuming that a full path may then match as well.
matchStart
in interface PathMatcher
pattern
- the pattern to match againstpath
- the path String to testtrue
if the supplied path
matched,
false
if it didn'tprotected boolean doMatch(java.lang.String pattern, java.lang.String path, boolean fullMatch, @Nullable java.util.Map<java.lang.String,java.lang.String> uriTemplateVariables)
path
against the given pattern
.pattern
- the pattern to match againstpath
- the path String to testfullMatch
- whether a full pattern match is required (else a pattern match
as far as the given base path goes is sufficient)true
if the supplied path
matched, false
if it didn'tprivate boolean isPotentialMatch(java.lang.String path, java.lang.String[] pattDirs)
private int skipSegment(java.lang.String path, int pos, java.lang.String prefix)
private int skipSeparator(java.lang.String path, int pos, java.lang.String separator)
private boolean isWildcardChar(char c)
protected java.lang.String[] tokenizePattern(java.lang.String pattern)
Performs caching based on setCachePatterns(boolean)
, delegating to
tokenizePath(String)
for the actual tokenization algorithm.
pattern
- the pattern to tokenizeprotected java.lang.String[] tokenizePath(java.lang.String path)
path
- the path to tokenizeprivate boolean matchStrings(java.lang.String pattern, java.lang.String str, @Nullable java.util.Map<java.lang.String,java.lang.String> uriTemplateVariables)
pattern
- the pattern to match against (never null
)str
- the String which must be matched against the pattern (never null
)true
if the string matches against the pattern, or false
otherwiseprotected AntPathMatcher.AntPathStringMatcher getStringMatcher(java.lang.String pattern)
AntPathMatcher.AntPathStringMatcher
for the given pattern.
The default implementation checks this AntPathMatcher's internal cache
(see setCachePatterns(boolean)
), creating a new AntPathStringMatcher instance
if no cached copy is found.
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.
This method may be overridden to implement a custom cache strategy.
pattern
- the pattern to match against (never null
)null
)setCachePatterns(boolean)
public java.lang.String extractPathWithinPattern(java.lang.String pattern, java.lang.String path)
For example:
/docs/cvs/commit.html
' and '/docs/cvs/commit.html
-> ''/docs/*
' and '/docs/cvs/commit
-> 'cvs/commit
'/docs/cvs/*.html
' and '/docs/cvs/commit.html
-> 'commit.html
'/docs/**
' and '/docs/cvs/commit
-> 'cvs/commit
'/docs/**\/*.html
' and '/docs/cvs/commit.html
-> 'cvs/commit.html
'/*.html
' and '/docs/cvs/commit.html
-> 'docs/cvs/commit.html
'*.html
' and '/docs/cvs/commit.html
-> '/docs/cvs/commit.html
'*
' and '/docs/cvs/commit.html
-> '/docs/cvs/commit.html
'Assumes that match(java.lang.String, java.lang.String)
returns true
for 'pattern
' and 'path
', but
does not enforce this.
extractPathWithinPattern
in interface PathMatcher
pattern
- the path patternpath
- the full path to introspectpath
(never null
)public java.util.Map<java.lang.String,java.lang.String> extractUriTemplateVariables(java.lang.String pattern, java.lang.String path)
PathMatcher
For example: For pattern "/hotels/{hotel}" and path "/hotels/1", this method will return a map containing "hotel"->"1".
extractUriTemplateVariables
in interface PathMatcher
pattern
- the path pattern, possibly containing URI templatespath
- the full path to extract template variables frompublic java.lang.String combine(java.lang.String pattern1, java.lang.String pattern2)
This implementation simply concatenates the two patterns, unless
the first pattern contains a file extension match (e.g., *.html
).
In that case, the second pattern will be merged into the first. Otherwise,
an IllegalArgumentException
will be thrown.
Pattern 1 | Pattern 2 | Result |
---|---|---|
null | null | |
/hotels | null | /hotels |
null | /hotels | /hotels |
/hotels | /bookings | /hotels/bookings |
/hotels | bookings | /hotels/bookings |
/hotels/* | /bookings | /hotels/bookings |
/hotels/** | /bookings | /hotels/**/bookings |
/hotels | {hotel} | /hotels/{hotel} |
/hotels/* | {hotel} | /hotels/{hotel} |
/hotels/** | {hotel} | /hotels/**/{hotel} |
/*.html | /hotels.html | /hotels.html |
/*.html | /hotels | /hotels.html |
/*.html | /*.txt | IllegalArgumentException |
combine
in interface PathMatcher
pattern1
- the first patternpattern2
- the second patternjava.lang.IllegalArgumentException
- if the two patterns cannot be combinedprivate java.lang.String concat(java.lang.String path1, java.lang.String path2)
public java.util.Comparator<java.lang.String> getPatternComparator(java.lang.String path)
Comparator
suitable for sorting patterns in order of
explicitness.
ThisComparator
will 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:
/hotels/new
/hotels/{hotel}
/hotels/*
The full path given as parameter is used to test for exact matches. So when the given path
is /hotels/2
, the pattern /hotels/2
will be sorted before /hotels/1
.
getPatternComparator
in interface PathMatcher
path
- the full path to use for comparison