WebUtilsTests.java 10.3 KB
Newer Older
1
/*
2
 * Copyright 2002-2018 the original author or authors.
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
 *
 * 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;

19
import java.util.Arrays;
20
import java.util.Collections;
21
import java.util.HashMap;
22
import java.util.List;
23
import java.util.Map;
24
import javax.servlet.http.HttpServletRequest;
25

26
import org.junit.Test;
27

28 29 30
import org.springframework.http.HttpHeaders;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServletServerHttpRequest;
31
import org.springframework.mock.web.test.MockFilterChain;
32
import org.springframework.mock.web.test.MockHttpServletRequest;
33
import org.springframework.mock.web.test.MockHttpServletResponse;
34
import org.springframework.util.MultiValueMap;
35
import org.springframework.web.filter.ForwardedHeaderFilter;
36

37
import static org.junit.Assert.*;
38

39 40
/**
 * @author Juergen Hoeller
41
 * @author Arjen Poutsma
42
 * @author Rossen Stoyanchev
43
 * @author Sebastien Deleuze
44
 */
45
public class WebUtilsTests {
46

47 48
	@Test
	public void findParameterValue() {
49
		Map<String, Object> params = new HashMap<>();
50 51 52
		params.put("myKey1", "myValue1");
		params.put("myKey2_myValue2", "xxx");
		params.put("myKey3_myValue3.x", "xxx");
J
Juergen Hoeller 已提交
53
		params.put("myKey4_myValue4.y", new String[] {"yyy"});
54 55 56 57 58 59 60 61

		assertNull(WebUtils.findParameterValue(params, "myKey0"));
		assertEquals("myValue1", WebUtils.findParameterValue(params, "myKey1"));
		assertEquals("myValue2", WebUtils.findParameterValue(params, "myKey2"));
		assertEquals("myValue3", WebUtils.findParameterValue(params, "myKey3"));
		assertEquals("myValue4", WebUtils.findParameterValue(params, "myKey4"));
	}

62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91
	@Test
	public void parseMatrixVariablesString() {
		MultiValueMap<String, String> variables;

		variables = WebUtils.parseMatrixVariables(null);
		assertEquals(0, variables.size());

		variables = WebUtils.parseMatrixVariables("year");
		assertEquals(1, variables.size());
		assertEquals("", variables.getFirst("year"));

		variables = WebUtils.parseMatrixVariables("year=2012");
		assertEquals(1, variables.size());
		assertEquals("2012", variables.getFirst("year"));

		variables = WebUtils.parseMatrixVariables("year=2012;colors=red,blue,green");
		assertEquals(2, variables.size());
		assertEquals(Arrays.asList("red", "blue", "green"), variables.get("colors"));
		assertEquals("2012", variables.getFirst("year"));

		variables = WebUtils.parseMatrixVariables(";year=2012;colors=red,blue,green;");
		assertEquals(2, variables.size());
		assertEquals(Arrays.asList("red", "blue", "green"), variables.get("colors"));
		assertEquals("2012", variables.getFirst("year"));

		variables = WebUtils.parseMatrixVariables("colors=red;colors=blue;colors=green");
		assertEquals(1, variables.size());
		assertEquals(Arrays.asList("red", "blue", "green"), variables.get("colors"));
	}

92
	@Test
93
	public void isValidOrigin() {
94
		List<String> allowed = Collections.emptyList();
S
Spring Operator 已提交
95 96
		assertTrue(checkValidOrigin("mydomain1.com", -1, "https://mydomain1.com", allowed));
		assertFalse(checkValidOrigin("mydomain1.com", -1, "/QTifZ/", allowed));
97 98

		allowed = Collections.singletonList("*");
S
Spring Operator 已提交
99
		assertTrue(checkValidOrigin("mydomain1.com", -1, "/QTifZ/", allowed));
100

S
Spring Operator 已提交
101 102
		allowed = Collections.singletonList("https://mydomain1.com");
		assertTrue(checkValidOrigin("mydomain2.com", -1, "https://mydomain1.com", allowed));
103
		assertFalse(checkValidOrigin("mydomain2.com", -1, "http://mydomain3.com", allowed));
104 105 106
	}

	@Test
107
	public void isSameOrigin() {
S
Spring Operator 已提交
108 109
		assertTrue(checkSameOrigin("http", "mydomain1.com", -1, "https://mydomain1.com"));
		assertTrue(checkSameOrigin("http", "mydomain1.com", -1, "https://www.mydomain1.com/"));
110 111
		assertTrue(checkSameOrigin("https", "mydomain1.com", 443, "https://mydomain1.com"));
		assertTrue(checkSameOrigin("https", "mydomain1.com", 443, "https://mydomain1.com:443"));
S
Spring Operator 已提交
112
		assertTrue(checkSameOrigin("http", "mydomain1.com", 123, "https://mydomain1.com:123"));
113 114 115
		assertTrue(checkSameOrigin("ws", "mydomain1.com", -1, "ws://mydomain1.com"));
		assertTrue(checkSameOrigin("wss", "mydomain1.com", 443, "wss://mydomain1.com"));

S
Spring Operator 已提交
116
		assertFalse(checkSameOrigin("http", "mydomain1.com", -1, "/QTifZ/"));
117 118
		assertFalse(checkSameOrigin("http", "mydomain1.com", -1, "https://mydomain1.com"));
		assertFalse(checkSameOrigin("http", "mydomain1.com", -1, "invalid-origin"));
S
Spring Operator 已提交
119
		assertFalse(checkSameOrigin("https", "mydomain1.com", -1, "https://mydomain1.com"));
120 121

		// Handling of invalid origins as described in SPR-13478
S
Spring Operator 已提交
122 123 124 125 126 127 128 129
		assertTrue(checkSameOrigin("http", "mydomain1.com", -1, "https://mydomain1.com/"));
		assertTrue(checkSameOrigin("http", "mydomain1.com", -1, "https://www.mydomain1.com/"));
		assertTrue(checkSameOrigin("http", "mydomain1.com", -1, "https://mydomain1.com/path"));
		assertTrue(checkSameOrigin("http", "mydomain1.com", -1, "https://www.mydomain1.com/path"));
		assertFalse(checkSameOrigin("http", "mydomain2.com", -1, "https://mydomain1.com/"));
		assertFalse(checkSameOrigin("http", "mydomain2.com", -1, "https://www.mydomain1.com/"));
		assertFalse(checkSameOrigin("http", "mydomain2.com", -1, "https://mydomain1.com/path"));
		assertFalse(checkSameOrigin("http", "mydomain2.com", -1, "https://www.mydomain1.com/path"));
130 131

		// Handling of IPv6 hosts as described in SPR-13525
132 133 134
		assertTrue(checkSameOrigin("http", "[::1]", -1, "http://[::1]"));
		assertTrue(checkSameOrigin("http", "[::1]", 8080, "http://[::1]:8080"));
		assertTrue(checkSameOrigin("http",
R
Rossen Stoyanchev 已提交
135 136
				"[2001:0db8:0000:85a3:0000:0000:ac1f:8001]", -1,
				"http://[2001:0db8:0000:85a3:0000:0000:ac1f:8001]"));
137
		assertTrue(checkSameOrigin("http",
R
Rossen Stoyanchev 已提交
138 139
				"[2001:0db8:0000:85a3:0000:0000:ac1f:8001]", 8080,
				"http://[2001:0db8:0000:85a3:0000:0000:ac1f:8001]:8080"));
140 141
		assertFalse(checkSameOrigin("http", "[::1]", -1, "http://[::1]:8080"));
		assertFalse(checkSameOrigin("http", "[::1]", 8080,
R
Rossen Stoyanchev 已提交
142
				"http://[2001:0db8:0000:85a3:0000:0000:ac1f:8001]:8080"));
143
	}
144

S
sdeleuze 已提交
145
	@Test  // SPR-16262
146 147 148 149 150 151 152 153
	public void isSameOriginWithXForwardedHeaders() throws Exception {
		String server = "mydomain1.com";
		testWithXForwardedHeaders(server, -1, "https", null, -1, "https://mydomain1.com");
		testWithXForwardedHeaders(server, 123, "https", null, -1, "https://mydomain1.com");
		testWithXForwardedHeaders(server, -1, "https", "mydomain2.com", -1, "https://mydomain2.com");
		testWithXForwardedHeaders(server, 123, "https", "mydomain2.com", -1, "https://mydomain2.com");
		testWithXForwardedHeaders(server, -1, "https", "mydomain2.com", 456, "https://mydomain2.com:456");
		testWithXForwardedHeaders(server, 123, "https", "mydomain2.com", 456, "https://mydomain2.com:456");
S
sdeleuze 已提交
154 155 156
	}

	@Test  // SPR-16262
157 158 159 160 161 162 163 164
	public void isSameOriginWithForwardedHeader() throws Exception {
		String server = "mydomain1.com";
		testWithForwardedHeader(server, -1, "proto=https", "https://mydomain1.com");
		testWithForwardedHeader(server, 123, "proto=https", "https://mydomain1.com");
		testWithForwardedHeader(server, -1, "proto=https; host=mydomain2.com", "https://mydomain2.com");
		testWithForwardedHeader(server, 123, "proto=https; host=mydomain2.com", "https://mydomain2.com");
		testWithForwardedHeader(server, -1, "proto=https; host=mydomain2.com:456", "https://mydomain2.com:456");
		testWithForwardedHeader(server, 123, "proto=https; host=mydomain2.com:456", "https://mydomain2.com:456");
S
sdeleuze 已提交
165 166
	}

167

168 169 170 171 172 173 174
	private boolean checkValidOrigin(String serverName, int port, String originHeader, List<String> allowed) {
		MockHttpServletRequest servletRequest = new MockHttpServletRequest();
		ServerHttpRequest request = new ServletServerHttpRequest(servletRequest);
		servletRequest.setServerName(serverName);
		if (port != -1) {
			servletRequest.setServerPort(port);
		}
S
sdeleuze 已提交
175
		servletRequest.addHeader(HttpHeaders.ORIGIN, originHeader);
176
		return WebUtils.isValidOrigin(request, allowed);
177 178
	}

179
	private boolean checkSameOrigin(String scheme, String serverName, int port, String originHeader) {
180 181
		MockHttpServletRequest servletRequest = new MockHttpServletRequest();
		ServerHttpRequest request = new ServletServerHttpRequest(servletRequest);
182
		servletRequest.setScheme(scheme);
183 184 185 186
		servletRequest.setServerName(serverName);
		if (port != -1) {
			servletRequest.setServerPort(port);
		}
S
sdeleuze 已提交
187
		servletRequest.addHeader(HttpHeaders.ORIGIN, originHeader);
188
		return WebUtils.isSameOrigin(request);
189 190
	}

191 192 193 194 195
	private void testWithXForwardedHeaders(String serverName, int port, String forwardedProto,
			String forwardedHost, int forwardedPort, String originHeader) throws Exception {

		MockHttpServletRequest request = new MockHttpServletRequest();
		request.setServerName(serverName);
S
sdeleuze 已提交
196
		if (port != -1) {
197
			request.setServerPort(port);
S
sdeleuze 已提交
198 199
		}
		if (forwardedProto != null) {
200
			request.addHeader("X-Forwarded-Proto", forwardedProto);
S
sdeleuze 已提交
201 202
		}
		if (forwardedHost != null) {
203
			request.addHeader("X-Forwarded-Host", forwardedHost);
S
sdeleuze 已提交
204 205
		}
		if (forwardedPort != -1) {
206
			request.addHeader("X-Forwarded-Port", String.valueOf(forwardedPort));
S
sdeleuze 已提交
207
		}
208 209 210 211 212 213
		request.addHeader(HttpHeaders.ORIGIN, originHeader);

		HttpServletRequest requestToUse = adaptFromForwardedHeaders(request);
		ServerHttpRequest httpRequest = new ServletServerHttpRequest(requestToUse);

		assertTrue(WebUtils.isSameOrigin(httpRequest));
S
sdeleuze 已提交
214 215
	}

216 217 218 219 220
	private void testWithForwardedHeader(String serverName, int port, String forwardedHeader,
			String originHeader) throws Exception {

		MockHttpServletRequest request = new MockHttpServletRequest();
		request.setServerName(serverName);
S
sdeleuze 已提交
221
		if (port != -1) {
222
			request.setServerPort(port);
S
sdeleuze 已提交
223
		}
224 225 226 227 228 229 230 231 232 233 234 235 236 237
		request.addHeader("Forwarded", forwardedHeader);
		request.addHeader(HttpHeaders.ORIGIN, originHeader);

		HttpServletRequest requestToUse = adaptFromForwardedHeaders(request);
		ServerHttpRequest httpRequest = new ServletServerHttpRequest(requestToUse);

		assertTrue(WebUtils.isSameOrigin(httpRequest));
	}

	// SPR-16668
	private HttpServletRequest adaptFromForwardedHeaders(HttpServletRequest request) throws Exception {
		MockFilterChain chain = new MockFilterChain();
		new ForwardedHeaderFilter().doFilter(request, new MockHttpServletResponse(), chain);
		return (HttpServletRequest) chain.getRequest();
S
sdeleuze 已提交
238 239
	}

240
}