SEC-2829: SecurityContextChannelInterceptor restores original SecurityContext
This commit is contained in:
parent
9149451b9d
commit
b97a5d3b53
|
@ -36,6 +36,9 @@ import org.springframework.util.Assert;
|
||||||
* @author Rob Winch
|
* @author Rob Winch
|
||||||
*/
|
*/
|
||||||
public final class SecurityContextChannelInterceptor extends ChannelInterceptorAdapter implements ExecutorChannelInterceptor {
|
public final class SecurityContextChannelInterceptor extends ChannelInterceptorAdapter implements ExecutorChannelInterceptor {
|
||||||
|
private final SecurityContext EMPTY_CONTEXT = SecurityContextHolder.createEmptyContext();
|
||||||
|
private static final ThreadLocal<SecurityContext> ORIGINAL_CONTEXT = new ThreadLocal<SecurityContext>();
|
||||||
|
|
||||||
private final String authenticationHeaderName;
|
private final String authenticationHeaderName;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -75,6 +78,9 @@ public final class SecurityContextChannelInterceptor extends ChannelInterceptorA
|
||||||
}
|
}
|
||||||
|
|
||||||
private void setup(Message<?> message) {
|
private void setup(Message<?> message) {
|
||||||
|
SecurityContext currentContext = SecurityContextHolder.getContext();
|
||||||
|
ORIGINAL_CONTEXT.set(currentContext);
|
||||||
|
|
||||||
Object user = message.getHeaders().get(authenticationHeaderName);
|
Object user = message.getHeaders().get(authenticationHeaderName);
|
||||||
if(!(user instanceof Authentication)) {
|
if(!(user instanceof Authentication)) {
|
||||||
return;
|
return;
|
||||||
|
@ -86,6 +92,17 @@ public final class SecurityContextChannelInterceptor extends ChannelInterceptorA
|
||||||
}
|
}
|
||||||
|
|
||||||
private void cleanup() {
|
private void cleanup() {
|
||||||
SecurityContextHolder.clearContext();
|
SecurityContext originalContext = ORIGINAL_CONTEXT.get();
|
||||||
|
ORIGINAL_CONTEXT.remove();
|
||||||
|
|
||||||
|
try {
|
||||||
|
if(EMPTY_CONTEXT.equals(originalContext)) {
|
||||||
|
SecurityContextHolder.clearContext();
|
||||||
|
} else {
|
||||||
|
SecurityContextHolder.setContext(originalContext);
|
||||||
|
}
|
||||||
|
} catch(Throwable t) {
|
||||||
|
SecurityContextHolder.clearContext();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -146,4 +146,19 @@ public class SecurityContextChannelInterceptorTests {
|
||||||
|
|
||||||
assertThat(SecurityContextHolder.getContext().getAuthentication()).isNull();
|
assertThat(SecurityContextHolder.getContext().getAuthentication()).isNull();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void restoresOriginalContext() throws Exception {
|
||||||
|
TestingAuthenticationToken original = new TestingAuthenticationToken("original", "original", "ROLE_USER");
|
||||||
|
SecurityContextHolder.getContext().setAuthentication(original);
|
||||||
|
|
||||||
|
messageBuilder.setHeader(SimpMessageHeaderAccessor.USER_HEADER, authentication);
|
||||||
|
interceptor.beforeHandle(messageBuilder.build(), channel, handler);
|
||||||
|
|
||||||
|
assertThat(SecurityContextHolder.getContext().getAuthentication()).isSameAs(authentication);
|
||||||
|
|
||||||
|
interceptor.afterMessageHandled(messageBuilder.build(), channel, handler, null);
|
||||||
|
|
||||||
|
assertThat(SecurityContextHolder.getContext().getAuthentication()).isSameAs(original);
|
||||||
|
}
|
||||||
}
|
}
|
Loading…
Reference in New Issue