UriComponentsBuilder allows more efficient (and less restrictive) CORS origin comparison
Also introduces a specifically narrowed cloneBuilder() method for UriComponentsBuilders. Issue: SPR-14080
This commit is contained in:
parent
abe7345008
commit
9a52c81443
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2015 the original author or authors.
|
* Copyright 2002-2016 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -24,6 +24,7 @@ import java.util.Map;
|
||||||
import java.util.regex.Matcher;
|
import java.util.regex.Matcher;
|
||||||
import java.util.regex.Pattern;
|
import java.util.regex.Pattern;
|
||||||
|
|
||||||
|
import org.springframework.http.HttpHeaders;
|
||||||
import org.springframework.http.HttpRequest;
|
import org.springframework.http.HttpRequest;
|
||||||
import org.springframework.util.Assert;
|
import org.springframework.util.Assert;
|
||||||
import org.springframework.util.LinkedMultiValueMap;
|
import org.springframework.util.LinkedMultiValueMap;
|
||||||
|
@ -130,7 +131,7 @@ public class UriComponentsBuilder implements Cloneable {
|
||||||
this.userInfo = other.userInfo;
|
this.userInfo = other.userInfo;
|
||||||
this.host = other.host;
|
this.host = other.host;
|
||||||
this.port = other.port;
|
this.port = other.port;
|
||||||
this.pathBuilder = (CompositePathComponentBuilder) other.pathBuilder.clone();
|
this.pathBuilder = other.pathBuilder.cloneBuilder();
|
||||||
this.queryParams.putAll(other.queryParams);
|
this.queryParams.putAll(other.queryParams);
|
||||||
this.fragment = other.fragment;
|
this.fragment = other.fragment;
|
||||||
}
|
}
|
||||||
|
@ -271,72 +272,17 @@ public class UriComponentsBuilder implements Cloneable {
|
||||||
/**
|
/**
|
||||||
* Create a new {@code UriComponents} object from the URI associated with
|
* Create a new {@code UriComponents} object from the URI associated with
|
||||||
* the given HttpRequest while also overlaying with values from the headers
|
* the given HttpRequest while also overlaying with values from the headers
|
||||||
* "Forwarded" (<a href="http://tools.ietf.org/html/rfc7239">RFC 7239</a>, or
|
* "Forwarded" (<a href="http://tools.ietf.org/html/rfc7239">RFC 7239</a>,
|
||||||
* "X-Forwarded-Host", "X-Forwarded-Port", and "X-Forwarded-Proto" if "Forwarded" is
|
* or "X-Forwarded-Host", "X-Forwarded-Port", and "X-Forwarded-Proto" if
|
||||||
* not found.
|
* "Forwarded" is not found.
|
||||||
* @param request the source request
|
* @param request the source request
|
||||||
* @return the URI components of the URI
|
* @return the URI components of the URI
|
||||||
* @since 4.1.5
|
* @since 4.1.5
|
||||||
*/
|
*/
|
||||||
public static UriComponentsBuilder fromHttpRequest(HttpRequest request) {
|
public static UriComponentsBuilder fromHttpRequest(HttpRequest request) {
|
||||||
URI uri = request.getURI();
|
return fromUri(request.getURI()).adaptFromForwardedHeaders(request.getHeaders());
|
||||||
UriComponentsBuilder builder = UriComponentsBuilder.fromUri(uri);
|
|
||||||
|
|
||||||
String scheme = uri.getScheme();
|
|
||||||
String host = uri.getHost();
|
|
||||||
int port = uri.getPort();
|
|
||||||
|
|
||||||
String forwardedHeader = request.getHeaders().getFirst("Forwarded");
|
|
||||||
if (StringUtils.hasText(forwardedHeader)) {
|
|
||||||
String forwardedToUse = StringUtils.commaDelimitedListToStringArray(forwardedHeader)[0];
|
|
||||||
Matcher m = FORWARDED_HOST_PATTERN.matcher(forwardedToUse);
|
|
||||||
if (m.find()) {
|
|
||||||
host = m.group(1).trim();
|
|
||||||
}
|
|
||||||
m = FORWARDED_PROTO_PATTERN.matcher(forwardedToUse);
|
|
||||||
if (m.find()) {
|
|
||||||
scheme = m.group(1).trim();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
String hostHeader = request.getHeaders().getFirst("X-Forwarded-Host");
|
|
||||||
if (StringUtils.hasText(hostHeader)) {
|
|
||||||
String[] hosts = StringUtils.commaDelimitedListToStringArray(hostHeader);
|
|
||||||
String hostToUse = hosts[0];
|
|
||||||
if (hostToUse.contains(":")) {
|
|
||||||
String[] hostAndPort = StringUtils.split(hostToUse, ":");
|
|
||||||
host = hostAndPort[0];
|
|
||||||
port = Integer.parseInt(hostAndPort[1]);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
host = hostToUse;
|
|
||||||
port = -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
String portHeader = request.getHeaders().getFirst("X-Forwarded-Port");
|
|
||||||
if (StringUtils.hasText(portHeader)) {
|
|
||||||
String[] ports = StringUtils.commaDelimitedListToStringArray(portHeader);
|
|
||||||
port = Integer.parseInt(ports[0]);
|
|
||||||
}
|
|
||||||
|
|
||||||
String protocolHeader = request.getHeaders().getFirst("X-Forwarded-Proto");
|
|
||||||
if (StringUtils.hasText(protocolHeader)) {
|
|
||||||
String[] protocols = StringUtils.commaDelimitedListToStringArray(protocolHeader);
|
|
||||||
scheme = protocols[0];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
builder.scheme(scheme);
|
|
||||||
builder.host(host);
|
|
||||||
builder.port(null);
|
|
||||||
if (scheme.equals("http") && port != 80 || scheme.equals("https") && port != 443) {
|
|
||||||
builder.port(port);
|
|
||||||
}
|
|
||||||
return builder;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create an instance by parsing the "Origin" header of an HTTP request.
|
* Create an instance by parsing the "Origin" header of an HTTP request.
|
||||||
* @see <a href="https://tools.ietf.org/html/rfc6454">RFC 6454</a>
|
* @see <a href="https://tools.ietf.org/html/rfc6454">RFC 6454</a>
|
||||||
|
@ -463,18 +409,6 @@ public class UriComponentsBuilder implements Cloneable {
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void resetHierarchicalComponents() {
|
|
||||||
this.userInfo = null;
|
|
||||||
this.host = null;
|
|
||||||
this.port = null;
|
|
||||||
this.pathBuilder = new CompositePathComponentBuilder();
|
|
||||||
this.queryParams.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void resetSchemeSpecificPart() {
|
|
||||||
this.ssp = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the URI scheme. The given scheme may contain URI template variables,
|
* Set the URI scheme. The given scheme may contain URI template variables,
|
||||||
* and may also be {@code null} to clear the scheme of this builder.
|
* and may also be {@code null} to clear the scheme of this builder.
|
||||||
|
@ -724,17 +658,103 @@ public class UriComponentsBuilder implements Cloneable {
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adapt this builder's scheme+host+port from the given headers, specifically
|
||||||
|
* "Forwarded" (<a href="http://tools.ietf.org/html/rfc7239">RFC 7239</a>,
|
||||||
|
* or "X-Forwarded-Host", "X-Forwarded-Port", and "X-Forwarded-Proto" if
|
||||||
|
* "Forwarded" is not found.
|
||||||
|
* @param headers the HTTP headers to consider
|
||||||
|
* @return this UriComponentsBuilder
|
||||||
|
* @since 4.3
|
||||||
|
*/
|
||||||
|
UriComponentsBuilder adaptFromForwardedHeaders(HttpHeaders headers) {
|
||||||
|
String forwardedHeader = headers.getFirst("Forwarded");
|
||||||
|
if (StringUtils.hasText(forwardedHeader)) {
|
||||||
|
String forwardedToUse = StringUtils.commaDelimitedListToStringArray(forwardedHeader)[0];
|
||||||
|
Matcher matcher = FORWARDED_HOST_PATTERN.matcher(forwardedToUse);
|
||||||
|
if (matcher.find()) {
|
||||||
|
host(matcher.group(1).trim());
|
||||||
|
}
|
||||||
|
matcher = FORWARDED_PROTO_PATTERN.matcher(forwardedToUse);
|
||||||
|
if (matcher.find()) {
|
||||||
|
scheme(matcher.group(1).trim());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
String hostHeader = headers.getFirst("X-Forwarded-Host");
|
||||||
|
if (StringUtils.hasText(hostHeader)) {
|
||||||
|
String[] hosts = StringUtils.commaDelimitedListToStringArray(hostHeader);
|
||||||
|
String hostToUse = hosts[0];
|
||||||
|
if (hostToUse.contains(":")) {
|
||||||
|
String[] hostAndPort = StringUtils.split(hostToUse, ":");
|
||||||
|
host(hostAndPort[0]);
|
||||||
|
port(Integer.parseInt(hostAndPort[1]));
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
host(hostToUse);
|
||||||
|
port(null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
String portHeader = headers.getFirst("X-Forwarded-Port");
|
||||||
|
if (StringUtils.hasText(portHeader)) {
|
||||||
|
String[] ports = StringUtils.commaDelimitedListToStringArray(portHeader);
|
||||||
|
port(Integer.parseInt(ports[0]));
|
||||||
|
}
|
||||||
|
|
||||||
|
String protocolHeader = headers.getFirst("X-Forwarded-Proto");
|
||||||
|
if (StringUtils.hasText(protocolHeader)) {
|
||||||
|
String[] protocols = StringUtils.commaDelimitedListToStringArray(protocolHeader);
|
||||||
|
scheme(protocols[0]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((this.scheme.equals("http") && "80".equals(this.port)) ||
|
||||||
|
(this.scheme.equals("https") && "443".equals(this.port))) {
|
||||||
|
this.port = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void resetHierarchicalComponents() {
|
||||||
|
this.userInfo = null;
|
||||||
|
this.host = null;
|
||||||
|
this.port = null;
|
||||||
|
this.pathBuilder = new CompositePathComponentBuilder();
|
||||||
|
this.queryParams.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void resetSchemeSpecificPart() {
|
||||||
|
this.ssp = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Public declaration of Object's {@code clone()} method.
|
||||||
|
* Delegates to {@link #cloneBuilder()}.
|
||||||
|
* @see Object#clone()
|
||||||
|
*/
|
||||||
@Override
|
@Override
|
||||||
public Object clone() {
|
public Object clone() {
|
||||||
|
return cloneBuilder();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clone this {@code UriComponentsBuilder}.
|
||||||
|
* @return the cloned {@code UriComponentsBuilder} object
|
||||||
|
* @since 4.3
|
||||||
|
*/
|
||||||
|
public UriComponentsBuilder cloneBuilder() {
|
||||||
return new UriComponentsBuilder(this);
|
return new UriComponentsBuilder(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private interface PathComponentBuilder extends Cloneable {
|
private interface PathComponentBuilder {
|
||||||
|
|
||||||
PathComponent build();
|
PathComponent build();
|
||||||
|
|
||||||
Object clone();
|
PathComponentBuilder cloneBuilder();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -810,10 +830,10 @@ public class UriComponentsBuilder implements Cloneable {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Object clone() {
|
public CompositePathComponentBuilder cloneBuilder() {
|
||||||
CompositePathComponentBuilder compositeBuilder = new CompositePathComponentBuilder();
|
CompositePathComponentBuilder compositeBuilder = new CompositePathComponentBuilder();
|
||||||
for (PathComponentBuilder builder : this.builders) {
|
for (PathComponentBuilder builder : this.builders) {
|
||||||
compositeBuilder.builders.add((PathComponentBuilder) builder.clone());
|
compositeBuilder.builders.add(builder.cloneBuilder());
|
||||||
}
|
}
|
||||||
return compositeBuilder;
|
return compositeBuilder;
|
||||||
}
|
}
|
||||||
|
@ -852,7 +872,7 @@ public class UriComponentsBuilder implements Cloneable {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Object clone() {
|
public FullPathComponentBuilder cloneBuilder() {
|
||||||
FullPathComponentBuilder builder = new FullPathComponentBuilder();
|
FullPathComponentBuilder builder = new FullPathComponentBuilder();
|
||||||
builder.append(this.path.toString());
|
builder.append(this.path.toString());
|
||||||
return builder;
|
return builder;
|
||||||
|
@ -879,7 +899,7 @@ public class UriComponentsBuilder implements Cloneable {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Object clone() {
|
public PathSegmentComponentBuilder cloneBuilder() {
|
||||||
PathSegmentComponentBuilder builder = new PathSegmentComponentBuilder();
|
PathSegmentComponentBuilder builder = new PathSegmentComponentBuilder();
|
||||||
builder.pathSegments.addAll(this.pathSegments);
|
builder.pathSegments.addAll(this.pathSegments);
|
||||||
return builder;
|
return builder;
|
||||||
|
|
|
@ -34,6 +34,7 @@ import javax.servlet.http.HttpServletResponse;
|
||||||
import javax.servlet.http.HttpSession;
|
import javax.servlet.http.HttpSession;
|
||||||
|
|
||||||
import org.springframework.http.HttpRequest;
|
import org.springframework.http.HttpRequest;
|
||||||
|
import org.springframework.http.server.ServletServerHttpRequest;
|
||||||
import org.springframework.util.Assert;
|
import org.springframework.util.Assert;
|
||||||
import org.springframework.util.CollectionUtils;
|
import org.springframework.util.CollectionUtils;
|
||||||
import org.springframework.util.LinkedMultiValueMap;
|
import org.springframework.util.LinkedMultiValueMap;
|
||||||
|
@ -811,18 +812,31 @@ public abstract class WebUtils {
|
||||||
if (origin == null) {
|
if (origin == null) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
UriComponents actualUrl = UriComponentsBuilder.fromHttpRequest(request).build();
|
UriComponentsBuilder urlBuilder;
|
||||||
|
if (request instanceof ServletServerHttpRequest) {
|
||||||
|
// Build more efficiently if we can: we only need scheme, host, port for origin comparison
|
||||||
|
HttpServletRequest servletRequest = ((ServletServerHttpRequest) request).getServletRequest();
|
||||||
|
urlBuilder = new UriComponentsBuilder().
|
||||||
|
scheme(servletRequest.getScheme()).
|
||||||
|
host(servletRequest.getServerName()).
|
||||||
|
port(servletRequest.getServerPort()).
|
||||||
|
adaptFromForwardedHeaders(request.getHeaders());
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
urlBuilder = UriComponentsBuilder.fromHttpRequest(request);
|
||||||
|
}
|
||||||
|
UriComponents actualUrl = urlBuilder.build();
|
||||||
UriComponents originUrl = UriComponentsBuilder.fromOriginHeader(origin).build();
|
UriComponents originUrl = UriComponentsBuilder.fromOriginHeader(origin).build();
|
||||||
return (actualUrl.getHost().equals(originUrl.getHost()) && getPort(actualUrl) == getPort(originUrl));
|
return (actualUrl.getHost().equals(originUrl.getHost()) && getPort(actualUrl) == getPort(originUrl));
|
||||||
}
|
}
|
||||||
|
|
||||||
private static int getPort(UriComponents component) {
|
private static int getPort(UriComponents uri) {
|
||||||
int port = component.getPort();
|
int port = uri.getPort();
|
||||||
if (port == -1) {
|
if (port == -1) {
|
||||||
if ("http".equals(component.getScheme()) || "ws".equals(component.getScheme())) {
|
if ("http".equals(uri.getScheme()) || "ws".equals(uri.getScheme())) {
|
||||||
port = 80;
|
port = 80;
|
||||||
}
|
}
|
||||||
else if ("https".equals(component.getScheme()) || "wss".equals(component.getScheme())) {
|
else if ("https".equals(uri.getScheme()) || "wss".equals(uri.getScheme())) {
|
||||||
port = 443;
|
port = 443;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2015 the original author or authors.
|
* Copyright 2002-2016 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -406,7 +406,7 @@ public class MvcUriComponentsBuilder {
|
||||||
|
|
||||||
private static UriComponentsBuilder getBaseUrlToUse(UriComponentsBuilder baseUrl) {
|
private static UriComponentsBuilder getBaseUrlToUse(UriComponentsBuilder baseUrl) {
|
||||||
if (baseUrl != null) {
|
if (baseUrl != null) {
|
||||||
return (UriComponentsBuilder) baseUrl.clone();
|
return baseUrl.cloneBuilder();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return ServletUriComponentsBuilder.fromCurrentServletMapping();
|
return ServletUriComponentsBuilder.fromCurrentServletMapping();
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2015 the original author or authors.
|
* Copyright 2002-2016 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -223,7 +223,7 @@ public class ServletUriComponentsBuilder extends UriComponentsBuilder {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Object clone() {
|
public ServletUriComponentsBuilder cloneBuilder() {
|
||||||
return new ServletUriComponentsBuilder(this);
|
return new ServletUriComponentsBuilder(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue