Added support for Anonymous Authentication
1. Created new WebFilter AnonymousAuthenticationWebFilter to for anonymous authentication 2. Created class AnonymousSpec, method anonymous to configure anonymous authentication in ServerHttpSecurity 3. Added ANONYMOUS_AUTHENTICATION order after AUTHENTICATION for anonymous authentication in SecurityWebFiltersOrder 4. Added tests for anonymous authentication in AnonymousAuthenticationWebFilterTests and ServerHttpSecurityTests 5. Added support for Controller in WebTestClientBuilder Fixes: gh-5934
This commit is contained in:
parent
60e3bf4093
commit
2b369cfe98
|
@ -48,6 +48,10 @@ public enum SecurityWebFiltersOrder {
|
||||||
*/
|
*/
|
||||||
FORM_LOGIN,
|
FORM_LOGIN,
|
||||||
AUTHENTICATION,
|
AUTHENTICATION,
|
||||||
|
/**
|
||||||
|
* Instance of AnonymousAuthenticationWebFilter
|
||||||
|
*/
|
||||||
|
ANONYMOUS_AUTHENTICATION,
|
||||||
OAUTH2_AUTHORIZATION_CODE,
|
OAUTH2_AUTHORIZATION_CODE,
|
||||||
LOGIN_PAGE_GENERATING,
|
LOGIN_PAGE_GENERATING,
|
||||||
LOGOUT_PAGE_GENERATING,
|
LOGOUT_PAGE_GENERATING,
|
||||||
|
|
|
@ -33,6 +33,7 @@ import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
import java.util.function.Function;
|
import java.util.function.Function;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
import reactor.core.publisher.Mono;
|
import reactor.core.publisher.Mono;
|
||||||
import reactor.util.context.Context;
|
import reactor.util.context.Context;
|
||||||
|
@ -158,6 +159,9 @@ import org.springframework.web.cors.reactive.DefaultCorsProcessor;
|
||||||
import org.springframework.web.server.ServerWebExchange;
|
import org.springframework.web.server.ServerWebExchange;
|
||||||
import org.springframework.web.server.WebFilter;
|
import org.springframework.web.server.WebFilter;
|
||||||
import org.springframework.web.server.WebFilterChain;
|
import org.springframework.web.server.WebFilterChain;
|
||||||
|
import org.springframework.security.web.server.authentication.AnonymousAuthenticationWebFilter;
|
||||||
|
import org.springframework.security.core.GrantedAuthority;
|
||||||
|
import org.springframework.security.core.authority.AuthorityUtils;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A {@link ServerHttpSecurity} is similar to Spring Security's {@code HttpSecurity} but for WebFlux.
|
* A {@link ServerHttpSecurity} is similar to Spring Security's {@code HttpSecurity} but for WebFlux.
|
||||||
|
@ -264,6 +268,8 @@ public class ServerHttpSecurity {
|
||||||
|
|
||||||
private Throwable built;
|
private Throwable built;
|
||||||
|
|
||||||
|
private AnonymousSpec anonymous;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The ServerExchangeMatcher that determines which requests apply to this HttpSecurity instance.
|
* The ServerExchangeMatcher that determines which requests apply to this HttpSecurity instance.
|
||||||
*
|
*
|
||||||
|
@ -425,6 +431,29 @@ public class ServerHttpSecurity {
|
||||||
return this.cors;
|
return this.cors;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @since 5.2.0
|
||||||
|
* @author Ankur Pathak
|
||||||
|
* Enables and Configures annonymous authentication. Anonymous Authentication is disabled by default.
|
||||||
|
*
|
||||||
|
* <pre class="code">
|
||||||
|
* @Bean
|
||||||
|
* public SecurityWebFilterChain springSecurityFilterChain(ServerHttpSecurity http) {
|
||||||
|
* http
|
||||||
|
* // ...
|
||||||
|
* .anonymous().key("key")
|
||||||
|
* .authorities("ROLE_ANONYMOUS");
|
||||||
|
* return http.build();
|
||||||
|
* }
|
||||||
|
* </pre>
|
||||||
|
*/
|
||||||
|
public AnonymousSpec anonymous(){
|
||||||
|
if (this.anonymous == null) {
|
||||||
|
this.anonymous = new AnonymousSpec();
|
||||||
|
}
|
||||||
|
return this.anonymous;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Configures CORS support within Spring Security. This ensures that the {@link CorsWebFilter} is place in the
|
* Configures CORS support within Spring Security. This ensures that the {@link CorsWebFilter} is place in the
|
||||||
* correct order.
|
* correct order.
|
||||||
|
@ -1356,6 +1385,9 @@ public class ServerHttpSecurity {
|
||||||
if (this.client != null) {
|
if (this.client != null) {
|
||||||
this.client.configure(this);
|
this.client.configure(this);
|
||||||
}
|
}
|
||||||
|
if (this.anonymous != null) {
|
||||||
|
this.anonymous.configure(this);
|
||||||
|
}
|
||||||
this.loginPage.configure(this);
|
this.loginPage.configure(this);
|
||||||
if (this.logout != null) {
|
if (this.logout != null) {
|
||||||
this.logout.configure(this);
|
this.logout.configure(this);
|
||||||
|
@ -2589,4 +2621,124 @@ public class ServerHttpSecurity {
|
||||||
.subscriberContext(Context.of(ServerWebExchange.class, exchange));
|
.subscriberContext(Context.of(ServerWebExchange.class, exchange));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configures annonymous authentication
|
||||||
|
* @author Ankur Pathak
|
||||||
|
* @since 5.2.0
|
||||||
|
*/
|
||||||
|
public final class AnonymousSpec {
|
||||||
|
private String key;
|
||||||
|
private AnonymousAuthenticationWebFilter authenticationFilter;
|
||||||
|
private Object principal = "anonymousUser";
|
||||||
|
private List<GrantedAuthority> authorities = AuthorityUtils.createAuthorityList("ROLE_ANONYMOUS");
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the key to identify tokens created for anonymous authentication. Default is a
|
||||||
|
* secure randomly generated key.
|
||||||
|
*
|
||||||
|
* @param key the key to identify tokens created for anonymous authentication. Default
|
||||||
|
* is a secure randomly generated key.
|
||||||
|
* @return the {@link AnonymousSpec} for further customization of anonymous
|
||||||
|
* authentication
|
||||||
|
*/
|
||||||
|
public AnonymousSpec key(String key) {
|
||||||
|
this.key = key;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the principal for {@link Authentication} objects of anonymous users
|
||||||
|
*
|
||||||
|
* @param principal used for the {@link Authentication} object of anonymous users
|
||||||
|
* @return the {@link AnonymousSpec} for further customization of anonymous
|
||||||
|
* authentication
|
||||||
|
*/
|
||||||
|
public AnonymousSpec principal(Object principal) {
|
||||||
|
this.principal = principal;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the {@link org.springframework.security.core.Authentication#getAuthorities()}
|
||||||
|
* for anonymous users
|
||||||
|
*
|
||||||
|
* @param authorities Sets the
|
||||||
|
* {@link org.springframework.security.core.Authentication#getAuthorities()} for
|
||||||
|
* anonymous users
|
||||||
|
* @return the {@link AnonymousSpec} for further customization of anonymous
|
||||||
|
* authentication
|
||||||
|
*/
|
||||||
|
public AnonymousSpec authorities(List<GrantedAuthority> authorities) {
|
||||||
|
this.authorities = authorities;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the {@link org.springframework.security.core.Authentication#getAuthorities()}
|
||||||
|
* for anonymous users
|
||||||
|
*
|
||||||
|
* @param authorities Sets the
|
||||||
|
* {@link org.springframework.security.core.Authentication#getAuthorities()} for
|
||||||
|
* anonymous users (i.e. "ROLE_ANONYMOUS")
|
||||||
|
* @return the {@link AnonymousSpec} for further customization of anonymous
|
||||||
|
* authentication
|
||||||
|
*/
|
||||||
|
public AnonymousSpec authorities(String... authorities) {
|
||||||
|
return authorities(AuthorityUtils.createAuthorityList(authorities));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the {@link AnonymousAuthenticationWebFilter} used to populate an anonymous user.
|
||||||
|
* If this is set, no attributes on the {@link AnonymousSpec} will be set on the
|
||||||
|
* {@link AnonymousAuthenticationWebFilter}.
|
||||||
|
*
|
||||||
|
* @param authenticationFilter the {@link AnonymousAuthenticationWebFilter} used to
|
||||||
|
* populate an anonymous user.
|
||||||
|
*
|
||||||
|
* @return the {@link AnonymousSpec} for further customization of anonymous
|
||||||
|
* authentication
|
||||||
|
*/
|
||||||
|
public AnonymousSpec authenticationFilter(
|
||||||
|
AnonymousAuthenticationWebFilter authenticationFilter) {
|
||||||
|
this.authenticationFilter = authenticationFilter;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Allows method chaining to continue configuring the {@link ServerHttpSecurity}
|
||||||
|
* @return the {@link ServerHttpSecurity} to continue configuring
|
||||||
|
*/
|
||||||
|
public ServerHttpSecurity and() {
|
||||||
|
return ServerHttpSecurity.this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Disables anonymous authentication.
|
||||||
|
* @return the {@link ServerHttpSecurity} to continue configuring
|
||||||
|
*/
|
||||||
|
public ServerHttpSecurity disable() {
|
||||||
|
ServerHttpSecurity.this.anonymous = null;
|
||||||
|
return ServerHttpSecurity.this;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void configure(ServerHttpSecurity http) {
|
||||||
|
if (authenticationFilter == null) {
|
||||||
|
authenticationFilter = new AnonymousAuthenticationWebFilter(getKey(), principal,
|
||||||
|
authorities);
|
||||||
|
}
|
||||||
|
http.addFilterAt(authenticationFilter, SecurityWebFiltersOrder.ANONYMOUS_AUTHENTICATION);
|
||||||
|
}
|
||||||
|
|
||||||
|
private String getKey() {
|
||||||
|
if (key == null) {
|
||||||
|
key = UUID.randomUUID().toString();
|
||||||
|
}
|
||||||
|
return key;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private AnonymousSpec() {}
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -34,8 +34,6 @@ import org.junit.runner.RunWith;
|
||||||
import org.mockito.Mock;
|
import org.mockito.Mock;
|
||||||
import org.mockito.junit.MockitoJUnitRunner;
|
import org.mockito.junit.MockitoJUnitRunner;
|
||||||
|
|
||||||
import org.springframework.security.web.server.context.SecurityContextServerWebExchangeWebFilter;
|
|
||||||
import org.springframework.web.server.WebFilterChain;
|
|
||||||
import reactor.core.publisher.Mono;
|
import reactor.core.publisher.Mono;
|
||||||
import reactor.test.publisher.TestPublisher;
|
import reactor.test.publisher.TestPublisher;
|
||||||
|
|
||||||
|
@ -63,6 +61,9 @@ import org.springframework.web.bind.annotation.GetMapping;
|
||||||
import org.springframework.web.bind.annotation.RestController;
|
import org.springframework.web.bind.annotation.RestController;
|
||||||
import org.springframework.web.server.ServerWebExchange;
|
import org.springframework.web.server.ServerWebExchange;
|
||||||
import org.springframework.web.server.WebFilter;
|
import org.springframework.web.server.WebFilter;
|
||||||
|
import org.springframework.security.web.server.context.SecurityContextServerWebExchangeWebFilter;
|
||||||
|
import org.springframework.web.server.WebFilterChain;
|
||||||
|
import org.springframework.security.web.server.authentication.AnonymousAuthenticationWebFilterTests;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author Rob Winch
|
* @author Rob Winch
|
||||||
|
@ -216,6 +217,44 @@ public class ServerHttpSecurityTests {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void anonymous(){
|
||||||
|
SecurityWebFilterChain securityFilterChain = this.http.anonymous().and().build();
|
||||||
|
WebTestClient client = WebTestClientBuilder.bindToControllerAndWebFilters(AnonymousAuthenticationWebFilterTests.HttpMeController.class,
|
||||||
|
securityFilterChain).build();
|
||||||
|
|
||||||
|
client.get()
|
||||||
|
.uri("/me")
|
||||||
|
.exchange()
|
||||||
|
.expectStatus().isOk()
|
||||||
|
.expectBody(String.class).isEqualTo("anonymousUser");
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void basicWithAnonymous() {
|
||||||
|
given(this.authenticationManager.authenticate(any())).willReturn(Mono.just(new TestingAuthenticationToken("rob", "rob", "ROLE_USER", "ROLE_ADMIN")));
|
||||||
|
|
||||||
|
this.http.securityContextRepository(new WebSessionServerSecurityContextRepository());
|
||||||
|
this.http.httpBasic().and().anonymous();
|
||||||
|
this.http.authenticationManager(this.authenticationManager);
|
||||||
|
ServerHttpSecurity.AuthorizeExchangeSpec authorize = this.http.authorizeExchange();
|
||||||
|
authorize.anyExchange().hasAuthority("ROLE_ADMIN");
|
||||||
|
|
||||||
|
WebTestClient client = buildClient();
|
||||||
|
|
||||||
|
EntityExchangeResult<String> result = client.get()
|
||||||
|
.uri("/")
|
||||||
|
.headers(headers -> headers.setBasicAuth("rob", "rob"))
|
||||||
|
.exchange()
|
||||||
|
.expectStatus().isOk()
|
||||||
|
.expectHeader().valueMatches(HttpHeaders.CACHE_CONTROL, ".+")
|
||||||
|
.expectBody(String.class).consumeWith(b -> assertThat(b.getResponseBody()).isEqualTo("ok"))
|
||||||
|
.returnResult();
|
||||||
|
|
||||||
|
assertThat(result.getResponseCookies().getFirst("SESSION")).isNull();
|
||||||
|
}
|
||||||
|
|
||||||
private <T extends WebFilter> Optional<T> getWebFilter(SecurityWebFilterChain filterChain, Class<T> filterClass) {
|
private <T extends WebFilter> Optional<T> getWebFilter(SecurityWebFilterChain filterChain, Class<T> filterClass) {
|
||||||
return (Optional<T>) filterChain.getWebFilters()
|
return (Optional<T>) filterChain.getWebFilters()
|
||||||
.filter(Objects::nonNull)
|
.filter(Objects::nonNull)
|
||||||
|
@ -242,7 +281,6 @@ public class ServerHttpSecurityTests {
|
||||||
}
|
}
|
||||||
|
|
||||||
private static class TestWebFilter implements WebFilter {
|
private static class TestWebFilter implements WebFilter {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Mono<Void> filter(ServerWebExchange exchange, WebFilterChain chain) {
|
public Mono<Void> filter(ServerWebExchange exchange, WebFilterChain chain) {
|
||||||
return chain.filter(exchange);
|
return chain.filter(exchange);
|
||||||
|
|
|
@ -0,0 +1,94 @@
|
||||||
|
/*
|
||||||
|
* 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.web.server.authentication;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import reactor.core.publisher.Mono;
|
||||||
|
|
||||||
|
import org.springframework.security.authentication.AnonymousAuthenticationToken;
|
||||||
|
import org.springframework.security.core.Authentication;
|
||||||
|
import org.springframework.security.core.GrantedAuthority;
|
||||||
|
import org.springframework.security.core.authority.AuthorityUtils;
|
||||||
|
import org.springframework.security.core.context.ReactiveSecurityContextHolder;
|
||||||
|
import org.springframework.security.core.context.SecurityContext;
|
||||||
|
import org.springframework.security.core.context.SecurityContextImpl;
|
||||||
|
import org.springframework.util.Assert;
|
||||||
|
import org.springframework.web.server.ServerWebExchange;
|
||||||
|
import org.springframework.web.server.WebFilter;
|
||||||
|
import org.springframework.web.server.WebFilterChain;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Detects if there is no {@code Authentication} object in the
|
||||||
|
* {@code ReactiveSecurityContextHolder}, and populates it with one if needed.
|
||||||
|
*
|
||||||
|
* @author Ankur Pathak
|
||||||
|
* @since 5.2.0
|
||||||
|
*/
|
||||||
|
public class AnonymousAuthenticationWebFilter implements WebFilter {
|
||||||
|
// ~ Instance fields
|
||||||
|
// ================================================================================================
|
||||||
|
|
||||||
|
private String key;
|
||||||
|
private Object principal;
|
||||||
|
private List<GrantedAuthority> authorities;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a filter with a principal named "anonymousUser" and the single authority
|
||||||
|
* "ROLE_ANONYMOUS".
|
||||||
|
*
|
||||||
|
* @param key the key to identify tokens created by this filter
|
||||||
|
*/
|
||||||
|
public AnonymousAuthenticationWebFilter(String key) {
|
||||||
|
this(key, "anonymousUser", AuthorityUtils.createAuthorityList("ROLE_ANONYMOUS"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param key key the key to identify tokens created by this filter
|
||||||
|
* @param principal the principal which will be used to represent anonymous users
|
||||||
|
* @param authorities the authority list for anonymous users
|
||||||
|
*/
|
||||||
|
public AnonymousAuthenticationWebFilter(String key, Object principal,
|
||||||
|
List<GrantedAuthority> authorities) {
|
||||||
|
Assert.hasLength(key, "key cannot be null or empty");
|
||||||
|
Assert.notNull(principal, "Anonymous authentication principal must be set");
|
||||||
|
Assert.notNull(authorities, "Anonymous authorities must be set");
|
||||||
|
this.key = key;
|
||||||
|
this.principal = principal;
|
||||||
|
this.authorities = authorities;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Mono<Void> filter(ServerWebExchange exchange, WebFilterChain chain) {
|
||||||
|
return ReactiveSecurityContextHolder.getContext()
|
||||||
|
.switchIfEmpty(Mono.defer(() -> {
|
||||||
|
SecurityContext securityContext = new SecurityContextImpl();
|
||||||
|
securityContext.setAuthentication(createAuthentication(exchange));
|
||||||
|
return chain.filter(exchange)
|
||||||
|
.subscriberContext(ReactiveSecurityContextHolder.withSecurityContext(Mono.just(securityContext)))
|
||||||
|
.then(Mono.empty());
|
||||||
|
})).flatMap(securityContext -> chain.filter(exchange));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Authentication createAuthentication(ServerWebExchange exchange) {
|
||||||
|
AnonymousAuthenticationToken auth = new AnonymousAuthenticationToken(key,
|
||||||
|
principal, authorities);
|
||||||
|
return auth;
|
||||||
|
}
|
||||||
|
}
|
|
@ -43,6 +43,14 @@ public class WebTestClientBuilder {
|
||||||
return bindToWebFilters(new WebFilterChainProxy(securityWebFilterChain));
|
return bindToWebFilters(new WebFilterChainProxy(securityWebFilterChain));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static Builder bindToControllerAndWebFilters(Class<?> controller, WebFilter... webFilters) {
|
||||||
|
return WebTestClient.bindToController(controller).webFilter(webFilters).configureClient();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Builder bindToControllerAndWebFilters(Class<?> controller, SecurityWebFilterChain securityWebFilterChain) {
|
||||||
|
return bindToControllerAndWebFilters(controller, new WebFilterChainProxy(securityWebFilterChain));
|
||||||
|
}
|
||||||
|
|
||||||
@RestController
|
@RestController
|
||||||
public static class Http200RestController {
|
public static class Http200RestController {
|
||||||
@RequestMapping("/**")
|
@RequestMapping("/**")
|
||||||
|
@ -51,4 +59,5 @@ public class WebTestClientBuilder {
|
||||||
return "ok";
|
return "ok";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,70 @@
|
||||||
|
/*
|
||||||
|
* 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.web.server.authentication;
|
||||||
|
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.mockito.junit.MockitoJUnitRunner;
|
||||||
|
|
||||||
|
import reactor.core.publisher.Mono;
|
||||||
|
|
||||||
|
import org.springframework.security.core.Authentication;
|
||||||
|
import org.springframework.security.core.context.ReactiveSecurityContextHolder;
|
||||||
|
import org.springframework.security.core.context.SecurityContext;
|
||||||
|
import org.springframework.test.web.reactive.server.WebTestClient;
|
||||||
|
import org.springframework.web.bind.annotation.GetMapping;
|
||||||
|
import org.springframework.web.bind.annotation.RequestMapping;
|
||||||
|
import org.springframework.web.bind.annotation.RestController;
|
||||||
|
import org.springframework.web.server.ServerWebExchange;
|
||||||
|
import org.springframework.security.test.web.reactive.server.WebTestClientBuilder;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author Ankur Pathak
|
||||||
|
* @since 5.2.0
|
||||||
|
*/
|
||||||
|
@RunWith(MockitoJUnitRunner.class)
|
||||||
|
public class AnonymousAuthenticationWebFilterTests {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void anonymousAuthenticationFilterWorking() {
|
||||||
|
|
||||||
|
WebTestClient client = WebTestClientBuilder.bindToControllerAndWebFilters(HttpMeController.class,
|
||||||
|
new AnonymousAuthenticationWebFilter(UUID.randomUUID().toString()))
|
||||||
|
.build();
|
||||||
|
|
||||||
|
client.get()
|
||||||
|
.uri("/me")
|
||||||
|
.exchange()
|
||||||
|
.expectStatus().isOk()
|
||||||
|
.expectBody(String.class).isEqualTo("anonymousUser");
|
||||||
|
}
|
||||||
|
|
||||||
|
@RestController
|
||||||
|
@RequestMapping("/me")
|
||||||
|
public static class HttpMeController {
|
||||||
|
@GetMapping
|
||||||
|
public Mono<String> me(ServerWebExchange exchange) {
|
||||||
|
return ReactiveSecurityContextHolder
|
||||||
|
.getContext()
|
||||||
|
.map(SecurityContext::getAuthentication)
|
||||||
|
.map(Authentication::getPrincipal)
|
||||||
|
.ofType(String.class);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue