Skip to content

Commit 702538e

Browse files
AuthorizationEventPublisher Accepts AuthorizationResult
Closes gh-15915 Co-authored-by: Max Batischev <[email protected]>
1 parent ef1226d commit 702538e

File tree

17 files changed

+196
-33
lines changed

17 files changed

+196
-33
lines changed

config/src/test/java/org/springframework/security/config/annotation/method/configuration/PrePostMethodSecurityConfigurationTests.java

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,7 @@
7575
import org.springframework.security.authorization.AuthorizationDecision;
7676
import org.springframework.security.authorization.AuthorizationEventPublisher;
7777
import org.springframework.security.authorization.AuthorizationManager;
78+
import org.springframework.security.authorization.AuthorizationResult;
7879
import org.springframework.security.authorization.method.AuthorizationAdvisor;
7980
import org.springframework.security.authorization.method.AuthorizationAdvisorProxyFactory;
8081
import org.springframework.security.authorization.method.AuthorizationAdvisorProxyFactory.TargetVisitor;
@@ -110,6 +111,7 @@
110111
import static org.mockito.ArgumentMatchers.any;
111112
import static org.mockito.Mockito.atLeastOnce;
112113
import static org.mockito.Mockito.clearInvocations;
114+
import static org.mockito.Mockito.doCallRealMethod;
113115
import static org.mockito.Mockito.mock;
114116
import static org.mockito.Mockito.never;
115117
import static org.mockito.Mockito.spy;
@@ -1293,6 +1295,8 @@ static class AuthorizationEventPublisherConfig {
12931295

12941296
@Bean
12951297
AuthorizationEventPublisher authorizationEventPublisher() {
1298+
doCallRealMethod().when(this.publisher)
1299+
.publishAuthorizationEvent(any(), any(), any(AuthorizationResult.class));
12961300
return this.publisher;
12971301
}
12981302

config/src/test/java/org/springframework/security/config/annotation/web/configurers/AuthorizeHttpRequestsConfigurerTests.java

Lines changed: 5 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2002-2023 the original author or authors.
2+
* Copyright 2002-2024 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -42,6 +42,7 @@
4242
import org.springframework.security.authorization.AuthorizationEventPublisher;
4343
import org.springframework.security.authorization.AuthorizationManager;
4444
import org.springframework.security.authorization.AuthorizationObservationContext;
45+
import org.springframework.security.authorization.AuthorizationResult;
4546
import org.springframework.security.config.ObjectPostProcessor;
4647
import org.springframework.security.config.annotation.web.AbstractRequestMatcherRegistry;
4748
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
@@ -78,6 +79,7 @@
7879
import static org.assertj.core.api.Assertions.assertThatExceptionOfType;
7980
import static org.mockito.Mockito.any;
8081
import static org.mockito.Mockito.atLeastOnce;
82+
import static org.mockito.Mockito.doCallRealMethod;
8183
import static org.mockito.Mockito.mock;
8284
import static org.mockito.Mockito.spy;
8385
import static org.mockito.Mockito.verify;
@@ -1241,6 +1243,8 @@ static class AuthorizationEventPublisherConfig {
12411243

12421244
@Bean
12431245
AuthorizationEventPublisher authorizationEventPublisher() {
1246+
doCallRealMethod().when(this.publisher)
1247+
.publishAuthorizationEvent(any(), any(), any(AuthorizationResult.class));
12441248
return this.publisher;
12451249
}
12461250

core/src/main/java/org/springframework/security/authorization/AuthorizationEventPublisher.java

Lines changed: 30 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2002-2022 the original author or authors.
2+
* Copyright 2002-2024 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -42,8 +42,37 @@ public interface AuthorizationEventPublisher {
4242
* @param object the secured object
4343
* @param decision the decision about whether the user may access the secured object
4444
* @param <T> the secured object's type
45+
* @deprecated use
46+
* {@link #publishAuthorizationEvent(Supplier, Object, AuthorizationResult)} instead
4547
*/
48+
@Deprecated
4649
<T> void publishAuthorizationEvent(Supplier<Authentication> authentication, T object,
4750
AuthorizationDecision decision);
4851

52+
/**
53+
* Publish the given details in the form of an event, typically
54+
* {@link AuthorizationGrantedEvent} or {@link AuthorizationDeniedEvent}.
55+
*
56+
* Note that success events can be very noisy if enabled by default. Because of this
57+
* implementations may choose to drop success events by default.
58+
* @param authentication a {@link Supplier} for the current user
59+
* @param object the secured object
60+
* @param result {@link AuthorizationResult} the result about whether the user may
61+
* access the secured object
62+
* @param <T> the secured object's type
63+
* @since 6.4
64+
*/
65+
default <T> void publishAuthorizationEvent(Supplier<Authentication> authentication, T object,
66+
AuthorizationResult result) {
67+
if (result == null) {
68+
publishAuthorizationEvent(authentication, object, null);
69+
return;
70+
}
71+
if (result instanceof AuthorizationDecision decision) {
72+
publishAuthorizationEvent(authentication, object, decision);
73+
return;
74+
}
75+
throw new UnsupportedOperationException("result must be of type AuthorizationDecision");
76+
}
77+
4978
}

core/src/main/java/org/springframework/security/authorization/SpringAuthorizationEventPublisher.java

Lines changed: 8 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -55,10 +55,16 @@ public SpringAuthorizationEventPublisher(ApplicationEventPublisher eventPublishe
5555
@Override
5656
public <T> void publishAuthorizationEvent(Supplier<Authentication> authentication, T object,
5757
AuthorizationDecision decision) {
58-
if (decision == null || decision.isGranted()) {
58+
publishAuthorizationEvent(authentication, object, (AuthorizationResult) decision);
59+
}
60+
61+
@Override
62+
public <T> void publishAuthorizationEvent(Supplier<Authentication> authentication, T object,
63+
AuthorizationResult result) {
64+
if (result == null || result.isGranted()) {
5965
return;
6066
}
61-
AuthorizationDeniedEvent<T> failure = new AuthorizationDeniedEvent<>(authentication, object, decision);
67+
AuthorizationDeniedEvent<T> failure = new AuthorizationDeniedEvent<>(authentication, object, result);
6268
this.eventPublisher.publishEvent(failure);
6369
}
6470

core/src/main/java/org/springframework/security/authorization/event/AuthorizationDeniedEvent.java

Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020

2121
import org.springframework.context.ApplicationEvent;
2222
import org.springframework.security.authorization.AuthorizationDecision;
23+
import org.springframework.security.authorization.AuthorizationResult;
2324
import org.springframework.security.core.Authentication;
2425

2526
/**
@@ -31,10 +32,21 @@
3132
*/
3233
public class AuthorizationDeniedEvent<T> extends AuthorizationEvent {
3334

35+
/**
36+
* @deprecated Please use an {@link AuthorizationResult} constructor instead
37+
*/
38+
@Deprecated
3439
public AuthorizationDeniedEvent(Supplier<Authentication> authentication, T object, AuthorizationDecision decision) {
3540
super(authentication, object, decision);
3641
}
3742

43+
/**
44+
* @since 6.4
45+
*/
46+
public AuthorizationDeniedEvent(Supplier<Authentication> authentication, T object, AuthorizationResult result) {
47+
super(authentication, object, result);
48+
}
49+
3850
/**
3951
* Get the object to which access was requested
4052
* @return the object to which access was requested

core/src/main/java/org/springframework/security/authorization/event/AuthorizationEvent.java

Lines changed: 37 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020

2121
import org.springframework.context.ApplicationEvent;
2222
import org.springframework.security.authorization.AuthorizationDecision;
23+
import org.springframework.security.authorization.AuthorizationResult;
2324
import org.springframework.security.core.Authentication;
2425
import org.springframework.util.Assert;
2526

@@ -34,19 +35,32 @@ public class AuthorizationEvent extends ApplicationEvent {
3435

3536
private final Supplier<Authentication> authentication;
3637

37-
private final AuthorizationDecision decision;
38+
private final AuthorizationResult result;
3839

3940
/**
4041
* Construct an {@link AuthorizationEvent}
4142
* @param authentication the principal requiring access
4243
* @param object the object to which access was requested
43-
* @param decision whether authorization was granted or denied
44+
* @param result whether authorization was granted or denied
4445
*/
45-
public AuthorizationEvent(Supplier<Authentication> authentication, Object object, AuthorizationDecision decision) {
46+
public AuthorizationEvent(Supplier<Authentication> authentication, Object object, AuthorizationDecision result) {
4647
super(object);
4748
Assert.notNull(authentication, "authentication supplier cannot be null");
4849
this.authentication = authentication;
49-
this.decision = decision;
50+
this.result = result;
51+
}
52+
53+
/**
54+
* Construct an {@link AuthorizationEvent}
55+
* @param authentication the principal requiring access
56+
* @param object the object to which access was requested
57+
* @param result whether authorization was granted or denied
58+
*/
59+
public AuthorizationEvent(Supplier<Authentication> authentication, Object object, AuthorizationResult result) {
60+
super(object);
61+
Assert.notNull(authentication, "authentication supplier cannot be null");
62+
this.authentication = authentication;
63+
this.result = result;
5064
}
5165

5266
/**
@@ -68,9 +82,27 @@ public Object getObject() {
6882
/**
6983
* Get the response to the principal's request
7084
* @return the response to the principal's request
85+
* @deprecated please use {@link #getAuthorizationResult()}
7186
*/
87+
@Deprecated
7288
public AuthorizationDecision getAuthorizationDecision() {
73-
return this.decision;
89+
if (this.result == null) {
90+
return null;
91+
}
92+
if (this.result instanceof AuthorizationDecision decision) {
93+
return decision;
94+
}
95+
throw new IllegalArgumentException(
96+
"Please either call getAuthorizationResult or ensure that the result is of type AuthorizationDecision");
97+
}
98+
99+
/**
100+
* Get the response to the principal's request
101+
* @return the response to the principal's request
102+
* @since 6.4
103+
*/
104+
public AuthorizationResult getAuthorizationResult() {
105+
return this.result;
74106
}
75107

76108
}

core/src/main/java/org/springframework/security/authorization/event/AuthorizationGrantedEvent.java

Lines changed: 13 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020

2121
import org.springframework.context.ApplicationEvent;
2222
import org.springframework.security.authorization.AuthorizationDecision;
23+
import org.springframework.security.authorization.AuthorizationResult;
2324
import org.springframework.security.core.Authentication;
2425

2526
/**
@@ -31,11 +32,23 @@
3132
*/
3233
public class AuthorizationGrantedEvent<T> extends AuthorizationEvent {
3334

35+
/**
36+
* @deprecated please use a constructor that takes an
37+
* {@link org.springframework.security.authorization.AuthorizationResult}
38+
*/
39+
@Deprecated
3440
public AuthorizationGrantedEvent(Supplier<Authentication> authentication, T object,
3541
AuthorizationDecision decision) {
3642
super(authentication, object, decision);
3743
}
3844

45+
/**
46+
* @since 6.4
47+
*/
48+
public AuthorizationGrantedEvent(Supplier<Authentication> authentication, T object, AuthorizationResult result) {
49+
super(authentication, object, result);
50+
}
51+
3952
/**
4053
* Get the object to which access was requested
4154
* @return the object to which access was requested

core/src/main/java/org/springframework/security/authorization/method/AuthorizationManagerAfterMethodInterceptor.java

Lines changed: 1 addition & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -60,7 +60,7 @@ public final class AuthorizationManagerAfterMethodInterceptor implements Authori
6060

6161
private int order;
6262

63-
private AuthorizationEventPublisher eventPublisher = AuthorizationManagerAfterMethodInterceptor::noPublish;
63+
private AuthorizationEventPublisher eventPublisher = new NoOpAuthorizationEventPublisher();
6464

6565
/**
6666
* Creates an instance.
@@ -209,9 +209,4 @@ private Authentication getAuthentication() {
209209
return authentication;
210210
}
211211

212-
private static <T> void noPublish(Supplier<Authentication> authentication, T object,
213-
AuthorizationDecision decision) {
214-
215-
}
216-
217212
}

core/src/main/java/org/springframework/security/authorization/method/AuthorizationManagerBeforeMethodInterceptor.java

Lines changed: 1 addition & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,7 @@ public final class AuthorizationManagerBeforeMethodInterceptor implements Author
6565

6666
private int order = AuthorizationInterceptorsOrder.FIRST.getOrder();
6767

68-
private AuthorizationEventPublisher eventPublisher = AuthorizationManagerBeforeMethodInterceptor::noPublish;
68+
private AuthorizationEventPublisher eventPublisher = new NoOpAuthorizationEventPublisher();
6969

7070
/**
7171
* Creates an instance.
@@ -299,9 +299,4 @@ private Authentication getAuthentication() {
299299
return authentication;
300300
}
301301

302-
private static <T> void noPublish(Supplier<Authentication> authentication, T object,
303-
AuthorizationDecision decision) {
304-
305-
}
306-
307302
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
/*
2+
* Copyright 2002-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 org.springframework.security.authorization.method;
18+
19+
import java.util.function.Supplier;
20+
21+
import org.springframework.security.authorization.AuthorizationDecision;
22+
import org.springframework.security.authorization.AuthorizationEventPublisher;
23+
import org.springframework.security.authorization.AuthorizationResult;
24+
import org.springframework.security.core.Authentication;
25+
26+
/**
27+
* An {@link AuthorizationEventPublisher} implementation that does nothing.
28+
*
29+
* @author Max Batischev
30+
* @since 6.4
31+
*/
32+
final class NoOpAuthorizationEventPublisher implements AuthorizationEventPublisher {
33+
34+
@Override
35+
public <T> void publishAuthorizationEvent(Supplier<Authentication> authentication, T object,
36+
AuthorizationDecision decision) {
37+
}
38+
39+
@Override
40+
public <T> void publishAuthorizationEvent(Supplier<Authentication> authentication, T object,
41+
AuthorizationResult result) {
42+
}
43+
44+
}

core/src/test/java/org/springframework/security/authorization/method/AuthorizationManagerAfterMethodInterceptorTests.java

Lines changed: 4 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2002-2023 the original author or authors.
2+
* Copyright 2002-2024 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -42,6 +42,7 @@
4242
import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException;
4343
import static org.mockito.ArgumentMatchers.any;
4444
import static org.mockito.BDDMockito.given;
45+
import static org.mockito.Mockito.doCallRealMethod;
4546
import static org.mockito.Mockito.mock;
4647
import static org.mockito.Mockito.verify;
4748

@@ -127,6 +128,8 @@ public void invokeWhenAuthorizationEventPublisherThenUses() throws Throwable {
127128
AuthorizationManagerAfterMethodInterceptor advice = new AuthorizationManagerAfterMethodInterceptor(
128129
Pointcut.TRUE, AuthenticatedAuthorizationManager.authenticated());
129130
AuthorizationEventPublisher eventPublisher = mock(AuthorizationEventPublisher.class);
131+
doCallRealMethod().when(eventPublisher)
132+
.publishAuthorizationEvent(any(Supplier.class), any(), any(AuthorizationResult.class));
130133
advice.setAuthorizationEventPublisher(eventPublisher);
131134

132135
SecurityContext securityContext = new SecurityContextImpl();

core/src/test/java/org/springframework/security/authorization/method/AuthorizationManagerBeforeMethodInterceptorTests.java

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright 2002-2023 the original author or authors.
2+
* Copyright 2002-2024 the original author or authors.
33
*
44
* Licensed under the Apache License, Version 2.0 (the "License");
55
* you may not use this file except in compliance with the License.
@@ -41,6 +41,7 @@
4141
import static org.mockito.ArgumentMatchers.any;
4242
import static org.mockito.ArgumentMatchers.eq;
4343
import static org.mockito.BDDMockito.given;
44+
import static org.mockito.Mockito.doCallRealMethod;
4445
import static org.mockito.Mockito.mock;
4546
import static org.mockito.Mockito.verify;
4647

@@ -121,6 +122,7 @@ public void invokeWhenAuthorizationEventPublisherThenUses() throws Throwable {
121122
AuthorizationManagerBeforeMethodInterceptor advice = new AuthorizationManagerBeforeMethodInterceptor(
122123
Pointcut.TRUE, AuthenticatedAuthorizationManager.authenticated());
123124
AuthorizationEventPublisher eventPublisher = mock(AuthorizationEventPublisher.class);
125+
doCallRealMethod().when(eventPublisher).publishAuthorizationEvent(any(), any(), any(AuthorizationResult.class));
124126
advice.setAuthorizationEventPublisher(eventPublisher);
125127

126128
SecurityContext securityContext = new SecurityContextImpl();

gradle.properties

Lines changed: 0 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,6 @@
1313
# See the License for the specific language governing permissions and
1414
# limitations under the License.
1515
#
16-
1716
springBootVersion=3.3.3
1817
version=6.4.0-SNAPSHOT
1918
samplesBranch=main

0 commit comments

Comments
 (0)