Skip to content

Commit 14b6bd4

Browse files
committed
Add PrincipalResolver to RestClient sample
Closes gh-328
1 parent ec4d7f9 commit 14b6bd4

File tree

4 files changed

+101
-5
lines changed

4 files changed

+101
-5
lines changed

servlet/spring-boot/java/oauth2/restclient/README.adoc

+8
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,14 @@ Activate one of the following profiles to try them out:
6060

6161
4. `authentication-required` - Demonstrates a custom `ClientRegistrationIdResolver` that requires authentication using OAuth 2.0 or Open ID Connect 1.0. Uses `login-client-with-messaging` to log in.
6262

63+
This sample also demonstrates alternate strategies for resolving a `principal` (see https://github.com/spring-projects/spring-security-samples/tree/main/servlet/spring-boot/java/oauth2/restclient/src/main/java/example/PrincipalResolverConfiguration.java[PrincipalResolverConfiguration] for more information).
64+
65+
Activate one of the following profiles to try them out:
66+
67+
1. `per-request` - Demonstrates an alternate setup with `RequestAttributePrincipalResolver` that resolves the principal using attributes.
68+
69+
2. `anonymous-user` - Demonstrates a custom `PrincipalResolver` that statically resolves a `principal`. Requires specifying the `principal` via `RequestAttributePrincipalResolver.principal(Authentication)`.
70+
6371
[TIP]
6472
====
6573
You can activate a profile with the `./gradlew bootRun` command by adding the argument `--args='--spring.profiles.active=xyz'`.

servlet/spring-boot/java/oauth2/restclient/src/main/java/example/ClientRegistrationIdResolverConfiguration.java

+3-3
Original file line numberDiff line numberDiff line change
@@ -116,7 +116,7 @@ private static ClientRegistrationIdResolver authenticationRequiredClientRegistra
116116

117117
@Configuration
118118
@Profile("current-user")
119-
public static class CurrentUserConfiguration {
119+
public static class CurrentUserClientRegistrationIdResolverConfiguration {
120120

121121
@Bean
122122
public ClientRegistrationIdResolver clientRegistrationIdResolver() {
@@ -127,7 +127,7 @@ public ClientRegistrationIdResolver clientRegistrationIdResolver() {
127127

128128
@Configuration
129129
@Profile("composite")
130-
public static class CompositeConfiguration {
130+
public static class CompositeClientRegistrationIdResolverConfiguration {
131131

132132
@Bean
133133
public ClientRegistrationIdResolver clientRegistrationIdResolver() {
@@ -138,7 +138,7 @@ public ClientRegistrationIdResolver clientRegistrationIdResolver() {
138138

139139
@Configuration
140140
@Profile("authentication-required")
141-
public static class AuthenticationRequiredConfiguration {
141+
public static class AuthenticationRequiredClientRegistrationIdResolverConfiguration {
142142

143143
@Bean
144144
public ClientRegistrationIdResolver clientRegistrationIdResolver() {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
/*
2+
* Copyright 2024 the original author or authors.
3+
*
4+
* Licensed under the Apache License, Version 2.0 (the "License");
5+
* you may not use this file except in compliance with the License.
6+
* You may obtain a copy of the License at
7+
*
8+
* https://www.apache.org/licenses/LICENSE-2.0
9+
*
10+
* Unless required by applicable law or agreed to in writing, software
11+
* distributed under the License is distributed on an "AS IS" BASIS,
12+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
* See the License for the specific language governing permissions and
14+
* limitations under the License.
15+
*/
16+
17+
package example;
18+
19+
import org.springframework.context.annotation.Bean;
20+
import org.springframework.context.annotation.Configuration;
21+
import org.springframework.context.annotation.Profile;
22+
import org.springframework.security.authentication.AnonymousAuthenticationToken;
23+
import org.springframework.security.core.Authentication;
24+
import org.springframework.security.core.authority.AuthorityUtils;
25+
import org.springframework.security.oauth2.client.web.client.OAuth2ClientHttpRequestInterceptor.PrincipalResolver;
26+
import org.springframework.security.oauth2.client.web.client.RequestAttributePrincipalResolver;
27+
import org.springframework.security.oauth2.client.web.client.SecurityContextHolderPrincipalResolver;
28+
29+
/**
30+
* Configuration for demonstrating additional strategies for resolving a {@code principal}
31+
* via the {@link PrincipalResolver}. This sample uses the following profiles to
32+
* demonstrate multiple configurations:
33+
*
34+
* <ol>
35+
* <li>{@code default} - Demonstrates the default setup with
36+
* {@link SecurityContextHolderPrincipalResolver}.</li>
37+
* <li>{@code per-request} - Demonstrates an alternate setup with
38+
* {@link RequestAttributePrincipalResolver}. Requires specifying the {@code principal}
39+
* via {@link RequestAttributePrincipalResolver#principal(Authentication)}.</li>
40+
* <li>{@code anonymous-user} - Demonstrates a custom {@link PrincipalResolver} that
41+
* statically resolves the {@code principal}.</li>
42+
* </ol>
43+
*
44+
* @author Steve Riesenberg
45+
*/
46+
public class PrincipalResolverConfiguration {
47+
48+
@Configuration
49+
@Profile("per-request")
50+
public static class PerRequestPrincipalResolverConfiguration {
51+
52+
@Bean
53+
public PrincipalResolver principalResolver() {
54+
return new RequestAttributePrincipalResolver();
55+
}
56+
57+
}
58+
59+
@Configuration
60+
@Profile("anonymous-user")
61+
public static class AnonymousUserPrincipalResolverConfiguration {
62+
63+
@Bean
64+
public PrincipalResolver principalResolver() {
65+
AnonymousAuthenticationToken anonymousUser = new AnonymousAuthenticationToken("anonymous", "anonymousUser",
66+
AuthorityUtils.createAuthorityList("ROLE_ANONYMOUS"));
67+
return (request) -> anonymousUser;
68+
}
69+
70+
}
71+
72+
}

servlet/spring-boot/java/oauth2/restclient/src/main/java/example/RestClientConfiguration.java

+18-2
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525
import org.springframework.security.oauth2.client.web.OAuth2AuthorizedClientRepository;
2626
import org.springframework.security.oauth2.client.web.client.OAuth2ClientHttpRequestInterceptor;
2727
import org.springframework.security.oauth2.client.web.client.RequestAttributeClientRegistrationIdResolver;
28+
import org.springframework.security.oauth2.client.web.client.SecurityContextHolderPrincipalResolver;
2829
import org.springframework.web.client.RestClient;
2930

3031
/**
@@ -45,11 +46,12 @@ public RestClientConfiguration(@Value("${messages.base-url}") String baseUrl) {
4546
public RestClient restClient(OAuth2AuthorizedClientManager authorizedClientManager,
4647
OAuth2AuthorizedClientRepository authorizedClientRepository,
4748
OAuth2ClientHttpRequestInterceptor.ClientRegistrationIdResolver clientRegistrationIdResolver,
48-
RestClient.Builder builder) {
49+
OAuth2ClientHttpRequestInterceptor.PrincipalResolver principalResolver, RestClient.Builder builder) {
4950

5051
OAuth2ClientHttpRequestInterceptor requestInterceptor = new OAuth2ClientHttpRequestInterceptor(
5152
authorizedClientManager);
5253
requestInterceptor.setClientRegistrationIdResolver(clientRegistrationIdResolver);
54+
requestInterceptor.setPrincipalResolver(principalResolver);
5355

5456
OAuth2AuthorizationFailureHandler authorizationFailureHandler = OAuth2ClientHttpRequestInterceptor
5557
.authorizationFailureHandler(authorizedClientRepository);
@@ -62,7 +64,8 @@ public RestClient restClient(OAuth2AuthorizedClientManager authorizedClientManag
6264
* This sample uses profiles to demonstrate additional strategies for resolving the
6365
* {@code clientRegistrationId}. See {@link ClientRegistrationIdResolverConfiguration}
6466
* for alternate implementations.
65-
* @return the default {@link ClientRegistrationIdResolverConfiguration}
67+
* @return the default
68+
* {@link OAuth2ClientHttpRequestInterceptor.ClientRegistrationIdResolver}
6669
* @see ClientRegistrationIdResolverConfiguration
6770
*/
6871
@Bean
@@ -71,4 +74,17 @@ public OAuth2ClientHttpRequestInterceptor.ClientRegistrationIdResolver clientReg
7174
return new RequestAttributeClientRegistrationIdResolver();
7275
}
7376

77+
/**
78+
* This sample uses profiles to demonstrate additional strategies for resolving the
79+
* {@code principal}. See {@link PrincipalResolverConfiguration} for alternate
80+
* implementations.
81+
* @return the default {@link OAuth2ClientHttpRequestInterceptor.PrincipalResolver}
82+
* @see PrincipalResolverConfiguration
83+
*/
84+
@Bean
85+
@ConditionalOnMissingBean
86+
public OAuth2ClientHttpRequestInterceptor.PrincipalResolver principalResolver() {
87+
return new SecurityContextHolderPrincipalResolver();
88+
}
89+
7490
}

0 commit comments

Comments
 (0)