Merge remote-tracking branch 'origin/jetty-9.4.x' into jetty-10.0.x
This commit is contained in:
commit
de76d44c97
|
@ -36,7 +36,7 @@ public class RoleInfo
|
||||||
/**
|
/**
|
||||||
* List of permitted roles
|
* List of permitted roles
|
||||||
*/
|
*/
|
||||||
private final Set<String> _roles = new CopyOnWriteArraySet<String>();
|
private final Set<String> _roles = new CopyOnWriteArraySet<>();
|
||||||
|
|
||||||
public RoleInfo()
|
public RoleInfo()
|
||||||
{
|
{
|
||||||
|
@ -135,26 +135,28 @@ public class RoleInfo
|
||||||
{
|
{
|
||||||
if (other._forbidden)
|
if (other._forbidden)
|
||||||
setForbidden(true);
|
setForbidden(true);
|
||||||
else if (!other._checked) // TODO is this the right way around???
|
else if (other._checked)
|
||||||
|
{
|
||||||
setChecked(true);
|
setChecked(true);
|
||||||
else if (other._isAnyRole)
|
if (other._isAnyAuth)
|
||||||
setAnyRole(true);
|
|
||||||
else if (other._isAnyAuth)
|
|
||||||
setAnyAuth(true);
|
setAnyAuth(true);
|
||||||
else if (!_isAnyRole)
|
if (other._isAnyRole)
|
||||||
{
|
setAnyRole(true);
|
||||||
for (String r : other._roles)
|
|
||||||
{
|
|
||||||
_roles.add(r);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
_roles.addAll(other._roles);
|
||||||
|
}
|
||||||
setUserDataConstraint(other._userDataConstraint);
|
setUserDataConstraint(other._userDataConstraint);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String toString()
|
public String toString()
|
||||||
{
|
{
|
||||||
return "{RoleInfo" + (_forbidden ? ",F" : "") + (_checked ? ",C" : "") + (_isAnyRole ? ",*" : _roles) + (_userDataConstraint != null ? "," + _userDataConstraint : "") + "}";
|
return String.format("RoleInfo@%x{%s%s%s%s,%s}",
|
||||||
|
hashCode(),
|
||||||
|
(_forbidden ? "Forbidden," : ""),
|
||||||
|
(_checked ? "Checked," : ""),
|
||||||
|
(_isAnyAuth ? "AnyAuth," : ""),
|
||||||
|
(_isAnyRole ? "*" : _roles),
|
||||||
|
_userDataConstraint);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,6 +26,7 @@ import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
import java.util.function.Consumer;
|
import java.util.function.Consumer;
|
||||||
|
import java.util.logging.Logger;
|
||||||
import java.util.regex.Matcher;
|
import java.util.regex.Matcher;
|
||||||
import java.util.regex.Pattern;
|
import java.util.regex.Pattern;
|
||||||
import java.util.stream.Stream;
|
import java.util.stream.Stream;
|
||||||
|
@ -69,6 +70,7 @@ import org.junit.jupiter.params.provider.MethodSource;
|
||||||
|
|
||||||
import static java.nio.charset.StandardCharsets.ISO_8859_1;
|
import static java.nio.charset.StandardCharsets.ISO_8859_1;
|
||||||
import static org.hamcrest.MatcherAssert.assertThat;
|
import static org.hamcrest.MatcherAssert.assertThat;
|
||||||
|
import static org.hamcrest.Matchers.contains;
|
||||||
import static org.hamcrest.Matchers.containsString;
|
import static org.hamcrest.Matchers.containsString;
|
||||||
import static org.hamcrest.Matchers.in;
|
import static org.hamcrest.Matchers.in;
|
||||||
import static org.hamcrest.Matchers.is;
|
import static org.hamcrest.Matchers.is;
|
||||||
|
@ -87,9 +89,17 @@ public class ConstraintTest
|
||||||
private LocalConnector _connector;
|
private LocalConnector _connector;
|
||||||
private ConstraintSecurityHandler _security;
|
private ConstraintSecurityHandler _security;
|
||||||
private HttpConfiguration _config;
|
private HttpConfiguration _config;
|
||||||
|
private Constraint _forbidConstraint;
|
||||||
|
private Constraint _authAnyRoleConstraint;
|
||||||
|
private Constraint _authAdminConstraint;
|
||||||
|
private Constraint _relaxConstraint;
|
||||||
|
private Constraint _loginPageConstraint;
|
||||||
|
private Constraint _noAuthConstraint;
|
||||||
|
private Constraint _confidentialDataConstraint;
|
||||||
|
private Constraint _anyUserAuthConstraint;
|
||||||
|
|
||||||
@BeforeEach
|
@BeforeEach
|
||||||
public void startServer()
|
public void setupServer()
|
||||||
{
|
{
|
||||||
_server = new Server();
|
_server = new Server();
|
||||||
_connector = new LocalConnector(_server);
|
_connector = new LocalConnector(_server);
|
||||||
|
@ -138,98 +148,80 @@ public class ConstraintTest
|
||||||
|
|
||||||
private List<ConstraintMapping> getConstraintMappings()
|
private List<ConstraintMapping> getConstraintMappings()
|
||||||
{
|
{
|
||||||
Constraint constraint0 = new Constraint();
|
_forbidConstraint = new Constraint();
|
||||||
constraint0.setAuthenticate(true);
|
_forbidConstraint.setAuthenticate(true);
|
||||||
constraint0.setName("forbid");
|
_forbidConstraint.setName("forbid");
|
||||||
ConstraintMapping mapping0 = new ConstraintMapping();
|
ConstraintMapping mapping0 = new ConstraintMapping();
|
||||||
mapping0.setPathSpec("/forbid/*");
|
mapping0.setPathSpec("/forbid/*");
|
||||||
mapping0.setConstraint(constraint0);
|
mapping0.setConstraint(_forbidConstraint);
|
||||||
|
|
||||||
Constraint constraint1 = new Constraint();
|
_authAnyRoleConstraint = new Constraint();
|
||||||
constraint1.setAuthenticate(true);
|
_authAnyRoleConstraint.setAuthenticate(true);
|
||||||
constraint1.setName("auth");
|
_authAnyRoleConstraint.setName("auth");
|
||||||
constraint1.setRoles(new String[]{Constraint.ANY_ROLE});
|
_authAnyRoleConstraint.setRoles(new String[]{Constraint.ANY_ROLE});
|
||||||
ConstraintMapping mapping1 = new ConstraintMapping();
|
ConstraintMapping mapping1 = new ConstraintMapping();
|
||||||
mapping1.setPathSpec("/auth/*");
|
mapping1.setPathSpec("/auth/*");
|
||||||
mapping1.setConstraint(constraint1);
|
mapping1.setConstraint(_authAnyRoleConstraint);
|
||||||
|
|
||||||
Constraint constraint2 = new Constraint();
|
_authAdminConstraint = new Constraint();
|
||||||
constraint2.setAuthenticate(true);
|
_authAdminConstraint.setAuthenticate(true);
|
||||||
constraint2.setName("admin");
|
_authAdminConstraint.setName("admin");
|
||||||
constraint2.setRoles(new String[]{"administrator"});
|
_authAdminConstraint.setRoles(new String[]{"administrator"});
|
||||||
ConstraintMapping mapping2 = new ConstraintMapping();
|
ConstraintMapping mapping2 = new ConstraintMapping();
|
||||||
mapping2.setPathSpec("/admin/*");
|
mapping2.setPathSpec("/admin/*");
|
||||||
mapping2.setConstraint(constraint2);
|
mapping2.setConstraint(_authAdminConstraint);
|
||||||
mapping2.setMethod("GET");
|
mapping2.setMethod("GET");
|
||||||
|
ConstraintMapping mapping2o = new ConstraintMapping();
|
||||||
|
mapping2o.setPathSpec("/admin/*");
|
||||||
|
mapping2o.setConstraint(_forbidConstraint);
|
||||||
|
mapping2o.setMethodOmissions(new String[]{"GET"});
|
||||||
|
|
||||||
Constraint constraint3 = new Constraint();
|
_relaxConstraint = new Constraint();
|
||||||
constraint3.setAuthenticate(false);
|
_relaxConstraint.setAuthenticate(false);
|
||||||
constraint3.setName("relax");
|
_relaxConstraint.setName("relax");
|
||||||
ConstraintMapping mapping3 = new ConstraintMapping();
|
ConstraintMapping mapping3 = new ConstraintMapping();
|
||||||
mapping3.setPathSpec("/admin/relax/*");
|
mapping3.setPathSpec("/admin/relax/*");
|
||||||
mapping3.setConstraint(constraint3);
|
mapping3.setConstraint(_relaxConstraint);
|
||||||
|
|
||||||
Constraint constraint4 = new Constraint();
|
_loginPageConstraint = new Constraint();
|
||||||
constraint4.setAuthenticate(true);
|
_loginPageConstraint.setAuthenticate(true);
|
||||||
constraint4.setName("loginpage");
|
_loginPageConstraint.setName("loginpage");
|
||||||
constraint4.setRoles(new String[]{"administrator"});
|
_loginPageConstraint.setRoles(new String[]{"administrator"});
|
||||||
ConstraintMapping mapping4 = new ConstraintMapping();
|
ConstraintMapping mapping4 = new ConstraintMapping();
|
||||||
mapping4.setPathSpec("/testLoginPage");
|
mapping4.setPathSpec("/testLoginPage");
|
||||||
mapping4.setConstraint(constraint4);
|
mapping4.setConstraint(_loginPageConstraint);
|
||||||
|
|
||||||
Constraint constraint5 = new Constraint();
|
_noAuthConstraint = new Constraint();
|
||||||
constraint5.setAuthenticate(false);
|
_noAuthConstraint.setAuthenticate(false);
|
||||||
constraint5.setName("allow forbidden POST");
|
_noAuthConstraint.setName("allow forbidden");
|
||||||
ConstraintMapping mapping5 = new ConstraintMapping();
|
ConstraintMapping mapping5 = new ConstraintMapping();
|
||||||
mapping5.setPathSpec("/forbid/post");
|
mapping5.setPathSpec("/forbid/post");
|
||||||
mapping5.setConstraint(constraint5);
|
mapping5.setConstraint(_noAuthConstraint);
|
||||||
mapping5.setMethod("POST");
|
mapping5.setMethod("POST");
|
||||||
|
ConstraintMapping mapping5o = new ConstraintMapping();
|
||||||
|
mapping5o.setPathSpec("/forbid/post");
|
||||||
|
mapping5o.setConstraint(_forbidConstraint);
|
||||||
|
mapping5o.setMethodOmissions(new String[]{"POST"});
|
||||||
|
|
||||||
Constraint constraint6 = new Constraint();
|
_confidentialDataConstraint = new Constraint();
|
||||||
constraint6.setAuthenticate(false);
|
_confidentialDataConstraint.setAuthenticate(false);
|
||||||
constraint6.setName("data constraint");
|
_confidentialDataConstraint.setName("data constraint");
|
||||||
constraint6.setDataConstraint(2);
|
_confidentialDataConstraint.setDataConstraint(Constraint.DC_CONFIDENTIAL);
|
||||||
ConstraintMapping mapping6 = new ConstraintMapping();
|
ConstraintMapping mapping6 = new ConstraintMapping();
|
||||||
mapping6.setPathSpec("/data/*");
|
mapping6.setPathSpec("/data/*");
|
||||||
mapping6.setConstraint(constraint6);
|
mapping6.setConstraint(_confidentialDataConstraint);
|
||||||
|
|
||||||
Constraint constraint7 = new Constraint();
|
_anyUserAuthConstraint = new Constraint();
|
||||||
constraint7.setAuthenticate(true);
|
_anyUserAuthConstraint.setAuthenticate(true);
|
||||||
constraint7.setName("** constraint");
|
_anyUserAuthConstraint.setName("** constraint");
|
||||||
constraint7.setRoles(new String[]{
|
_anyUserAuthConstraint.setRoles(new String[]{
|
||||||
Constraint.ANY_AUTH, "user"
|
Constraint.ANY_AUTH, "user"
|
||||||
}); //the "user" role is superfluous once ** has been defined
|
}); //the "user" role is superfluous once ** has been defined
|
||||||
ConstraintMapping mapping7 = new ConstraintMapping();
|
ConstraintMapping mapping7 = new ConstraintMapping();
|
||||||
mapping7.setPathSpec("/starstar/*");
|
mapping7.setPathSpec("/starstar/*");
|
||||||
mapping7.setConstraint(constraint7);
|
mapping7.setConstraint(_anyUserAuthConstraint);
|
||||||
|
|
||||||
return Arrays.asList(mapping0, mapping1, mapping2, mapping3, mapping4, mapping5, mapping6, mapping7);
|
return Arrays.asList(mapping0, mapping1, mapping2, mapping2o, mapping3, mapping4, mapping5, mapping5o, mapping6, mapping7);
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void testConstraints() throws Exception
|
|
||||||
{
|
|
||||||
List<ConstraintMapping> mappings = new ArrayList<>(_security.getConstraintMappings());
|
|
||||||
|
|
||||||
assertTrue(mappings.get(0).getConstraint().isForbidden());
|
|
||||||
assertFalse(mappings.get(1).getConstraint().isForbidden());
|
|
||||||
assertFalse(mappings.get(2).getConstraint().isForbidden());
|
|
||||||
assertFalse(mappings.get(3).getConstraint().isForbidden());
|
|
||||||
|
|
||||||
assertFalse(mappings.get(0).getConstraint().isAnyRole());
|
|
||||||
assertTrue(mappings.get(1).getConstraint().isAnyRole());
|
|
||||||
assertFalse(mappings.get(2).getConstraint().isAnyRole());
|
|
||||||
assertFalse(mappings.get(3).getConstraint().isAnyRole());
|
|
||||||
|
|
||||||
assertFalse(mappings.get(0).getConstraint().hasRole("administrator"));
|
|
||||||
assertTrue(mappings.get(1).getConstraint().hasRole("administrator"));
|
|
||||||
assertTrue(mappings.get(2).getConstraint().hasRole("administrator"));
|
|
||||||
assertFalse(mappings.get(3).getConstraint().hasRole("administrator"));
|
|
||||||
|
|
||||||
assertTrue(mappings.get(0).getConstraint().getAuthenticate());
|
|
||||||
assertTrue(mappings.get(1).getConstraint().getAuthenticate());
|
|
||||||
assertTrue(mappings.get(2).getConstraint().getAuthenticate());
|
|
||||||
assertFalse(mappings.get(3).getConstraint().getAuthenticate());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1803,9 +1795,80 @@ public class ConstraintTest
|
||||||
assertThat(response, startsWith("HTTP/1.1 200 "));
|
assertThat(response, startsWith("HTTP/1.1 200 "));
|
||||||
|
|
||||||
response = _connector.getResponse("GET /ctx/forbid/post HTTP/1.0\r\n\r\n");
|
response = _connector.getResponse("GET /ctx/forbid/post HTTP/1.0\r\n\r\n");
|
||||||
|
assertThat(response, startsWith("HTTP/1.1 403 "));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testUncoveredMethod() throws Exception
|
||||||
|
{
|
||||||
|
ConstraintMapping specificMethod = new ConstraintMapping();
|
||||||
|
specificMethod.setMethod("GET");
|
||||||
|
specificMethod.setPathSpec("/specific/method");
|
||||||
|
specificMethod.setConstraint(_forbidConstraint);
|
||||||
|
_security.addConstraintMapping(specificMethod);
|
||||||
|
_security.setAuthenticator(new BasicAuthenticator());
|
||||||
|
Logger.getAnonymousLogger().info("Uncovered method for /specific/method is expected");
|
||||||
|
_server.start();
|
||||||
|
|
||||||
|
assertThat(_security.getPathsWithUncoveredHttpMethods(), contains("/specific/method"));
|
||||||
|
|
||||||
|
String response;
|
||||||
|
response = _connector.getResponse("GET /ctx/specific/method HTTP/1.0\r\n\r\n");
|
||||||
|
assertThat(response, startsWith("HTTP/1.1 403 "));
|
||||||
|
|
||||||
|
response = _connector.getResponse("POST /ctx/specific/method HTTP/1.0\r\n\r\n");
|
||||||
assertThat(response, startsWith("HTTP/1.1 200 ")); // This is so stupid, but it is the S P E C
|
assertThat(response, startsWith("HTTP/1.1 200 ")); // This is so stupid, but it is the S P E C
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testForbidTraceAndOptions() throws Exception
|
||||||
|
{
|
||||||
|
ConstraintMapping forbidTrace = new ConstraintMapping();
|
||||||
|
forbidTrace.setMethod("TRACE");
|
||||||
|
forbidTrace.setPathSpec("/");
|
||||||
|
forbidTrace.setConstraint(_forbidConstraint);
|
||||||
|
ConstraintMapping allowOmitTrace = new ConstraintMapping();
|
||||||
|
allowOmitTrace.setMethodOmissions(new String[] {"TRACE"});
|
||||||
|
allowOmitTrace.setPathSpec("/");
|
||||||
|
allowOmitTrace.setConstraint(_relaxConstraint);
|
||||||
|
|
||||||
|
ConstraintMapping forbidOptions = new ConstraintMapping();
|
||||||
|
forbidOptions.setMethod("OPTIONS");
|
||||||
|
forbidOptions.setPathSpec("/");
|
||||||
|
forbidOptions.setConstraint(_forbidConstraint);
|
||||||
|
ConstraintMapping allowOmitOptions = new ConstraintMapping();
|
||||||
|
allowOmitOptions.setMethodOmissions(new String[] {"OPTIONS"});
|
||||||
|
allowOmitOptions.setPathSpec("/");
|
||||||
|
allowOmitOptions.setConstraint(_relaxConstraint);
|
||||||
|
|
||||||
|
ConstraintMapping someConstraint = new ConstraintMapping();
|
||||||
|
someConstraint.setPathSpec("/some/constaint/*");
|
||||||
|
someConstraint.setConstraint(_noAuthConstraint);
|
||||||
|
|
||||||
|
_security.setConstraintMappings(new ConstraintMapping[] {forbidTrace, allowOmitTrace, forbidOptions, allowOmitOptions, someConstraint});
|
||||||
|
|
||||||
|
_security.setAuthenticator(new BasicAuthenticator());
|
||||||
|
_server.start();
|
||||||
|
|
||||||
|
assertThat(_security.getPathsWithUncoveredHttpMethods(), Matchers.empty());
|
||||||
|
|
||||||
|
String response;
|
||||||
|
response = _connector.getResponse("TRACE /ctx/some/path HTTP/1.0\r\n\r\n");
|
||||||
|
assertThat(response, startsWith("HTTP/1.1 403 "));
|
||||||
|
|
||||||
|
response = _connector.getResponse("OPTIONS /ctx/some/path HTTP/1.0\r\n\r\n");
|
||||||
|
assertThat(response, startsWith("HTTP/1.1 403 "));
|
||||||
|
|
||||||
|
response = _connector.getResponse("GET /ctx/some/path HTTP/1.0\r\n\r\n");
|
||||||
|
assertThat(response, startsWith("HTTP/1.1 200 "));
|
||||||
|
|
||||||
|
response = _connector.getResponse("GET /ctx/some/constraint/info HTTP/1.0\r\n\r\n");
|
||||||
|
assertThat(response, startsWith("HTTP/1.1 200 "));
|
||||||
|
|
||||||
|
response = _connector.getResponse("OPTIONS /ctx/some/constraint/info HTTP/1.0\r\n\r\n");
|
||||||
|
assertThat(response, startsWith("HTTP/1.1 403 "));
|
||||||
|
}
|
||||||
|
|
||||||
private static String authBase64(String authorization)
|
private static String authBase64(String authorization)
|
||||||
{
|
{
|
||||||
byte[] raw = authorization.getBytes(ISO_8859_1);
|
byte[] raw = authorization.getBytes(ISO_8859_1);
|
||||||
|
|
|
@ -684,7 +684,7 @@ public class ResourceService
|
||||||
putHeaders(response, content, Response.USE_KNOWN_CONTENT_LENGTH);
|
putHeaders(response, content, Response.USE_KNOWN_CONTENT_LENGTH);
|
||||||
|
|
||||||
// write the content asynchronously if supported
|
// write the content asynchronously if supported
|
||||||
if (request.isAsyncSupported() && content.getContentLengthValue() > response.getBufferSize())
|
if (request.isAsyncSupported())
|
||||||
{
|
{
|
||||||
final AsyncContext context = request.startAsync();
|
final AsyncContext context = request.startAsync();
|
||||||
context.setTimeout(0);
|
context.setTimeout(0);
|
||||||
|
|
Loading…
Reference in New Issue