-
Notifications
You must be signed in to change notification settings - Fork 6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Sorting in AuthorizationAdvisorProxyFactory should be thread-safe #16819
Comments
Thanks for the detailed report, @wbxz987! We'll target fixing this in the next maintenance release. In the meantime, you can workaround this by providing your own proxy factory instance like so: @Primary
@Bean
@Role(BeanDefinition.ROLE_INFRASTRUCTURE)
static AuthorizationProxyFactory proxyFactory() {
AuthorizationAdvisorProxyFactory proxyFactory = new AuthorizationAdvisorProxyFactory(new ArrayList<>());
List<AuthorizationAdvisor> advisors = new ArrayList<>();
advisors.add(AuthorizationManagerBeforeMethodInterceptor.preAuthorize());
advisors.add(AuthorizationManagerAfterMethodInterceptor.postAuthorize());
advisors.add(new PreFilterAuthorizationMethodInterceptor());
advisors.add(new PostFilterAuthorizationMethodInterceptor());
advisors.add(new AuthorizeReturnObjectMethodInterceptor(proxyFactory));
AnnotationAwareOrderComparator.sort(advisors);
for (AuthorizationAdvisor advisor : advisors) {
proxyFactory.addAdvisor(advisor);
}
return proxyFactory;
} Will you please let me know if this addresses the issue for the time being? |
In order to make so that authorization advisors are sorted only one time and also as part of the configuration lifecycle, AuthorizationAdvisorProxyFactory now implements SmartInitializingBean. Closes spring-projectsgh-16819 Signed-off-by: Tran Ngoc Nhan <ngocnhan.tran1996@gmail.com>
This commit ensures that the default advisors and added advisors are sorted in the event that this component is not being published as a Spring bean. Issue spring-projectsgh-16819
This commit ensures that the default advisors and added advisors are sorted in the event that this component is not being published as a Spring bean. Issue spring-projectsgh-16819
This commit ensures that the default advisors and added advisors are sorted in the event that this component is not being published as a Spring bean. Issue spring-projectsgh-16819
This commit ensures that the default advisors and added advisors are sorted in the event that this component is not being published as a Spring bean. Issue gh-16819
Thanks for the quick fix @jzheaux and @ngocnhan-tran1996, looking forward for the release. I updated the sample repository with the proposed workaround: https://github.com/wbxz987/ConcurrentModificationException/blob/main/src/main/java/org/example/concurrentmodificationexception/config/SecurityConfig.java#L38
Am i missing something? |
I see, my mistake, @wbxz987. I incorrectly anticipated that What you can do instead is publish a thread safe version of the proxy like so: @Component
@Primary
public class ThreadsafeAuthorizationProxyFactory implements AuthorizationProxyFactory {
private AuthorizationAdvisorProxyFactory delegate;
private final ReentrantLock lock = new ReentrantLock();
@Override
public Object proxy(Object object) {
try {
this.lock.lock();
return this.delegate.proxy(object);
} finally {
this.lock.unlock();
}
}
public void setDelegate(AuthorizationAdvisorProxyFactory delegate) {
this.delegate = delegate;
}
} And then provide that proxy to an @Bean
@Role(BeanDefinition.ROLE_INFRASTRUCTURE)
static Advisor authorizeReturnObjectAdvisor(ObjectProvider<AuthorizationAdvisor> provider,
ThreadsafeAuthorizationProxyFactory proxyFactory) {
AuthorizationAdvisorProxyFactory delegate = new AuthorizationAdvisorProxyFactory(new ArrayList<>());
provider.forEach(delegate::addAdvisor);
AuthorizeReturnObjectMethodInterceptor interceptor = new AuthorizeReturnObjectMethodInterceptor(proxyFactory);
delegate.addAdvisor(interceptor);
proxyFactory.setDelegate(delegate);
return interceptor;
} I updated the provided sample with this change: wbxz987/ConcurrentModificationException#1 Does this solution work better? |
Yes, the workaround works perfectly. Thank you @jzheaux! |
Using Spring Security 6.4.4 (via Spring Boot 3.4.4)
Describe the bug
When a endpoint annotated with
@AuthorizeReturnObject
returns an object that has an object authorized with e.g.@PreAuthorize
,ConcurrentModificationException
is thrown when multiple requests come in parallel.To Reproduce
See the sample repository below for a reproducible test case.
Expected behavior
Parallel requests should be handled without errors.
Sample
https://github.com/wbxz987/ConcurrentModificationException
The sample repository contains a test, that simulates multiple requests coming in parallel. The test fails because a
ConcurrentModificationException
is thrown.The test works when downgrading Spring Security to version 6.3.2, and breaks after this commit 0cab7c8
The text was updated successfully, but these errors were encountered: