parent
5155719877
commit
e0a6a9efa9
|
@ -107,6 +107,18 @@ public class StrictHttpFirewall implements HttpFirewall {
|
||||||
|
|
||||||
private static final List<String> FORBIDDEN_NULL = Collections.unmodifiableList(Arrays.asList("\0", "%00"));
|
private static final List<String> FORBIDDEN_NULL = Collections.unmodifiableList(Arrays.asList("\0", "%00"));
|
||||||
|
|
||||||
|
private static final List<String> FORBIDDEN_LF = Collections
|
||||||
|
.unmodifiableList(Arrays.asList("\r", "%0a", "%0A"));
|
||||||
|
|
||||||
|
private static final List<String> FORBIDDEN_CR = Collections
|
||||||
|
.unmodifiableList(Arrays.asList("\n", "%0d", "%0D"));
|
||||||
|
|
||||||
|
private static final List<String> FORBIDDEN_LINE_SEPARATOR = Collections
|
||||||
|
.unmodifiableList(Arrays.asList("\u2028"));
|
||||||
|
|
||||||
|
private static final List<String> FORBIDDEN_PARAGRAPH_SEPARATOR = Collections
|
||||||
|
.unmodifiableList(Arrays.asList("\u2029"));
|
||||||
|
|
||||||
private Set<String> encodedUrlBlocklist = new HashSet<>();
|
private Set<String> encodedUrlBlocklist = new HashSet<>();
|
||||||
|
|
||||||
private Set<String> decodedUrlBlocklist = new HashSet<>();
|
private Set<String> decodedUrlBlocklist = new HashSet<>();
|
||||||
|
@ -135,10 +147,14 @@ public class StrictHttpFirewall implements HttpFirewall {
|
||||||
urlBlocklistsAddAll(FORBIDDEN_DOUBLE_FORWARDSLASH);
|
urlBlocklistsAddAll(FORBIDDEN_DOUBLE_FORWARDSLASH);
|
||||||
urlBlocklistsAddAll(FORBIDDEN_BACKSLASH);
|
urlBlocklistsAddAll(FORBIDDEN_BACKSLASH);
|
||||||
urlBlocklistsAddAll(FORBIDDEN_NULL);
|
urlBlocklistsAddAll(FORBIDDEN_NULL);
|
||||||
|
urlBlocklistsAddAll(FORBIDDEN_LF);
|
||||||
|
urlBlocklistsAddAll(FORBIDDEN_CR);
|
||||||
|
|
||||||
this.encodedUrlBlocklist.add(ENCODED_PERCENT);
|
this.encodedUrlBlocklist.add(ENCODED_PERCENT);
|
||||||
this.encodedUrlBlocklist.addAll(FORBIDDEN_ENCODED_PERIOD);
|
this.encodedUrlBlocklist.addAll(FORBIDDEN_ENCODED_PERIOD);
|
||||||
this.decodedUrlBlocklist.add(PERCENT);
|
this.decodedUrlBlocklist.add(PERCENT);
|
||||||
|
this.decodedUrlBlocklist.addAll(FORBIDDEN_LINE_SEPARATOR);
|
||||||
|
this.decodedUrlBlocklist.addAll(FORBIDDEN_PARAGRAPH_SEPARATOR);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -432,9 +448,6 @@ public class StrictHttpFirewall implements HttpFirewall {
|
||||||
throw new RequestRejectedException("The request was rejected because the URL was not normalized.");
|
throw new RequestRejectedException("The request was rejected because the URL was not normalized.");
|
||||||
}
|
}
|
||||||
rejectNonPrintableAsciiCharactersInFieldName(request.getRequestURI(), "requestURI");
|
rejectNonPrintableAsciiCharactersInFieldName(request.getRequestURI(), "requestURI");
|
||||||
rejectNonPrintableAsciiCharactersInFieldName(request.getServletPath(), "servletPath");
|
|
||||||
rejectNonPrintableAsciiCharactersInFieldName(request.getPathInfo(), "pathInfo");
|
|
||||||
rejectNonPrintableAsciiCharactersInFieldName(request.getContextPath(), "contextPath");
|
|
||||||
return new StrictFirewalledRequest(request);
|
return new StrictFirewalledRequest(request);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -343,6 +343,12 @@ public class StrictHttpFirewallTests {
|
||||||
this.firewall.getFirewalledRequest(this.request);
|
this.firewall.getFirewalledRequest(this.request);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void getFirewalledRequestWhenJapaneseCharacterThenNoException() {
|
||||||
|
this.request.setServletPath("/\u3042");
|
||||||
|
this.firewall.getFirewalledRequest(this.request);
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void getFirewalledRequestWhenExceedsUpperboundAsciiThenException() {
|
public void getFirewalledRequestWhenExceedsUpperboundAsciiThenException() {
|
||||||
this.request.setRequestURI("/\u007f");
|
this.request.setRequestURI("/\u007f");
|
||||||
|
@ -364,6 +370,20 @@ public class StrictHttpFirewallTests {
|
||||||
.isThrownBy(() -> this.firewall.getFirewalledRequest(this.request));
|
.isThrownBy(() -> this.firewall.getFirewalledRequest(this.request));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void getFirewalledRequestWhenContainsLowercaseEncodedLineFeedThenException() {
|
||||||
|
this.request.setRequestURI("/something%0a/");
|
||||||
|
assertThatExceptionOfType(RequestRejectedException.class)
|
||||||
|
.isThrownBy(() -> this.firewall.getFirewalledRequest(this.request));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void getFirewalledRequestWhenContainsUppercaseEncodedLineFeedThenException() {
|
||||||
|
this.request.setRequestURI("/something%0A/");
|
||||||
|
assertThatExceptionOfType(RequestRejectedException.class)
|
||||||
|
.isThrownBy(() -> this.firewall.getFirewalledRequest(this.request));
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void getFirewalledRequestWhenContainsLineFeedThenException() {
|
public void getFirewalledRequestWhenContainsLineFeedThenException() {
|
||||||
this.request.setRequestURI("/something\n/");
|
this.request.setRequestURI("/something\n/");
|
||||||
|
@ -378,6 +398,20 @@ public class StrictHttpFirewallTests {
|
||||||
.isThrownBy(() -> this.firewall.getFirewalledRequest(this.request));
|
.isThrownBy(() -> this.firewall.getFirewalledRequest(this.request));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void getFirewalledRequestWhenContainsLowercaseEncodedCarriageReturnThenException() {
|
||||||
|
this.request.setRequestURI("/something%0d/");
|
||||||
|
assertThatExceptionOfType(RequestRejectedException.class)
|
||||||
|
.isThrownBy(() -> this.firewall.getFirewalledRequest(this.request));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void getFirewalledRequestWhenContainsUppercaseEncodedCarriageReturnThenException() {
|
||||||
|
this.request.setRequestURI("/something%0D/");
|
||||||
|
assertThatExceptionOfType(RequestRejectedException.class)
|
||||||
|
.isThrownBy(() -> this.firewall.getFirewalledRequest(this.request));
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void getFirewalledRequestWhenContainsCarriageReturnThenException() {
|
public void getFirewalledRequestWhenContainsCarriageReturnThenException() {
|
||||||
this.request.setRequestURI("/something\r/");
|
this.request.setRequestURI("/something\r/");
|
||||||
|
@ -392,6 +426,20 @@ public class StrictHttpFirewallTests {
|
||||||
.isThrownBy(() -> this.firewall.getFirewalledRequest(this.request));
|
.isThrownBy(() -> this.firewall.getFirewalledRequest(this.request));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void getFirewalledRequestWhenServletPathContainsLineSeparatorThenException() {
|
||||||
|
this.request.setServletPath("/something\u2028/");
|
||||||
|
assertThatExceptionOfType(RequestRejectedException.class)
|
||||||
|
.isThrownBy(() -> this.firewall.getFirewalledRequest(this.request));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void getFirewalledRequestWhenServletPathContainsParagraphSeparatorThenException() {
|
||||||
|
this.request.setServletPath("/something\u2029/");
|
||||||
|
assertThatExceptionOfType(RequestRejectedException.class)
|
||||||
|
.isThrownBy(() -> this.firewall.getFirewalledRequest(this.request));
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* On WebSphere 8.5 a URL like /context-root/a/b;%2f1/c can bypass a rule on /a/b/c
|
* On WebSphere 8.5 a URL like /context-root/a/b;%2f1/c can bypass a rule on /a/b/c
|
||||||
* because the pathInfo is /a/b;/1/c which ends up being /a/b/1/c while Spring MVC
|
* because the pathInfo is /a/b;/1/c which ends up being /a/b/1/c while Spring MVC
|
||||||
|
|
Loading…
Reference in New Issue