-
Notifications
You must be signed in to change notification settings - Fork 1.9k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Fixes #6263 - Review URI encoding in ConcatServlet & WelcomeFilter.
Review URI encoding in ConcatServlet & WelcomeFilter and improve testing. Signed-off-by: Lachlan Roberts <lachlan@webtide.com> Signed-off-by: Simone Bordet <simone.bordet@gmail.com> Co-authored-by: Simone Bordet <simone.bordet@gmail.com>
- Loading branch information
1 parent
9cb9343
commit 1c05b0b
Showing
6 changed files
with
353 additions
and
31 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
143 changes: 143 additions & 0 deletions
143
jetty-servlets/src/test/java/org/eclipse/jetty/servlets/WelcomeFilterTest.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,143 @@ | ||
// | ||
// ======================================================================== | ||
// Copyright (c) 1995-2021 Mort Bay Consulting Pty Ltd and others. | ||
// ------------------------------------------------------------------------ | ||
// All rights reserved. This program and the accompanying materials | ||
// are made available under the terms of the Eclipse Public License v1.0 | ||
// and Apache License v2.0 which accompanies this distribution. | ||
// | ||
// The Eclipse Public License is available at | ||
// http://www.eclipse.org/legal/epl-v10.html | ||
// | ||
// The Apache License v2.0 is available at | ||
// http://www.opensource.org/licenses/apache2.0.php | ||
// | ||
// You may elect to redistribute this code under either of these licenses. | ||
// ======================================================================== | ||
// | ||
|
||
package org.eclipse.jetty.servlets; | ||
|
||
import java.io.OutputStream; | ||
import java.nio.charset.StandardCharsets; | ||
import java.nio.file.Files; | ||
import java.nio.file.Path; | ||
import java.util.EnumSet; | ||
import java.util.stream.Stream; | ||
import javax.servlet.DispatcherType; | ||
|
||
import org.eclipse.jetty.server.LocalConnector; | ||
import org.eclipse.jetty.server.Server; | ||
import org.eclipse.jetty.servlet.FilterHolder; | ||
import org.eclipse.jetty.toolchain.test.MavenTestingUtils; | ||
import org.eclipse.jetty.webapp.WebAppContext; | ||
import org.junit.jupiter.api.AfterEach; | ||
import org.junit.jupiter.api.BeforeEach; | ||
import org.junit.jupiter.params.ParameterizedTest; | ||
import org.junit.jupiter.params.provider.Arguments; | ||
import org.junit.jupiter.params.provider.MethodSource; | ||
|
||
import static org.hamcrest.MatcherAssert.assertThat; | ||
import static org.hamcrest.Matchers.containsString; | ||
import static org.junit.jupiter.api.Assertions.assertNotNull; | ||
|
||
public class WelcomeFilterTest | ||
{ | ||
private Server server; | ||
private LocalConnector connector; | ||
|
||
@BeforeEach | ||
public void prepareServer() throws Exception | ||
{ | ||
server = new Server(); | ||
connector = new LocalConnector(server); | ||
server.addConnector(connector); | ||
|
||
Path directoryPath = MavenTestingUtils.getTargetTestingDir().toPath(); | ||
Files.createDirectories(directoryPath); | ||
Path welcomeResource = directoryPath.resolve("welcome.html"); | ||
try (OutputStream output = Files.newOutputStream(welcomeResource)) | ||
{ | ||
output.write("<h1>welcome page</h1>".getBytes(StandardCharsets.UTF_8)); | ||
} | ||
|
||
Path otherResource = directoryPath.resolve("other.html"); | ||
try (OutputStream output = Files.newOutputStream(otherResource)) | ||
{ | ||
output.write("<h1>other resource</h1>".getBytes(StandardCharsets.UTF_8)); | ||
} | ||
|
||
Path hiddenDirectory = directoryPath.resolve("WEB-INF"); | ||
Files.createDirectories(hiddenDirectory); | ||
Path hiddenResource = hiddenDirectory.resolve("one.js"); | ||
try (OutputStream output = Files.newOutputStream(hiddenResource)) | ||
{ | ||
output.write("CONFIDENTIAL".getBytes(StandardCharsets.UTF_8)); | ||
} | ||
|
||
Path hiddenWelcome = hiddenDirectory.resolve("index.html"); | ||
try (OutputStream output = Files.newOutputStream(hiddenWelcome)) | ||
{ | ||
output.write("CONFIDENTIAL".getBytes(StandardCharsets.UTF_8)); | ||
} | ||
|
||
WebAppContext context = new WebAppContext(server, directoryPath.toString(), "/"); | ||
server.setHandler(context); | ||
String concatPath = "/*"; | ||
|
||
FilterHolder filterHolder = new FilterHolder(new WelcomeFilter()); | ||
filterHolder.setInitParameter("welcome", "welcome.html"); | ||
context.addFilter(filterHolder, concatPath, EnumSet.of(DispatcherType.REQUEST)); | ||
server.start(); | ||
|
||
// Verify that I can get the file programmatically, as required by the spec. | ||
assertNotNull(context.getServletContext().getResource("/WEB-INF/one.js")); | ||
} | ||
|
||
@AfterEach | ||
public void destroy() throws Exception | ||
{ | ||
if (server != null) | ||
server.stop(); | ||
} | ||
|
||
public static Stream<Arguments> argumentsStream() | ||
{ | ||
return Stream.of( | ||
// Normal requests for the directory are redirected to the welcome page. | ||
Arguments.of("/", new String[]{"HTTP/1.1 200 ", "<h1>welcome page</h1>"}), | ||
|
||
// Try a normal resource (will bypass the filter). | ||
Arguments.of("/other.html", new String[]{"HTTP/1.1 200 ", "<h1>other resource</h1>"}), | ||
|
||
// Cannot access files in WEB-INF. | ||
Arguments.of("/WEB-INF/one.js", new String[]{"HTTP/1.1 404 "}), | ||
|
||
// Cannot serve welcome from WEB-INF. | ||
Arguments.of("/WEB-INF/", new String[]{"HTTP/1.1 404 "}), | ||
|
||
// Try to trick the filter into serving a protected resource. | ||
Arguments.of("/WEB-INF/one.js#/", new String[]{"HTTP/1.1 404 "}), | ||
Arguments.of("/js/../WEB-INF/one.js#/", new String[]{"HTTP/1.1 404 "}), | ||
|
||
// Test the URI is not double decoded in the dispatcher. | ||
Arguments.of("/%2557EB-INF/one.js%23/", new String[]{"HTTP/1.1 404 "}) | ||
); | ||
} | ||
|
||
@ParameterizedTest | ||
@MethodSource("argumentsStream") | ||
public void testWelcomeFilter(String uri, String[] contains) throws Exception | ||
{ | ||
String request = | ||
"GET " + uri + " HTTP/1.1\r\n" + | ||
"Host: localhost\r\n" + | ||
"Connection: close\r\n" + | ||
"\r\n"; | ||
String response = connector.getResponse(request); | ||
for (String s : contains) | ||
{ | ||
assertThat(response, containsString(s)); | ||
} | ||
} | ||
} |
Oops, something went wrong.