spring - Spring WebFlux 中的 ReactiveSecurityContextHolder 为空

标签 spring spring-boot spring-security spring-webflux reactive

我正在尝试将 ReactiveSecurityContextHolder 与 Spring WebFlux 一起使用。不幸的是,SecurityContext 是空的:

@Configuration
public class Router {

    @Bean
    public RouterFunction<ServerResponse> routes(Handler handler) {
        return nest(
                path("/bill"),
                route(
                        GET("/").and(accept(APPLICATION_JSON)), handler::all));
    }

    @Component
    class Handler {

        Mono<ServerResponse> all(ServerRequest request) {

            ReactiveSecurityContextHolder.getContext()
                .switchIfEmpty(Mono.error(new IllegalStateException("ReactiveSecurityContext is empty")))
                .map(SecurityContext::getAuthentication)
                .map(Authentication::getName)
                .flatMap(s -> Mono.just("Hi " + s))
                .subscribe(
                        System.out::println,
                        Throwable::printStackTrace,
                        () -> System.out.println("completed without a value")
                );

            return ok().build();
        }

    }

}

此代码始终抛出 IllegalStateException。

如果我添加一个像所示的subscriberContext here :

Authentication authentication = new TestingAuthenticationToken("admin", "password", "ROLE_ADMIN");

ReactiveSecurityContextHolder.getContext()
        .switchIfEmpty(Mono.error(new IllegalStateException("ReactiveSecurityContext is empty")))
        .map(SecurityContext::getAuthentication)
        .map(Authentication::getName)
        .flatMap(s -> Mono.just("Hi " + s))
        .subscriberContext(ReactiveSecurityContextHolder.withAuthentication(authentication))
        .subscribe(
                System.out::println,
                Throwable::printStackTrace,
                () -> System.out.println("completed without a value")
        );

它工作正常并打印“Hi admin”。但这不是重点,文章说“在 WebFlux 应用程序中,subscriberContext 是使用 ReactorContextWebFilter 自动设置的”。所以我应该能够获取登录的用户。

我有这个配置:

@EnableWebFluxSecurity
@EnableReactiveMethodSecurity
public class SecurityConfig {

    @Bean
    public SecurityWebFilterChain securityWebFilterChain(ServerHttpSecurity http) {
        return http.authorizeExchange()
            .anyExchange().authenticated()
            .and().formLogin()
            .and().build();
    }

    @Bean
    public MapReactiveUserDetailsService userDetailsService() {
        UserDetails user = User.withDefaultPasswordEncoder()
            .username("user")
            .password("password")
            .roles("USER")
            .build();

        UserDetails admin = User.withDefaultPasswordEncoder()
            .username("admin")
            .password("password")
            .roles("ADMIN")
            .build();

        return new MapReactiveUserDetailsService(user, admin);
    }

}

我在这里遗漏了什么吗?如果我在 ReactorContextWebFilter 中放置断点,我可以看到它在每个请求之前被正确调用。但我的 ReactiveSecurityContextHolder 始终为空...

最佳答案

您必须返回要访问ReactiveSecurityContextHolder的流。ReactiveSecurityContextHolder。您不能在另一个流中订阅或者您必须手动进行 Reactor 上下文切换。

@Component
class Handler {
    Mono<ServerResponse> all(ServerRequest request) {
        return ReactiveSecurityContextHolder.getContext()
                .switchIfEmpty(Mono.error(new IllegalStateException("ReactiveSecurityContext is empty")))
                .map(SecurityContext::getAuthentication)
                .map(Authentication::getName)
                .flatMap(s -> Mono.just("Hi " + s))
                .doOnNext(System.out::println)
                .doOnError(Throwable::printStackTrace)
                .doOnSuccess(s -> System.out.println("completed without value: " + s))
                .flatMap(s -> ServerResponse.ok().build());
    }
}

关于spring - Spring WebFlux 中的 ReactiveSecurityContextHolder 为空,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50905505/

相关文章:

java - Spring Kafka - 代理重新连接事件

java - Spring MVC 和 "magic string"返回类型

docker - GitLab CI 因 maven-surefire-plugin 和 VM 崩溃而失败

java - 如何根据另一个主题的响应从 Kafka 主题读取消息

java - Spring Security 配置过滤除特定端点之外的任何请求?

java - Oauth2 Spring 实现

java - 请求 : Spring configured CXF ws client -- replacing hardcoded variables like 'address'

java - 如何创建通用类以将 JPARepository 用于多个实体

java - 将验证码与 Spring Security 集成

java - 我的 postman 中出现此错误(POST 和 DELETE 出现此错误),希望有人可以解决此问题