5 changed files with 120 additions and 303 deletions
@ -1,248 +0,0 @@ |
|||||||
/* |
|
||||||
* 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.web; |
|
||||||
|
|
||||||
import java.util.Map; |
|
||||||
import java.util.function.Consumer; |
|
||||||
|
|
||||||
import org.reactivestreams.Subscription; |
|
||||||
import reactor.core.CoreSubscriber; |
|
||||||
import reactor.core.publisher.Hooks; |
|
||||||
import reactor.core.publisher.Mono; |
|
||||||
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.lang.Nullable; |
|
||||||
import org.springframework.security.core.Authentication; |
|
||||||
import org.springframework.security.core.context.SecurityContextHolder; |
|
||||||
import org.springframework.security.oauth2.core.AbstractOAuth2Token; |
|
||||||
import org.springframework.web.reactive.function.client.ClientRequest; |
|
||||||
import org.springframework.web.reactive.function.client.ClientResponse; |
|
||||||
import org.springframework.web.reactive.function.client.ExchangeFilterFunction; |
|
||||||
import org.springframework.web.reactive.function.client.ExchangeFunction; |
|
||||||
import org.springframework.web.reactive.function.client.WebClient; |
|
||||||
|
|
||||||
/** |
|
||||||
* An {@link ExchangeFilterFunction} that adds the |
|
||||||
* <a href="https://tools.ietf.org/html/rfc6750#section-1.2" target="_blank">Bearer Token</a> |
|
||||||
* from an existing {@link AbstractOAuth2Token} tied to the current {@link Authentication}. |
|
||||||
* |
|
||||||
* Suitable for Servlet applications, applying it to a typical {@link org.springframework.web.reactive.function.client.WebClient} |
|
||||||
* configuration: |
|
||||||
* |
|
||||||
* <pre> |
|
||||||
* @Bean |
|
||||||
* WebClient webClient() { |
|
||||||
* ServletBearerExchangeFilterFunction bearer = new ServletBearerExchangeFilterFunction(); |
|
||||||
* return WebClient.builder() |
|
||||||
* .apply(bearer.oauth2Configuration()) |
|
||||||
* .build(); |
|
||||||
* } |
|
||||||
* </pre> |
|
||||||
* |
|
||||||
* @author Josh Cummings |
|
||||||
* @since 5.2 |
|
||||||
*/ |
|
||||||
public class ServletBearerExchangeFilterFunction |
|
||||||
implements ExchangeFilterFunction, InitializingBean, DisposableBean { |
|
||||||
|
|
||||||
private static final String AUTHENTICATION_ATTR_NAME = Authentication.class.getName(); |
|
||||||
|
|
||||||
private static final String REQUEST_CONTEXT_OPERATOR_KEY = RequestContextSubscriber.class.getName(); |
|
||||||
|
|
||||||
/** |
|
||||||
* {@inheritDoc} |
|
||||||
*/ |
|
||||||
@Override |
|
||||||
public void afterPropertiesSet() throws Exception { |
|
||||||
Hooks.onLastOperator(REQUEST_CONTEXT_OPERATOR_KEY, |
|
||||||
Operators.liftPublisher((s, sub) -> createRequestContextSubscriber(sub))); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* {@inheritDoc} |
|
||||||
*/ |
|
||||||
@Override |
|
||||||
public void destroy() throws Exception { |
|
||||||
Hooks.resetOnLastOperator(REQUEST_CONTEXT_OPERATOR_KEY); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Configures the builder with {@link #defaultRequest()} and adds this as a {@link ExchangeFilterFunction} |
|
||||||
* @return the {@link Consumer} to configure the builder |
|
||||||
*/ |
|
||||||
public Consumer<WebClient.Builder> oauth2Configuration() { |
|
||||||
return builder -> builder.defaultRequest(defaultRequest()).filter(this); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Provides defaults for the {@link Authentication} using |
|
||||||
* {@link SecurityContextHolder}. It also can default the {@link AbstractOAuth2Token} using the |
|
||||||
* {@link #authentication(Authentication)}. |
|
||||||
* @return the {@link Consumer} to populate the attributes |
|
||||||
*/ |
|
||||||
public Consumer<WebClient.RequestHeadersSpec<?>> defaultRequest() { |
|
||||||
return spec -> spec.attributes(attrs -> { |
|
||||||
populateDefaultAuthentication(attrs); |
|
||||||
}); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Modifies the {@link ClientRequest#attributes()} to include the {@link Authentication} used to |
|
||||||
* look up and save the {@link AbstractOAuth2Token}. The value is defaulted in |
|
||||||
* {@link ServletBearerExchangeFilterFunction#defaultRequest()} |
|
||||||
* |
|
||||||
* @param authentication the {@link Authentication} to use. |
|
||||||
* @return the {@link Consumer} to populate the attributes |
|
||||||
*/ |
|
||||||
public static Consumer<Map<String, Object>> authentication(Authentication authentication) { |
|
||||||
return attributes -> attributes.put(AUTHENTICATION_ATTR_NAME, authentication); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* {@inheritDoc} |
|
||||||
*/ |
|
||||||
@Override |
|
||||||
public Mono<ClientResponse> filter(ClientRequest request, ExchangeFunction next) { |
|
||||||
return mergeRequestAttributesIfNecessary(request) |
|
||||||
.filter(req -> req.attribute(AUTHENTICATION_ATTR_NAME).isPresent()) |
|
||||||
.map(req -> getOAuth2Token(req.attributes())) |
|
||||||
.map(token -> bearer(request, token)) |
|
||||||
.flatMap(next::exchange) |
|
||||||
.switchIfEmpty(Mono.defer(() -> next.exchange(request))); |
|
||||||
} |
|
||||||
|
|
||||||
private Mono<ClientRequest> mergeRequestAttributesIfNecessary(ClientRequest request) { |
|
||||||
if (request.attribute(AUTHENTICATION_ATTR_NAME).isPresent()) { |
|
||||||
return Mono.just(request); |
|
||||||
} |
|
||||||
|
|
||||||
return mergeRequestAttributesFromContext(request); |
|
||||||
} |
|
||||||
|
|
||||||
private Mono<ClientRequest> mergeRequestAttributesFromContext(ClientRequest request) { |
|
||||||
ClientRequest.Builder builder = ClientRequest.from(request); |
|
||||||
return Mono.subscriberContext() |
|
||||||
.map(ctx -> builder.attributes(attrs -> populateRequestAttributes(attrs, ctx))) |
|
||||||
.map(ClientRequest.Builder::build); |
|
||||||
} |
|
||||||
|
|
||||||
private void populateRequestAttributes(Map<String, Object> attrs, Context ctx) { |
|
||||||
RequestContextDataHolder holder = RequestContextSubscriber.getRequestContext(ctx); |
|
||||||
if (holder == null) { |
|
||||||
return; |
|
||||||
} |
|
||||||
if (holder.getAuthentication() != null) { |
|
||||||
attrs.putIfAbsent(AUTHENTICATION_ATTR_NAME, holder.getAuthentication()); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
private AbstractOAuth2Token getOAuth2Token(Map<String, Object> attrs) { |
|
||||||
Authentication authentication = (Authentication) attrs.get(AUTHENTICATION_ATTR_NAME); |
|
||||||
if (authentication.getCredentials() instanceof AbstractOAuth2Token) { |
|
||||||
return (AbstractOAuth2Token) authentication.getCredentials(); |
|
||||||
} |
|
||||||
return null; |
|
||||||
} |
|
||||||
|
|
||||||
private ClientRequest bearer(ClientRequest request, AbstractOAuth2Token token) { |
|
||||||
return ClientRequest.from(request) |
|
||||||
.headers(headers -> headers.setBearerAuth(token.getTokenValue())) |
|
||||||
.build(); |
|
||||||
} |
|
||||||
|
|
||||||
private <T> CoreSubscriber<T> createRequestContextSubscriber(CoreSubscriber<T> delegate) { |
|
||||||
Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); |
|
||||||
return new RequestContextSubscriber<>(delegate, authentication); |
|
||||||
} |
|
||||||
|
|
||||||
private void populateDefaultAuthentication(Map<String, Object> attrs) { |
|
||||||
if (attrs.containsKey(AUTHENTICATION_ATTR_NAME)) { |
|
||||||
return; |
|
||||||
} |
|
||||||
Authentication authentication = SecurityContextHolder.getContext().getAuthentication(); |
|
||||||
attrs.putIfAbsent(AUTHENTICATION_ATTR_NAME, authentication); |
|
||||||
} |
|
||||||
|
|
||||||
private static class RequestContextDataHolder { |
|
||||||
private final Authentication authentication; |
|
||||||
|
|
||||||
RequestContextDataHolder(Authentication authentication) { |
|
||||||
this.authentication = authentication; |
|
||||||
} |
|
||||||
|
|
||||||
public Authentication getAuthentication() { |
|
||||||
return this.authentication; |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
private static class RequestContextSubscriber<T> implements CoreSubscriber<T> { |
|
||||||
private static final String REQUEST_CONTEXT_DATA_HOLDER_ATTR_NAME = |
|
||||||
RequestContextSubscriber.class.getName().concat(".REQUEST_CONTEXT_DATA_HOLDER"); |
|
||||||
|
|
||||||
private CoreSubscriber<T> delegate; |
|
||||||
private final Context context; |
|
||||||
|
|
||||||
private RequestContextSubscriber(CoreSubscriber<T> delegate, |
|
||||||
Authentication authentication) { |
|
||||||
|
|
||||||
this.delegate = delegate; |
|
||||||
Context parentContext = this.delegate.currentContext(); |
|
||||||
Context context; |
|
||||||
if (authentication == null || parentContext.hasKey(REQUEST_CONTEXT_DATA_HOLDER_ATTR_NAME)) { |
|
||||||
context = parentContext; |
|
||||||
} else { |
|
||||||
context = parentContext.put(REQUEST_CONTEXT_DATA_HOLDER_ATTR_NAME, |
|
||||||
new RequestContextDataHolder(authentication)); |
|
||||||
} |
|
||||||
|
|
||||||
this.context = context; |
|
||||||
} |
|
||||||
|
|
||||||
@Nullable |
|
||||||
static RequestContextDataHolder getRequestContext(Context ctx) { |
|
||||||
return ctx.getOrDefault(REQUEST_CONTEXT_DATA_HOLDER_ATTR_NAME, null); |
|
||||||
} |
|
||||||
|
|
||||||
@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(); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
42
oauth2/oauth2-resource-server/src/main/java/org/springframework/security/oauth2/server/resource/web/server/ServerBearerExchangeFilterFunction.java → oauth2/oauth2-resource-server/src/main/java/org/springframework/security/oauth2/server/resource/web/reactive/function/client/ServerBearerExchangeFilterFunction.java
42
oauth2/oauth2-resource-server/src/main/java/org/springframework/security/oauth2/server/resource/web/server/ServerBearerExchangeFilterFunction.java → oauth2/oauth2-resource-server/src/main/java/org/springframework/security/oauth2/server/resource/web/reactive/function/client/ServerBearerExchangeFilterFunction.java
@ -0,0 +1,85 @@ |
|||||||
|
/* |
||||||
|
* 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.web.reactive.function.client; |
||||||
|
|
||||||
|
import reactor.core.publisher.Mono; |
||||||
|
|
||||||
|
import org.springframework.security.authentication.AnonymousAuthenticationToken; |
||||||
|
import org.springframework.security.core.Authentication; |
||||||
|
import org.springframework.security.core.authority.AuthorityUtils; |
||||||
|
import org.springframework.security.core.context.SecurityContextHolder; |
||||||
|
import org.springframework.security.oauth2.core.AbstractOAuth2Token; |
||||||
|
import org.springframework.web.reactive.function.client.ClientRequest; |
||||||
|
import org.springframework.web.reactive.function.client.ClientResponse; |
||||||
|
import org.springframework.web.reactive.function.client.ExchangeFilterFunction; |
||||||
|
import org.springframework.web.reactive.function.client.ExchangeFunction; |
||||||
|
|
||||||
|
/** |
||||||
|
* An {@link ExchangeFilterFunction} that adds the |
||||||
|
* <a href="https://tools.ietf.org/html/rfc6750#section-1.2" target="_blank">Bearer Token</a> |
||||||
|
* from an existing {@link AbstractOAuth2Token} tied to the current {@link Authentication}. |
||||||
|
* |
||||||
|
* Suitable for Servlet applications, applying it to a typical {@link org.springframework.web.reactive.function.client.WebClient} |
||||||
|
* configuration: |
||||||
|
* |
||||||
|
* <pre> |
||||||
|
* @Bean |
||||||
|
* WebClient webClient() { |
||||||
|
* ServletBearerExchangeFilterFunction bearer = new ServletBearerExchangeFilterFunction(); |
||||||
|
* return WebClient.builder() |
||||||
|
* .filter(bearer).build(); |
||||||
|
* } |
||||||
|
* </pre> |
||||||
|
* |
||||||
|
* @author Josh Cummings |
||||||
|
* @since 5.2 |
||||||
|
*/ |
||||||
|
public final class ServletBearerExchangeFilterFunction |
||||||
|
implements ExchangeFilterFunction { |
||||||
|
|
||||||
|
private static final AnonymousAuthenticationToken ANONYMOUS_USER_TOKEN = new AnonymousAuthenticationToken("anonymous", "anonymousUser", |
||||||
|
AuthorityUtils.createAuthorityList("ROLE_USER")); |
||||||
|
|
||||||
|
/** |
||||||
|
* {@inheritDoc} |
||||||
|
*/ |
||||||
|
@Override |
||||||
|
public Mono<ClientResponse> filter(ClientRequest request, ExchangeFunction next) { |
||||||
|
return oauth2Token() |
||||||
|
.map(token -> bearer(request, token)) |
||||||
|
.defaultIfEmpty(request) |
||||||
|
.flatMap(next::exchange); |
||||||
|
} |
||||||
|
|
||||||
|
private Mono<AbstractOAuth2Token> oauth2Token() { |
||||||
|
return currentAuthentication() |
||||||
|
.filter(authentication -> authentication.getCredentials() instanceof AbstractOAuth2Token) |
||||||
|
.map(Authentication::getCredentials) |
||||||
|
.cast(AbstractOAuth2Token.class); |
||||||
|
} |
||||||
|
|
||||||
|
private Mono<Authentication> currentAuthentication() { |
||||||
|
return Mono.justOrEmpty(SecurityContextHolder.getContext().getAuthentication()) |
||||||
|
.defaultIfEmpty(ANONYMOUS_USER_TOKEN); |
||||||
|
} |
||||||
|
|
||||||
|
private ClientRequest bearer(ClientRequest request, AbstractOAuth2Token token) { |
||||||
|
return ClientRequest.from(request) |
||||||
|
.headers(headers -> headers.setBearerAuth(token.getTokenValue())) |
||||||
|
.build(); |
||||||
|
} |
||||||
|
} |
||||||
20
oauth2/oauth2-resource-server/src/test/java/org/springframework/security/oauth2/server/resource/web/server/ServerBearerExchangeFilterFunctionTests.java → oauth2/oauth2-resource-server/src/test/java/org/springframework/security/oauth2/server/resource/web/reactive/function/client/ServerBearerExchangeFilterFunctionTests.java
20
oauth2/oauth2-resource-server/src/test/java/org/springframework/security/oauth2/server/resource/web/server/ServerBearerExchangeFilterFunctionTests.java → oauth2/oauth2-resource-server/src/test/java/org/springframework/security/oauth2/server/resource/web/reactive/function/client/ServerBearerExchangeFilterFunctionTests.java
28
oauth2/oauth2-resource-server/src/test/java/org/springframework/security/oauth2/server/resource/web/ServletBearerExchangeFilterFunctionTests.java → oauth2/oauth2-resource-server/src/test/java/org/springframework/security/oauth2/server/resource/web/reactive/function/client/ServletBearerExchangeFilterFunctionTests.java
28
oauth2/oauth2-resource-server/src/test/java/org/springframework/security/oauth2/server/resource/web/ServletBearerExchangeFilterFunctionTests.java → oauth2/oauth2-resource-server/src/test/java/org/springframework/security/oauth2/server/resource/web/reactive/function/client/ServletBearerExchangeFilterFunctionTests.java
Loading…
Reference in new issue