7 changed files with 407 additions and 30 deletions
@ -0,0 +1,144 @@
@@ -0,0 +1,144 @@
|
||||
/* |
||||
* Copyright 2002-2019 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 |
||||
* |
||||
* https://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.configuration; |
||||
|
||||
import org.reactivestreams.Subscription; |
||||
import reactor.core.CoreSubscriber; |
||||
import reactor.core.publisher.Hooks; |
||||
import reactor.core.publisher.Operators; |
||||
import reactor.util.context.Context; |
||||
|
||||
import org.springframework.beans.factory.DisposableBean; |
||||
import org.springframework.beans.factory.InitializingBean; |
||||
import org.springframework.context.annotation.Bean; |
||||
import org.springframework.context.annotation.Configuration; |
||||
import org.springframework.context.annotation.Import; |
||||
import org.springframework.context.annotation.ImportSelector; |
||||
import org.springframework.core.type.AnnotationMetadata; |
||||
import org.springframework.security.core.Authentication; |
||||
import org.springframework.security.core.context.SecurityContextHolder; |
||||
import org.springframework.util.ClassUtils; |
||||
|
||||
/** |
||||
* {@link Configuration} for OAuth 2.0 Resource Server support. |
||||
* |
||||
* <p> |
||||
* This {@code Configuration} is conditionally imported by {@link OAuth2ImportSelector} |
||||
* when the {@code spring-security-oauth2-resource-server} module is present on the classpath. |
||||
* |
||||
* @author Josh Cummings |
||||
* @since 5.2 |
||||
* @see OAuth2ImportSelector |
||||
*/ |
||||
@Import(OAuth2ResourceServerConfiguration.OAuth2ClientWebFluxImportSelector.class) |
||||
final class OAuth2ResourceServerConfiguration { |
||||
|
||||
static class OAuth2ClientWebFluxImportSelector implements ImportSelector { |
||||
|
||||
@Override |
||||
public String[] selectImports(AnnotationMetadata importingClassMetadata) { |
||||
boolean webfluxPresent = ClassUtils.isPresent( |
||||
"org.springframework.web.reactive.function.client.WebClient", getClass().getClassLoader()); |
||||
|
||||
return webfluxPresent ? |
||||
new String[] { "org.springframework.security.config.annotation.web.configuration.OAuth2ResourceServerConfiguration.OAuth2ResourceServerWebFluxSecurityConfiguration" } : |
||||
new String[] {}; |
||||
} |
||||
} |
||||
|
||||
@Configuration(proxyBeanMethods = false) |
||||
static class OAuth2ResourceServerWebFluxSecurityConfiguration { |
||||
@Bean |
||||
BearerRequestContextSubscriberRegistrar bearerRequestContextSubscriberRegistrar() { |
||||
return new BearerRequestContextSubscriberRegistrar(); |
||||
} |
||||
|
||||
/** |
||||
* Registers a {@link CoreSubscriber} that provides the current {@link Authentication} |
||||
* to the correct {@link Context}. |
||||
* |
||||
* This is published as a {@code @Bean} automatically, so long as `spring-security-oauth2-resource-server` |
||||
* and `spring-webflux` are on the classpath. |
||||
*/ |
||||
static class BearerRequestContextSubscriberRegistrar |
||||
implements InitializingBean, DisposableBean { |
||||
|
||||
private static final String REQUEST_CONTEXT_OPERATOR_KEY = BearerRequestContextSubscriber.class.getName(); |
||||
|
||||
@Override |
||||
public void afterPropertiesSet() throws Exception { |
||||
Hooks.onLastOperator(REQUEST_CONTEXT_OPERATOR_KEY, |
||||
Operators.liftPublisher((s, sub) -> createRequestContextSubscriber(sub))); |
||||
} |
||||
|
||||
@Override |
||||
public void destroy() throws Exception { |
||||
Hooks.resetOnLastOperator(REQUEST_CONTEXT_OPERATOR_KEY); |
||||
} |
||||
|
||||
private <T> CoreSubscriber<T> createRequestContextSubscriber(CoreSubscriber<T> delegate) { |
||||
Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); |
||||
return new BearerRequestContextSubscriber<>(delegate, authentication); |
||||
} |
||||
|
||||
static class BearerRequestContextSubscriber<T> implements CoreSubscriber<T> { |
||||
private CoreSubscriber<T> delegate; |
||||
private final Context context; |
||||
|
||||
private BearerRequestContextSubscriber(CoreSubscriber<T> delegate, |
||||
Authentication authentication) { |
||||
|
||||
this.delegate = delegate; |
||||
Context parentContext = this.delegate.currentContext(); |
||||
Context context; |
||||
if (authentication == null || parentContext.hasKey(Authentication.class)) { |
||||
context = parentContext; |
||||
} else { |
||||
context = parentContext.put(Authentication.class, authentication); |
||||
} |
||||
|
||||
this.context = context; |
||||
} |
||||
|
||||
@Override |
||||
public Context currentContext() { |
||||
return this.context; |
||||
} |
||||
|
||||
@Override |
||||
public void onSubscribe(Subscription s) { |
||||
this.delegate.onSubscribe(s); |
||||
} |
||||
|
||||
@Override |
||||
public void onNext(T t) { |
||||
this.delegate.onNext(t); |
||||
} |
||||
|
||||
@Override |
||||
public void onError(Throwable t) { |
||||
this.delegate.onError(t); |
||||
} |
||||
|
||||
@Override |
||||
public void onComplete() { |
||||
this.delegate.onComplete(); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
||||
@ -0,0 +1,165 @@
@@ -0,0 +1,165 @@
|
||||
/* |
||||
* Copyright 2002-2019 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 |
||||
* |
||||
* https://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.configuration; |
||||
|
||||
import javax.annotation.PreDestroy; |
||||
|
||||
import okhttp3.mockwebserver.Dispatcher; |
||||
import okhttp3.mockwebserver.MockResponse; |
||||
import okhttp3.mockwebserver.MockWebServer; |
||||
import okhttp3.mockwebserver.RecordedRequest; |
||||
import org.apache.commons.lang.StringUtils; |
||||
import org.junit.Rule; |
||||
import org.junit.Test; |
||||
|
||||
import org.springframework.beans.factory.annotation.Autowired; |
||||
import org.springframework.context.annotation.Bean; |
||||
import org.springframework.context.annotation.Configuration; |
||||
import org.springframework.security.config.annotation.web.builders.HttpSecurity; |
||||
import org.springframework.security.config.test.SpringTestRule; |
||||
import org.springframework.security.oauth2.server.resource.authentication.BearerTokenAuthentication; |
||||
import org.springframework.security.oauth2.server.resource.web.reactive.function.client.ServletBearerExchangeFilterFunction; |
||||
import org.springframework.test.web.servlet.MockMvc; |
||||
import org.springframework.web.bind.annotation.GetMapping; |
||||
import org.springframework.web.bind.annotation.RestController; |
||||
import org.springframework.web.reactive.function.client.WebClient; |
||||
|
||||
import static org.springframework.security.oauth2.server.resource.authentication.TestBearerTokenAuthentications.bearer; |
||||
import static org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.authentication; |
||||
import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; |
||||
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.content; |
||||
import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; |
||||
|
||||
/** |
||||
* Tests for {@link OAuth2ResourceServerConfiguration}. |
||||
* |
||||
* @author Josh Cummings |
||||
*/ |
||||
public class OAuth2ResourceServerConfigurationTests { |
||||
@Rule |
||||
public final SpringTestRule spring = new SpringTestRule(); |
||||
|
||||
@Autowired |
||||
private MockMvc mockMvc; |
||||
|
||||
// gh-7418
|
||||
@Test |
||||
public void requestWhenUsingFilterThenBearerTokenPropagated() throws Exception { |
||||
BearerTokenAuthentication authentication = bearer(); |
||||
this.spring.register(BearerFilterConfig.class, WebServerConfig.class, Controller.class).autowire(); |
||||
|
||||
this.mockMvc.perform(get("/token") |
||||
.with(authentication(authentication))) |
||||
.andExpect(status().isOk()) |
||||
.andExpect(content().string("Bearer token")); |
||||
} |
||||
|
||||
// gh-7418
|
||||
@Test |
||||
public void requestWhenNotUsingFilterThenBearerTokenNotPropagated() throws Exception { |
||||
BearerTokenAuthentication authentication = bearer(); |
||||
this.spring.register(BearerFilterlessConfig.class, WebServerConfig.class, Controller.class).autowire(); |
||||
|
||||
this.mockMvc.perform(get("/token") |
||||
.with(authentication(authentication))) |
||||
.andExpect(status().isOk()) |
||||
.andExpect(content().string("")); |
||||
} |
||||
|
||||
|
||||
@EnableWebSecurity |
||||
static class BearerFilterConfig extends WebSecurityConfigurerAdapter { |
||||
@Override |
||||
protected void configure(HttpSecurity http) throws Exception { |
||||
} |
||||
|
||||
@Bean |
||||
WebClient rest() { |
||||
ServletBearerExchangeFilterFunction bearer = |
||||
new ServletBearerExchangeFilterFunction(); |
||||
return WebClient.builder() |
||||
.filter(bearer).build(); |
||||
} |
||||
} |
||||
|
||||
@EnableWebSecurity |
||||
static class BearerFilterlessConfig extends WebSecurityConfigurerAdapter { |
||||
@Override |
||||
protected void configure(HttpSecurity http) throws Exception { |
||||
} |
||||
|
||||
@Bean |
||||
WebClient rest() { |
||||
return WebClient.create(); |
||||
} |
||||
} |
||||
|
||||
@RestController |
||||
static class Controller { |
||||
private final WebClient rest; |
||||
private final String uri; |
||||
|
||||
@Autowired |
||||
Controller(MockWebServer server, WebClient rest) { |
||||
this.uri = server.url("/").toString(); |
||||
this.rest = rest; |
||||
} |
||||
|
||||
@GetMapping("/token") |
||||
public String token() { |
||||
return this.rest.get() |
||||
.uri(this.uri) |
||||
.retrieve() |
||||
.bodyToMono(String.class) |
||||
.flatMap(result -> this.rest.get() |
||||
.uri(this.uri) |
||||
.retrieve() |
||||
.bodyToMono(String.class)) |
||||
.block(); |
||||
} |
||||
} |
||||
|
||||
@Configuration |
||||
static class WebServerConfig { |
||||
private final MockWebServer server = new MockWebServer(); |
||||
|
||||
@Bean |
||||
MockWebServer server() throws Exception { |
||||
this.server.setDispatcher(new AuthorizationHeaderDispatcher()); |
||||
this.server.start(); |
||||
return this.server; |
||||
} |
||||
|
||||
@PreDestroy |
||||
void shutdown() throws Exception { |
||||
this.server.shutdown(); |
||||
} |
||||
} |
||||
|
||||
static class AuthorizationHeaderDispatcher extends Dispatcher { |
||||
|
||||
@Override |
||||
public MockResponse dispatch(RecordedRequest request) { |
||||
MockResponse response = new MockResponse().setResponseCode(200); |
||||
String header = request.getHeader("Authorization"); |
||||
if (StringUtils.isBlank(header)) { |
||||
return response; |
||||
|
||||
} |
||||
return response.setBody(header); |
||||
} |
||||
} |
||||
} |
||||
@ -0,0 +1,51 @@
@@ -0,0 +1,51 @@
|
||||
/* |
||||
* Copyright 2002-2019 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 |
||||
* |
||||
* https://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.oauth2.server.resource.authentication; |
||||
|
||||
import java.time.Instant; |
||||
import java.util.Arrays; |
||||
import java.util.Collection; |
||||
import java.util.Collections; |
||||
import java.util.HashSet; |
||||
|
||||
import org.springframework.security.core.GrantedAuthority; |
||||
import org.springframework.security.core.authority.AuthorityUtils; |
||||
import org.springframework.security.oauth2.core.DefaultOAuth2AuthenticatedPrincipal; |
||||
import org.springframework.security.oauth2.core.OAuth2AccessToken; |
||||
import org.springframework.security.oauth2.core.OAuth2AuthenticatedPrincipal; |
||||
|
||||
/** |
||||
* Test instances of {@link BearerTokenAuthentication} |
||||
* |
||||
* @author Josh Cummings |
||||
*/ |
||||
public class TestBearerTokenAuthentications { |
||||
public static BearerTokenAuthentication bearer() { |
||||
Collection<GrantedAuthority> authorities = |
||||
AuthorityUtils.createAuthorityList("SCOPE_USER"); |
||||
OAuth2AuthenticatedPrincipal principal = |
||||
new DefaultOAuth2AuthenticatedPrincipal( |
||||
Collections.singletonMap("sub", "user"), |
||||
authorities); |
||||
OAuth2AccessToken token = |
||||
new OAuth2AccessToken(OAuth2AccessToken.TokenType.BEARER, |
||||
"token", Instant.now(), Instant.now().plusSeconds(86400), |
||||
new HashSet<>(Arrays.asList("USER"))); |
||||
|
||||
return new BearerTokenAuthentication(principal, token, authorities); |
||||
} |
||||
} |
||||
Loading…
Reference in new issue