and() {
}
}
+
+ /**
+ * Allows configuration for Feature
+ * Policy .
+ *
+ * Calling this method automatically enables (includes) the {@code Feature-Policy}
+ * header in the response using the supplied policy directive(s).
+ *
+ * Configuration is provided to the {@link FeaturePolicyHeaderWriter} which is
+ * responsible for writing the header.
+ *
+ * @see FeaturePolicyHeaderWriter
+ * @since 5.1
+ * @return the {@link FeaturePolicyHeaderWriter} for additional configuration
+ * @throws IllegalArgumentException if policyDirectives is {@code null} or empty
+ */
+ public FeaturePolicyConfig featurePolicy(String policyDirectives) {
+ this.featurePolicy.writer = new FeaturePolicyHeaderWriter(policyDirectives);
+ return featurePolicy;
+ }
+
+ public final class FeaturePolicyConfig {
+
+ private FeaturePolicyHeaderWriter writer;
+
+ private FeaturePolicyConfig() {
+ }
+
+ /**
+ * Allows completing configuration of Feature Policy and continuing configuration
+ * of headers.
+ *
+ * @return the {@link HeadersConfigurer} for additional configuration
+ */
+ public HeadersConfigurer and() {
+ return HeadersConfigurer.this;
+ }
+
+ }
+
}
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/PortMapperConfigurer.java b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/PortMapperConfigurer.java
index 262bb1436c7..0a64f0e5ab8 100644
--- a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/PortMapperConfigurer.java
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/PortMapperConfigurer.java
@@ -45,7 +45,7 @@ public PortMapperConfigurer() {
/**
* Allows specifying the {@link PortMapper} instance.
* @param portMapper
- * @return
+ * @return the {@link PortMapperConfigurer} for further customizations
*/
public PortMapperConfigurer portMapper(PortMapper portMapper) {
this.portMapper = portMapper;
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/RequestCacheConfigurer.java b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/RequestCacheConfigurer.java
index 628b9eca21d..4cc62c887b3 100644
--- a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/RequestCacheConfigurer.java
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/RequestCacheConfigurer.java
@@ -19,6 +19,8 @@
import java.util.Collections;
import java.util.List;
+import org.springframework.beans.factory.NoSuchBeanDefinitionException;
+import org.springframework.context.ApplicationContext;
import org.springframework.http.MediaType;
import org.springframework.security.config.annotation.web.HttpSecurityBuilder;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
@@ -113,11 +115,26 @@ private RequestCache getRequestCache(H http) {
if (result != null) {
return result;
}
+ result = getBeanOrNull(RequestCache.class);
+ if (result != null) {
+ return result;
+ }
HttpSessionRequestCache defaultCache = new HttpSessionRequestCache();
defaultCache.setRequestMatcher(createDefaultSavedRequestMatcher(http));
return defaultCache;
}
+ private T getBeanOrNull(Class type) {
+ ApplicationContext context = getBuilder().getSharedObject(ApplicationContext.class);
+ if (context == null) {
+ return null;
+ }
+ try {
+ return context.getBean(type);
+ } catch (NoSuchBeanDefinitionException e) {
+ return null;
+ }
+ }
@SuppressWarnings("unchecked")
private RequestMatcher createDefaultSavedRequestMatcher(H http) {
ContentNegotiationStrategy contentNegotiationStrategy = http
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/SessionManagementConfigurer.java b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/SessionManagementConfigurer.java
index fd814edc7d7..43ea86d7f01 100644
--- a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/SessionManagementConfigurer.java
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/SessionManagementConfigurer.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2002-2016 the original author or authors.
+ * Copyright 2002-2018 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
@@ -18,7 +18,6 @@
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
-
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
@@ -105,7 +104,7 @@ public final class SessionManagementConfigurer>
private Integer maximumSessions;
private String expiredUrl;
private boolean maxSessionsPreventsLogin;
- private SessionCreationPolicy sessionPolicy = SessionCreationPolicy.IF_REQUIRED;
+ private SessionCreationPolicy sessionPolicy;
private boolean enableSessionUrlRewriting;
private String invalidSessionUrl;
private String sessionAuthenticationErrorUrl;
@@ -549,7 +548,14 @@ AuthenticationFailureHandler getSessionAuthenticationFailureHandler() {
* @return the {@link SessionCreationPolicy}
*/
SessionCreationPolicy getSessionCreationPolicy() {
- return this.sessionPolicy;
+ if (this.sessionPolicy != null) {
+ return this.sessionPolicy;
+ }
+
+ SessionCreationPolicy sessionPolicy =
+ getBuilder().getSharedObject(SessionCreationPolicy.class);
+ return sessionPolicy == null ?
+ SessionCreationPolicy.IF_REQUIRED : sessionPolicy;
}
/**
@@ -558,8 +564,9 @@ SessionCreationPolicy getSessionCreationPolicy() {
* @return true if the {@link SessionCreationPolicy} allows session creation
*/
private boolean isAllowSessionCreation() {
- return SessionCreationPolicy.ALWAYS == this.sessionPolicy
- || SessionCreationPolicy.IF_REQUIRED == this.sessionPolicy;
+ SessionCreationPolicy sessionPolicy = getSessionCreationPolicy();
+ return SessionCreationPolicy.ALWAYS == sessionPolicy
+ || SessionCreationPolicy.IF_REQUIRED == sessionPolicy;
}
/**
@@ -567,7 +574,8 @@ private boolean isAllowSessionCreation() {
* @return
*/
private boolean isStateless() {
- return SessionCreationPolicy.STATELESS == this.sessionPolicy;
+ SessionCreationPolicy sessionPolicy = getSessionCreationPolicy();
+ return SessionCreationPolicy.STATELESS == sessionPolicy;
}
/**
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/UrlAuthorizationConfigurer.java b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/UrlAuthorizationConfigurer.java
index edd17b07119..88ac8eb547d 100644
--- a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/UrlAuthorizationConfigurer.java
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/UrlAuthorizationConfigurer.java
@@ -99,7 +99,7 @@ public UrlAuthorizationConfigurer(ApplicationContext context) {
* The StandardInterceptUrlRegistry is what users will interact with after applying
* the {@link UrlAuthorizationConfigurer}.
*
- * @return
+ * @return the {@link ExpressionUrlAuthorizationConfigurer} for further customizations
*/
public StandardInterceptUrlRegistry getRegistry() {
return REGISTRY;
@@ -198,7 +198,7 @@ FilterInvocationSecurityMetadataSource createMetadataSource(H http) {
* provided {@link ConfigAttribute} instances
* @param configAttributes the {@link ConfigAttribute} instances that should be mapped
* by the {@link RequestMatcher} instances
- * @return the {@link UrlAuthorizationConfigurer} for further customizations
+ * @return the {@link ExpressionUrlAuthorizationConfigurer} for further customizations
*/
private StandardInterceptUrlRegistry addMapping(
Iterable extends RequestMatcher> requestMatchers,
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/OAuth2Configurer.java b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/OAuth2Configurer.java
index 60b5e498558..433119b453c 100644
--- a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/OAuth2Configurer.java
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/OAuth2Configurer.java
@@ -15,9 +15,14 @@
*/
package org.springframework.security.config.annotation.web.configurers.oauth2;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.context.ApplicationContext;
+import org.springframework.security.config.annotation.ObjectPostProcessor;
+import org.springframework.security.config.annotation.web.HttpSecurityBuilder;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.config.annotation.web.configurers.AbstractHttpConfigurer;
import org.springframework.security.config.annotation.web.configurers.oauth2.client.OAuth2ClientConfigurer;
+import org.springframework.security.config.annotation.web.configurers.oauth2.server.resource.OAuth2ResourceServerConfigurer;
/**
* An {@link AbstractHttpConfigurer} that provides support for the
@@ -29,24 +34,72 @@
* @see OAuth2ClientConfigurer
* @see AbstractHttpConfigurer
*/
-public final class OAuth2Configurer extends AbstractHttpConfigurer {
+public final class OAuth2Configurer>
+ extends AbstractHttpConfigurer, B> {
+
+ @Autowired
+ private ObjectPostProcessor objectPostProcessor;
+
+ private OAuth2ClientConfigurer clientConfigurer;
+
+ private OAuth2ResourceServerConfigurer resourceServerConfigurer;
/**
* Returns the {@link OAuth2ClientConfigurer} for configuring OAuth 2.0 Client support.
*
* @return the {@link OAuth2ClientConfigurer}
- * @throws Exception
*/
- public OAuth2ClientConfigurer client() throws Exception {
- return this.getOrApply(new OAuth2ClientConfigurer<>());
+ public OAuth2ClientConfigurer client() {
+ if (this.clientConfigurer == null) {
+ this.initClientConfigurer();
+ }
+ return this.clientConfigurer;
}
- @SuppressWarnings("unchecked")
- private > C getOrApply(C configurer) throws Exception {
- C existingConfigurer = (C) this.getBuilder().getConfigurer(configurer.getClass());
- if (existingConfigurer != null) {
- return existingConfigurer;
+ /**
+ * Returns the {@link OAuth2ResourceServerConfigurer} for configuring OAuth 2.0 Resource Server support.
+ *
+ * @return the {@link OAuth2ResourceServerConfigurer}
+ */
+ public OAuth2ResourceServerConfigurer resourceServer() {
+ if (this.resourceServerConfigurer == null) {
+ this.initResourceServerConfigurer();
}
- return this.getBuilder().apply(configurer);
+ return this.resourceServerConfigurer;
+ }
+
+ @Override
+ public void init(B builder) throws Exception {
+ if (this.clientConfigurer != null) {
+ this.clientConfigurer.init(builder);
+ }
+
+ if (this.resourceServerConfigurer != null) {
+ this.resourceServerConfigurer.init(builder);
+ }
+ }
+
+ @Override
+ public void configure(B builder) throws Exception {
+ if (this.clientConfigurer != null) {
+ this.clientConfigurer.configure(builder);
+ }
+
+ if (this.resourceServerConfigurer != null) {
+ this.resourceServerConfigurer.configure(builder);
+ }
+ }
+
+ private void initClientConfigurer() {
+ this.clientConfigurer = new OAuth2ClientConfigurer<>();
+ this.clientConfigurer.setBuilder(this.getBuilder());
+ this.clientConfigurer.addObjectPostProcessor(this.objectPostProcessor);
+ }
+
+ private void initResourceServerConfigurer() {
+ ApplicationContext context = getBuilder().getSharedObject(ApplicationContext.class);
+ this.resourceServerConfigurer = new OAuth2ResourceServerConfigurer<>(context);
+ this.resourceServerConfigurer.setBuilder(this.getBuilder());
+ this.resourceServerConfigurer.addObjectPostProcessor(this.objectPostProcessor);
}
}
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/client/OAuth2ClientConfigurer.java b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/client/OAuth2ClientConfigurer.java
index e7e6b47a80b..5ff7e2975c6 100644
--- a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/client/OAuth2ClientConfigurer.java
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/client/OAuth2ClientConfigurer.java
@@ -24,9 +24,13 @@
import org.springframework.security.oauth2.client.endpoint.OAuth2AccessTokenResponseClient;
import org.springframework.security.oauth2.client.endpoint.OAuth2AuthorizationCodeGrantRequest;
import org.springframework.security.oauth2.client.registration.ClientRegistrationRepository;
+import org.springframework.security.oauth2.client.web.AuthenticatedPrincipalOAuth2AuthorizedClientRepository;
import org.springframework.security.oauth2.client.web.AuthorizationRequestRepository;
+import org.springframework.security.oauth2.client.web.DefaultOAuth2AuthorizationRequestResolver;
import org.springframework.security.oauth2.client.web.OAuth2AuthorizationCodeGrantFilter;
import org.springframework.security.oauth2.client.web.OAuth2AuthorizationRequestRedirectFilter;
+import org.springframework.security.oauth2.client.web.OAuth2AuthorizationRequestResolver;
+import org.springframework.security.oauth2.client.web.OAuth2AuthorizedClientRepository;
import org.springframework.security.oauth2.core.endpoint.OAuth2AuthorizationRequest;
import org.springframework.security.web.savedrequest.RequestCache;
import org.springframework.util.Assert;
@@ -61,7 +65,7 @@
*
*
* {@link ClientRegistrationRepository} (required)
- * {@link OAuth2AuthorizedClientService} (optional)
+ * {@link OAuth2AuthorizedClientRepository} (optional)
*
*
* Shared Objects Used
@@ -70,7 +74,7 @@
*
*
* {@link ClientRegistrationRepository}
- * {@link OAuth2AuthorizedClientService}
+ * {@link OAuth2AuthorizedClientRepository}
*
*
* @author Joe Grandja
@@ -78,7 +82,7 @@
* @see OAuth2AuthorizationRequestRedirectFilter
* @see OAuth2AuthorizationCodeGrantFilter
* @see ClientRegistrationRepository
- * @see OAuth2AuthorizedClientService
+ * @see OAuth2AuthorizedClientRepository
* @see AbstractHttpConfigurer
*/
public final class OAuth2ClientConfigurer> extends
@@ -98,6 +102,18 @@ public OAuth2ClientConfigurer clientRegistrationRepository(ClientRegistration
return this;
}
+ /**
+ * Sets the repository for authorized client(s).
+ *
+ * @param authorizedClientRepository the authorized client repository
+ * @return the {@link OAuth2ClientConfigurer} for further configuration
+ */
+ public OAuth2ClientConfigurer authorizedClientRepository(OAuth2AuthorizedClientRepository authorizedClientRepository) {
+ Assert.notNull(authorizedClientRepository, "authorizedClientRepository cannot be null");
+ this.getBuilder().setSharedObject(OAuth2AuthorizedClientRepository.class, authorizedClientRepository);
+ return this;
+ }
+
/**
* Sets the service for authorized client(s).
*
@@ -106,7 +122,7 @@ public OAuth2ClientConfigurer clientRegistrationRepository(ClientRegistration
*/
public OAuth2ClientConfigurer authorizedClientService(OAuth2AuthorizedClientService authorizedClientService) {
Assert.notNull(authorizedClientService, "authorizedClientService cannot be null");
- this.getBuilder().setSharedObject(OAuth2AuthorizedClientService.class, authorizedClientService);
+ this.authorizedClientRepository(new AuthenticatedPrincipalOAuth2AuthorizedClientRepository(authorizedClientService));
return this;
}
@@ -145,7 +161,8 @@ public AuthorizationEndpointConfig authorizationEndpoint() {
* Configuration options for the Authorization Server's Authorization Endpoint.
*/
public class AuthorizationEndpointConfig {
- private String authorizationRequestBaseUri;
+ private String authorizationRequestBaseUri = OAuth2AuthorizationRequestRedirectFilter.DEFAULT_AUTHORIZATION_REQUEST_BASE_URI;
+ private OAuth2AuthorizationRequestResolver authorizationRequestResolver;
private AuthorizationRequestRepository authorizationRequestRepository;
private AuthorizationEndpointConfig() {
@@ -163,6 +180,18 @@ public AuthorizationEndpointConfig baseUri(String authorizationRequestBaseUri) {
return this;
}
+ /**
+ * Sets the resolver used for resolving {@link OAuth2AuthorizationRequest}'s.
+ *
+ * @param authorizationRequestResolver the resolver used for resolving {@link OAuth2AuthorizationRequest}'s
+ * @return the {@link AuthorizationEndpointConfig} for further configuration
+ */
+ public AuthorizationEndpointConfig authorizationRequestResolver(OAuth2AuthorizationRequestResolver authorizationRequestResolver) {
+ Assert.notNull(authorizationRequestResolver, "authorizationRequestResolver cannot be null");
+ this.authorizationRequestResolver = authorizationRequestResolver;
+ return this;
+ }
+
/**
* Sets the repository used for storing {@link OAuth2AuthorizationRequest}'s.
*
@@ -185,6 +214,52 @@ public AuthorizationEndpointConfig authorizationRequestRepository(
public AuthorizationCodeGrantConfigurer and() {
return AuthorizationCodeGrantConfigurer.this;
}
+
+ private OAuth2AuthorizationRequestResolver getAuthorizationRequestResolver() {
+ if (this.authorizationRequestResolver != null) {
+ return this.authorizationRequestResolver;
+ }
+ ClientRegistrationRepository clientRegistrationRepository = OAuth2ClientConfigurerUtils
+ .getClientRegistrationRepository(getBuilder());
+ return new DefaultOAuth2AuthorizationRequestResolver(clientRegistrationRepository,
+ this.authorizationRequestBaseUri);
+ }
+
+ private OAuth2AuthorizationRequestRedirectFilter createAuthorizationRequestRedirectFilter(B builder) {
+ OAuth2AuthorizationRequestResolver resolver = getAuthorizationRequestResolver();
+ OAuth2AuthorizationRequestRedirectFilter authorizationRequestFilter = new OAuth2AuthorizationRequestRedirectFilter(resolver);
+
+ if (this.authorizationRequestRepository != null) {
+ authorizationRequestFilter.setAuthorizationRequestRepository(
+ this.authorizationRequestRepository);
+ }
+ RequestCache requestCache = builder.getSharedObject(RequestCache.class);
+ if (requestCache != null) {
+ authorizationRequestFilter.setRequestCache(requestCache);
+ }
+ return authorizationRequestFilter;
+ }
+
+ private OAuth2AuthorizationCodeGrantFilter createAuthorizationCodeGrantFilter(B builder) {
+ AuthenticationManager authenticationManager = builder.getSharedObject(AuthenticationManager.class);
+ OAuth2AuthorizationCodeGrantFilter authorizationCodeGrantFilter = new OAuth2AuthorizationCodeGrantFilter(
+ OAuth2ClientConfigurerUtils.getClientRegistrationRepository(builder),
+ OAuth2ClientConfigurerUtils.getAuthorizedClientRepository(builder),
+ authenticationManager);
+
+ if (this.authorizationRequestRepository != null) {
+ authorizationCodeGrantFilter.setAuthorizationRequestRepository(
+ this.authorizationRequestRepository);
+ }
+ return authorizationCodeGrantFilter;
+ }
+
+ private void configure(B builder) {
+ OAuth2AuthorizationRequestRedirectFilter authorizationRequestFilter = createAuthorizationRequestRedirectFilter(builder);
+ builder.addFilter(postProcess(authorizationRequestFilter));
+ OAuth2AuthorizationCodeGrantFilter authorizationCodeGrantFilter = createAuthorizationCodeGrantFilter(builder);
+ builder.addFilter(postProcess(authorizationCodeGrantFilter));
+ }
}
/**
@@ -237,6 +312,10 @@ public AuthorizationCodeGrantConfigurer and() {
public OAuth2ClientConfigurer and() {
return OAuth2ClientConfigurer.this;
}
+
+ private void configure(B builder) {
+ this.authorizationEndpointConfig.configure(builder);
+ }
}
@Override
@@ -249,7 +328,7 @@ public void init(B builder) throws Exception {
@Override
public void configure(B builder) throws Exception {
if (this.authorizationCodeGrantConfigurer != null) {
- this.configure(builder, this.authorizationCodeGrantConfigurer);
+ this.authorizationCodeGrantConfigurer.configure(builder);
}
}
@@ -264,37 +343,4 @@ private void init(B builder, AuthorizationCodeGrantConfigurer authorizationCodeG
new OAuth2AuthorizationCodeAuthenticationProvider(accessTokenResponseClient);
builder.authenticationProvider(this.postProcess(authorizationCodeAuthenticationProvider));
}
-
- private void configure(B builder, AuthorizationCodeGrantConfigurer authorizationCodeGrantConfigurer) throws Exception {
- String authorizationRequestBaseUri = authorizationCodeGrantConfigurer.authorizationEndpointConfig.authorizationRequestBaseUri;
- if (authorizationRequestBaseUri == null) {
- authorizationRequestBaseUri = OAuth2AuthorizationRequestRedirectFilter.DEFAULT_AUTHORIZATION_REQUEST_BASE_URI;
- }
-
- OAuth2AuthorizationRequestRedirectFilter authorizationRequestFilter = new OAuth2AuthorizationRequestRedirectFilter(
- OAuth2ClientConfigurerUtils.getClientRegistrationRepository(builder), authorizationRequestBaseUri);
-
- if (authorizationCodeGrantConfigurer.authorizationEndpointConfig.authorizationRequestRepository != null) {
- authorizationRequestFilter.setAuthorizationRequestRepository(
- authorizationCodeGrantConfigurer.authorizationEndpointConfig.authorizationRequestRepository);
- }
- RequestCache requestCache = builder.getSharedObject(RequestCache.class);
- if (requestCache != null) {
- authorizationRequestFilter.setRequestCache(requestCache);
- }
- builder.addFilter(this.postProcess(authorizationRequestFilter));
-
- AuthenticationManager authenticationManager = builder.getSharedObject(AuthenticationManager.class);
-
- OAuth2AuthorizationCodeGrantFilter authorizationCodeGrantFilter = new OAuth2AuthorizationCodeGrantFilter(
- OAuth2ClientConfigurerUtils.getClientRegistrationRepository(builder),
- OAuth2ClientConfigurerUtils.getAuthorizedClientService(builder),
- authenticationManager);
-
- if (authorizationCodeGrantConfigurer.authorizationEndpointConfig.authorizationRequestRepository != null) {
- authorizationCodeGrantFilter.setAuthorizationRequestRepository(
- authorizationCodeGrantConfigurer.authorizationEndpointConfig.authorizationRequestRepository);
- }
- builder.addFilter(this.postProcess(authorizationCodeGrantFilter));
- }
}
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/client/OAuth2ClientConfigurerUtils.java b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/client/OAuth2ClientConfigurerUtils.java
index 646c32accfd..140ad70c321 100644
--- a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/client/OAuth2ClientConfigurerUtils.java
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/client/OAuth2ClientConfigurerUtils.java
@@ -23,6 +23,9 @@
import org.springframework.security.oauth2.client.InMemoryOAuth2AuthorizedClientService;
import org.springframework.security.oauth2.client.OAuth2AuthorizedClientService;
import org.springframework.security.oauth2.client.registration.ClientRegistrationRepository;
+import org.springframework.security.oauth2.client.web.AuthenticatedPrincipalOAuth2AuthorizedClientRepository;
+import org.springframework.security.oauth2.client.web.OAuth2AuthorizedClientRepository;
+import org.springframework.util.StringUtils;
import java.util.Map;
@@ -50,14 +53,35 @@ private static > ClientRegistrationRepository g
return builder.getSharedObject(ApplicationContext.class).getBean(ClientRegistrationRepository.class);
}
- static > OAuth2AuthorizedClientService getAuthorizedClientService(B builder) {
- OAuth2AuthorizedClientService authorizedClientService = builder.getSharedObject(OAuth2AuthorizedClientService.class);
- if (authorizedClientService == null) {
- authorizedClientService = getAuthorizedClientServiceBean(builder);
- if (authorizedClientService == null) {
- authorizedClientService = new InMemoryOAuth2AuthorizedClientService(getClientRegistrationRepository(builder));
+ static > OAuth2AuthorizedClientRepository getAuthorizedClientRepository(B builder) {
+ OAuth2AuthorizedClientRepository authorizedClientRepository = builder.getSharedObject(OAuth2AuthorizedClientRepository.class);
+ if (authorizedClientRepository == null) {
+ authorizedClientRepository = getAuthorizedClientRepositoryBean(builder);
+ if (authorizedClientRepository == null) {
+ authorizedClientRepository = new AuthenticatedPrincipalOAuth2AuthorizedClientRepository(
+ getAuthorizedClientService((builder)));
}
- builder.setSharedObject(OAuth2AuthorizedClientService.class, authorizedClientService);
+ builder.setSharedObject(OAuth2AuthorizedClientRepository.class, authorizedClientRepository);
+ }
+ return authorizedClientRepository;
+ }
+
+ private static > OAuth2AuthorizedClientRepository getAuthorizedClientRepositoryBean(B builder) {
+ Map authorizedClientRepositoryMap = BeanFactoryUtils.beansOfTypeIncludingAncestors(
+ builder.getSharedObject(ApplicationContext.class), OAuth2AuthorizedClientRepository.class);
+ if (authorizedClientRepositoryMap.size() > 1) {
+ throw new NoUniqueBeanDefinitionException(OAuth2AuthorizedClientRepository.class, authorizedClientRepositoryMap.size(),
+ "Expected single matching bean of type '" + OAuth2AuthorizedClientRepository.class.getName() + "' but found " +
+ authorizedClientRepositoryMap.size() + ": " + StringUtils.collectionToCommaDelimitedString(authorizedClientRepositoryMap.keySet()));
+ }
+ return (!authorizedClientRepositoryMap.isEmpty() ? authorizedClientRepositoryMap.values().iterator().next() : null);
+ }
+
+
+ private static > OAuth2AuthorizedClientService getAuthorizedClientService(B builder) {
+ OAuth2AuthorizedClientService authorizedClientService = getAuthorizedClientServiceBean(builder);
+ if (authorizedClientService == null) {
+ authorizedClientService = new InMemoryOAuth2AuthorizedClientService(getClientRegistrationRepository(builder));
}
return authorizedClientService;
}
@@ -67,7 +91,8 @@ private static > OAuth2AuthorizedClientService
builder.getSharedObject(ApplicationContext.class), OAuth2AuthorizedClientService.class);
if (authorizedClientServiceMap.size() > 1) {
throw new NoUniqueBeanDefinitionException(OAuth2AuthorizedClientService.class, authorizedClientServiceMap.size(),
- "Only one matching @Bean of type " + OAuth2AuthorizedClientService.class.getName() + " should be registered.");
+ "Expected single matching bean of type '" + OAuth2AuthorizedClientService.class.getName() + "' but found " +
+ authorizedClientServiceMap.size() + ": " + StringUtils.collectionToCommaDelimitedString(authorizedClientServiceMap.keySet()));
}
return (!authorizedClientServiceMap.isEmpty() ? authorizedClientServiceMap.values().iterator().next() : null);
}
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/client/OAuth2LoginConfigurer.java b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/client/OAuth2LoginConfigurer.java
index 972afc1443a..b84f4bb390b 100644
--- a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/client/OAuth2LoginConfigurer.java
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/client/OAuth2LoginConfigurer.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2002-2017 the original author or authors.
+ * Copyright 2002-2018 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
@@ -42,8 +42,11 @@
import org.springframework.security.oauth2.client.userinfo.DelegatingOAuth2UserService;
import org.springframework.security.oauth2.client.userinfo.OAuth2UserRequest;
import org.springframework.security.oauth2.client.userinfo.OAuth2UserService;
+import org.springframework.security.oauth2.client.web.AuthenticatedPrincipalOAuth2AuthorizedClientRepository;
import org.springframework.security.oauth2.client.web.AuthorizationRequestRepository;
import org.springframework.security.oauth2.client.web.OAuth2AuthorizationRequestRedirectFilter;
+import org.springframework.security.oauth2.client.web.OAuth2AuthorizationRequestResolver;
+import org.springframework.security.oauth2.client.web.OAuth2AuthorizedClientRepository;
import org.springframework.security.oauth2.client.web.OAuth2LoginAuthenticationFilter;
import org.springframework.security.oauth2.core.OAuth2AuthenticationException;
import org.springframework.security.oauth2.core.OAuth2Error;
@@ -51,15 +54,23 @@
import org.springframework.security.oauth2.core.oidc.OidcScopes;
import org.springframework.security.oauth2.core.oidc.user.OidcUser;
import org.springframework.security.oauth2.core.user.OAuth2User;
+import org.springframework.security.web.AuthenticationEntryPoint;
+import org.springframework.security.web.authentication.DelegatingAuthenticationEntryPoint;
+import org.springframework.security.web.authentication.LoginUrlAuthenticationEntryPoint;
import org.springframework.security.web.authentication.ui.DefaultLoginPageGeneratingFilter;
import org.springframework.security.web.savedrequest.RequestCache;
+import org.springframework.security.web.util.matcher.AndRequestMatcher;
import org.springframework.security.web.util.matcher.AntPathRequestMatcher;
+import org.springframework.security.web.util.matcher.NegatedRequestMatcher;
+import org.springframework.security.web.util.matcher.OrRequestMatcher;
import org.springframework.security.web.util.matcher.RequestMatcher;
import org.springframework.util.Assert;
import org.springframework.util.ClassUtils;
import java.util.ArrayList;
+import java.util.Collections;
import java.util.HashMap;
+import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
@@ -91,7 +102,7 @@
*
*
* {@link ClientRegistrationRepository} (required)
- * {@link OAuth2AuthorizedClientService} (optional)
+ * {@link OAuth2AuthorizedClientRepository} (optional)
* {@link GrantedAuthoritiesMapper} (optional)
*
*
@@ -101,7 +112,7 @@
*
*
* {@link ClientRegistrationRepository}
- * {@link OAuth2AuthorizedClientService}
+ * {@link OAuth2AuthorizedClientRepository}
* {@link GrantedAuthoritiesMapper}
* {@link DefaultLoginPageGeneratingFilter} - if {@link #loginPage(String)} is not configured
* and {@code DefaultLoginPageGeneratingFilter} is available, than a default login page will be made available
@@ -114,6 +125,7 @@
* @see OAuth2AuthorizationRequestRedirectFilter
* @see OAuth2LoginAuthenticationFilter
* @see ClientRegistrationRepository
+ * @see OAuth2AuthorizedClientRepository
* @see AbstractAuthenticationFilterConfigurer
*/
public final class OAuth2LoginConfigurer> extends
@@ -124,6 +136,7 @@ public final class OAuth2LoginConfigurer> exten
private final RedirectionEndpointConfig redirectionEndpointConfig = new RedirectionEndpointConfig();
private final UserInfoEndpointConfig userInfoEndpointConfig = new UserInfoEndpointConfig();
private String loginPage;
+ private String loginProcessingUrl = OAuth2LoginAuthenticationFilter.DEFAULT_FILTER_PROCESSES_URI;
/**
* Sets the repository of client registrations.
@@ -137,6 +150,19 @@ public OAuth2LoginConfigurer clientRegistrationRepository(ClientRegistrationR
return this;
}
+ /**
+ * Sets the repository for authorized client(s).
+ *
+ * @since 5.1
+ * @param authorizedClientRepository the authorized client repository
+ * @return the {@link OAuth2LoginConfigurer} for further configuration
+ */
+ public OAuth2LoginConfigurer authorizedClientRepository(OAuth2AuthorizedClientRepository authorizedClientRepository) {
+ Assert.notNull(authorizedClientRepository, "authorizedClientRepository cannot be null");
+ this.getBuilder().setSharedObject(OAuth2AuthorizedClientRepository.class, authorizedClientRepository);
+ return this;
+ }
+
/**
* Sets the service for authorized client(s).
*
@@ -145,7 +171,7 @@ public OAuth2LoginConfigurer clientRegistrationRepository(ClientRegistrationR
*/
public OAuth2LoginConfigurer authorizedClientService(OAuth2AuthorizedClientService authorizedClientService) {
Assert.notNull(authorizedClientService, "authorizedClientService cannot be null");
- this.getBuilder().setSharedObject(OAuth2AuthorizedClientService.class, authorizedClientService);
+ this.authorizedClientRepository(new AuthenticatedPrincipalOAuth2AuthorizedClientRepository(authorizedClientService));
return this;
}
@@ -156,6 +182,13 @@ public OAuth2LoginConfigurer loginPage(String loginPage) {
return this;
}
+ @Override
+ public OAuth2LoginConfigurer loginProcessingUrl(String loginProcessingUrl) {
+ Assert.hasText(loginProcessingUrl, "loginProcessingUrl cannot be empty");
+ this.loginProcessingUrl = loginProcessingUrl;
+ return this;
+ }
+
/**
* Returns the {@link AuthorizationEndpointConfig} for configuring the Authorization Server's Authorization Endpoint.
*
@@ -170,6 +203,7 @@ public AuthorizationEndpointConfig authorizationEndpoint() {
*/
public class AuthorizationEndpointConfig {
private String authorizationRequestBaseUri;
+ private OAuth2AuthorizationRequestResolver authorizationRequestResolver;
private AuthorizationRequestRepository authorizationRequestRepository;
private AuthorizationEndpointConfig() {
@@ -187,6 +221,19 @@ public AuthorizationEndpointConfig baseUri(String authorizationRequestBaseUri) {
return this;
}
+ /**
+ * Sets the resolver used for resolving {@link OAuth2AuthorizationRequest}'s.
+ *
+ * @since 5.1
+ * @param authorizationRequestResolver the resolver used for resolving {@link OAuth2AuthorizationRequest}'s
+ * @return the {@link AuthorizationEndpointConfig} for further configuration
+ */
+ public AuthorizationEndpointConfig authorizationRequestResolver(OAuth2AuthorizationRequestResolver authorizationRequestResolver) {
+ Assert.notNull(authorizationRequestResolver, "authorizationRequestResolver cannot be null");
+ this.authorizationRequestResolver = authorizationRequestResolver;
+ return this;
+ }
+
/**
* Sets the repository used for storing {@link OAuth2AuthorizationRequest}'s.
*
@@ -377,14 +424,28 @@ public void init(B http) throws Exception {
OAuth2LoginAuthenticationFilter authenticationFilter =
new OAuth2LoginAuthenticationFilter(
OAuth2ClientConfigurerUtils.getClientRegistrationRepository(this.getBuilder()),
- OAuth2ClientConfigurerUtils.getAuthorizedClientService(this.getBuilder()),
- OAuth2LoginAuthenticationFilter.DEFAULT_FILTER_PROCESSES_URI);
+ OAuth2ClientConfigurerUtils.getAuthorizedClientRepository(this.getBuilder()),
+ this.loginProcessingUrl);
this.setAuthenticationFilter(authenticationFilter);
- this.loginProcessingUrl(OAuth2LoginAuthenticationFilter.DEFAULT_FILTER_PROCESSES_URI);
+ super.loginProcessingUrl(this.loginProcessingUrl);
+
if (this.loginPage != null) {
+ // Set custom login page
super.loginPage(this.loginPage);
+ super.init(http);
+ } else {
+ Map loginUrlToClientName = this.getLoginLinks();
+ if (loginUrlToClientName.size() == 1) {
+ // Setup auto-redirect to provider login page
+ // when only 1 client is configured
+ this.updateAuthenticationDefaults();
+ this.updateAccessDefaults(http);
+ String providerLoginPage = loginUrlToClientName.keySet().iterator().next();
+ this.registerAuthenticationEntryPoint(http, this.getLoginEntryPoint(http, providerLoginPage));
+ } else {
+ super.init(http);
+ }
}
- super.init(http);
OAuth2AccessTokenResponseClient accessTokenResponseClient =
this.tokenEndpointConfig.accessTokenResponseClient;
@@ -436,13 +497,19 @@ public void init(B http) throws Exception {
@Override
public void configure(B http) throws Exception {
- String authorizationRequestBaseUri = this.authorizationEndpointConfig.authorizationRequestBaseUri;
- if (authorizationRequestBaseUri == null) {
- authorizationRequestBaseUri = OAuth2AuthorizationRequestRedirectFilter.DEFAULT_AUTHORIZATION_REQUEST_BASE_URI;
- }
+ OAuth2AuthorizationRequestRedirectFilter authorizationRequestFilter;
- OAuth2AuthorizationRequestRedirectFilter authorizationRequestFilter = new OAuth2AuthorizationRequestRedirectFilter(
- OAuth2ClientConfigurerUtils.getClientRegistrationRepository(this.getBuilder()), authorizationRequestBaseUri);
+ if (this.authorizationEndpointConfig.authorizationRequestResolver != null) {
+ authorizationRequestFilter = new OAuth2AuthorizationRequestRedirectFilter(
+ this.authorizationEndpointConfig.authorizationRequestResolver);
+ } else {
+ String authorizationRequestBaseUri = this.authorizationEndpointConfig.authorizationRequestBaseUri;
+ if (authorizationRequestBaseUri == null) {
+ authorizationRequestBaseUri = OAuth2AuthorizationRequestRedirectFilter.DEFAULT_AUTHORIZATION_REQUEST_BASE_URI;
+ }
+ authorizationRequestFilter = new OAuth2AuthorizationRequestRedirectFilter(
+ OAuth2ClientConfigurerUtils.getClientRegistrationRepository(this.getBuilder()), authorizationRequestBaseUri);
+ }
if (this.authorizationEndpointConfig.authorizationRequestRepository != null) {
authorizationRequestFilter.setAuthorizationRequestRepository(
@@ -484,9 +551,9 @@ private GrantedAuthoritiesMapper getGrantedAuthoritiesMapper() {
private GrantedAuthoritiesMapper getGrantedAuthoritiesMapperBean() {
Map grantedAuthoritiesMapperMap =
- BeanFactoryUtils.beansOfTypeIncludingAncestors(
- this.getBuilder().getSharedObject(ApplicationContext.class),
- GrantedAuthoritiesMapper.class);
+ BeanFactoryUtils.beansOfTypeIncludingAncestors(
+ this.getBuilder().getSharedObject(ApplicationContext.class),
+ GrantedAuthoritiesMapper.class);
return (!grantedAuthoritiesMapperMap.isEmpty() ? grantedAuthoritiesMapperMap.values().iterator().next() : null);
}
@@ -496,29 +563,51 @@ private void initDefaultLoginFilter(B http) {
return;
}
+ loginPageGeneratingFilter.setOauth2LoginEnabled(true);
+ loginPageGeneratingFilter.setOauth2AuthenticationUrlToClientName(this.getLoginLinks());
+ loginPageGeneratingFilter.setLoginPageUrl(this.getLoginPage());
+ loginPageGeneratingFilter.setFailureUrl(this.getFailureUrl());
+ }
+
+ @SuppressWarnings("unchecked")
+ private Map getLoginLinks() {
Iterable clientRegistrations = null;
ClientRegistrationRepository clientRegistrationRepository =
- OAuth2ClientConfigurerUtils.getClientRegistrationRepository(this.getBuilder());
+ OAuth2ClientConfigurerUtils.getClientRegistrationRepository(this.getBuilder());
ResolvableType type = ResolvableType.forInstance(clientRegistrationRepository).as(Iterable.class);
if (type != ResolvableType.NONE && ClientRegistration.class.isAssignableFrom(type.resolveGenerics()[0])) {
clientRegistrations = (Iterable) clientRegistrationRepository;
}
if (clientRegistrations == null) {
- return;
+ return Collections.emptyMap();
}
String authorizationRequestBaseUri = this.authorizationEndpointConfig.authorizationRequestBaseUri != null ?
- this.authorizationEndpointConfig.authorizationRequestBaseUri :
- OAuth2AuthorizationRequestRedirectFilter.DEFAULT_AUTHORIZATION_REQUEST_BASE_URI;
- Map authenticationUrlToClientName = new HashMap<>();
+ this.authorizationEndpointConfig.authorizationRequestBaseUri :
+ OAuth2AuthorizationRequestRedirectFilter.DEFAULT_AUTHORIZATION_REQUEST_BASE_URI;
+ Map loginUrlToClientName = new HashMap<>();
+ clientRegistrations.forEach(registration -> loginUrlToClientName.put(
+ authorizationRequestBaseUri + "/" + registration.getRegistrationId(),
+ registration.getClientName()));
+
+ return loginUrlToClientName;
+ }
- clientRegistrations.forEach(registration -> authenticationUrlToClientName.put(
- authorizationRequestBaseUri + "/" + registration.getRegistrationId(),
- registration.getClientName()));
- loginPageGeneratingFilter.setOauth2LoginEnabled(true);
- loginPageGeneratingFilter.setOauth2AuthenticationUrlToClientName(authenticationUrlToClientName);
- loginPageGeneratingFilter.setLoginPageUrl(this.getLoginPage());
- loginPageGeneratingFilter.setFailureUrl(this.getFailureUrl());
+ private AuthenticationEntryPoint getLoginEntryPoint(B http, String providerLoginPage) {
+ RequestMatcher loginPageMatcher = new AntPathRequestMatcher(this.getLoginPage());
+ RequestMatcher faviconMatcher = new AntPathRequestMatcher("/favicon.ico");
+ RequestMatcher defaultEntryPointMatcher = this.getAuthenticationEntryPointMatcher(http);
+ RequestMatcher defaultLoginPageMatcher = new AndRequestMatcher(
+ new OrRequestMatcher(loginPageMatcher, faviconMatcher), defaultEntryPointMatcher);
+
+ LinkedHashMap entryPoints = new LinkedHashMap<>();
+ entryPoints.put(new NegatedRequestMatcher(defaultLoginPageMatcher),
+ new LoginUrlAuthenticationEntryPoint(providerLoginPage));
+
+ DelegatingAuthenticationEntryPoint loginEntryPoint = new DelegatingAuthenticationEntryPoint(entryPoints);
+ loginEntryPoint.setDefaultEntryPoint(this.getAuthenticationEntryPoint());
+
+ return loginEntryPoint;
}
private static class OidcAuthenticationRequestChecker implements AuthenticationProvider {
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/server/resource/OAuth2ResourceServerConfigurer.java b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/server/resource/OAuth2ResourceServerConfigurer.java
new file mode 100644
index 00000000000..d03662615e2
--- /dev/null
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/oauth2/server/resource/OAuth2ResourceServerConfigurer.java
@@ -0,0 +1,294 @@
+/*
+ * Copyright 2002-2018 the original author or authors.
+ *
+ * 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.security.config.annotation.web.configurers.oauth2.server.resource;
+
+import javax.servlet.http.HttpServletRequest;
+
+import org.springframework.context.ApplicationContext;
+import org.springframework.security.authentication.AuthenticationManager;
+import org.springframework.security.config.annotation.web.HttpSecurityBuilder;
+import org.springframework.security.config.annotation.web.configurers.AbstractHttpConfigurer;
+import org.springframework.security.config.annotation.web.configurers.CsrfConfigurer;
+import org.springframework.security.config.annotation.web.configurers.ExceptionHandlingConfigurer;
+import org.springframework.security.config.http.SessionCreationPolicy;
+import org.springframework.security.oauth2.core.OAuth2AuthenticationException;
+import org.springframework.security.oauth2.jwt.JwtDecoder;
+import org.springframework.security.oauth2.jwt.NimbusJwtDecoderJwkSupport;
+import org.springframework.security.oauth2.server.resource.authentication.JwtAuthenticationProvider;
+import org.springframework.security.oauth2.server.resource.web.BearerTokenAuthenticationEntryPoint;
+import org.springframework.security.oauth2.server.resource.web.BearerTokenAuthenticationFilter;
+import org.springframework.security.oauth2.server.resource.web.BearerTokenResolver;
+import org.springframework.security.oauth2.server.resource.web.DefaultBearerTokenResolver;
+import org.springframework.security.oauth2.server.resource.web.access.BearerTokenAccessDeniedHandler;
+import org.springframework.security.web.AuthenticationEntryPoint;
+import org.springframework.security.web.access.AccessDeniedHandler;
+import org.springframework.security.web.util.matcher.RequestMatcher;
+import org.springframework.util.Assert;
+
+/**
+ *
+ * An {@link AbstractHttpConfigurer} for OAuth 2.0 Resource Server Support.
+ *
+ * By default, this wires a {@link BearerTokenAuthenticationFilter}, which can be used to parse the request
+ * for bearer tokens and make an authentication attempt.
+ *
+ *
+ * The following configuration options are available:
+ *
+ *
+ * {@link #accessDeniedHandler(AccessDeniedHandler)} - customizes how access denied errors are handled
+ * {@link #authenticationEntryPoint(AuthenticationEntryPoint)} - customizes how authentication failures are handled
+ * {@link #bearerTokenResolver(BearerTokenResolver)} - customizes how to resolve a bearer token from the request
+ * {@link #jwt()} - enables Jwt-encoded bearer token support
+ *
+ *
+ *
+ * When using {@link #jwt()}, either
+ *
+ *
+ *
+ * supply a Jwk Set Uri via {@link JwtConfigurer#jwkSetUri}, or
+ *
+ *
+ * supply a {@link JwtDecoder} instance via {@link JwtConfigurer#decoder}, or
+ *
+ *
+ * expose a {@link JwtDecoder} bean
+ *
+ *
+ *
+ * Security Filters
+ *
+ * The following {@code Filter}s are populated when {@link #jwt()} is configured:
+ *
+ *
+ * {@link BearerTokenAuthenticationFilter}
+ *
+ *
+ * Shared Objects Created
+ *
+ * The following shared objects are populated:
+ *
+ *
+ * {@link SessionCreationPolicy} (optional)
+ *
+ *
+ * Shared Objects Used
+ *
+ * The following shared objects are used:
+ *
+ *
+ * {@link AuthenticationManager}
+ *
+ *
+ * @author Josh Cummings
+ * @since 5.1
+ * @see BearerTokenAuthenticationFilter
+ * @see JwtAuthenticationProvider
+ * @see NimbusJwtDecoderJwkSupport
+ * @see AbstractHttpConfigurer
+ */
+public final class OAuth2ResourceServerConfigurer> extends
+ AbstractHttpConfigurer, H> {
+
+ private final ApplicationContext context;
+
+ private BearerTokenResolver bearerTokenResolver;
+ private JwtConfigurer jwtConfigurer;
+
+ private AccessDeniedHandler accessDeniedHandler = new BearerTokenAccessDeniedHandler();
+ private AuthenticationEntryPoint authenticationEntryPoint = new BearerTokenAuthenticationEntryPoint();
+ private BearerTokenRequestMatcher requestMatcher = new BearerTokenRequestMatcher();
+
+ public OAuth2ResourceServerConfigurer(ApplicationContext context) {
+ Assert.notNull(context, "context cannot be null");
+ this.context = context;
+ }
+
+ public OAuth2ResourceServerConfigurer accessDeniedHandler(AccessDeniedHandler accessDeniedHandler) {
+ Assert.notNull(accessDeniedHandler, "accessDeniedHandler cannot be null");
+ this.accessDeniedHandler = accessDeniedHandler;
+ return this;
+ }
+
+ public OAuth2ResourceServerConfigurer authenticationEntryPoint(AuthenticationEntryPoint entryPoint) {
+ Assert.notNull(entryPoint, "entryPoint cannot be null");
+ this.authenticationEntryPoint = entryPoint;
+ return this;
+ }
+
+ public OAuth2ResourceServerConfigurer bearerTokenResolver(BearerTokenResolver bearerTokenResolver) {
+ Assert.notNull(bearerTokenResolver, "bearerTokenResolver cannot be null");
+ this.bearerTokenResolver = bearerTokenResolver;
+ return this;
+ }
+
+ public JwtConfigurer jwt() {
+ if ( this.jwtConfigurer == null ) {
+ this.jwtConfigurer = new JwtConfigurer(this.context);
+ }
+
+ return this.jwtConfigurer;
+ }
+
+ @Override
+ public void setBuilder(H http) {
+ super.setBuilder(http);
+ initSessionCreationPolicy(http);
+ }
+
+ @Override
+ public void init(H http) throws Exception {
+ registerDefaultAccessDeniedHandler(http);
+ registerDefaultEntryPoint(http);
+ registerDefaultCsrfOverride(http);
+ }
+
+ @Override
+ public void configure(H http) throws Exception {
+ BearerTokenResolver bearerTokenResolver = getBearerTokenResolver();
+ this.requestMatcher.setBearerTokenResolver(bearerTokenResolver);
+
+ AuthenticationManager manager = http.getSharedObject(AuthenticationManager.class);
+
+ BearerTokenAuthenticationFilter filter =
+ new BearerTokenAuthenticationFilter(manager);
+ filter.setBearerTokenResolver(bearerTokenResolver);
+ filter.setAuthenticationEntryPoint(this.authenticationEntryPoint);
+ filter = postProcess(filter);
+
+ http.addFilter(filter);
+
+ if ( this.jwtConfigurer == null ) {
+ throw new IllegalStateException("Jwt is the only supported format for bearer tokens " +
+ "in Spring Security and no Jwt configuration was found. Make sure to specify " +
+ "a jwk set uri by doing http.oauth2().resourceServer().jwt().jwkSetUri(uri), or wire a " +
+ "JwtDecoder instance by doing http.oauth2().resourceServer().jwt().decoder(decoder), or " +
+ "expose a JwtDecoder instance as a bean and do http.oauth2().resourceServer().jwt().");
+ }
+
+ JwtDecoder decoder = this.jwtConfigurer.getJwtDecoder();
+
+ JwtAuthenticationProvider provider =
+ new JwtAuthenticationProvider(decoder);
+ provider = postProcess(provider);
+
+ http.authenticationProvider(provider);
+ }
+
+ public class JwtConfigurer {
+ private final ApplicationContext context;
+
+ private JwtDecoder decoder;
+
+ JwtConfigurer(ApplicationContext context) {
+ this.context = context;
+ }
+
+ public JwtConfigurer decoder(JwtDecoder decoder) {
+ this.decoder = decoder;
+ return this;
+ }
+
+ public JwtConfigurer jwkSetUri(String uri) {
+ this.decoder = new NimbusJwtDecoderJwkSupport(uri);
+ return this;
+ }
+
+ public OAuth2ResourceServerConfigurer and() {
+ return OAuth2ResourceServerConfigurer.this;
+ }
+
+ JwtDecoder getJwtDecoder() {
+ if ( this.decoder == null ) {
+ return this.context.getBean(JwtDecoder.class);
+ }
+
+ return this.decoder;
+ }
+ }
+
+ private void initSessionCreationPolicy(H http) {
+ if (http.getSharedObject(SessionCreationPolicy.class) == null) {
+ http.setSharedObject(SessionCreationPolicy.class, SessionCreationPolicy.STATELESS);
+ }
+ }
+
+ private void registerDefaultAccessDeniedHandler(H http) {
+ ExceptionHandlingConfigurer exceptionHandling = http
+ .getConfigurer(ExceptionHandlingConfigurer.class);
+ if (exceptionHandling == null) {
+ return;
+ }
+
+ exceptionHandling.defaultAccessDeniedHandlerFor(
+ this.accessDeniedHandler,
+ this.requestMatcher);
+ }
+
+ private void registerDefaultEntryPoint(H http) {
+ ExceptionHandlingConfigurer exceptionHandling = http
+ .getConfigurer(ExceptionHandlingConfigurer.class);
+ if (exceptionHandling == null) {
+ return;
+ }
+
+ exceptionHandling.defaultAuthenticationEntryPointFor(
+ this.authenticationEntryPoint,
+ this.requestMatcher);
+ }
+
+ private void registerDefaultCsrfOverride(H http) {
+ CsrfConfigurer csrf = http
+ .getConfigurer(CsrfConfigurer.class);
+ if (csrf == null) {
+ return;
+ }
+
+ csrf.ignoringRequestMatchers(this.requestMatcher);
+ }
+
+ BearerTokenResolver getBearerTokenResolver() {
+ if ( this.bearerTokenResolver == null ) {
+ if ( this.context.getBeanNamesForType(BearerTokenResolver.class).length > 0 ) {
+ this.bearerTokenResolver = this.context.getBean(BearerTokenResolver.class);
+ } else {
+ this.bearerTokenResolver = new DefaultBearerTokenResolver();
+ }
+ }
+
+ return this.bearerTokenResolver;
+ }
+
+ private static final class BearerTokenRequestMatcher implements RequestMatcher {
+ private BearerTokenResolver bearerTokenResolver;
+
+ @Override
+ public boolean matches(HttpServletRequest request) {
+ try {
+ return this.bearerTokenResolver.resolve(request) != null;
+ } catch ( OAuth2AuthenticationException e ) {
+ return false;
+ }
+ }
+
+ public void setBearerTokenResolver(BearerTokenResolver tokenResolver) {
+ Assert.notNull(tokenResolver, "resolver cannot be null");
+ this.bearerTokenResolver = tokenResolver;
+ }
+ }
+}
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/openid/OpenIDLoginConfigurer.java b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/openid/OpenIDLoginConfigurer.java
index a1c8ae02253..e29ccdee920 100644
--- a/config/src/main/java/org/springframework/security/config/annotation/web/configurers/openid/OpenIDLoginConfigurer.java
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/configurers/openid/OpenIDLoginConfigurer.java
@@ -479,7 +479,7 @@ public AttributeConfigurer required(boolean required) {
/**
* The OpenID attribute type.
* @param type
- * @return
+ * @return the {@link AttributeConfigurer} for further customizations
*/
public AttributeConfigurer type(String type) {
this.type = type;
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/reactive/ReactiveOAuth2ClientImportSelector.java b/config/src/main/java/org/springframework/security/config/annotation/web/reactive/ReactiveOAuth2ClientImportSelector.java
index c5d2ddb6dac..6c0603049dc 100644
--- a/config/src/main/java/org/springframework/security/config/annotation/web/reactive/ReactiveOAuth2ClientImportSelector.java
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/reactive/ReactiveOAuth2ClientImportSelector.java
@@ -16,19 +16,18 @@
package org.springframework.security.config.annotation.web.reactive;
-import java.util.List;
-
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.ImportSelector;
import org.springframework.core.type.AnnotationMetadata;
import org.springframework.security.oauth2.client.ReactiveOAuth2AuthorizedClientService;
-import org.springframework.security.oauth2.client.registration.ReactiveClientRegistrationRepository;
-import org.springframework.security.oauth2.client.web.reactive.result.method.annotation.OAuth2ClientArgumentResolver;
+import org.springframework.security.oauth2.client.web.reactive.result.method.annotation.OAuth2AuthorizedClientArgumentResolver;
import org.springframework.util.ClassUtils;
import org.springframework.web.reactive.config.WebFluxConfigurer;
import org.springframework.web.reactive.result.method.annotation.ArgumentResolverConfigurer;
+import java.util.List;
+
/**
* {@link Configuration} for OAuth 2.0 Client support.
*
@@ -52,21 +51,12 @@ public String[] selectImports(AnnotationMetadata importingClassMetadata) {
@Configuration
static class OAuth2ClientWebFluxSecurityConfiguration implements WebFluxConfigurer {
- private ReactiveClientRegistrationRepository clientRegistrationRepository;
-
private ReactiveOAuth2AuthorizedClientService authorizedClientService;
@Override
public void configureArgumentResolvers(ArgumentResolverConfigurer configurer) {
- if (this.clientRegistrationRepository != null && this.authorizedClientService != null) {
- configurer.addCustomResolver(new OAuth2ClientArgumentResolver(this.clientRegistrationRepository, this.authorizedClientService));
- }
- }
-
- @Autowired(required = false)
- public void setClientRegistrationRepository(List clientRegistrationRepository) {
- if (clientRegistrationRepository.size() == 1) {
- this.clientRegistrationRepository = clientRegistrationRepository.get(0);
+ if (this.authorizedClientService != null) {
+ configurer.addCustomResolver(new OAuth2AuthorizedClientArgumentResolver(this.authorizedClientService));
}
}
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/reactive/ServerHttpSecurityConfiguration.java b/config/src/main/java/org/springframework/security/config/annotation/web/reactive/ServerHttpSecurityConfiguration.java
index 8e069900924..d5c63002a88 100644
--- a/config/src/main/java/org/springframework/security/config/annotation/web/reactive/ServerHttpSecurityConfiguration.java
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/reactive/ServerHttpSecurityConfiguration.java
@@ -22,12 +22,14 @@
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Scope;
import org.springframework.context.expression.BeanFactoryResolver;
import org.springframework.core.ReactiveAdapterRegistry;
import org.springframework.security.authentication.ReactiveAuthenticationManager;
import org.springframework.security.authentication.UserDetailsRepositoryReactiveAuthenticationManager;
import org.springframework.security.config.web.server.ServerHttpSecurity;
+import org.springframework.security.core.userdetails.ReactiveUserDetailsPasswordService;
import org.springframework.security.core.userdetails.ReactiveUserDetailsService;
import org.springframework.security.crypto.password.PasswordEncoder;
import org.springframework.security.web.reactive.result.method.annotation.AuthenticationPrincipalArgumentResolver;
@@ -38,6 +40,7 @@
* @author Rob Winch
* @since 5.0
*/
+@Configuration
class ServerHttpSecurityConfiguration implements WebFluxConfigurer {
private static final String BEAN_NAME_PREFIX = "org.springframework.security.config.annotation.web.reactive.HttpSecurityConfiguration.";
private static final String HTTPSECURITY_BEAN_NAME = BEAN_NAME_PREFIX + "httpSecurity";
@@ -54,6 +57,9 @@ class ServerHttpSecurityConfiguration implements WebFluxConfigurer {
@Autowired(required = false)
private PasswordEncoder passwordEncoder;
+ @Autowired(required = false)
+ private ReactiveUserDetailsPasswordService userDetailsPasswordService;
+
@Autowired(required = false)
private BeanFactory beanFactory;
@@ -92,6 +98,7 @@ private ReactiveAuthenticationManager authenticationManager() {
if(this.passwordEncoder != null) {
manager.setPasswordEncoder(this.passwordEncoder);
}
+ manager.setUserDetailsPasswordService(this.userDetailsPasswordService);
return manager;
}
return null;
diff --git a/config/src/main/java/org/springframework/security/config/annotation/web/reactive/WebFluxSecurityConfiguration.java b/config/src/main/java/org/springframework/security/config/annotation/web/reactive/WebFluxSecurityConfiguration.java
index bed91d4f973..30b2b44b7f4 100644
--- a/config/src/main/java/org/springframework/security/config/annotation/web/reactive/WebFluxSecurityConfiguration.java
+++ b/config/src/main/java/org/springframework/security/config/annotation/web/reactive/WebFluxSecurityConfiguration.java
@@ -87,13 +87,14 @@ private SecurityWebFilterChain springSecurityFilterChain() {
private SecurityWebFilterChain springSecurityFilterChain(ServerHttpSecurity http) {
http
.authorizeExchange()
- .anyExchange().authenticated()
- .and()
- .httpBasic().and()
- .formLogin();
+ .anyExchange().authenticated();
- if (isOAuth2Present) {
+ if (isOAuth2Present && OAuth2ClasspathGuard.shouldConfigure(this.context)) {
OAuth2ClasspathGuard.configure(this.context, http);
+ } else {
+ http
+ .httpBasic().and()
+ .formLogin();
}
SecurityWebFilterChain result = http.build();
@@ -102,11 +103,13 @@ private SecurityWebFilterChain springSecurityFilterChain(ServerHttpSecurity http
private static class OAuth2ClasspathGuard {
static void configure(ApplicationContext context, ServerHttpSecurity http) {
+ http.oauth2Login();
+ }
+
+ static boolean shouldConfigure(ApplicationContext context) {
ClassLoader loader = context.getClassLoader();
Class> reactiveClientRegistrationRepositoryClass = ClassUtils.resolveClassName(REACTIVE_CLIENT_REGISTRATION_REPOSITORY_CLASSNAME, loader);
- if (context.getBeanNamesForType(reactiveClientRegistrationRepositoryClass).length == 1) {
- http.oauth2Login();
- }
+ return context.getBeanNamesForType(reactiveClientRegistrationRepositoryClass).length == 1;
}
}
}
diff --git a/config/src/main/java/org/springframework/security/config/core/userdetails/ReactiveUserDetailsServiceResourceFactoryBean.java b/config/src/main/java/org/springframework/security/config/core/userdetails/ReactiveUserDetailsServiceResourceFactoryBean.java
index e932cc17c15..2f74d105a3d 100644
--- a/config/src/main/java/org/springframework/security/config/core/userdetails/ReactiveUserDetailsServiceResourceFactoryBean.java
+++ b/config/src/main/java/org/springframework/security/config/core/userdetails/ReactiveUserDetailsServiceResourceFactoryBean.java
@@ -57,7 +57,6 @@ public void setResourceLoader(ResourceLoader resourceLoader) {
* Sets the location of a Resource that is a Properties file in the format defined in {@link UserDetailsResourceFactoryBean}.
*
* @param resourceLocation the location of the properties file that contains the users (i.e. "classpath:users.properties")
- * @return the UserDetailsResourceFactoryBean
*/
public void setResourceLocation(String resourceLocation) {
this.userDetails.setResourceLocation(resourceLocation);
diff --git a/config/src/main/java/org/springframework/security/config/http/AuthenticationConfigBuilder.java b/config/src/main/java/org/springframework/security/config/http/AuthenticationConfigBuilder.java
index 39bcdbf434c..3ad241ccb31 100644
--- a/config/src/main/java/org/springframework/security/config/http/AuthenticationConfigBuilder.java
+++ b/config/src/main/java/org/springframework/security/config/http/AuthenticationConfigBuilder.java
@@ -48,6 +48,7 @@
import org.springframework.security.web.authentication.preauth.x509.SubjectDnX509PrincipalExtractor;
import org.springframework.security.web.authentication.preauth.x509.X509AuthenticationFilter;
import org.springframework.security.web.authentication.ui.DefaultLoginPageGeneratingFilter;
+import org.springframework.security.web.authentication.ui.DefaultLogoutPageGeneratingFilter;
import org.springframework.security.web.authentication.www.BasicAuthenticationEntryPoint;
import org.springframework.security.web.authentication.www.BasicAuthenticationFilter;
import org.springframework.security.web.csrf.CsrfToken;
@@ -123,6 +124,7 @@ final class AuthenticationConfigBuilder {
@SuppressWarnings("rawtypes")
private ManagedList logoutHandlers;
private BeanDefinition loginPageGenerationFilter;
+ private BeanDefinition logoutPageGenerationFilter;
private BeanDefinition etf;
private final BeanReference requestCache;
private final BeanReference portMapper;
@@ -544,6 +546,10 @@ void createLoginPageFilterIfNeeded() {
.rootBeanDefinition(DefaultLoginPageGeneratingFilter.class);
loginPageFilter.addPropertyValue("resolveHiddenInputs", new CsrfTokenHiddenInputFunction());
+ BeanDefinitionBuilder logoutPageFilter = BeanDefinitionBuilder
+ .rootBeanDefinition(DefaultLogoutPageGeneratingFilter.class);
+ logoutPageFilter.addPropertyValue("resolveHiddenInputs", new CsrfTokenHiddenInputFunction());
+
if (formFilterId != null) {
loginPageFilter.addConstructorArgReference(formFilterId);
loginPageFilter.addPropertyValue("authenticationUrl", loginProcessingUrl);
@@ -556,6 +562,7 @@ void createLoginPageFilterIfNeeded() {
}
loginPageGenerationFilter = loginPageFilter.getBeanDefinition();
+ this.logoutPageGenerationFilter = logoutPageFilter.getBeanDefinition();
}
}
@@ -798,6 +805,7 @@ List getFilters() {
if (loginPageGenerationFilter != null) {
filters.add(new OrderDecorator(loginPageGenerationFilter, LOGIN_PAGE_FILTER));
+ filters.add(new OrderDecorator(this.logoutPageGenerationFilter, LOGOUT_PAGE_FILTER));
}
if (basicFilter != null) {
diff --git a/config/src/main/java/org/springframework/security/config/http/HeadersBeanDefinitionParser.java b/config/src/main/java/org/springframework/security/config/http/HeadersBeanDefinitionParser.java
index 20ac10dab7e..dffeab94a09 100644
--- a/config/src/main/java/org/springframework/security/config/http/HeadersBeanDefinitionParser.java
+++ b/config/src/main/java/org/springframework/security/config/http/HeadersBeanDefinitionParser.java
@@ -1,5 +1,5 @@
/*
- * Copyright 2002-2016 the original author or authors.
+ * Copyright 2002-2018 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
@@ -13,6 +13,7 @@
* See the License for the specific language governing permissions and
* limitations under the License.
*/
+
package org.springframework.security.config.http;
import java.net.URI;
@@ -47,6 +48,7 @@
* @author Marten Deinum
* @author Tim Ysewyn
* @author EddĂș MelĂ©ndez
+ * @author Vedran Pavic
* @since 3.2
*/
public class HeadersBeanDefinitionParser implements BeanDefinitionParser {
@@ -85,6 +87,7 @@ public class HeadersBeanDefinitionParser implements BeanDefinitionParser {
private static final String CONTENT_SECURITY_POLICY_ELEMENT = "content-security-policy";
private static final String REFERRER_POLICY_ELEMENT = "referrer-policy";
+ private static final String FEATURE_POLICY_ELEMENT = "feature-policy";
private static final String ALLOW_FROM = "ALLOW-FROM";
@@ -114,6 +117,8 @@ public BeanDefinition parse(Element element, ParserContext parserContext) {
parseReferrerPolicyElement(element, parserContext);
+ parseFeaturePolicyElement(element, parserContext);
+
parseHeaderElements(element);
boolean noWriters = headerWriters.isEmpty();
@@ -313,6 +318,32 @@ private void addReferrerPolicy(Element referrerPolicyElement, ParserContext cont
headerWriters.add(headersWriter.getBeanDefinition());
}
+ private void parseFeaturePolicyElement(Element element, ParserContext context) {
+ Element featurePolicyElement = (element == null) ? null
+ : DomUtils.getChildElementByTagName(element, FEATURE_POLICY_ELEMENT);
+ if (featurePolicyElement != null) {
+ addFeaturePolicy(featurePolicyElement, context);
+ }
+ }
+
+ private void addFeaturePolicy(Element featurePolicyElement, ParserContext context) {
+ BeanDefinitionBuilder headersWriter = BeanDefinitionBuilder
+ .genericBeanDefinition(FeaturePolicyHeaderWriter.class);
+
+ String policyDirectives = featurePolicyElement
+ .getAttribute(ATT_POLICY_DIRECTIVES);
+ if (!StringUtils.hasText(policyDirectives)) {
+ context.getReaderContext().error(
+ ATT_POLICY_DIRECTIVES + " requires a 'value' to be set.",
+ featurePolicyElement);
+ }
+ else {
+ headersWriter.addConstructorArgValue(policyDirectives);
+ }
+
+ headerWriters.add(headersWriter.getBeanDefinition());
+ }
+
private void attrNotAllowed(ParserContext context, String attrName,
String otherAttrName, Element element) {
context.getReaderContext().error(
diff --git a/config/src/main/java/org/springframework/security/config/http/SecurityFilters.java b/config/src/main/java/org/springframework/security/config/http/SecurityFilters.java
index 3c01a48f94e..516cf31cfc8 100644
--- a/config/src/main/java/org/springframework/security/config/http/SecurityFilters.java
+++ b/config/src/main/java/org/springframework/security/config/http/SecurityFilters.java
@@ -40,7 +40,9 @@ enum SecurityFilters {
FORM_LOGIN_FILTER,
OPENID_FILTER,
LOGIN_PAGE_FILTER,
+ LOGOUT_PAGE_FILTER,
DIGEST_AUTH_FILTER,
+ BEARER_TOKEN_AUTH_FILTER,
BASIC_AUTH_FILTER,
REQUEST_CACHE_FILTER,
SERVLET_API_SUPPORT_FILTER,
diff --git a/config/src/main/java/org/springframework/security/config/oauth2/client/CommonOAuth2Provider.java b/config/src/main/java/org/springframework/security/config/oauth2/client/CommonOAuth2Provider.java
index 526a05cc5fb..a98aff17261 100644
--- a/config/src/main/java/org/springframework/security/config/oauth2/client/CommonOAuth2Provider.java
+++ b/config/src/main/java/org/springframework/security/config/oauth2/client/CommonOAuth2Provider.java
@@ -73,7 +73,7 @@ public Builder getBuilder(String registrationId) {
builder.scope("public_profile", "email");
builder.authorizationUri("https://www.facebook.com/v2.8/dialog/oauth");
builder.tokenUri("https://graph.facebook.com/v2.8/oauth/access_token");
- builder.userInfoUri("https://graph.facebook.com/me");
+ builder.userInfoUri("https://graph.facebook.com/me?fields=id,name,email");
builder.userNameAttributeName("id");
builder.clientName("Facebook");
return builder;
diff --git a/config/src/main/java/org/springframework/security/config/web/server/SecurityWebFiltersOrder.java b/config/src/main/java/org/springframework/security/config/web/server/SecurityWebFiltersOrder.java
index fc68e28f7d0..3c3d8acf9c4 100644
--- a/config/src/main/java/org/springframework/security/config/web/server/SecurityWebFiltersOrder.java
+++ b/config/src/main/java/org/springframework/security/config/web/server/SecurityWebFiltersOrder.java
@@ -23,6 +23,10 @@
public enum SecurityWebFiltersOrder {
FIRST(Integer.MIN_VALUE),
HTTP_HEADERS_WRITER,
+ /**
+ * {@link org.springframework.web.cors.reactive.CorsWebFilter}
+ */
+ CORS,
/**
* {@link org.springframework.security.web.server.csrf.CsrfWebFilter}
*/
diff --git a/config/src/main/java/org/springframework/security/config/web/server/ServerHttpSecurity.java b/config/src/main/java/org/springframework/security/config/web/server/ServerHttpSecurity.java
index 6c8c40622cf..0d2e848269e 100644
--- a/config/src/main/java/org/springframework/security/config/web/server/ServerHttpSecurity.java
+++ b/config/src/main/java/org/springframework/security/config/web/server/ServerHttpSecurity.java
@@ -20,6 +20,7 @@
import java.io.IOException;
import java.io.PrintWriter;
import java.io.StringWriter;
+import java.security.interfaces.RSAPublicKey;
import java.time.Duration;
import java.util.ArrayList;
import java.util.Arrays;
@@ -35,21 +36,30 @@
import org.springframework.core.annotation.AnnotationAwareOrderComparator;
import org.springframework.http.HttpMethod;
import org.springframework.http.MediaType;
+import org.springframework.security.authentication.DelegatingReactiveAuthenticationManager;
import org.springframework.security.authentication.ReactiveAuthenticationManager;
import org.springframework.security.authorization.AuthenticatedReactiveAuthorizationManager;
import org.springframework.security.authorization.AuthorityReactiveAuthorizationManager;
import org.springframework.security.authorization.AuthorizationDecision;
import org.springframework.security.authorization.ReactiveAuthorizationManager;
import org.springframework.security.core.AuthenticationException;
+import org.springframework.security.oauth2.client.InMemoryReactiveOAuth2AuthorizedClientService;
import org.springframework.security.oauth2.client.ReactiveOAuth2AuthorizedClientService;
import org.springframework.security.oauth2.client.authentication.OAuth2LoginReactiveAuthenticationManager;
-import org.springframework.security.oauth2.client.endpoint.NimbusReactiveAuthorizationCodeTokenResponseClient;
+import org.springframework.security.oauth2.client.endpoint.WebClientReactiveAuthorizationCodeTokenResponseClient;
+import org.springframework.security.oauth2.client.oidc.authentication.OidcAuthorizationCodeReactiveAuthenticationManager;
+import org.springframework.security.oauth2.client.oidc.userinfo.OidcReactiveOAuth2UserService;
import org.springframework.security.oauth2.client.registration.ClientRegistration;
import org.springframework.security.oauth2.client.registration.ReactiveClientRegistrationRepository;
import org.springframework.security.oauth2.client.userinfo.DefaultReactiveOAuth2UserService;
import org.springframework.security.oauth2.client.userinfo.ReactiveOAuth2UserService;
import org.springframework.security.oauth2.client.web.OAuth2AuthorizationRequestRedirectWebFilter;
import org.springframework.security.oauth2.client.web.ServerOAuth2LoginAuthenticationTokenConverter;
+import org.springframework.security.oauth2.jwt.NimbusReactiveJwtDecoder;
+import org.springframework.security.oauth2.jwt.ReactiveJwtDecoder;
+import org.springframework.security.oauth2.server.resource.authentication.JwtReactiveAuthenticationManager;
+import org.springframework.security.oauth2.server.resource.web.server.BearerTokenServerAuthenticationEntryPoint;
+import org.springframework.security.oauth2.server.resource.web.server.ServerBearerTokenAuthenticationConverter;
import org.springframework.security.web.server.DelegatingServerAuthenticationEntryPoint;
import org.springframework.security.web.server.MatcherSecurityWebFilterChain;
import org.springframework.security.web.server.SecurityWebFilterChain;
@@ -100,6 +110,11 @@
import org.springframework.security.web.server.util.matcher.ServerWebExchangeMatcherEntry;
import org.springframework.security.web.server.util.matcher.ServerWebExchangeMatchers;
import org.springframework.util.Assert;
+import org.springframework.util.ClassUtils;
+import org.springframework.web.cors.reactive.CorsConfigurationSource;
+import org.springframework.web.cors.reactive.CorsProcessor;
+import org.springframework.web.cors.reactive.CorsWebFilter;
+import org.springframework.web.cors.reactive.DefaultCorsProcessor;
import org.springframework.web.server.ServerWebExchange;
import org.springframework.web.server.WebFilter;
import org.springframework.web.server.WebFilterChain;
@@ -170,6 +185,8 @@ public class ServerHttpSecurity {
private CsrfSpec csrf = new CsrfSpec();
+ private CorsSpec cors = new CorsSpec();
+
private ExceptionHandlingSpec exceptionHandling = new ExceptionHandlingSpec();
private HttpBasicSpec httpBasic;
@@ -180,8 +197,12 @@ public class ServerHttpSecurity {
private OAuth2LoginSpec oauth2Login;
+ private OAuth2Spec oauth2;
+
private LogoutSpec logout = new LogoutSpec();
+ private LoginPageSpec loginPage = new LoginPageSpec();
+
private ReactiveAuthenticationManager authenticationManager;
private ServerSecurityContextRepository securityContextRepository = new WebSessionServerSecurityContextRepository();
@@ -203,6 +224,7 @@ public class ServerHttpSecurity {
*
* @param matcher the ServerExchangeMatcher that determines which requests apply to this HttpSecurity instance.
* Default is all requests.
+ * @return the {@link ServerHttpSecurity} to continue configuring
*/
public ServerHttpSecurity securityMatcher(ServerWebExchangeMatcher matcher) {
Assert.notNull(matcher, "matcher cannot be null");
@@ -283,6 +305,80 @@ public CsrfSpec csrf() {
return this.csrf;
}
+ /**
+ * Configures CORS headers. By default if a {@link CorsConfigurationSource} Bean is found, it will be used
+ * to create a {@link CorsWebFilter}. If {@link CorsSpec#configurationSource(CorsConfigurationSource)} is invoked
+ * it will be used instead. If neither has been configured, the Cors configuration will do nothing.
+ * @return the {@link CorsSpec} to customize
+ */
+ public CorsSpec cors() {
+ if (this.cors == null) {
+ this.cors = new CorsSpec();
+ }
+ return this.cors;
+ }
+
+ /**
+ * Configures CORS support within Spring Security. This ensures that the {@link CorsWebFilter} is place in the
+ * correct order.
+ */
+ public class CorsSpec {
+ private CorsWebFilter corsFilter;
+
+ /**
+ * Configures the {@link CorsConfigurationSource} to be used
+ * @param source the source to use
+ * @return the {@link CorsSpec} for additional configuration
+ */
+ public CorsSpec configurationSource(CorsConfigurationSource source) {
+ this.corsFilter = new CorsWebFilter(source);
+ return this;
+ }
+
+ /**
+ * Disables CORS support within Spring Security.
+ * @return the {@link ServerHttpSecurity} to continue configuring
+ */
+ public ServerHttpSecurity disable() {
+ ServerHttpSecurity.this.cors = null;
+ return ServerHttpSecurity.this;
+ }
+
+ /**
+ * Allows method chaining to continue configuring the {@link ServerHttpSecurity}
+ * @return the {@link ServerHttpSecurity} to continue configuring
+ */
+ public ServerHttpSecurity and() {
+ return ServerHttpSecurity.this;
+ }
+
+ protected void configure(ServerHttpSecurity http) {
+ CorsWebFilter corsFilter = getCorsFilter();
+ if (corsFilter != null) {
+ http.addFilterAt(this.corsFilter, SecurityWebFiltersOrder.CORS);
+ }
+ }
+
+ private CorsWebFilter getCorsFilter() {
+ if (this.corsFilter != null) {
+ return this.corsFilter;
+ }
+
+ CorsConfigurationSource source = getBeanOrNull(CorsConfigurationSource.class);
+ if (source == null) {
+ return null;
+ }
+ CorsProcessor processor = getBeanOrNull(CorsProcessor.class);
+ if (processor == null) {
+ processor = new DefaultCorsProcessor();
+ }
+ this.corsFilter = new CorsWebFilter(source, processor);
+ return this.corsFilter;
+ }
+
+ private CorsSpec() {}
+ }
+
/**
* Configures HTTP Basic authentication. An example configuration is provided below:
*
@@ -361,19 +457,32 @@ public OAuth2LoginSpec authorizedClientService(ReactiveOAuth2AuthorizedClientSer
return this;
}
- protected void configure(LoginPageGeneratingWebFilter loginPageFilter, ServerHttpSecurity http) {
- if (loginPageFilter != null) {
- loginPageFilter.setOauth2AuthenticationUrlToClientName(getLinks());
- }
+ /**
+ * Allows method chaining to continue configuring the {@link ServerHttpSecurity}
+ * @return the {@link ServerHttpSecurity} to continue configuring
+ */
+ public ServerHttpSecurity and() {
+ return ServerHttpSecurity.this;
+ }
+
+ protected void configure(ServerHttpSecurity http) {
ReactiveClientRegistrationRepository clientRegistrationRepository = getClientRegistrationRepository();
ReactiveOAuth2AuthorizedClientService authorizedClientService = getAuthorizedClientService();
OAuth2AuthorizationRequestRedirectWebFilter oauthRedirectFilter = new OAuth2AuthorizationRequestRedirectWebFilter(clientRegistrationRepository);
- NimbusReactiveAuthorizationCodeTokenResponseClient client = new NimbusReactiveAuthorizationCodeTokenResponseClient();
+ WebClientReactiveAuthorizationCodeTokenResponseClient client = new WebClientReactiveAuthorizationCodeTokenResponseClient();
ReactiveOAuth2UserService userService = new DefaultReactiveOAuth2UserService();
- OAuth2LoginReactiveAuthenticationManager manager = new OAuth2LoginReactiveAuthenticationManager(client, userService,
+ ReactiveAuthenticationManager manager = new OAuth2LoginReactiveAuthenticationManager(client, userService,
authorizedClientService);
+
+ boolean oidcAuthenticationProviderEnabled = ClassUtils.isPresent(
+ "org.springframework.security.oauth2.jwt.JwtDecoder", this.getClass().getClassLoader());
+ if (oidcAuthenticationProviderEnabled) {
+ OidcAuthorizationCodeReactiveAuthenticationManager oidc = new OidcAuthorizationCodeReactiveAuthenticationManager(client, new OidcReactiveOAuth2UserService(), authorizedClientService);
+ manager = new DelegatingReactiveAuthenticationManager(oidc, manager);
+ }
+
AuthenticationWebFilter authenticationFilter = new AuthenticationWebFilter(manager);
authenticationFilter.setRequiresAuthenticationMatcher(new PathPatternParserServerWebExchangeMatcher("/login/oauth2/code/{registrationId}"));
authenticationFilter.setAuthenticationConverter(new ServerOAuth2LoginAuthenticationTokenConverter(clientRegistrationRepository));
@@ -390,6 +499,16 @@ public Mono onAuthenticationFailure(WebFilterExchange webFilterExchange,
});
authenticationFilter.setSecurityContextRepository(new WebSessionServerSecurityContextRepository());
+ MediaTypeServerWebExchangeMatcher htmlMatcher = new MediaTypeServerWebExchangeMatcher(
+ MediaType.TEXT_HTML);
+ htmlMatcher.setIgnoredMediaTypes(Collections.singleton(MediaType.ALL));
+ Map urlToText = http.oauth2Login.getLinks();
+ if (urlToText.size() == 1) {
+ http.defaultEntryPoints.add(new DelegateEntry(htmlMatcher, new RedirectServerAuthenticationEntryPoint(urlToText.keySet().iterator().next())));
+ } else {
+ http.defaultEntryPoints.add(new DelegateEntry(htmlMatcher, new RedirectServerAuthenticationEntryPoint("/login")));
+ }
+
http.addFilterAt(oauthRedirectFilter, SecurityWebFiltersOrder.HTTP_BASIC);
http.addFilterAt(authenticationFilter, SecurityWebFiltersOrder.AUTHENTICATION);
}
@@ -417,12 +536,149 @@ private ReactiveOAuth2AuthorizedClientService getAuthorizedClientService() {
if (this.authorizedClientService == null) {
this.authorizedClientService = getBeanOrNull(ReactiveOAuth2AuthorizedClientService.class);
}
+ if (this.authorizedClientService == null) {
+ this.authorizedClientService = new InMemoryReactiveOAuth2AuthorizedClientService(getClientRegistrationRepository());
+ }
return this.authorizedClientService;
}
private OAuth2LoginSpec() {}
}
+ /**
+ * Configures OAuth2 support. An example configuration is provided below:
+ *
+ *
+ * @Bean
+ * public SecurityWebFilterChain springSecurityFilterChain(ServerHttpSecurity http) {
+ * http
+ * // ...
+ * .oauth2()
+ * .resourceServer()
+ * .jwt()
+ * .jwkSeturi(jwkSetUri);
+ * return http.build();
+ * }
+ *
+ *
+ * @return the {@link HttpBasicSpec} to customize
+ */
+ public OAuth2Spec oauth2() {
+ if (this.oauth2 == null) {
+ this.oauth2 = new OAuth2Spec();
+ }
+ return this.oauth2;
+ }
+
+ /**
+ * Configures OAuth2 Support
+ *
+ * @since 5.1
+ */
+ public class OAuth2Spec {
+ private ResourceServerSpec resourceServer;
+
+ public ResourceServerSpec resourceServer() {
+ if (this.resourceServer == null) {
+ this.resourceServer = new ResourceServerSpec();
+ }
+ return this.resourceServer;
+ }
+
+ /**
+ * Configures OAuth2 Resource Server Support
+ */
+ public class ResourceServerSpec {
+ private JwtSpec jwt;
+
+ public JwtSpec jwt() {
+ if (this.jwt == null) {
+ this.jwt = new JwtSpec();
+ }
+ return this.jwt;
+ }
+
+ protected void configure(ServerHttpSecurity http) {
+ if (this.jwt != null) {
+ this.jwt.configure(http);
+ }
+ }
+
+ /**
+ * Configures JWT Resource Server Support
+ */
+ public class JwtSpec {
+ private ReactiveJwtDecoder jwtDecoder;
+
+ /**
+ * Configures the {@link ReactiveJwtDecoder} to use
+ * @param jwtDecoder the decoder to use
+ * @return the {@code JwtSpec} for additional configuration
+ */
+ public JwtSpec jwtDecoder(ReactiveJwtDecoder jwtDecoder) {
+ this.jwtDecoder = jwtDecoder;
+ return this;
+ }
+
+ /**
+ * Configures a {@link ReactiveJwtDecoder} that leverages the provided {@link RSAPublicKey}
+ *
+ * @param publicKey the public key to use.
+ * @return the {@code JwtSpec} for additional configuration
+ */
+ public JwtSpec publicKey(RSAPublicKey publicKey) {
+ this.jwtDecoder = new NimbusReactiveJwtDecoder(publicKey);
+ return this;
+ }
+
+ /**
+ * Configures a {@link ReactiveJwtDecoder} using
+ * JSON Web Key (JWK) URL
+ * @param jwkSetUri the URL to use.
+ * @return the {@code JwtSpec} for additional configuration
+ */
+ public JwtSpec jwkSetUri(String jwkSetUri) {
+ this.jwtDecoder = new NimbusReactiveJwtDecoder(jwkSetUri);
+ return this;
+ }
+
+ public ResourceServerSpec and() {
+ return ResourceServerSpec.this;
+ }
+
+ protected void configure(ServerHttpSecurity http) {
+ BearerTokenServerAuthenticationEntryPoint entryPoint = new BearerTokenServerAuthenticationEntryPoint();
+ JwtReactiveAuthenticationManager authenticationManager = new JwtReactiveAuthenticationManager(
+ this.jwtDecoder);
+ AuthenticationWebFilter oauth2 = new AuthenticationWebFilter(authenticationManager);
+ oauth2.setAuthenticationConverter(new ServerBearerTokenAuthenticationConverter());
+ oauth2.setAuthenticationFailureHandler(new ServerAuthenticationEntryPointFailureHandler(entryPoint));
+ http
+ .exceptionHandling()
+ .authenticationEntryPoint(entryPoint)
+ .and()
+ .addFilterAt(oauth2, SecurityWebFiltersOrder.AUTHENTICATION);
+ }
+ }
+
+ public OAuth2Spec and() {
+ return OAuth2Spec.this;
+ }
+ }
+
+ public ServerHttpSecurity and() {
+ return ServerHttpSecurity.this;
+ }
+
+ protected void configure(ServerHttpSecurity http) {
+ if (this.resourceServer != null) {
+ this.resourceServer.configure(http);
+ }
+ }
+
+ private OAuth2Spec() {}
+ }
+
/**
* Configures HTTP Response Headers. The default headers are:
*
@@ -606,26 +862,27 @@ public SecurityWebFilterChain build() {
if(this.csrf != null) {
this.csrf.configure(this);
}
+ if (this.cors != null) {
+ this.cors.configure(this);
+ }
if(this.httpBasic != null) {
this.httpBasic.authenticationManager(this.authenticationManager);
this.httpBasic.configure(this);
}
- LoginPageGeneratingWebFilter loginPageFilter = null;
if(this.formLogin != null) {
this.formLogin.authenticationManager(this.authenticationManager);
if(this.securityContextRepository != null) {
this.formLogin.securityContextRepository(this.securityContextRepository);
}
- if(this.formLogin.authenticationEntryPoint == null) {
- loginPageFilter = new LoginPageGeneratingWebFilter();
- this.webFilters.add(new OrderedWebFilter(loginPageFilter, SecurityWebFiltersOrder.LOGIN_PAGE_GENERATING.getOrder()));
- this.webFilters.add(new OrderedWebFilter(new LogoutPageGeneratingWebFilter(), SecurityWebFiltersOrder.LOGOUT_PAGE_GENERATING.getOrder()));
- }
this.formLogin.configure(this);
}
if (this.oauth2Login != null) {
- this.oauth2Login.configure(loginPageFilter, this);
+ this.oauth2Login.configure(this);
+ }
+ if (this.oauth2 != null) {
+ this.oauth2.configure(this);
}
+ this.loginPage.configure(this);
if(this.logout != null) {
this.logout.configure(this);
}
@@ -638,8 +895,8 @@ public SecurityWebFilterChain build() {
exceptionTranslationWebFilter.setAuthenticationEntryPoint(
authenticationEntryPoint);
}
- if(accessDeniedHandler != null) {
- exceptionTranslationWebFilter.setAccessDeniedHandler(accessDeniedHandler);
+ if(this.accessDeniedHandler != null) {
+ exceptionTranslationWebFilter.setAccessDeniedHandler(this.accessDeniedHandler);
}
this.addFilterAt(exceptionTranslationWebFilter, SecurityWebFiltersOrder.EXCEPTION_TRANSLATION);
this.authorizeExchange.configure(this);
@@ -724,7 +981,7 @@ public ServerHttpSecurity and() {
/**
* Disables authorization.
- * @return the {@link ServerHttpSecurity} to continue configuring
+ * @return the {@link Access} to continue configuring
*/
@Override
public Access anyExchange() {
@@ -1075,6 +1332,8 @@ public class FormLoginSpec {
private ServerAuthenticationEntryPoint authenticationEntryPoint;
+ private boolean isEntryPointExplicit;
+
private ServerWebExchangeMatcher requiresAuthenticationMatcher;
private ServerAuthenticationFailureHandler authenticationFailureHandler;
@@ -1197,7 +1456,10 @@ public ServerHttpSecurity disable() {
protected void configure(ServerHttpSecurity http) {
if(this.authenticationEntryPoint == null) {
+ this.isEntryPointExplicit = false;
loginPage("/login");
+ } else {
+ this.isEntryPointExplicit = true;
}
if(http.requestCache != null) {
ServerRequestCache requestCache = http.requestCache.requestCache;
@@ -1224,6 +1486,35 @@ private FormLoginSpec() {
}
}
+ private class LoginPageSpec {
+ protected void configure(ServerHttpSecurity http) {
+ if (http.authenticationEntryPoint != null) {
+ return;
+ }
+ if (http.formLogin != null && http.formLogin.isEntryPointExplicit) {
+ return;
+ }
+ LoginPageGeneratingWebFilter loginPage = null;
+ if (http.formLogin != null && !http.formLogin.isEntryPointExplicit) {
+ loginPage = new LoginPageGeneratingWebFilter();
+ loginPage.setFormLoginEnabled(true);
+ }
+ if (http.oauth2Login != null) {
+ Map urlToText = http.oauth2Login.getLinks();
+ if (loginPage == null) {
+ loginPage = new LoginPageGeneratingWebFilter();
+ }
+ loginPage.setOauth2AuthenticationUrlToClientName(urlToText);
+ }
+ if (loginPage != null) {
+ http.addFilterAt(loginPage, SecurityWebFiltersOrder.LOGIN_PAGE_GENERATING);
+ http.addFilterAt(new LogoutPageGeneratingWebFilter(), SecurityWebFiltersOrder.LOGOUT_PAGE_GENERATING);
+ }
+ }
+
+ private LoginPageSpec() {}
+ }
+
/**
* Configures HTTP Response Headers.
*
@@ -1367,7 +1658,7 @@ public HeaderSpec and() {
/**
* Disables frame options response header
- * @return the {@link ServerHttpSecurity} to continue configuring
+ * @return the {@link HeaderSpec} to continue configuring
*/
public HeaderSpec disable() {
HeaderSpec.this.writers.remove(HeaderSpec.this.frameOptions);
@@ -1385,17 +1676,21 @@ public class HstsSpec {
/**
* Configures the max age. Default is one year.
* @param maxAge the max age
+ * @return the {@link HstsSpec} to continue configuring
*/
- public void maxAge(Duration maxAge) {
+ public HstsSpec maxAge(Duration maxAge) {
HeaderSpec.this.hsts.setMaxAge(maxAge);
+ return this;
}
/**
* Configures if subdomains should be included. Default is true
* @param includeSubDomains if subdomains should be included
+ * @return the {@link HstsSpec} to continue configuring
*/
- public void includeSubdomains(boolean includeSubDomains) {
+ public HstsSpec includeSubdomains(boolean includeSubDomains) {
HeaderSpec.this.hsts.setIncludeSubDomains(includeSubDomains);
+ return this;
}
/**
@@ -1408,7 +1703,7 @@ public HeaderSpec and() {
/**
* Disables strict transport security response header
- * @return the {@link ServerHttpSecurity} to continue configuring
+ * @return the {@link HeaderSpec} to continue configuring
*/
public HeaderSpec disable() {
HeaderSpec.this.writers.remove(HeaderSpec.this.hsts);
@@ -1425,7 +1720,7 @@ private HstsSpec() {}
public class XssProtectionSpec {
/**
* Disables the x-xss-protection response header
- * @return
+ * @return the {@link HeaderSpec} to continue configuring
*/
public HeaderSpec disable() {
HeaderSpec.this.writers.remove(HeaderSpec.this.xss);
@@ -1454,7 +1749,7 @@ public final class LogoutSpec {
/**
* Configures the logout handler. Default is {@code SecurityContextServerLogoutHandler}
* @param logoutHandler
- * @return
+ * @return the {@link LogoutSpec} to configure
*/
public LogoutSpec logoutHandler(ServerLogoutHandler logoutHandler) {
this.logoutWebFilter.setLogoutHandler(logoutHandler);
diff --git a/config/src/main/java/org/springframework/security/config/websocket/WebSocketMessageBrokerSecurityBeanDefinitionParser.java b/config/src/main/java/org/springframework/security/config/websocket/WebSocketMessageBrokerSecurityBeanDefinitionParser.java
index afa29975af6..8165cca7e35 100644
--- a/config/src/main/java/org/springframework/security/config/websocket/WebSocketMessageBrokerSecurityBeanDefinitionParser.java
+++ b/config/src/main/java/org/springframework/security/config/websocket/WebSocketMessageBrokerSecurityBeanDefinitionParser.java
@@ -110,7 +110,7 @@ public final class WebSocketMessageBrokerSecurityBeanDefinitionParser implements
/**
* @param element
* @param parserContext
- * @return
+ * @return the {@link BeanDefinition}
*/
public BeanDefinition parse(Element element, ParserContext parserContext) {
BeanDefinitionRegistry registry = parserContext.getRegistry();
diff --git a/config/src/main/resources/META-INF/spring.schemas b/config/src/main/resources/META-INF/spring.schemas
index 97d4844af25..fe5518e1347 100644
--- a/config/src/main/resources/META-INF/spring.schemas
+++ b/config/src/main/resources/META-INF/spring.schemas
@@ -1,4 +1,5 @@
-http\://www.springframework.org/schema/security/spring-security.xsd=org/springframework/security/config/spring-security-5.0.xsd
+http\://www.springframework.org/schema/security/spring-security.xsd=org/springframework/security/config/spring-security-5.1.xsd
+http\://www.springframework.org/schema/security/spring-security-5.1.xsd=org/springframework/security/config/spring-security-5.1.xsd
http\://www.springframework.org/schema/security/spring-security-5.0.xsd=org/springframework/security/config/spring-security-5.0.xsd
http\://www.springframework.org/schema/security/spring-security-4.2.xsd=org/springframework/security/config/spring-security-4.2.xsd
http\://www.springframework.org/schema/security/spring-security-4.1.xsd=org/springframework/security/config/spring-security-4.1.xsd
diff --git a/config/src/main/resources/org/springframework/security/config/spring-security-5.1.rnc b/config/src/main/resources/org/springframework/security/config/spring-security-5.1.rnc
new file mode 100644
index 00000000000..af761497c96
--- /dev/null
+++ b/config/src/main/resources/org/springframework/security/config/spring-security-5.1.rnc
@@ -0,0 +1,916 @@
+namespace a = "http://relaxng.org/ns/compatibility/annotations/1.0"
+datatypes xsd = "http://www.w3.org/2001/XMLSchema-datatypes"
+
+default namespace = "http://www.springframework.org/schema/security"
+
+start = http | ldap-server | authentication-provider | ldap-authentication-provider | any-user-service | ldap-server | ldap-authentication-provider
+
+hash =
+ ## Defines the hashing algorithm used on user passwords. Bcrypt is recommended.
+ attribute hash {"bcrypt"}
+base64 =
+ ## Whether a string should be base64 encoded
+ attribute base64 {xsd:boolean}
+request-matcher =
+ ## Defines the strategy use for matching incoming requests. Currently the options are 'mvc' (for Spring MVC matcher), 'ant' (for ant path patterns), 'regex' for regular expressions and 'ciRegex' for case-insensitive regular expressions.
+ attribute request-matcher {"mvc" | "ant" | "regex" | "ciRegex"}
+port =
+ ## Specifies an IP port number. Used to configure an embedded LDAP server, for example.
+ attribute port { xsd:positiveInteger }
+url =
+ ## Specifies a URL.
+ attribute url { xsd:token }
+id =
+ ## A bean identifier, used for referring to the bean elsewhere in the context.
+ attribute id {xsd:token}
+name =
+ ## A bean identifier, used for referring to the bean elsewhere in the context.
+ attribute name {xsd:token}
+ref =
+ ## Defines a reference to a Spring bean Id.
+ attribute ref {xsd:token}
+
+cache-ref =
+ ## Defines a reference to a cache for use with a UserDetailsService.
+ attribute cache-ref {xsd:token}
+
+user-service-ref =
+ ## A reference to a user-service (or UserDetailsService bean) Id
+ attribute user-service-ref {xsd:token}
+
+authentication-manager-ref =
+ ## A reference to an AuthenticationManager bean
+ attribute authentication-manager-ref {xsd:token}
+
+data-source-ref =
+ ## A reference to a DataSource bean
+ attribute data-source-ref {xsd:token}
+
+
+
+debug =
+ ## Enables Spring Security debugging infrastructure. This will provide human-readable (multi-line) debugging information to monitor requests coming into the security filters. This may include sensitive information, such as request parameters or headers, and should only be used in a development environment.
+ element debug {empty}
+
+password-encoder =
+ ## element which defines a password encoding strategy. Used by an authentication provider to convert submitted passwords to hashed versions, for example.
+ element password-encoder {password-encoder.attlist}
+password-encoder.attlist &=
+ ref | (hash)
+
+role-prefix =
+ ## A non-empty string prefix that will be added to role strings loaded from persistent storage (e.g. "ROLE_"). Use the value "none" for no prefix in cases where the default is non-empty.
+ attribute role-prefix {xsd:token}
+
+use-expressions =
+ ## Enables the use of expressions in the 'access' attributes in elements rather than the traditional list of configuration attributes. Defaults to 'true'. If enabled, each attribute should contain a single boolean expression. If the expression evaluates to 'true', access will be granted.
+ attribute use-expressions {xsd:boolean}
+
+ldap-server =
+ ## Defines an LDAP server location or starts an embedded server. The url indicates the location of a remote server. If no url is given, an embedded server will be started, listening on the supplied port number. The port is optional and defaults to 33389. A Spring LDAP ContextSource bean will be registered for the server with the id supplied.
+ element ldap-server {ldap-server.attlist}
+ldap-server.attlist &= id?
+ldap-server.attlist &= (url | port)?
+ldap-server.attlist &=
+ ## Username (DN) of the "manager" user identity which will be used to authenticate to a (non-embedded) LDAP server. If omitted, anonymous access will be used.
+ attribute manager-dn {xsd:string}?
+ldap-server.attlist &=
+ ## The password for the manager DN. This is required if the manager-dn is specified.
+ attribute manager-password {xsd:string}?
+ldap-server.attlist &=
+ ## Explicitly specifies an ldif file resource to load into an embedded LDAP server. The default is classpath*:*.ldiff
+ attribute ldif { xsd:string }?
+ldap-server.attlist &=
+ ## Optional root suffix for the embedded LDAP server. Default is "dc=springframework,dc=org"
+ attribute root { xsd:string }?
+
+ldap-server-ref-attribute =
+ ## The optional server to use. If omitted, and a default LDAP server is registered (using with no Id), that server will be used.
+ attribute server-ref {xsd:token}
+
+
+group-search-filter-attribute =
+ ## Group search filter. Defaults to (uniqueMember={0}). The substituted parameter is the DN of the user.
+ attribute group-search-filter {xsd:token}
+group-search-base-attribute =
+ ## Search base for group membership searches. Defaults to "" (searching from the root).
+ attribute group-search-base {xsd:token}
+user-search-filter-attribute =
+ ## The LDAP filter used to search for users (optional). For example "(uid={0})". The substituted parameter is the user's login name.
+ attribute user-search-filter {xsd:token}
+user-search-base-attribute =
+ ## Search base for user searches. Defaults to "". Only used with a 'user-search-filter'.
+ attribute user-search-base {xsd:token}
+group-role-attribute-attribute =
+ ## The LDAP attribute name which contains the role name which will be used within Spring Security. Defaults to "cn".
+ attribute group-role-attribute {xsd:token}
+user-details-class-attribute =
+ ## Allows the objectClass of the user entry to be specified. If set, the framework will attempt to load standard attributes for the defined class into the returned UserDetails object
+ attribute user-details-class {"person" | "inetOrgPerson"}
+user-context-mapper-attribute =
+ ## Allows explicit customization of the loaded user object by specifying a UserDetailsContextMapper bean which will be called with the context information from the user's directory entry
+ attribute user-context-mapper-ref {xsd:token}
+
+
+ldap-user-service =
+ ## This element configures a LdapUserDetailsService which is a combination of a FilterBasedLdapUserSearch and a DefaultLdapAuthoritiesPopulator.
+ element ldap-user-service {ldap-us.attlist}
+ldap-us.attlist &= id?
+ldap-us.attlist &=
+ ldap-server-ref-attribute?
+ldap-us.attlist &=
+ user-search-filter-attribute?
+ldap-us.attlist &=
+ user-search-base-attribute?
+ldap-us.attlist &=
+ group-search-filter-attribute?
+ldap-us.attlist &=
+ group-search-base-attribute?
+ldap-us.attlist &=
+ group-role-attribute-attribute?
+ldap-us.attlist &=
+ cache-ref?
+ldap-us.attlist &=
+ role-prefix?
+ldap-us.attlist &=
+ (user-details-class-attribute | user-context-mapper-attribute)?
+
+ldap-authentication-provider =
+ ## Sets up an ldap authentication provider
+ element ldap-authentication-provider {ldap-ap.attlist, password-compare-element?}
+ldap-ap.attlist &=
+ ldap-server-ref-attribute?
+ldap-ap.attlist &=
+ user-search-base-attribute?
+ldap-ap.attlist &=
+ user-search-filter-attribute?
+ldap-ap.attlist &=
+ group-search-base-attribute?
+ldap-ap.attlist &=
+ group-search-filter-attribute?
+ldap-ap.attlist &=
+ group-role-attribute-attribute?
+ldap-ap.attlist &=
+ ## A specific pattern used to build the user's DN, for example "uid={0},ou=people". The key "{0}" must be present and will be substituted with the username.
+ attribute user-dn-pattern {xsd:token}?
+ldap-ap.attlist &=
+ role-prefix?
+ldap-ap.attlist &=
+ (user-details-class-attribute | user-context-mapper-attribute)?
+
+password-compare-element =
+ ## Specifies that an LDAP provider should use an LDAP compare operation of the user's password to authenticate the user
+ element password-compare {password-compare.attlist, password-encoder?}
+
+password-compare.attlist &=
+ ## The attribute in the directory which contains the user password. Defaults to "userPassword".
+ attribute password-attribute {xsd:token}?
+password-compare.attlist &=
+ hash?
+
+intercept-methods =
+ ## Can be used inside a bean definition to add a security interceptor to the bean and set up access configuration attributes for the bean's methods
+ element intercept-methods {intercept-methods.attlist, protect+}
+intercept-methods.attlist &=
+ ## Optional AccessDecisionManager bean ID to be used by the created method security interceptor.
+ attribute access-decision-manager-ref {xsd:token}?
+
+
+protect =
+ ## Defines a protected method and the access control configuration attributes that apply to it. We strongly advise you NOT to mix "protect" declarations with any services provided "global-method-security".
+ element protect {protect.attlist, empty}
+protect.attlist &=
+ ## A method name
+ attribute method {xsd:token}
+protect.attlist &=
+ ## Access configuration attributes list that applies to the method, e.g. "ROLE_A,ROLE_B".
+ attribute access {xsd:token}
+
+method-security-metadata-source =
+ ## Creates a MethodSecurityMetadataSource instance
+ element method-security-metadata-source {msmds.attlist, protect+}
+msmds.attlist &= id?
+
+msmds.attlist &= use-expressions?
+
+global-method-security =
+ ## Provides method security for all beans registered in the Spring application context. Specifically, beans will be scanned for matches with the ordered list of "protect-pointcut" sub-elements, Spring Security annotations and/or. Where there is a match, the beans will automatically be proxied and security authorization applied to the methods accordingly. If you use and enable all four sources of method security metadata (ie "protect-pointcut" declarations, expression annotations, @Secured and also JSR250 security annotations), the metadata sources will be queried in that order. In practical terms, this enables you to use XML to override method security metadata expressed in annotations. If using annotations, the order of precedence is EL-based (@PreAuthorize etc.), @Secured and finally JSR-250.
+ element global-method-security {global-method-security.attlist, (pre-post-annotation-handling | expression-handler)?, protect-pointcut*, after-invocation-provider*}
+global-method-security.attlist &=
+ ## Specifies whether the use of Spring Security's pre and post invocation annotations (@PreFilter, @PreAuthorize, @PostFilter, @PostAuthorize) should be enabled for this application context. Defaults to "disabled".
+ attribute pre-post-annotations {"disabled" | "enabled" }?
+global-method-security.attlist &=
+ ## Specifies whether the use of Spring Security's @Secured annotations should be enabled for this application context. Defaults to "disabled".
+ attribute secured-annotations {"disabled" | "enabled" }?
+global-method-security.attlist &=
+ ## Specifies whether JSR-250 style attributes are to be used (for example "RolesAllowed"). This will require the javax.annotation.security classes on the classpath. Defaults to "disabled".
+ attribute jsr250-annotations {"disabled" | "enabled" }?
+global-method-security.attlist &=
+ ## Optional AccessDecisionManager bean ID to override the default used for method security.
+ attribute access-decision-manager-ref {xsd:token}?
+global-method-security.attlist &=
+ ## Optional RunAsmanager implementation which will be used by the configured MethodSecurityInterceptor
+ attribute run-as-manager-ref {xsd:token}?
+global-method-security.attlist &=
+ ## Allows the advice "order" to be set for the method security interceptor.
+ attribute order {xsd:token}?
+global-method-security.attlist &=
+ ## If true, class based proxying will be used instead of interface based proxying.
+ attribute proxy-target-class {xsd:boolean}?
+global-method-security.attlist &=
+ ## Can be used to specify that AspectJ should be used instead of the default Spring AOP. If set, secured classes must be woven with the AnnotationSecurityAspect from the spring-security-aspects module.
+ attribute mode {"aspectj"}?
+global-method-security.attlist &=
+ ## An external MethodSecurityMetadataSource instance can be supplied which will take priority over other sources (such as the default annotations).
+ attribute metadata-source-ref {xsd:token}?
+global-method-security.attlist &=
+ authentication-manager-ref?
+
+
+after-invocation-provider =
+ ## Allows addition of extra AfterInvocationProvider beans which should be called by the MethodSecurityInterceptor created by global-method-security.
+ element after-invocation-provider {ref}
+
+pre-post-annotation-handling =
+ ## Allows the default expression-based mechanism for handling Spring Security's pre and post invocation annotations (@PreFilter, @PreAuthorize, @PostFilter, @PostAuthorize) to be replace entirely. Only applies if these annotations are enabled.
+ element pre-post-annotation-handling {invocation-attribute-factory, pre-invocation-advice, post-invocation-advice}
+
+invocation-attribute-factory =
+ ## Defines the PrePostInvocationAttributeFactory instance which is used to generate pre and post invocation metadata from the annotated methods.
+ element invocation-attribute-factory {ref}
+
+pre-invocation-advice =
+ ## Customizes the PreInvocationAuthorizationAdviceVoter with the ref as the PreInvocationAuthorizationAdviceVoter for the element.
+ element pre-invocation-advice {ref}
+
+post-invocation-advice =
+ ## Customizes the PostInvocationAdviceProvider with the ref as the PostInvocationAuthorizationAdvice for the element.
+ element post-invocation-advice {ref}
+
+
+expression-handler =
+ ## Defines the SecurityExpressionHandler instance which will be used if expression-based access-control is enabled. A default implementation (with no ACL support) will be used if not supplied.
+ element expression-handler {ref}
+
+protect-pointcut =
+ ## Defines a protected pointcut and the access control configuration attributes that apply to it. Every bean registered in the Spring application context that provides a method that matches the pointcut will receive security authorization.
+ element protect-pointcut {protect-pointcut.attlist, empty}
+protect-pointcut.attlist &=
+ ## An AspectJ expression, including the 'execution' keyword. For example, 'execution(int com.foo.TargetObject.countLength(String))' (without the quotes).
+ attribute expression {xsd:string}
+protect-pointcut.attlist &=
+ ## Access configuration attributes list that applies to all methods matching the pointcut, e.g. "ROLE_A,ROLE_B"
+ attribute access {xsd:token}
+
+websocket-message-broker =
+ ## Allows securing a Message Broker. There are two modes. If no id is specified: ensures that any SimpAnnotationMethodMessageHandler has the AuthenticationPrincipalArgumentResolver registered as a custom argument resolver; ensures that the SecurityContextChannelInterceptor is automatically registered for the clientInboundChannel; and that a ChannelSecurityInterceptor is registered with the clientInboundChannel. If the id is specified, creates a ChannelSecurityInterceptor that can be manually registered with the clientInboundChannel.
+ element websocket-message-broker { websocket-message-broker.attrlist, (intercept-message* & expression-handler?) }
+
+websocket-message-broker.attrlist &=
+ ## A bean identifier, used for referring to the bean elsewhere in the context. If specified, explicit configuration within clientInboundChannel is required. If not specified, ensures that any SimpAnnotationMethodMessageHandler has the AuthenticationPrincipalArgumentResolver registered as a custom argument resolver; ensures that the SecurityContextChannelInterceptor is automatically registered for the clientInboundChannel; and that a ChannelSecurityInterceptor is registered with the clientInboundChannel.
+ attribute id {xsd:token}?
+websocket-message-broker.attrlist &=
+ ## Disables the requirement for CSRF token to be present in the Stomp headers (default false). Changing the default is useful if it is necessary to allow other origins to make SockJS connections.
+ attribute same-origin-disabled {xsd:boolean}?
+
+intercept-message =
+ ## Creates an authorization rule for a websocket message.
+ element intercept-message {intercept-message.attrlist}
+
+intercept-message.attrlist &=
+ ## The destination ant pattern which will be mapped to the access attribute. For example, /** matches any message with a destination, /admin/** matches any message that has a destination that starts with admin.
+ attribute pattern {xsd:token}?
+intercept-message.attrlist &=
+ ## The access configuration attributes that apply for the configured message. For example, permitAll grants access to anyone, hasRole('ROLE_ADMIN') requires the user have the role 'ROLE_ADMIN'.
+ attribute access {xsd:token}?
+intercept-message.attrlist &=
+ ## The type of message to match on. Valid values are defined in SimpMessageType (i.e. CONNECT, CONNECT_ACK, HEARTBEAT, MESSAGE, SUBSCRIBE, UNSUBSCRIBE, DISCONNECT, DISCONNECT_ACK, OTHER).
+ attribute type {"CONNECT" | "CONNECT_ACK" | "HEARTBEAT" | "MESSAGE" | "SUBSCRIBE"| "UNSUBSCRIBE" | "DISCONNECT" | "DISCONNECT_ACK" | "OTHER"}?
+
+http-firewall =
+ ## Allows a custom instance of HttpFirewall to be injected into the FilterChainProxy created by the namespace.
+ element http-firewall {ref}
+
+http =
+ ## Container element for HTTP security configuration. Multiple elements can now be defined, each with a specific pattern to which the enclosed security configuration applies. A pattern can also be configured to bypass Spring Security's filters completely by setting the "security" attribute to "none".
+ element http {http.attlist, (intercept-url* & access-denied-handler? & form-login? & openid-login? & x509? & jee? & http-basic? & logout? & session-management & remember-me? & anonymous? & port-mappings & custom-filter* & request-cache? & expression-handler? & headers? & csrf? & cors?) }
+http.attlist &=
+ ## The request URL pattern which will be mapped to the filter chain created by this element. If omitted, the filter chain will match all requests.
+ attribute pattern {xsd:token}?
+http.attlist &=
+ ## When set to 'none', requests matching the pattern attribute will be ignored by Spring Security. No security filters will be applied and no SecurityContext will be available. If set, the element must be empty, with no children.
+ attribute security {"none"}?
+http.attlist &=
+ ## Allows a RequestMatcher instance to be used, as an alternative to pattern-matching.
+ attribute request-matcher-ref { xsd:token }?
+http.attlist &=
+ ## A legacy attribute which automatically registers a login form, BASIC authentication and a logout URL and logout services. If unspecified, defaults to "false". We'd recommend you avoid using this and instead explicitly configure the services you require.
+ attribute auto-config {xsd:boolean}?
+http.attlist &=
+ use-expressions?
+http.attlist &=
+ ## Controls the eagerness with which an HTTP session is created by Spring Security classes. If not set, defaults to "ifRequired". If "stateless" is used, this implies that the application guarantees that it will not create a session. This differs from the use of "never" which means that Spring Security will not create a session, but will make use of one if the application does.
+ attribute create-session {"ifRequired" | "always" | "never" | "stateless"}?
+http.attlist &=
+ ## A reference to a SecurityContextRepository bean. This can be used to customize how the SecurityContext is stored between requests.
+ attribute security-context-repository-ref {xsd:token}?
+http.attlist &=
+ request-matcher?
+http.attlist &=
+ ## Provides versions of HttpServletRequest security methods such as isUserInRole() and getPrincipal() which are implemented by accessing the Spring SecurityContext. Defaults to "true".
+ attribute servlet-api-provision {xsd:boolean}?
+http.attlist &=
+ ## If available, runs the request as the Subject acquired from the JaasAuthenticationToken. Defaults to "false".
+ attribute jaas-api-provision {xsd:boolean}?
+http.attlist &=
+ ## Optional attribute specifying the ID of the AccessDecisionManager implementation which should be used for authorizing HTTP requests.
+ attribute access-decision-manager-ref {xsd:token}?
+http.attlist &=
+ ## Optional attribute specifying the realm name that will be used for all authentication features that require a realm name (eg BASIC and Digest authentication). If unspecified, defaults to "Spring Security Application".
+ attribute realm {xsd:token}?
+http.attlist &=
+ ## Allows a customized AuthenticationEntryPoint to be set on the ExceptionTranslationFilter.
+ attribute entry-point-ref {xsd:token}?
+http.attlist &=
+ ## Corresponds to the observeOncePerRequest property of FilterSecurityInterceptor. Defaults to "true"
+ attribute once-per-request {xsd:boolean}?
+http.attlist &=
+ ## Prevents the jsessionid parameter from being added to rendered URLs. Defaults to "true" (rewriting is disabled).
+ attribute disable-url-rewriting {xsd:boolean}?
+http.attlist &=
+ ## Exposes the list of filters defined by this configuration under this bean name in the application context.
+ name?
+http.attlist &=
+ authentication-manager-ref?
+
+access-denied-handler =
+ ## Defines the access-denied strategy that should be used. An access denied page can be defined or a reference to an AccessDeniedHandler instance.
+ element access-denied-handler {access-denied-handler.attlist, empty}
+access-denied-handler.attlist &= (ref | access-denied-handler-page)
+
+access-denied-handler-page =
+ ## The access denied page that an authenticated user will be redirected to if they request a page which they don't have the authority to access.
+ attribute error-page {xsd:token}
+
+intercept-url =
+ ## Specifies the access attributes and/or filter list for a particular set of URLs.
+ element intercept-url {intercept-url.attlist, empty}
+intercept-url.attlist &=
+ (pattern | request-matcher-ref)
+intercept-url.attlist &=
+ ## The access configuration attributes that apply for the configured path.
+ attribute access {xsd:token}?
+intercept-url.attlist &=
+ ## The HTTP Method for which the access configuration attributes should apply. If not specified, the attributes will apply to any method.
+ attribute method {"GET" | "DELETE" | "HEAD" | "OPTIONS" | "POST" | "PUT" | "PATCH" | "TRACE"}?
+
+intercept-url.attlist &=
+ ## The filter list for the path. Currently can be set to "none" to remove a path from having any filters applied. The full filter stack (consisting of all filters created by the namespace configuration, and any added using 'custom-filter'), will be applied to any other paths.
+ attribute filters {"none"}?
+intercept-url.attlist &=
+ ## Used to specify that a URL must be accessed over http or https, or that there is no preference. The value should be "http", "https" or "any", respectively.
+ attribute requires-channel {xsd:token}?
+intercept-url.attlist &=
+ ## The path to the servlet. This attribute is only applicable when 'request-matcher' is 'mvc'. In addition, the value is only required in the following 2 use cases: 1) There are 2 or more HttpServlet's registered in the ServletContext that have mappings starting with '/' and are different; 2) The pattern starts with the same value of a registered HttpServlet path, excluding the default (root) HttpServlet '/'.
+ attribute servlet-path {xsd:token}?
+
+logout =
+ ## Incorporates a logout processing filter. Most web applications require a logout filter, although you may not require one if you write a controller to provider similar logic.
+ element logout {logout.attlist, empty}
+logout.attlist &=
+ ## Specifies the URL that will cause a logout. Spring Security will initialize a filter that responds to this particular URL. Defaults to /logout if unspecified.
+ attribute logout-url {xsd:token}?
+logout.attlist &=
+ ## Specifies the URL to display once the user has logged out. If not specified, defaults to /?logout (i.e. /login?logout).
+ attribute logout-success-url {xsd:token}?
+logout.attlist &=
+ ## Specifies whether a logout also causes HttpSession invalidation, which is generally desirable. If unspecified, defaults to true.
+ attribute invalidate-session {xsd:boolean}?
+logout.attlist &=
+ ## A reference to a LogoutSuccessHandler implementation which will be used to determine the destination to which the user is taken after logging out.
+ attribute success-handler-ref {xsd:token}?
+logout.attlist &=
+ ## A comma-separated list of the names of cookies which should be deleted when the user logs out
+ attribute delete-cookies {xsd:token}?
+
+request-cache =
+ ## Allow the RequestCache used for saving requests during the login process to be set
+ element request-cache {ref}
+
+form-login =
+ ## Sets up a form login configuration for authentication with a username and password
+ element form-login {form-login.attlist, empty}
+form-login.attlist &=
+ ## The URL that the login form is posted to. If unspecified, it defaults to /login.
+ attribute login-processing-url {xsd:token}?
+form-login.attlist &=
+ ## The name of the request parameter which contains the username. Defaults to 'username'.
+ attribute username-parameter {xsd:token}?
+form-login.attlist &=
+ ## The name of the request parameter which contains the password. Defaults to 'password'.
+ attribute password-parameter {xsd:token}?
+form-login.attlist &=
+ ## The URL that will be redirected to after successful authentication, if the user's previous action could not be resumed. This generally happens if the user visits a login page without having first requested a secured operation that triggers authentication. If unspecified, defaults to the root of the application.
+ attribute default-target-url {xsd:token}?
+form-login.attlist &=
+ ## Whether the user should always be redirected to the default-target-url after login.
+ attribute always-use-default-target {xsd:boolean}?
+form-login.attlist &=
+ ## The URL for the login page. If no login URL is specified, Spring Security will automatically create a login URL at GET /login and a corresponding filter to render that login URL when requested.
+ attribute login-page {xsd:token}?
+form-login.attlist &=
+ ## The URL for the login failure page. If no login failure URL is specified, Spring Security will automatically create a failure login URL at /login?error and a corresponding filter to render that login failure URL when requested.
+ attribute authentication-failure-url {xsd:token}?
+form-login.attlist &=
+ ## Reference to an AuthenticationSuccessHandler bean which should be used to handle a successful authentication request. Should not be used in combination with default-target-url (or always-use-default-target-url) as the implementation should always deal with navigation to the subsequent destination
+ attribute authentication-success-handler-ref {xsd:token}?
+form-login.attlist &=
+ ## Reference to an AuthenticationFailureHandler bean which should be used to handle a failed authentication request. Should not be used in combination with authentication-failure-url as the implementation should always deal with navigation to the subsequent destination
+ attribute authentication-failure-handler-ref {xsd:token}?
+form-login.attlist &=
+ ## Reference to an AuthenticationDetailsSource which will be used by the authentication filter
+ attribute authentication-details-source-ref {xsd:token}?
+form-login.attlist &=
+ ## The URL for the ForwardAuthenticationFailureHandler
+ attribute authentication-failure-forward-url {xsd:token}?
+form-login.attlist &=
+ ## The URL for the ForwardAuthenticationSuccessHandler
+ attribute authentication-success-forward-url {xsd:token}?
+
+
+openid-login =
+ ## Sets up form login for authentication with an Open ID identity
+ element openid-login {form-login.attlist, user-service-ref?, attribute-exchange*}
+
+attribute-exchange =
+ ## Sets up an attribute exchange configuration to request specified attributes from the OpenID identity provider. When multiple elements are used, each must have an identifier-attribute attribute. Each configuration will be matched in turn against the supplied login identifier until a match is found.
+ element attribute-exchange {attribute-exchange.attlist, openid-attribute+}
+
+attribute-exchange.attlist &=
+ ## A regular expression which will be compared against the claimed identity, when deciding which attribute-exchange configuration to use during authentication.
+ attribute identifier-match {xsd:token}?
+
+openid-attribute =
+ ## Attributes used when making an OpenID AX Fetch Request
+ element openid-attribute {openid-attribute.attlist}
+
+openid-attribute.attlist &=
+ ## Specifies the name of the attribute that you wish to get back. For example, email.
+ attribute name {xsd:token}
+openid-attribute.attlist &=
+ ## Specifies the attribute type. For example, http://axschema.org/contact/email. See your OP's documentation for valid attribute types.
+ attribute type {xsd:token}
+openid-attribute.attlist &=
+ ## Specifies if this attribute is required to the OP, but does not error out if the OP does not return the attribute. Default is false.
+ attribute required {xsd:boolean}?
+openid-attribute.attlist &=
+ ## Specifies the number of attributes that you wish to get back. For example, return 3 emails. The default value is 1.
+ attribute count {xsd:int}?
+
+
+filter-chain-map =
+ ## Used to explicitly configure a FilterChainProxy instance with a FilterChainMap
+ element filter-chain-map {filter-chain-map.attlist, filter-chain+}
+filter-chain-map.attlist &=
+ request-matcher?
+
+filter-chain =
+ ## Used within to define a specific URL pattern and the list of filters which apply to the URLs matching that pattern. When multiple filter-chain elements are assembled in a list in order to configure a FilterChainProxy, the most specific patterns must be placed at the top of the list, with most general ones at the bottom.
+ element filter-chain {filter-chain.attlist, empty}
+filter-chain.attlist &=
+ (pattern | request-matcher-ref)
+filter-chain.attlist &=
+ ## A comma separated list of bean names that implement Filter that should be processed for this FilterChain. If the value is none, then no Filters will be used for this FilterChain.
+ attribute filters {xsd:token}
+
+pattern =
+ ## The request URL pattern which will be mapped to the FilterChain.
+ attribute pattern {xsd:token}
+request-matcher-ref =
+ ## Allows a RequestMatcher instance to be used, as an alternative to pattern-matching.
+ attribute request-matcher-ref {xsd:token}
+
+filter-security-metadata-source =
+ ## Used to explicitly configure a FilterSecurityMetadataSource bean for use with a FilterSecurityInterceptor. Usually only needed if you are configuring a FilterChainProxy explicitly, rather than using the element. The intercept-url elements used should only contain pattern, method and access attributes. Any others will result in a configuration error.
+ element filter-security-metadata-source {fsmds.attlist, intercept-url+}
+fsmds.attlist &=
+ use-expressions?
+fsmds.attlist &=
+ id?
+fsmds.attlist &=
+ request-matcher?
+
+http-basic =
+ ## Adds support for basic authentication
+ element http-basic {http-basic.attlist, empty}
+
+http-basic.attlist &=
+ ## Sets the AuthenticationEntryPoint which is used by the BasicAuthenticationFilter.
+ attribute entry-point-ref {xsd:token}?
+http-basic.attlist &=
+ ## Reference to an AuthenticationDetailsSource which will be used by the authentication filter
+ attribute authentication-details-source-ref {xsd:token}?
+
+session-management =
+ ## Session-management related functionality is implemented by the addition of a SessionManagementFilter to the filter stack.
+ element session-management {session-management.attlist, concurrency-control?}
+
+session-management.attlist &=
+ ## Indicates how session fixation protection will be applied when a user authenticates. If set to "none", no protection will be applied. "newSession" will create a new empty session, with only Spring Security-related attributes migrated. "migrateSession" will create a new session and copy all session attributes to the new session. In Servlet 3.1 (Java EE 7) and newer containers, specifying "changeSessionId" will keep the existing session and use the container-supplied session fixation protection (HttpServletRequest#changeSessionId()). Defaults to "changeSessionId" in Servlet 3.1 and newer containers, "migrateSession" in older containers. Throws an exception if "changeSessionId" is used in older containers.
+ attribute session-fixation-protection {"none" | "newSession" | "migrateSession" | "changeSessionId" }?
+session-management.attlist &=
+ ## The URL to which a user will be redirected if they submit an invalid session indentifier. Typically used to detect session timeouts.
+ attribute invalid-session-url {xsd:token}?
+session-management.attlist &=
+ ## Allows injection of the InvalidSessionStrategy instance used by the SessionManagementFilter
+ attribute invalid-session-strategy-ref {xsd:token}?
+session-management.attlist &=
+ ## Allows injection of the SessionAuthenticationStrategy instance used by the SessionManagementFilter
+ attribute session-authentication-strategy-ref {xsd:token}?
+session-management.attlist &=
+ ## Defines the URL of the error page which should be shown when the SessionAuthenticationStrategy raises an exception. If not set, an unauthorized (401) error code will be returned to the client. Note that this attribute doesn't apply if the error occurs during a form-based login, where the URL for authentication failure will take precedence.
+ attribute session-authentication-error-url {xsd:token}?
+
+
+concurrency-control =
+ ## Enables concurrent session control, limiting the number of authenticated sessions a user may have at the same time.
+ element concurrency-control {concurrency-control.attlist, empty}
+
+concurrency-control.attlist &=
+ ## The maximum number of sessions a single authenticated user can have open at the same time. Defaults to "1". A negative value denotes unlimited sessions.
+ attribute max-sessions {xsd:integer}?
+concurrency-control.attlist &=
+ ## The URL a user will be redirected to if they attempt to use a session which has been "expired" because they have logged in again.
+ attribute expired-url {xsd:token}?
+concurrency-control.attlist &=
+ ## Allows injection of the SessionInformationExpiredStrategy instance used by the ConcurrentSessionFilter
+ attribute expired-session-strategy-ref {xsd:token}?
+concurrency-control.attlist &=
+ ## Specifies that an unauthorized error should be reported when a user attempts to login when they already have the maximum configured sessions open. The default behaviour is to expire the original session. If the session-authentication-error-url attribute is set on the session-management URL, the user will be redirected to this URL.
+ attribute error-if-maximum-exceeded {xsd:boolean}?
+concurrency-control.attlist &=
+ ## Allows you to define an alias for the SessionRegistry bean in order to access it in your own configuration.
+ attribute session-registry-alias {xsd:token}?
+concurrency-control.attlist &=
+ ## Allows you to define an external SessionRegistry bean to be used by the concurrency control setup.
+ attribute session-registry-ref {xsd:token}?
+
+
+remember-me =
+ ## Sets up remember-me authentication. If used with the "key" attribute (or no attributes) the cookie-only implementation will be used. Specifying "token-repository-ref" or "remember-me-data-source-ref" will use the more secure, persisten token approach.
+ element remember-me {remember-me.attlist}
+remember-me.attlist &=
+ ## The "key" used to identify cookies from a specific token-based remember-me application. You should set this to a unique value for your application. If unset, it will default to a random value generated by SecureRandom.
+ attribute key {xsd:token}?
+
+remember-me.attlist &=
+ (token-repository-ref | remember-me-data-source-ref | remember-me-services-ref)
+
+remember-me.attlist &=
+ user-service-ref?
+
+remember-me.attlist &=
+ ## Exports the internally defined RememberMeServices as a bean alias, allowing it to be used by other beans in the application context.
+ attribute services-alias {xsd:token}?
+
+remember-me.attlist &=
+ ## Determines whether the "secure" flag will be set on the remember-me cookie. If set to true, the cookie will only be submitted over HTTPS (recommended). By default, secure cookies will be used if the request is made on a secure connection.
+ attribute use-secure-cookie {xsd:boolean}?
+
+remember-me.attlist &=
+ ## The period (in seconds) for which the remember-me cookie should be valid.
+ attribute token-validity-seconds {xsd:string}?
+
+remember-me.attlist &=
+ ## Reference to an AuthenticationSuccessHandler bean which should be used to handle a successful remember-me authentication.
+ attribute authentication-success-handler-ref {xsd:token}?
+remember-me.attlist &=
+ ## The name of the request parameter which toggles remember-me authentication. Defaults to 'remember-me'.
+ attribute remember-me-parameter {xsd:token}?
+remember-me.attlist &=
+ ## The name of cookie which store the token for remember-me authentication. Defaults to 'remember-me'.
+ attribute remember-me-cookie {xsd:token}?
+
+token-repository-ref =
+ ## Reference to a PersistentTokenRepository bean for use with the persistent token remember-me implementation.
+ attribute token-repository-ref {xsd:token}
+remember-me-services-ref =
+ ## Allows a custom implementation of RememberMeServices to be used. Note that this implementation should return RememberMeAuthenticationToken instances with the same "key" value as specified in the remember-me element. Alternatively it should register its own AuthenticationProvider. It should also implement the LogoutHandler interface, which will be invoked when a user logs out. Typically the remember-me cookie would be removed on logout.
+ attribute services-ref {xsd:token}?
+remember-me-data-source-ref =
+ ## DataSource bean for the database that contains the token repository schema.
+ data-source-ref
+
+anonymous =
+ ## Adds support for automatically granting all anonymous web requests a particular principal identity and a corresponding granted authority.
+ element anonymous {anonymous.attlist}
+anonymous.attlist &=
+ ## The key shared between the provider and filter. This generally does not need to be set. If unset, it will default to a random value generated by SecureRandom.
+ attribute key {xsd:token}?
+anonymous.attlist &=
+ ## The username that should be assigned to the anonymous request. This allows the principal to be identified, which may be important for logging and auditing. if unset, defaults to "anonymousUser".
+ attribute username {xsd:token}?
+anonymous.attlist &=
+ ## The granted authority that should be assigned to the anonymous request. Commonly this is used to assign the anonymous request particular roles, which can subsequently be used in authorization decisions. If unset, defaults to "ROLE_ANONYMOUS".
+ attribute granted-authority {xsd:token}?
+anonymous.attlist &=
+ ## With the default namespace setup, the anonymous "authentication" facility is automatically enabled. You can disable it using this property.
+ attribute enabled {xsd:boolean}?
+
+
+port-mappings =
+ ## Defines the list of mappings between http and https ports for use in redirects
+ element port-mappings {port-mappings.attlist, port-mapping+}
+
+port-mappings.attlist &= empty
+
+port-mapping =
+ ## Provides a method to map http ports to https ports when forcing a redirect.
+ element port-mapping {http-port, https-port}
+
+http-port =
+ ## The http port to use.
+ attribute http {xsd:token}
+
+https-port =
+ ## The https port to use.
+ attribute https {xsd:token}
+
+
+x509 =
+ ## Adds support for X.509 client authentication.
+ element x509 {x509.attlist}
+x509.attlist &=
+ ## The regular expression used to obtain the username from the certificate's subject. Defaults to matching on the common name using the pattern "CN=(.*?),".
+ attribute subject-principal-regex {xsd:token}?
+x509.attlist &=
+ ## Explicitly specifies which user-service should be used to load user data for X.509 authenticated clients. If ommitted, the default user-service will be used.
+ user-service-ref?
+x509.attlist &=
+ ## Reference to an AuthenticationDetailsSource which will be used by the authentication filter
+ attribute authentication-details-source-ref {xsd:token}?
+
+jee =
+ ## Adds a J2eePreAuthenticatedProcessingFilter to the filter chain to provide integration with container authentication.
+ element jee {jee.attlist}
+jee.attlist &=
+ ## A comma-separate list of roles to look for in the incoming HttpServletRequest.
+ attribute mappable-roles {xsd:token}
+jee.attlist &=
+ ## Explicitly specifies which user-service should be used to load user data for container authenticated clients. If ommitted, the set of mappable-roles will be used to construct the authorities for the user.
+ user-service-ref?
+
+authentication-manager =
+ ## Registers the AuthenticationManager instance and allows its list of AuthenticationProviders to be defined. Also allows you to define an alias to allow you to reference the AuthenticationManager in your own beans.
+ element authentication-manager {authman.attlist & authentication-provider* & ldap-authentication-provider*}
+authman.attlist &=
+ id?
+authman.attlist &=
+ ## An alias you wish to use for the AuthenticationManager bean (not required it you are using a specific id)
+ attribute alias {xsd:token}?
+authman.attlist &=
+ ## If set to true, the AuthenticationManger will attempt to clear any credentials data in the returned Authentication object, once the user has been authenticated.
+ attribute erase-credentials {xsd:boolean}?
+
+authentication-provider =
+ ## Indicates that the contained user-service should be used as an authentication source.
+ element authentication-provider {ap.attlist & any-user-service & password-encoder?}
+ap.attlist &=
+ ## Specifies a reference to a separately configured AuthenticationProvider instance which should be registered within the AuthenticationManager.
+ ref?
+ap.attlist &=
+ ## Specifies a reference to a separately configured UserDetailsService from which to obtain authentication data.
+ user-service-ref?
+
+user-service =
+ ## Creates an in-memory UserDetailsService from a properties file or a list of "user" child elements. Usernames are converted to lower-case internally to allow for case-insensitive lookups, so this should not be used if case-sensitivity is required.
+ element user-service {id? & (properties-file | (user*))}
+properties-file =
+ ## The location of a Properties file where each line is in the format of username=password,grantedAuthority[,grantedAuthority][,enabled|disabled]
+ attribute properties {xsd:token}?
+
+user =
+ ## Represents a user in the application.
+ element user {user.attlist, empty}
+user.attlist &=
+ ## The username assigned to the user.
+ attribute name {xsd:token}
+user.attlist &=
+ ## The password assigned to the user. This may be hashed if the corresponding authentication provider supports hashing (remember to set the "hash" attribute of the "user-service" element). This attribute be omitted in the case where the data will not be used for authentication, but only for accessing authorities. If omitted, the namespace will generate a random value, preventing its accidental use for authentication. Cannot be empty.
+ attribute password {xsd:string}?
+user.attlist &=
+ ## One of more authorities granted to the user. Separate authorities with a comma (but no space). For example, "ROLE_USER,ROLE_ADMINISTRATOR"
+ attribute authorities {xsd:token}
+user.attlist &=
+ ## Can be set to "true" to mark an account as locked and unusable.
+ attribute locked {xsd:boolean}?
+user.attlist &=
+ ## Can be set to "true" to mark an account as disabled and unusable.
+ attribute disabled {xsd:boolean}?
+
+jdbc-user-service =
+ ## Causes creation of a JDBC-based UserDetailsService.
+ element jdbc-user-service {id? & jdbc-user-service.attlist}
+jdbc-user-service.attlist &=
+ ## The bean ID of the DataSource which provides the required tables.
+ attribute data-source-ref {xsd:token}
+jdbc-user-service.attlist &=
+ cache-ref?
+jdbc-user-service.attlist &=
+ ## An SQL statement to query a username, password, and enabled status given a username. Default is "select username,password,enabled from users where username = ?"
+ attribute users-by-username-query {xsd:token}?
+jdbc-user-service.attlist &=
+ ## An SQL statement to query for a user's granted authorities given a username. The default is "select username, authority from authorities where username = ?"
+ attribute authorities-by-username-query {xsd:token}?
+jdbc-user-service.attlist &=
+ ## An SQL statement to query user's group authorities given a username. The default is "select g.id, g.group_name, ga.authority from groups g, group_members gm, group_authorities ga where gm.username = ? and g.id = ga.group_id and g.id = gm.group_id"
+ attribute group-authorities-by-username-query {xsd:token}?
+jdbc-user-service.attlist &=
+ role-prefix?
+
+csrf =
+## Element for configuration of the CsrfFilter for protection against CSRF. It also updates the default RequestCache to only replay "GET" requests.
+ element csrf {csrf-options.attlist}
+csrf-options.attlist &=
+ ## Specifies if csrf protection should be disabled. Default false (i.e. CSRF protection is enabled).
+ attribute disabled {xsd:boolean}?
+csrf-options.attlist &=
+ ## The RequestMatcher instance to be used to determine if CSRF should be applied. Default is any HTTP method except "GET", "TRACE", "HEAD", "OPTIONS"
+ attribute request-matcher-ref { xsd:token }?
+csrf-options.attlist &=
+ ## The CsrfTokenRepository to use. The default is HttpSessionCsrfTokenRepository
+ attribute token-repository-ref { xsd:token }?
+
+headers =
+## Element for configuration of the HeaderWritersFilter. Enables easy setting for the X-Frame-Options, X-XSS-Protection and X-Content-Type-Options headers.
+element headers { headers-options.attlist, (cache-control? & xss-protection? & hsts? & frame-options? & content-type-options? & hpkp? & content-security-policy? & referrer-policy? & feature-policy? & header*)}
+headers-options.attlist &=
+ ## Specifies if the default headers should be disabled. Default false.
+ attribute defaults-disabled {xsd:boolean}?
+headers-options.attlist &=
+ ## Specifies if headers should be disabled. Default false.
+ attribute disabled {xsd:boolean}?
+hsts =
+ ## Adds support for HTTP Strict Transport Security (HSTS)
+ element hsts {hsts-options.attlist}
+hsts-options.attlist &=
+ ## Specifies if HTTP Strict Transport Security (HSTS) should be disabled. Default false.
+ attribute disabled {xsd:boolean}?
+hsts-options.attlist &=
+ ## Specifies if subdomains should be included. Default true.
+ attribute include-subdomains {xsd:boolean}?
+hsts-options.attlist &=
+ ## Specifies the maximum ammount of time the host should be considered a Known HSTS Host. Default one year.
+ attribute max-age-seconds {xsd:integer}?
+hsts-options.attlist &=
+ ## The RequestMatcher instance to be used to determine if the header should be set. Default is if HttpServletRequest.isSecure() is true.
+ attribute request-matcher-ref { xsd:token }?
+
+cors =
+## Element for configuration of CorsFilter. If no CorsFilter or CorsConfigurationSource is specified a HandlerMappingIntrospector is used as the CorsConfigurationSource
+element cors { cors-options.attlist }
+cors-options.attlist &=
+ ref?
+cors-options.attlist &=
+ ## Specifies a bean id that is a CorsConfigurationSource used to construct the CorsFilter to use
+ attribute configuration-source-ref {xsd:token}?
+
+hpkp =
+ ## Adds support for HTTP Public Key Pinning (HPKP).
+ element hpkp {hpkp.pins,hpkp.attlist}
+hpkp.pins =
+ ## The list with pins
+ element pins {hpkp.pin+}
+hpkp.pin =
+ ## A pin is specified using the base64-encoded SPKI fingerprint as value and the cryptographic hash algorithm as attribute
+ element pin {
+ ## The cryptographic hash algorithm
+ attribute algorithm { xsd:string }?,
+ text
+ }
+hpkp.attlist &=
+ ## Specifies if HTTP Public Key Pinning (HPKP) should be disabled. Default false.
+ attribute disabled {xsd:boolean}?
+hpkp.attlist &=
+ ## Specifies if subdomains should be included. Default false.
+ attribute include-subdomains {xsd:boolean}?
+hpkp.attlist &=
+ ## Sets the value for the max-age directive of the Public-Key-Pins header. Default 60 days.
+ attribute max-age-seconds {xsd:integer}?
+hpkp.attlist &=
+ ## Specifies if the browser should only report pin validation failures. Default true.
+ attribute report-only {xsd:boolean}?
+hpkp.attlist &=
+ ## Specifies the URI to which the browser should report pin validation failures.
+ attribute report-uri {xsd:string}?
+
+content-security-policy =
+ ## Adds support for Content Security Policy (CSP)
+ element content-security-policy {csp-options.attlist}
+csp-options.attlist &=
+ ## The security policy directive(s) for the Content-Security-Policy header or if report-only is set to true, then the Content-Security-Policy-Report-Only header is used.
+ attribute policy-directives {xsd:token}?
+csp-options.attlist &=
+ ## Set to true, to enable the Content-Security-Policy-Report-Only header for reporting policy violations only. Defaults to false.
+ attribute report-only {xsd:boolean}?
+
+referrer-policy =
+ ## Adds support for Referrer Policy
+ element referrer-policy {referrer-options.attlist}
+referrer-options.attlist &=
+ ## The policies for the Referrer-Policy header.
+ attribute policy {"no-referrer","no-referrer-when-downgrade","same-origin","origin","strict-origin","origin-when-cross-origin","strict-origin-when-cross-origin","unsafe-url"}?
+
+feature-policy =
+ ## Adds support for Feature Policy
+ element feature-policy {feature-options.attlist}
+feature-options.attlist &=
+ ## The security policy directive(s) for the Feature-Policy header.
+ attribute policy-directives {xsd:token}?
+
+cache-control =
+ ## Adds Cache-Control no-cache, no-store, must-revalidate, Pragma no-cache, and Expires 0 for every request
+ element cache-control {cache-control.attlist}
+cache-control.attlist &=
+ ## Specifies if Cache Control should be disabled. Default false.
+ attribute disabled {xsd:boolean}?
+
+frame-options =
+ ## Enable basic clickjacking support for newer browsers (IE8+), will set the X-Frame-Options header.
+ element frame-options {frame-options.attlist,empty}
+frame-options.attlist &=
+ ## If disabled, the X-Frame-Options header will not be included. Default false.
+ attribute disabled {xsd:boolean}?
+frame-options.attlist &=
+ ## Specify the policy to use for the X-Frame-Options-Header.
+ attribute policy {"DENY","SAMEORIGIN","ALLOW-FROM"}?
+frame-options.attlist &=
+ ## Specify the strategy to use when ALLOW-FROM is chosen.
+ attribute strategy {"static","whitelist","regexp"}?
+frame-options.attlist &=
+ ## Specify a reference to the custom AllowFromStrategy to use when ALLOW-FROM is chosen.
+ ref?
+frame-options.attlist &=
+ ## Specify a value to use for the chosen strategy.
+ attribute value {xsd:string}?
+frame-options.attlist &=
+ ## Specify the request parameter to use for the origin when using a 'whitelist' or 'regexp' based strategy. Default is 'from'.
+ attribute from-parameter {xsd:string}?
+
+
+xss-protection =
+ ## Enable basic XSS browser protection, supported by newer browsers (IE8+), will set the X-XSS-Protection header.
+ element xss-protection {xss-protection.attlist,empty}
+xss-protection.attlist &=
+ ## disable the X-XSS-Protection header. Default is 'false' meaning it is enabled.
+ attribute disabled {xsd:boolean}?
+xss-protection.attlist &=
+ ## specify that XSS Protection should be explicitly enabled or disabled. Default is 'true' meaning it is enabled.
+ attribute enabled {xsd:boolean}?
+xss-protection.attlist &=
+ ## Add mode=block to the header or not, default is on.
+ attribute block {xsd:boolean}?
+
+content-type-options =
+ ## Add a X-Content-Type-Options header to the resopnse. Value is always 'nosniff'.
+ element content-type-options {content-type-options.attlist, empty}
+content-type-options.attlist &=
+ ## If disabled, the X-Content-Type-Options header will not be included. Default false.
+ attribute disabled {xsd:boolean}?
+
+header=
+ ## Add additional headers to the response.
+ element header {header.attlist}
+header.attlist &=
+ ## The name of the header to add.
+ attribute name {xsd:token}?
+header.attlist &=
+ ## The value for the header.
+ attribute value {xsd:token}?
+header.attlist &=
+ ## Reference to a custom HeaderWriter implementation.
+ ref?
+
+any-user-service = user-service | jdbc-user-service | ldap-user-service
+
+custom-filter =
+ ## Used to indicate that a filter bean declaration should be incorporated into the security filter chain.
+ element custom-filter {custom-filter.attlist}
+
+custom-filter.attlist &=
+ ref
+
+custom-filter.attlist &=
+ (after | before | position)
+
+after =
+ ## The filter immediately after which the custom-filter should be placed in the chain. This feature will only be needed by advanced users who wish to mix their own filters into the security filter chain and have some knowledge of the standard Spring Security filters. The filter names map to specific Spring Security implementation filters.
+ attribute after {named-security-filter}
+before =
+ ## The filter immediately before which the custom-filter should be placed in the chain
+ attribute before {named-security-filter}
+position =
+ ## The explicit position at which the custom-filter should be placed in the chain. Use if you are replacing a standard filter.
+ attribute position {named-security-filter}
+
+named-security-filter = "FIRST" | "CHANNEL_FILTER" | "SECURITY_CONTEXT_FILTER" | "CONCURRENT_SESSION_FILTER" | "WEB_ASYNC_MANAGER_FILTER" | "HEADERS_FILTER" | "CORS_FILTER" | "CSRF_FILTER" | "LOGOUT_FILTER" | "X509_FILTER" | "PRE_AUTH_FILTER" | "CAS_FILTER" | "FORM_LOGIN_FILTER" | "OPENID_FILTER" | "LOGIN_PAGE_FILTER" |"LOGOUT_PAGE_FILTER" | "DIGEST_AUTH_FILTER" | "BEARER_TOKEN_AUTH_FILTER" | "BASIC_AUTH_FILTER" | "REQUEST_CACHE_FILTER" | "SERVLET_API_SUPPORT_FILTER" | "JAAS_API_SUPPORT_FILTER" | "REMEMBER_ME_FILTER" | "ANONYMOUS_FILTER" | "SESSION_MANAGEMENT_FILTER" | "EXCEPTION_TRANSLATION_FILTER" | "FILTER_SECURITY_INTERCEPTOR" | "SWITCH_USER_FILTER" | "LAST"
diff --git a/config/src/main/resources/org/springframework/security/config/spring-security-5.1.xsd b/config/src/main/resources/org/springframework/security/config/spring-security-5.1.xsd
new file mode 100644
index 00000000000..e6c0f5a52ac
--- /dev/null
+++ b/config/src/main/resources/org/springframework/security/config/spring-security-5.1.xsd
@@ -0,0 +1,2738 @@
+
+
+
+
+
+ Defines the hashing algorithm used on user passwords. Bcrypt is recommended.
+
+
+
+
+
+
+
+
+
+
+
+
+ Whether a string should be base64 encoded
+
+
+
+
+
+
+
+ Defines the strategy use for matching incoming requests. Currently the options are 'mvc'
+ (for Spring MVC matcher), 'ant' (for ant path patterns), 'regex' for regular expressions
+ and 'ciRegex' for case-insensitive regular expressions.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Specifies an IP port number. Used to configure an embedded LDAP server, for example.
+
+
+
+
+
+
+
+ Specifies a URL.
+
+
+
+
+
+
+
+ A bean identifier, used for referring to the bean elsewhere in the context.
+
+
+
+
+
+
+
+ A bean identifier, used for referring to the bean elsewhere in the context.
+
+
+
+
+
+
+
+ Defines a reference to a Spring bean Id.
+
+
+
+
+
+
+
+ Defines a reference to a cache for use with a UserDetailsService.
+
+
+
+
+
+
+
+ A reference to a user-service (or UserDetailsService bean) Id
+
+
+
+
+
+
+
+ A reference to an AuthenticationManager bean
+
+
+
+
+
+
+
+ A reference to a DataSource bean
+
+
+
+
+
+
+ Enables Spring Security debugging infrastructure. This will provide human-readable
+ (multi-line) debugging information to monitor requests coming into the security filters.
+ This may include sensitive information, such as request parameters or headers, and should
+ only be used in a development environment.
+
+
+
+
+
+
+
+
+ Defines a reference to a Spring bean Id.
+
+
+
+
+
+ Defines the hashing algorithm used on user passwords. Bcrypt is recommended.
+
+
+
+
+
+
+
+
+
+
+
+
+ A non-empty string prefix that will be added to role strings loaded from persistent
+ storage (e.g. "ROLE_"). Use the value "none" for no prefix in cases where the default is
+ non-empty.
+
+
+
+
+
+
+
+ Enables the use of expressions in the 'access' attributes in <intercept-url> elements
+ rather than the traditional list of configuration attributes. Defaults to 'true'. If
+ enabled, each attribute should contain a single boolean expression. If the expression
+ evaluates to 'true', access will be granted.
+
+
+
+
+
+
+ Defines an LDAP server location or starts an embedded server. The url indicates the
+ location of a remote server. If no url is given, an embedded server will be started,
+ listening on the supplied port number. The port is optional and defaults to 33389. A
+ Spring LDAP ContextSource bean will be registered for the server with the id supplied.
+
+
+
+
+
+
+
+
+
+ A bean identifier, used for referring to the bean elsewhere in the context.
+
+
+
+
+
+ Specifies a URL.
+
+
+
+
+
+ Specifies an IP port number. Used to configure an embedded LDAP server, for example.
+
+
+
+
+
+ Username (DN) of the "manager" user identity which will be used to authenticate to a
+ (non-embedded) LDAP server. If omitted, anonymous access will be used.
+
+
+
+
+
+ The password for the manager DN. This is required if the manager-dn is specified.
+
+
+
+
+
+ Explicitly specifies an ldif file resource to load into an embedded LDAP server. The
+ default is classpath*:*.ldiff
+
+
+
+
+
+ Optional root suffix for the embedded LDAP server. Default is "dc=springframework,dc=org"
+
+
+
+
+
+
+
+ The optional server to use. If omitted, and a default LDAP server is registered (using
+ <ldap-server> with no Id), that server will be used.
+
+
+
+
+
+
+
+ Group search filter. Defaults to (uniqueMember={0}). The substituted parameter is the DN
+ of the user.
+
+
+
+
+
+
+
+ Search base for group membership searches. Defaults to "" (searching from the root).
+
+
+
+
+
+
+
+ The LDAP filter used to search for users (optional). For example "(uid={0})". The
+ substituted parameter is the user's login name.
+
+
+
+
+
+
+
+ Search base for user searches. Defaults to "". Only used with a 'user-search-filter'.
+
+
+
+
+
+
+
+ The LDAP attribute name which contains the role name which will be used within Spring
+ Security. Defaults to "cn".
+
+
+
+
+
+
+
+ Allows the objectClass of the user entry to be specified. If set, the framework will
+ attempt to load standard attributes for the defined class into the returned UserDetails
+ object
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Allows explicit customization of the loaded user object by specifying a
+ UserDetailsContextMapper bean which will be called with the context information from the
+ user's directory entry
+
+
+
+
+
+
+ This element configures a LdapUserDetailsService which is a combination of a
+ FilterBasedLdapUserSearch and a DefaultLdapAuthoritiesPopulator.
+
+
+
+
+
+
+
+
+
+ A bean identifier, used for referring to the bean elsewhere in the context.
+
+
+
+
+
+ The optional server to use. If omitted, and a default LDAP server is registered (using
+ <ldap-server> with no Id), that server will be used.
+
+
+
+
+
+ The LDAP filter used to search for users (optional). For example "(uid={0})". The
+ substituted parameter is the user's login name.
+
+
+
+
+
+ Search base for user searches. Defaults to "". Only used with a 'user-search-filter'.
+
+
+
+
+
+ Group search filter. Defaults to (uniqueMember={0}). The substituted parameter is the DN
+ of the user.
+
+
+
+
+
+ Search base for group membership searches. Defaults to "" (searching from the root).
+
+
+
+
+
+ The LDAP attribute name which contains the role name which will be used within Spring
+ Security. Defaults to "cn".
+
+
+
+
+
+ Defines a reference to a cache for use with a UserDetailsService.
+
+
+
+
+
+ A non-empty string prefix that will be added to role strings loaded from persistent
+ storage (e.g. "ROLE_"). Use the value "none" for no prefix in cases where the default is
+ non-empty.
+
+
+
+
+
+ Allows the objectClass of the user entry to be specified. If set, the framework will
+ attempt to load standard attributes for the defined class into the returned UserDetails
+ object
+
+
+
+
+
+
+
+
+
+
+
+ Allows explicit customization of the loaded user object by specifying a
+ UserDetailsContextMapper bean which will be called with the context information from the
+ user's directory entry
+
+
+
+
+
+
+
+
+ The optional server to use. If omitted, and a default LDAP server is registered (using
+ <ldap-server> with no Id), that server will be used.
+
+
+
+
+
+ Search base for user searches. Defaults to "". Only used with a 'user-search-filter'.
+
+
+
+
+
+ The LDAP filter used to search for users (optional). For example "(uid={0})". The
+ substituted parameter is the user's login name.
+
+
+
+
+
+ Search base for group membership searches. Defaults to "" (searching from the root).
+
+
+
+
+
+ Group search filter. Defaults to (uniqueMember={0}). The substituted parameter is the DN
+ of the user.
+
+
+
+
+
+ The LDAP attribute name which contains the role name which will be used within Spring
+ Security. Defaults to "cn".
+
+
+
+
+
+ A specific pattern used to build the user's DN, for example "uid={0},ou=people". The key
+ "{0}" must be present and will be substituted with the username.
+
+
+
+
+
+ A non-empty string prefix that will be added to role strings loaded from persistent
+ storage (e.g. "ROLE_"). Use the value "none" for no prefix in cases where the default is
+ non-empty.
+
+
+
+
+
+ Allows the objectClass of the user entry to be specified. If set, the framework will
+ attempt to load standard attributes for the defined class into the returned UserDetails
+ object
+
+
+
+
+
+
+
+
+
+
+
+ Allows explicit customization of the loaded user object by specifying a
+ UserDetailsContextMapper bean which will be called with the context information from the
+ user's directory entry
+
+
+
+
+
+
+
+
+ The attribute in the directory which contains the user password. Defaults to
+ "userPassword".
+
+
+
+
+
+ Defines the hashing algorithm used on user passwords. Bcrypt is recommended.
+
+
+
+
+
+
+
+
+
+
+
+ Can be used inside a bean definition to add a security interceptor to the bean and set up
+ access configuration attributes for the bean's methods
+
+
+
+
+
+
+ Defines a protected method and the access control configuration attributes that apply to
+ it. We strongly advise you NOT to mix "protect" declarations with any services provided
+ "global-method-security".
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Optional AccessDecisionManager bean ID to be used by the created method security
+ interceptor.
+
+
+
+
+
+
+
+
+ A method name
+
+
+
+
+
+ Access configuration attributes list that applies to the method, e.g. "ROLE_A,ROLE_B".
+
+
+
+
+
+
+ Creates a MethodSecurityMetadataSource instance
+
+
+
+
+
+
+ Defines a protected method and the access control configuration attributes that apply to
+ it. We strongly advise you NOT to mix "protect" declarations with any services provided
+ "global-method-security".
+
+
+
+
+
+
+
+
+
+
+
+
+
+ A bean identifier, used for referring to the bean elsewhere in the context.
+
+
+
+
+
+ Enables the use of expressions in the 'access' attributes in <intercept-url> elements
+ rather than the traditional list of configuration attributes. Defaults to 'true'. If
+ enabled, each attribute should contain a single boolean expression. If the expression
+ evaluates to 'true', access will be granted.
+
+
+
+
+
+
+ Provides method security for all beans registered in the Spring application context.
+ Specifically, beans will be scanned for matches with the ordered list of
+ "protect-pointcut" sub-elements, Spring Security annotations and/or. Where there is a
+ match, the beans will automatically be proxied and security authorization applied to the
+ methods accordingly. If you use and enable all four sources of method security metadata
+ (ie "protect-pointcut" declarations, expression annotations, @Secured and also JSR250
+ security annotations), the metadata sources will be queried in that order. In practical
+ terms, this enables you to use XML to override method security metadata expressed in
+ annotations. If using annotations, the order of precedence is EL-based (@PreAuthorize
+ etc.), @Secured and finally JSR-250.
+
+
+
+
+
+
+
+ Allows the default expression-based mechanism for handling Spring Security's pre and post
+ invocation annotations (@PreFilter, @PreAuthorize, @PostFilter, @PostAuthorize) to be
+ replace entirely. Only applies if these annotations are enabled.
+
+
+
+
+
+
+ Defines the PrePostInvocationAttributeFactory instance which is used to generate pre and
+ post invocation metadata from the annotated methods.
+
+
+
+
+
+
+
+
+ Customizes the PreInvocationAuthorizationAdviceVoter with the ref as the
+ PreInvocationAuthorizationAdviceVoter for the <pre-post-annotation-handling> element.
+
+
+
+
+
+
+
+
+ Customizes the PostInvocationAdviceProvider with the ref as the
+ PostInvocationAuthorizationAdvice for the <pre-post-annotation-handling> element.
+
+
+
+
+
+
+
+
+
+
+
+ Defines the SecurityExpressionHandler instance which will be used if expression-based
+ access-control is enabled. A default implementation (with no ACL support) will be used if
+ not supplied.
+
+
+
+
+
+
+
+
+
+ Defines a protected pointcut and the access control configuration attributes that apply to
+ it. Every bean registered in the Spring application context that provides a method that
+ matches the pointcut will receive security authorization.
+
+
+
+
+
+
+
+
+ Allows addition of extra AfterInvocationProvider beans which should be called by the
+ MethodSecurityInterceptor created by global-method-security.
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Specifies whether the use of Spring Security's pre and post invocation annotations
+ (@PreFilter, @PreAuthorize, @PostFilter, @PostAuthorize) should be enabled for this
+ application context. Defaults to "disabled".
+
+
+
+
+
+
+
+
+
+
+
+ Specifies whether the use of Spring Security's @Secured annotations should be enabled for
+ this application context. Defaults to "disabled".
+
+
+
+
+
+
+
+
+
+
+
+ Specifies whether JSR-250 style attributes are to be used (for example "RolesAllowed").
+ This will require the javax.annotation.security classes on the classpath. Defaults to
+ "disabled".
+
+
+
+
+
+
+
+
+
+
+
+ Optional AccessDecisionManager bean ID to override the default used for method security.
+
+
+
+
+
+ Optional RunAsmanager implementation which will be used by the configured
+ MethodSecurityInterceptor
+
+
+
+
+
+ Allows the advice "order" to be set for the method security interceptor.
+
+
+
+
+
+ If true, class based proxying will be used instead of interface based proxying.
+
+
+
+
+
+ Can be used to specify that AspectJ should be used instead of the default Spring AOP. If
+ set, secured classes must be woven with the AnnotationSecurityAspect from the
+ spring-security-aspects module.
+
+
+
+
+
+
+
+
+
+
+ An external MethodSecurityMetadataSource instance can be supplied which will take priority
+ over other sources (such as the default annotations).
+
+
+
+
+
+ A reference to an AuthenticationManager bean
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ An AspectJ expression, including the 'execution' keyword. For example, 'execution(int
+ com.foo.TargetObject.countLength(String))' (without the quotes).
+
+
+
+
+
+ Access configuration attributes list that applies to all methods matching the pointcut,
+ e.g. "ROLE_A,ROLE_B"
+
+
+
+
+
+
+ Allows securing a Message Broker. There are two modes. If no id is specified: ensures that
+ any SimpAnnotationMethodMessageHandler has the AuthenticationPrincipalArgumentResolver
+ registered as a custom argument resolver; ensures that the
+ SecurityContextChannelInterceptor is automatically registered for the
+ clientInboundChannel; and that a ChannelSecurityInterceptor is registered with the
+ clientInboundChannel. If the id is specified, creates a ChannelSecurityInterceptor that
+ can be manually registered with the clientInboundChannel.
+
+
+
+
+
+
+
+ Defines the SecurityExpressionHandler instance which will be used if expression-based
+ access-control is enabled. A default implementation (with no ACL support) will be used if
+ not supplied.
+
+
+
+
+
+
+
+
+
+
+
+
+
+ A bean identifier, used for referring to the bean elsewhere in the context. If specified,
+ explicit configuration within clientInboundChannel is required. If not specified, ensures
+ that any SimpAnnotationMethodMessageHandler has the
+ AuthenticationPrincipalArgumentResolver registered as a custom argument resolver; ensures
+ that the SecurityContextChannelInterceptor is automatically registered for the
+ clientInboundChannel; and that a ChannelSecurityInterceptor is registered with the
+ clientInboundChannel.
+
+
+
+
+
+ Disables the requirement for CSRF token to be present in the Stomp headers (default
+ false). Changing the default is useful if it is necessary to allow other origins to make
+ SockJS connections.
+
+
+
+
+
+
+ Creates an authorization rule for a websocket message.
+
+
+
+
+
+
+
+
+
+ The destination ant pattern which will be mapped to the access attribute. For example, /**
+ matches any message with a destination, /admin/** matches any message that has a
+ destination that starts with admin.
+
+
+
+
+
+ The access configuration attributes that apply for the configured message. For example,
+ permitAll grants access to anyone, hasRole('ROLE_ADMIN') requires the user have the role
+ 'ROLE_ADMIN'.
+
+
+
+
+
+ The type of message to match on. Valid values are defined in SimpMessageType (i.e.
+ CONNECT, CONNECT_ACK, HEARTBEAT, MESSAGE, SUBSCRIBE, UNSUBSCRIBE, DISCONNECT,
+ DISCONNECT_ACK, OTHER).
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Allows a custom instance of HttpFirewall to be injected into the FilterChainProxy created
+ by the namespace.
+
+
+
+
+
+
+
+
+ Container element for HTTP security configuration. Multiple elements can now be defined,
+ each with a specific pattern to which the enclosed security configuration applies. A
+ pattern can also be configured to bypass Spring Security's filters completely by setting
+ the "security" attribute to "none".
+
+
+
+
+
+
+ Specifies the access attributes and/or filter list for a particular set of URLs.
+
+
+
+
+
+
+
+
+ Defines the access-denied strategy that should be used. An access denied page can be
+ defined or a reference to an AccessDeniedHandler instance.
+
+
+
+
+
+
+
+
+ Sets up a form login configuration for authentication with a username and password
+
+
+
+
+
+
+
+
+ Sets up form login for authentication with an Open ID identity
+
+
+
+
+
+
+
+
+
+ A reference to a user-service (or UserDetailsService bean) Id
+
+
+
+
+
+
+
+ Adds support for X.509 client authentication.
+
+
+
+
+
+
+
+
+
+ Adds support for basic authentication
+
+
+
+
+
+
+
+
+ Incorporates a logout processing filter. Most web applications require a logout filter,
+ although you may not require one if you write a controller to provider similar logic.
+
+
+
+
+
+
+
+
+ Session-management related functionality is implemented by the addition of a
+ SessionManagementFilter to the filter stack.
+
+
+
+
+
+
+ Enables concurrent session control, limiting the number of authenticated sessions a user
+ may have at the same time.
+
+
+
+
+
+
+
+
+
+
+
+
+ Sets up remember-me authentication. If used with the "key" attribute (or no attributes)
+ the cookie-only implementation will be used. Specifying "token-repository-ref" or
+ "remember-me-data-source-ref" will use the more secure, persisten token approach.
+
+
+
+
+
+
+
+
+ Adds support for automatically granting all anonymous web requests a particular principal
+ identity and a corresponding granted authority.
+
+
+
+
+
+
+
+
+ Defines the list of mappings between http and https ports for use in redirects
+
+
+
+
+
+
+ Provides a method to map http ports to https ports when forcing a redirect.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Defines the SecurityExpressionHandler instance which will be used if expression-based
+ access-control is enabled. A default implementation (with no ACL support) will be used if
+ not supplied.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ The request URL pattern which will be mapped to the filter chain created by this <http>
+ element. If omitted, the filter chain will match all requests.
+
+
+
+
+
+ When set to 'none', requests matching the pattern attribute will be ignored by Spring
+ Security. No security filters will be applied and no SecurityContext will be available. If
+ set, the <http> element must be empty, with no children.
+
+
+
+
+
+
+
+
+
+
+ Allows a RequestMatcher instance to be used, as an alternative to pattern-matching.
+
+
+
+
+
+ A legacy attribute which automatically registers a login form, BASIC authentication and a
+ logout URL and logout services. If unspecified, defaults to "false". We'd recommend you
+ avoid using this and instead explicitly configure the services you require.
+
+
+
+
+
+ Enables the use of expressions in the 'access' attributes in <intercept-url> elements
+ rather than the traditional list of configuration attributes. Defaults to 'true'. If
+ enabled, each attribute should contain a single boolean expression. If the expression
+ evaluates to 'true', access will be granted.
+
+
+
+
+
+ Controls the eagerness with which an HTTP session is created by Spring Security classes.
+ If not set, defaults to "ifRequired". If "stateless" is used, this implies that the
+ application guarantees that it will not create a session. This differs from the use of
+ "never" which means that Spring Security will not create a session, but will make use of
+ one if the application does.
+
+
+
+
+
+
+
+
+
+
+
+
+
+ A reference to a SecurityContextRepository bean. This can be used to customize how the
+ SecurityContext is stored between requests.
+
+
+
+
+
+ Defines the strategy use for matching incoming requests. Currently the options are 'mvc'
+ (for Spring MVC matcher), 'ant' (for ant path patterns), 'regex' for regular expressions
+ and 'ciRegex' for case-insensitive regular expressions.
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Provides versions of HttpServletRequest security methods such as isUserInRole() and
+ getPrincipal() which are implemented by accessing the Spring SecurityContext. Defaults to
+ "true".
+
+
+
+
+
+ If available, runs the request as the Subject acquired from the JaasAuthenticationToken.
+ Defaults to "false".
+
+
+
+
+
+ Optional attribute specifying the ID of the AccessDecisionManager implementation which
+ should be used for authorizing HTTP requests.
+
+
+
+
+
+ Optional attribute specifying the realm name that will be used for all authentication
+ features that require a realm name (eg BASIC and Digest authentication). If unspecified,
+ defaults to "Spring Security Application".
+
+
+
+
+
+ Allows a customized AuthenticationEntryPoint to be set on the ExceptionTranslationFilter.
+
+
+
+
+
+ Corresponds to the observeOncePerRequest property of FilterSecurityInterceptor. Defaults
+ to "true"
+
+
+
+
+
+ Prevents the jsessionid parameter from being added to rendered URLs. Defaults to "true"
+ (rewriting is disabled).
+
+
+
+
+
+ A bean identifier, used for referring to the bean elsewhere in the context.
+
+
+
+
+
+ A reference to an AuthenticationManager bean
+
+
+
+
+
+
+
+
+ Defines a reference to a Spring bean Id.
+
+
+
+
+
+ The access denied page that an authenticated user will be redirected to if they request a
+ page which they don't have the authority to access.
+
+
+
+
+
+
+
+ The access denied page that an authenticated user will be redirected to if they request a
+ page which they don't have the authority to access.
+
+
+
+
+
+
+
+
+ The request URL pattern which will be mapped to the FilterChain.
+
+
+
+
+
+ Allows a RequestMatcher instance to be used, as an alternative to pattern-matching.
+
+
+
+
+
+ The access configuration attributes that apply for the configured path.
+
+
+
+
+
+ The HTTP Method for which the access configuration attributes should apply. If not
+ specified, the attributes will apply to any method.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ The filter list for the path. Currently can be set to "none" to remove a path from having
+ any filters applied. The full filter stack (consisting of all filters created by the
+ namespace configuration, and any added using 'custom-filter'), will be applied to any
+ other paths.
+
+
+
+
+
+
+
+
+
+
+ Used to specify that a URL must be accessed over http or https, or that there is no
+ preference. The value should be "http", "https" or "any", respectively.
+
+
+
+
+
+ The path to the servlet. This attribute is only applicable when 'request-matcher' is
+ 'mvc'. In addition, the value is only required in the following 2 use cases: 1) There are
+ 2 or more HttpServlet's registered in the ServletContext that have mappings starting with
+ '/' and are different; 2) The pattern starts with the same value of a registered
+ HttpServlet path, excluding the default (root) HttpServlet '/'.
+
+
+
+
+
+
+
+
+ Specifies the URL that will cause a logout. Spring Security will initialize a filter that
+ responds to this particular URL. Defaults to /logout if unspecified.
+
+
+
+
+
+ Specifies the URL to display once the user has logged out. If not specified, defaults to
+ <form-login-login-page>/?logout (i.e. /login?logout).
+
+
+
+
+
+ Specifies whether a logout also causes HttpSession invalidation, which is generally
+ desirable. If unspecified, defaults to true.
+
+
+
+
+
+ A reference to a LogoutSuccessHandler implementation which will be used to determine the
+ destination to which the user is taken after logging out.
+
+
+
+
+
+ A comma-separated list of the names of cookies which should be deleted when the user logs
+ out
+
+
+
+
+
+
+ Allow the RequestCache used for saving requests during the login process to be set
+
+
+
+
+
+
+
+
+
+
+ The URL that the login form is posted to. If unspecified, it defaults to /login.
+
+
+
+
+
+ The name of the request parameter which contains the username. Defaults to 'username'.
+
+
+
+
+
+ The name of the request parameter which contains the password. Defaults to 'password'.
+
+
+
+
+
+ The URL that will be redirected to after successful authentication, if the user's previous
+ action could not be resumed. This generally happens if the user visits a login page
+ without having first requested a secured operation that triggers authentication. If
+ unspecified, defaults to the root of the application.
+
+
+
+
+
+ Whether the user should always be redirected to the default-target-url after login.
+
+
+
+
+
+ The URL for the login page. If no login URL is specified, Spring Security will
+ automatically create a login URL at GET /login and a corresponding filter to render that
+ login URL when requested.
+
+
+
+
+
+ The URL for the login failure page. If no login failure URL is specified, Spring Security
+ will automatically create a failure login URL at /login?error and a corresponding filter
+ to render that login failure URL when requested.
+
+
+
+
+
+ Reference to an AuthenticationSuccessHandler bean which should be used to handle a
+ successful authentication request. Should not be used in combination with
+ default-target-url (or always-use-default-target-url) as the implementation should always
+ deal with navigation to the subsequent destination
+
+
+
+
+
+ Reference to an AuthenticationFailureHandler bean which should be used to handle a failed
+ authentication request. Should not be used in combination with authentication-failure-url
+ as the implementation should always deal with navigation to the subsequent destination
+
+
+
+
+
+ Reference to an AuthenticationDetailsSource which will be used by the authentication
+ filter
+
+
+
+
+
+ The URL for the ForwardAuthenticationFailureHandler
+
+
+
+
+
+ The URL for the ForwardAuthenticationSuccessHandler
+
+
+
+
+
+
+
+ Sets up an attribute exchange configuration to request specified attributes from the
+ OpenID identity provider. When multiple elements are used, each must have an
+ identifier-attribute attribute. Each configuration will be matched in turn against the
+ supplied login identifier until a match is found.
+
+
+
+
+
+
+
+
+
+
+
+
+ A regular expression which will be compared against the claimed identity, when deciding
+ which attribute-exchange configuration to use during authentication.
+
+
+
+
+
+
+ Attributes used when making an OpenID AX Fetch Request
+
+
+
+
+
+
+
+
+
+ Specifies the name of the attribute that you wish to get back. For example, email.
+
+
+
+
+
+ Specifies the attribute type. For example, http://axschema.org/contact/email. See your
+ OP's documentation for valid attribute types.
+
+
+
+
+
+ Specifies if this attribute is required to the OP, but does not error out if the OP does
+ not return the attribute. Default is false.
+
+
+
+
+
+ Specifies the number of attributes that you wish to get back. For example, return 3
+ emails. The default value is 1.
+
+
+
+
+
+
+ Used to explicitly configure a FilterChainProxy instance with a FilterChainMap
+
+
+
+
+
+
+
+
+
+
+
+
+ Defines the strategy use for matching incoming requests. Currently the options are 'mvc'
+ (for Spring MVC matcher), 'ant' (for ant path patterns), 'regex' for regular expressions
+ and 'ciRegex' for case-insensitive regular expressions.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Used within to define a specific URL pattern and the list of filters which apply to the
+ URLs matching that pattern. When multiple filter-chain elements are assembled in a list in
+ order to configure a FilterChainProxy, the most specific patterns must be placed at the
+ top of the list, with most general ones at the bottom.
+
+
+
+
+
+
+
+
+
+ The request URL pattern which will be mapped to the FilterChain.
+
+
+
+
+
+ Allows a RequestMatcher instance to be used, as an alternative to pattern-matching.
+
+
+
+
+
+ A comma separated list of bean names that implement Filter that should be processed for
+ this FilterChain. If the value is none, then no Filters will be used for this FilterChain.
+
+
+
+
+
+
+
+ The request URL pattern which will be mapped to the FilterChain.
+
+
+
+
+
+
+
+ Allows a RequestMatcher instance to be used, as an alternative to pattern-matching.
+
+
+
+
+
+
+ Used to explicitly configure a FilterSecurityMetadataSource bean for use with a
+ FilterSecurityInterceptor. Usually only needed if you are configuring a FilterChainProxy
+ explicitly, rather than using the <http> element. The intercept-url elements used should
+ only contain pattern, method and access attributes. Any others will result in a
+ configuration error.
+
+
+
+
+
+
+ Specifies the access attributes and/or filter list for a particular set of URLs.
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Enables the use of expressions in the 'access' attributes in <intercept-url> elements
+ rather than the traditional list of configuration attributes. Defaults to 'true'. If
+ enabled, each attribute should contain a single boolean expression. If the expression
+ evaluates to 'true', access will be granted.
+
+
+
+
+
+ A bean identifier, used for referring to the bean elsewhere in the context.
+
+
+
+
+
+ Defines the strategy use for matching incoming requests. Currently the options are 'mvc'
+ (for Spring MVC matcher), 'ant' (for ant path patterns), 'regex' for regular expressions
+ and 'ciRegex' for case-insensitive regular expressions.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Sets the AuthenticationEntryPoint which is used by the BasicAuthenticationFilter.
+
+
+
+
+
+ Reference to an AuthenticationDetailsSource which will be used by the authentication
+ filter
+
+
+
+
+
+
+
+
+ Indicates how session fixation protection will be applied when a user authenticates. If
+ set to "none", no protection will be applied. "newSession" will create a new empty
+ session, with only Spring Security-related attributes migrated. "migrateSession" will
+ create a new session and copy all session attributes to the new session. In Servlet 3.1
+ (Java EE 7) and newer containers, specifying "changeSessionId" will keep the existing
+ session and use the container-supplied session fixation protection
+ (HttpServletRequest#changeSessionId()). Defaults to "changeSessionId" in Servlet 3.1 and
+ newer containers, "migrateSession" in older containers. Throws an exception if
+ "changeSessionId" is used in older containers.
+
+
+
+
+
+
+
+
+
+
+
+
+
+ The URL to which a user will be redirected if they submit an invalid session indentifier.
+ Typically used to detect session timeouts.
+
+
+
+
+
+ Allows injection of the InvalidSessionStrategy instance used by the
+ SessionManagementFilter
+
+
+
+
+
+ Allows injection of the SessionAuthenticationStrategy instance used by the
+ SessionManagementFilter
+
+
+
+
+
+ Defines the URL of the error page which should be shown when the
+ SessionAuthenticationStrategy raises an exception. If not set, an unauthorized (401) error
+ code will be returned to the client. Note that this attribute doesn't apply if the error
+ occurs during a form-based login, where the URL for authentication failure will take
+ precedence.
+
+
+
+
+
+
+
+
+ The maximum number of sessions a single authenticated user can have open at the same time.
+ Defaults to "1". A negative value denotes unlimited sessions.
+
+
+
+
+
+ The URL a user will be redirected to if they attempt to use a session which has been
+ "expired" because they have logged in again.
+
+
+
+
+
+ Allows injection of the SessionInformationExpiredStrategy instance used by the
+ ConcurrentSessionFilter
+
+
+
+
+
+ Specifies that an unauthorized error should be reported when a user attempts to login when
+ they already have the maximum configured sessions open. The default behaviour is to expire
+ the original session. If the session-authentication-error-url attribute is set on the
+ session-management URL, the user will be redirected to this URL.
+
+
+
+
+
+ Allows you to define an alias for the SessionRegistry bean in order to access it in your
+ own configuration.
+
+
+
+
+
+ Allows you to define an external SessionRegistry bean to be used by the concurrency
+ control setup.
+
+
+
+
+
+
+
+
+ The "key" used to identify cookies from a specific token-based remember-me application.
+ You should set this to a unique value for your application. If unset, it will default to a
+ random value generated by SecureRandom.
+
+
+
+
+
+ Reference to a PersistentTokenRepository bean for use with the persistent token
+ remember-me implementation.
+
+
+
+
+
+ A reference to a DataSource bean
+
+
+
+
+
+
+ A reference to a user-service (or UserDetailsService bean) Id
+
+
+
+
+
+ Exports the internally defined RememberMeServices as a bean alias, allowing it to be used
+ by other beans in the application context.
+
+
+
+
+
+ Determines whether the "secure" flag will be set on the remember-me cookie. If set to
+ true, the cookie will only be submitted over HTTPS (recommended). By default, secure
+ cookies will be used if the request is made on a secure connection.
+
+
+
+
+
+ The period (in seconds) for which the remember-me cookie should be valid.
+
+
+
+
+
+ Reference to an AuthenticationSuccessHandler bean which should be used to handle a
+ successful remember-me authentication.
+
+
+
+
+
+ The name of the request parameter which toggles remember-me authentication. Defaults to
+ 'remember-me'.
+
+
+
+
+
+ The name of cookie which store the token for remember-me authentication. Defaults to
+ 'remember-me'.
+
+
+
+
+
+
+
+ Reference to a PersistentTokenRepository bean for use with the persistent token
+ remember-me implementation.
+
+
+
+
+
+
+
+ Allows a custom implementation of RememberMeServices to be used. Note that this
+ implementation should return RememberMeAuthenticationToken instances with the same "key"
+ value as specified in the remember-me element. Alternatively it should register its own
+ AuthenticationProvider. It should also implement the LogoutHandler interface, which will
+ be invoked when a user logs out. Typically the remember-me cookie would be removed on
+ logout.
+
+
+
+
+
+
+
+
+
+
+
+ The key shared between the provider and filter. This generally does not need to be set. If
+ unset, it will default to a random value generated by SecureRandom.
+
+
+
+
+
+ The username that should be assigned to the anonymous request. This allows the principal
+ to be identified, which may be important for logging and auditing. if unset, defaults to
+ "anonymousUser".
+
+
+
+
+
+ The granted authority that should be assigned to the anonymous request. Commonly this is
+ used to assign the anonymous request particular roles, which can subsequently be used in
+ authorization decisions. If unset, defaults to "ROLE_ANONYMOUS".
+
+
+
+
+
+ With the default namespace setup, the anonymous "authentication" facility is automatically
+ enabled. You can disable it using this property.
+
+
+
+
+
+
+
+
+
+ The http port to use.
+
+
+
+
+
+
+
+ The https port to use.
+
+
+
+
+
+
+
+
+ The regular expression used to obtain the username from the certificate's subject.
+ Defaults to matching on the common name using the pattern "CN=(.*?),".
+
+
+
+
+
+ A reference to a user-service (or UserDetailsService bean) Id
+
+
+
+
+
+ Reference to an AuthenticationDetailsSource which will be used by the authentication
+ filter
+
+
+
+
+
+
+ Adds a J2eePreAuthenticatedProcessingFilter to the filter chain to provide integration
+ with container authentication.
+
+
+
+
+
+
+
+
+
+ A comma-separate list of roles to look for in the incoming HttpServletRequest.
+
+
+
+
+
+ A reference to a user-service (or UserDetailsService bean) Id
+
+
+
+
+
+
+ Registers the AuthenticationManager instance and allows its list of
+ AuthenticationProviders to be defined. Also allows you to define an alias to allow you to
+ reference the AuthenticationManager in your own beans.
+
+
+
+
+
+
+ Indicates that the contained user-service should be used as an authentication source.
+
+
+
+
+
+
+
+ element which defines a password encoding strategy. Used by an authentication provider to
+ convert submitted passwords to hashed versions, for example.
+
+
+
+
+
+
+
+
+
+
+
+
+ Sets up an ldap authentication provider
+
+
+
+
+
+
+ Specifies that an LDAP provider should use an LDAP compare operation of the user's
+ password to authenticate the user
+
+
+
+
+
+
+ element which defines a password encoding strategy. Used by an authentication provider to
+ convert submitted passwords to hashed versions, for example.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ A bean identifier, used for referring to the bean elsewhere in the context.
+
+
+
+
+
+ An alias you wish to use for the AuthenticationManager bean (not required it you are using
+ a specific id)
+
+
+
+
+
+ If set to true, the AuthenticationManger will attempt to clear any credentials data in the
+ returned Authentication object, once the user has been authenticated.
+
+
+
+
+
+
+
+
+ Defines a reference to a Spring bean Id.
+
+
+
+
+
+ A reference to a user-service (or UserDetailsService bean) Id
+
+
+
+
+
+
+ Creates an in-memory UserDetailsService from a properties file or a list of "user" child
+ elements. Usernames are converted to lower-case internally to allow for case-insensitive
+ lookups, so this should not be used if case-sensitivity is required.
+
+
+
+
+
+
+ Represents a user in the application.
+
+
+
+
+
+
+
+
+
+ A bean identifier, used for referring to the bean elsewhere in the context.
+
+
+
+
+
+
+
+
+
+ The location of a Properties file where each line is in the format of
+ username=password,grantedAuthority[,grantedAuthority][,enabled|disabled]
+
+
+
+
+
+
+
+
+ The username assigned to the user.
+
+
+
+
+
+ The password assigned to the user. This may be hashed if the corresponding authentication
+ provider supports hashing (remember to set the "hash" attribute of the "user-service"
+ element). This attribute be omitted in the case where the data will not be used for
+ authentication, but only for accessing authorities. If omitted, the namespace will
+ generate a random value, preventing its accidental use for authentication. Cannot be
+ empty.
+
+
+
+
+
+ One of more authorities granted to the user. Separate authorities with a comma (but no
+ space). For example, "ROLE_USER,ROLE_ADMINISTRATOR"
+
+
+
+
+
+ Can be set to "true" to mark an account as locked and unusable.
+
+
+
+
+
+ Can be set to "true" to mark an account as disabled and unusable.
+
+
+
+
+
+
+ Causes creation of a JDBC-based UserDetailsService.
+
+
+
+
+
+ A bean identifier, used for referring to the bean elsewhere in the context.
+
+
+
+
+
+
+
+
+
+ The bean ID of the DataSource which provides the required tables.
+
+
+
+
+
+ Defines a reference to a cache for use with a UserDetailsService.
+
+
+
+
+
+ An SQL statement to query a username, password, and enabled status given a username.
+ Default is "select username,password,enabled from users where username = ?"
+
+
+
+
+
+ An SQL statement to query for a user's granted authorities given a username. The default
+ is "select username, authority from authorities where username = ?"
+
+
+
+
+
+ An SQL statement to query user's group authorities given a username. The default is
+ "select g.id, g.group_name, ga.authority from groups g, group_members gm,
+ group_authorities ga where gm.username = ? and g.id = ga.group_id and g.id = gm.group_id"
+
+
+
+
+
+ A non-empty string prefix that will be added to role strings loaded from persistent
+ storage (e.g. "ROLE_"). Use the value "none" for no prefix in cases where the default is
+ non-empty.
+
+
+
+
+
+
+ Element for configuration of the CsrfFilter for protection against CSRF. It also updates
+ the default RequestCache to only replay "GET" requests.
+
+
+
+
+
+
+
+
+
+ Specifies if csrf protection should be disabled. Default false (i.e. CSRF protection is
+ enabled).
+
+
+
+
+
+ The RequestMatcher instance to be used to determine if CSRF should be applied. Default is
+ any HTTP method except "GET", "TRACE", "HEAD", "OPTIONS"
+
+
+
+
+
+ The CsrfTokenRepository to use. The default is HttpSessionCsrfTokenRepository
+
+
+
+
+
+
+ Element for configuration of the HeaderWritersFilter. Enables easy setting for the
+ X-Frame-Options, X-XSS-Protection and X-Content-Type-Options headers.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Specifies if the default headers should be disabled. Default false.
+
+
+
+
+
+ Specifies if headers should be disabled. Default false.
+
+
+
+
+
+
+ Adds support for HTTP Strict Transport Security (HSTS)
+
+
+
+
+
+
+
+
+
+ Specifies if HTTP Strict Transport Security (HSTS) should be disabled. Default false.
+
+
+
+
+
+ Specifies if subdomains should be included. Default true.
+
+
+
+
+
+ Specifies the maximum ammount of time the host should be considered a Known HSTS Host.
+ Default one year.
+
+
+
+
+
+ The RequestMatcher instance to be used to determine if the header should be set. Default
+ is if HttpServletRequest.isSecure() is true.
+
+
+
+
+
+
+ Element for configuration of CorsFilter. If no CorsFilter or CorsConfigurationSource is
+ specified a HandlerMappingIntrospector is used as the CorsConfigurationSource
+
+
+
+
+
+
+
+
+
+ Defines a reference to a Spring bean Id.
+
+
+
+
+
+ Specifies a bean id that is a CorsConfigurationSource used to construct the CorsFilter to
+ use
+
+
+
+
+
+
+ Adds support for HTTP Public Key Pinning (HPKP).
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ The list with pins
+
+
+
+
+
+
+
+
+
+
+ A pin is specified using the base64-encoded SPKI fingerprint as value and the
+ cryptographic hash algorithm as attribute
+
+
+
+
+
+ The cryptographic hash algorithm
+
+
+
+
+
+
+
+
+ Specifies if HTTP Public Key Pinning (HPKP) should be disabled. Default false.
+
+
+
+
+
+ Specifies if subdomains should be included. Default false.
+
+
+
+
+
+ Sets the value for the max-age directive of the Public-Key-Pins header. Default 60 days.
+
+
+
+
+
+ Specifies if the browser should only report pin validation failures. Default true.
+
+
+
+
+
+ Specifies the URI to which the browser should report pin validation failures.
+
+
+
+
+
+
+ Adds support for Content Security Policy (CSP)
+
+
+
+
+
+
+
+
+
+ The security policy directive(s) for the Content-Security-Policy header or if report-only
+ is set to true, then the Content-Security-Policy-Report-Only header is used.
+
+
+
+
+
+ Set to true, to enable the Content-Security-Policy-Report-Only header for reporting policy
+ violations only. Defaults to false.
+
+
+
+
+
+
+ Adds support for Referrer Policy
+
+
+
+
+
+
+
+
+
+ The policies for the Referrer-Policy header.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Adds support for Feature Policy
+
+
+
+
+
+
+
+
+ The security policy directive(s) for the Feature-Policy header.
+
+
+
+
+
+ Adds Cache-Control no-cache, no-store, must-revalidate, Pragma no-cache, and Expires 0 for
+ every request
+
+
+
+
+
+
+
+
+
+ Specifies if Cache Control should be disabled. Default false.
+
+
+
+
+
+
+ Enable basic clickjacking support for newer browsers (IE8+), will set the X-Frame-Options
+ header.
+
+
+
+
+
+
+
+
+
+ If disabled, the X-Frame-Options header will not be included. Default false.
+
+
+
+
+
+ Specify the policy to use for the X-Frame-Options-Header.
+
+
+
+
+
+
+
+
+
+
+
+
+ Specify the strategy to use when ALLOW-FROM is chosen.
+
+
+
+
+
+
+
+
+
+
+
+
+ Defines a reference to a Spring bean Id.
+
+
+
+
+
+ Specify a value to use for the chosen strategy.
+
+
+
+
+
+ Specify the request parameter to use for the origin when using a 'whitelist' or 'regexp'
+ based strategy. Default is 'from'.
+
+
+
+
+
+
+ Enable basic XSS browser protection, supported by newer browsers (IE8+), will set the
+ X-XSS-Protection header.
+
+
+
+
+
+
+
+
+
+ disable the X-XSS-Protection header. Default is 'false' meaning it is enabled.
+
+
+
+
+
+ specify that XSS Protection should be explicitly enabled or disabled. Default is 'true'
+ meaning it is enabled.
+
+
+
+
+
+ Add mode=block to the header or not, default is on.
+
+
+
+
+
+
+ Add a X-Content-Type-Options header to the resopnse. Value is always 'nosniff'.
+
+
+
+
+
+
+
+
+
+ If disabled, the X-Content-Type-Options header will not be included. Default false.
+
+
+
+
+
+
+ Add additional headers to the response.
+
+
+
+
+
+
+
+
+
+ The name of the header to add.
+
+
+
+
+
+ The value for the header.
+
+
+
+
+
+ Defines a reference to a Spring bean Id.
+
+
+
+
+
+
+
+ Used to indicate that a filter bean declaration should be incorporated into the security
+ filter chain.
+
+
+
+
+
+
+
+
+
+
+ The filter immediately after which the custom-filter should be placed in the chain. This
+ feature will only be needed by advanced users who wish to mix their own filters into the
+ security filter chain and have some knowledge of the standard Spring Security filters. The
+ filter names map to specific Spring Security implementation filters.
+
+
+
+
+
+ The filter immediately before which the custom-filter should be placed in the chain
+
+
+
+
+
+ The explicit position at which the custom-filter should be placed in the chain. Use if you
+ are replacing a standard filter.
+
+
+
+
+
+
+
+ The filter immediately after which the custom-filter should be placed in the chain. This
+ feature will only be needed by advanced users who wish to mix their own filters into the
+ security filter chain and have some knowledge of the standard Spring Security filters. The
+ filter names map to specific Spring Security implementation filters.
+
+
+
+
+
+
+
+ The filter immediately before which the custom-filter should be placed in the chain
+
+
+
+
+
+
+
+ The explicit position at which the custom-filter should be placed in the chain. Use if you
+ are replacing a standard filter.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/config/src/test/groovy/org/springframework/security/config/annotation/web/configurers/CsrfConfigurerTests.groovy b/config/src/test/groovy/org/springframework/security/config/annotation/web/configurers/CsrfConfigurerTests.groovy
index 05ddb22564d..0628529318d 100644
--- a/config/src/test/groovy/org/springframework/security/config/annotation/web/configurers/CsrfConfigurerTests.groovy
+++ b/config/src/test/groovy/org/springframework/security/config/annotation/web/configurers/CsrfConfigurerTests.groovy
@@ -15,7 +15,10 @@
*/
package org.springframework.security.config.annotation.web.configurers
+import org.springframework.context.annotation.Bean
+import org.springframework.context.annotation.Configuration
import org.springframework.security.core.userdetails.PasswordEncodedUser
+import org.springframework.security.web.firewall.StrictHttpFirewall
import javax.servlet.http.HttpServletResponse
@@ -44,7 +47,7 @@ class CsrfConfigurerTests extends BaseSpringSpec {
@Unroll
def "csrf applied by default"() {
setup:
- loadConfig(CsrfAppliedDefaultConfig)
+ loadConfig(CsrfAppliedDefaultConfig, AllowHttpMethodsFirewallConfig)
request.method = httpMethod
clearCsrfToken()
when:
@@ -66,11 +69,21 @@ class CsrfConfigurerTests extends BaseSpringSpec {
def "csrf default creates CsrfRequestDataValueProcessor"() {
when:
- loadConfig(CsrfAppliedDefaultConfig)
+ loadConfig(CsrfAppliedDefaultConfig, AllowHttpMethodsFirewallConfig)
then:
context.getBean(RequestDataValueProcessor)
}
+ @Configuration
+ static class AllowHttpMethodsFirewallConfig {
+ @Bean
+ StrictHttpFirewall strictHttpFirewall() {
+ StrictHttpFirewall result = new StrictHttpFirewall();
+ result.setAllowedHttpMethods(StrictHttpFirewall.ALLOW_ANY_HTTP_METHOD);
+ return result;
+ }
+ }
+
@EnableWebSecurity
static class CsrfAppliedDefaultConfig extends WebSecurityConfigurerAdapter {
diff --git a/config/src/test/groovy/org/springframework/security/config/annotation/web/configurers/DefaultLoginPageConfigurerTests.groovy b/config/src/test/groovy/org/springframework/security/config/annotation/web/configurers/DefaultLoginPageConfigurerTests.groovy
index 25f10dbba79..3364d9b104f 100644
--- a/config/src/test/groovy/org/springframework/security/config/annotation/web/configurers/DefaultLoginPageConfigurerTests.groovy
+++ b/config/src/test/groovy/org/springframework/security/config/annotation/web/configurers/DefaultLoginPageConfigurerTests.groovy
@@ -53,15 +53,33 @@ public class DefaultLoginPageConfigurerTests extends BaseSpringSpec {
request.requestURI = "/login"
springSecurityFilterChain.doFilter(request,response,chain)
then:
- response.getContentAsString() == """Login Page
-Login with Username and Password """
+ response.getContentAsString() == """
+
+
+
+
+
+
+ Please sign in
+
+
+
+
+
+
+"""
when: "fail to log in"
super.setup()
request.servletPath = "/login"
@@ -77,15 +95,33 @@ public class DefaultLoginPageConfigurerTests extends BaseSpringSpec {
request.queryString = "error"
springSecurityFilterChain.doFilter(request,response,chain)
then:
- response.getContentAsString() == """
Login Page
-
Your login attempt was not successful, try again. Reason: Bad credentials
Login with Username and Password """
+ response.getContentAsString() == """
+
+
+
+
+
+
+
Please sign in
+
+
+
+
+
+
+"""
when: "login success"
super.setup()
request.servletPath = "/login"
@@ -106,15 +142,33 @@ public class DefaultLoginPageConfigurerTests extends BaseSpringSpec {
request.method = "GET"
springSecurityFilterChain.doFilter(request,response,chain)
then: "sent to default success page"
- response.getContentAsString() == """
Login Page
-
You have been logged out
Login with Username and Password """
+ response.getContentAsString() == """
+
+
+
+
+
+
+
Please sign in
+
+
+
+
+
+
+"""
}
@Configuration
@@ -191,16 +245,34 @@ public class DefaultLoginPageConfigurerTests extends BaseSpringSpec {
request.requestURI = "/login"
springSecurityFilterChain.doFilter(request,response,chain)
then:
- response.getContentAsString() == """
Login Page
-
Login with Username and Password """
+ response.getContentAsString() == """
+
+
+
+
+
+
+
Please sign in
+
+
+
+
+
+
+"""
}
@Configuration
@@ -224,13 +296,29 @@ public class DefaultLoginPageConfigurerTests extends BaseSpringSpec {
request.requestURI = "/login"
springSecurityFilterChain.doFilter(request,response,chain)
then:
- response.getContentAsString() == """
Login Page Login with OpenID Identity