Browse Source
Now that we also have RequestPath and PathContainer with the latter as the required input, the ParsingPathMatcher adapter can be removed.pull/1489/head
7 changed files with 29 additions and 260 deletions
@ -1,170 +0,0 @@
@@ -1,170 +0,0 @@
|
||||
/* |
||||
* Copyright 2002-2017 the original author or authors. |
||||
* |
||||
* Licensed under the Apache License, Version 2.0 (the "License"); |
||||
* you may not use this file except in compliance with the License. |
||||
* You may obtain a copy of the License at |
||||
* |
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
* |
||||
* Unless required by applicable law or agreed to in writing, software |
||||
* distributed under the License is distributed on an "AS IS" BASIS, |
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||
* See the License for the specific language governing permissions and |
||||
* limitations under the License. |
||||
*/ |
||||
|
||||
package org.springframework.web.util.pattern; |
||||
|
||||
import java.util.Collections; |
||||
import java.util.Comparator; |
||||
import java.util.Map; |
||||
import java.util.concurrent.ConcurrentHashMap; |
||||
import java.util.concurrent.ConcurrentMap; |
||||
import java.util.stream.Collectors; |
||||
|
||||
import org.springframework.http.server.PathContainer; |
||||
import org.springframework.lang.Nullable; |
||||
import org.springframework.util.PathMatcher; |
||||
import org.springframework.web.util.pattern.PathPattern.PathMatchResult; |
||||
|
||||
/** |
||||
* {@link PathMatcher} implementation for path patterns parsed |
||||
* as {@link PathPatternParser} and compiled as {@link PathPattern}s. |
||||
* |
||||
* <p>Once parsed, {@link PathPattern}s are tailored for fast matching |
||||
* and quick comparison. |
||||
* |
||||
* <p>Calls this {@link PathMatcher} implementation can lead to |
||||
* {@link PatternParseException} if the provided patterns are |
||||
* illegal. |
||||
* |
||||
* @author Andy Clement |
||||
* @author Juergen Hoeller |
||||
* @since 5.0 |
||||
* @see PathPattern |
||||
*/ |
||||
public class ParsingPathMatcher implements PathMatcher { |
||||
|
||||
private final PathPatternParser parser = new PathPatternParser(); |
||||
|
||||
private final ConcurrentMap<String, PathPattern> cache = new ConcurrentHashMap<>(256); |
||||
|
||||
|
||||
@Override |
||||
public boolean isPattern(String path) { |
||||
// TODO crude, should be smarter, lookup pattern and ask it
|
||||
return (path.indexOf('*') != -1 || path.indexOf('?') != -1); |
||||
} |
||||
|
||||
@Override |
||||
public boolean match(String pattern, String path) { |
||||
PathPattern pathPattern = getPathPattern(pattern); |
||||
return pathPattern.matches(PathContainer.parsePath(path)); |
||||
} |
||||
|
||||
@Override |
||||
public boolean matchStart(String pattern, String path) { |
||||
PathPattern pathPattern = getPathPattern(pattern); |
||||
return pathPattern.matchStart(PathContainer.parsePath(path)); |
||||
} |
||||
|
||||
@Override |
||||
public String extractPathWithinPattern(String pattern, String path) { |
||||
PathPattern pathPattern = getPathPattern(pattern); |
||||
PathContainer pathContainer = PathContainer.parsePath(path); |
||||
return pathPattern.extractPathWithinPattern(pathContainer).value(); |
||||
} |
||||
|
||||
@Override |
||||
public Map<String, String> extractUriTemplateVariables(String pattern, String path) { |
||||
PathPattern pathPattern = getPathPattern(pattern); |
||||
PathContainer pathContainer = PathContainer.parsePath(path); |
||||
PathMatchResult results = pathPattern.matchAndExtract(pathContainer); |
||||
// Collapse PathMatchResults to simple value results
|
||||
// TODO: (path parameters are lost in this translation)
|
||||
if (results.getUriVariables().size() == 0) { |
||||
return Collections.emptyMap(); |
||||
} |
||||
else { |
||||
return results.getUriVariables().entrySet().stream() |
||||
.collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue)); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public Comparator<String> getPatternComparator(String path) { |
||||
return new PathPatternStringComparatorConsideringPath(path); |
||||
} |
||||
|
||||
@Override |
||||
public String combine(String pattern1, String pattern2) { |
||||
PathPattern pathPattern = getPathPattern(pattern1); |
||||
return pathPattern.combine(getPathPattern(pattern2)).getPatternString(); |
||||
} |
||||
|
||||
private PathPattern getPathPattern(String pattern) { |
||||
PathPattern pathPattern = this.cache.get(pattern); |
||||
if (pathPattern == null) { |
||||
pathPattern = this.parser.parse(pattern); |
||||
this.cache.put(pattern, pathPattern); |
||||
} |
||||
return pathPattern; |
||||
} |
||||
|
||||
|
||||
private class PathPatternStringComparatorConsideringPath implements Comparator<String> { |
||||
|
||||
private final PatternComparatorConsideringPath ppcp; |
||||
|
||||
public PathPatternStringComparatorConsideringPath(String path) { |
||||
this.ppcp = new PatternComparatorConsideringPath(path); |
||||
} |
||||
|
||||
@Override |
||||
public int compare(@Nullable String o1, @Nullable String o2) { |
||||
if (o1 == null) { |
||||
return (o2 == null ? 0 : +1); |
||||
} |
||||
else if (o2 == null) { |
||||
return -1; |
||||
} |
||||
PathPattern p1 = getPathPattern(o1); |
||||
PathPattern p2 = getPathPattern(o2); |
||||
return this.ppcp.compare(p1, p2); |
||||
} |
||||
} |
||||
|
||||
|
||||
/** |
||||
* {@link PathPattern} comparator that takes account of a specified |
||||
* path and sorts anything that exactly matches it to be first. |
||||
*/ |
||||
static class PatternComparatorConsideringPath implements Comparator<PathPattern> { |
||||
|
||||
private final String path; |
||||
|
||||
public PatternComparatorConsideringPath(String path) { |
||||
this.path = path; |
||||
} |
||||
|
||||
@Override |
||||
public int compare(@Nullable PathPattern o1, @Nullable PathPattern o2) { |
||||
// Nulls get sorted to the end
|
||||
if (o1 == null) { |
||||
return (o2 == null ? 0 : +1); |
||||
} |
||||
else if (o2 == null) { |
||||
return -1; |
||||
} |
||||
if (o1.getPatternString().equals(this.path)) { |
||||
return (o2.getPatternString().equals(this.path)) ? 0 : -1; |
||||
} |
||||
else if (o2.getPatternString().equals(this.path)) { |
||||
return +1; |
||||
} |
||||
return o1.compareTo(o2); |
||||
} |
||||
} |
||||
|
||||
} |
||||
@ -1,49 +0,0 @@
@@ -1,49 +0,0 @@
|
||||
/* |
||||
* Copyright 2002-2017 the original author or authors. |
||||
* |
||||
* Licensed under the Apache License, Version 2.0 (the "License"); |
||||
* you may not use this file except in compliance with the License. |
||||
* You may obtain a copy of the License at |
||||
* |
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
* |
||||
* Unless required by applicable law or agreed to in writing, software |
||||
* distributed under the License is distributed on an "AS IS" BASIS, |
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||
* See the License for the specific language governing permissions and |
||||
* limitations under the License. |
||||
*/ |
||||
|
||||
package org.springframework.web.servlet.config.annotation; |
||||
|
||||
import org.hamcrest.Matchers; |
||||
import org.junit.Rule; |
||||
import org.junit.Test; |
||||
import org.junit.rules.ExpectedException; |
||||
|
||||
import org.springframework.web.util.pattern.ParsingPathMatcher; |
||||
|
||||
/** |
||||
* Unit tests for {@link PathMatchConfigurer} |
||||
* @author Brian Clozel |
||||
*/ |
||||
public class PathMatchConfigurerTests { |
||||
|
||||
@Rule |
||||
public ExpectedException thrown = ExpectedException.none(); |
||||
|
||||
// SPR-15303
|
||||
@Test |
||||
public void illegalConfigurationParsingPathMatcher() { |
||||
PathMatchConfigurer configurer = new PathMatchConfigurer(); |
||||
configurer.setPathMatcher(new ParsingPathMatcher()); |
||||
configurer.setUseSuffixPatternMatch(true); |
||||
configurer.setUseTrailingSlashMatch(true); |
||||
|
||||
this.thrown.expect(IllegalStateException.class); |
||||
this.thrown.expectMessage(Matchers.containsString("useSuffixPatternMatch")); |
||||
this.thrown.expectMessage(Matchers.containsString("useTrailingSlashMatch")); |
||||
|
||||
configurer.getPathMatcher(); |
||||
} |
||||
} |
||||
Loading…
Reference in new issue