提问者:小点点

将Spring Cloud负载平衡与KeyCloakResTemboard集成


我有一个配置了Spring Cloud discovery的微服务环境,因此我能够只使用其他服务实例的id来访问它们:

public class MyClass {

    @Autowired
    @LoadBalanced
    private RestTemplate restTemplate;

    public String doOtherStuff() {
        String results = restTemplate.getForObject("http://stores/stores", String.class);
        return results;
    }
}

现在我想访问一个需要OAuth2授权的服务。我使用了一个Keycloak服务器来提供它,Keycloak已经提供了一个带有特定KeycloakRestTemplate的适配器。反正怎么用负载均衡增强呢?


共2个答案

匿名用户

我们需要创建一个特定的KeycloakRestTemplate,它将使用LoadBalancerInterceptor:

@Configuration
@EnableWebSecurity
@ComponentScan(basePackageClasses = KeycloakSecurityComponents.class)
public class SecurityConfig extends KeycloakWebSecurityConfigurerAdapter {

    @Bean
    @Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
    @LoadBalanced
    public KeycloakRestTemplate keycloakRestTemplate(
            KeycloakClientRequestFactory keycloakClientRequestFactory,
            LoadBalancerInterceptor interceptor) {
        KeycloakRestTemplate result = new KeycloakRestTemplate(
            keycloakClientRequestFactory);
        // Add the interceptor for load balancing
        result.getInterceptors().add(interceptor);
        return result;
    }

    //More configurations for keycloak

}

因此,有机会获得授权/负载平衡模板:

@Autowired
@LoadBalanced
protected KeycloakRestTemplate restTemplate;

另请参见:

    < li >向普通RestTemplate添加功能区负载平衡

匿名用户

你的解决方案不是很好,因为

@Bean
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
@LoadBalanced
public KeycloakRestTemplate keycloakRestTemplate(
        KeycloakClientRequestFactory keycloakClientRequestFactory,
        LoadBalancerInterceptor interceptor) {
    KeycloakRestTemplate result = new KeycloakRestTemplate(
        keycloakClientRequestFactory);
    // Add the interceptor for load balancing
    result.getInterceptors().add(interceptor);
    return result;
}

不起作用,因为你有例外

The dependencies of some of the beans in the application context form a cycle:
...
┌─────┐
|  keycloakRestTemplate defined in class path resource [...]
↑     ↓
|  ribbonInterceptor defined in class path resource [org/springframework/cloud/client/loadbalancer/LoadBalancerAutoConfiguration$LoadBalancerInterceptorConfig.class]
↑     ↓
|  org.springframework.cloud.client.loadbalancer.LoadBalancerAutoConfiguration     (field private java.util.List org.springframework.cloud.client.loadbalancer.LoadBalancerAutoConfiguration.restTemplates)
└─────┘

你要做什么?

@Bean
@LoadBalanced
public KeycloakRestTemplate keycloakRestTemplate(
        KeycloakClientRequestFactory keycloakClientRequestFactory) {
    return new KeycloakRestTemplate(keycloakClientRequestFactory);
}

没有@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE),那么你就会得到单例,你可以像这样使用

@Autowired
protected KeycloakRestTemplate restTemplate;

或者您可以像原型一样定义rest Temboard

@Bean
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
@LoadBalanced
public KeycloakRestTemplate keycloakRestTemplate(
        KeycloakClientRequestFactory keycloakClientRequestFactory) {
    return new KeycloakRestTemplate(keycloakClientRequestFactory);
}

然后像这样使用

@Autowired
@LoadBalanced
protected KeycloakRestTemplate restTemplate;

编辑:由于周期问题,Xtreme Biker的解决方案不适用于SpringBoot 2和Keycloak 6,我的第一个主张不是线程/sesions保存,第二个不起作用,因为将在运行之前创建@LoadBalanced和restTemplate wchich是基于原型创建的,没有设置拦截器:|