web-dev-qa-db-ja.com

Spring Boot 2のRedisヘルスチェックがハングする

Spring Boot 2.1.5アプリケーションにいくつかのredis機能を実装しました。正常に動作します。また、redisのヘルスチェックも必要です。 Redisサーバーのスイッチを切ると、ヘルスチェック(アクチュエータ/ヘルス)が永久にハングします。 適切なタイムアウトを構成するにはどうすればよいですか?

私はこの問題の小さなデモをここに作成しました: https://github.com/markuskruse/demo-redis-health-bug

複製、実行、redisの停止、ヘルスのチェック(永久に待機)、redisの開始(ヘルスの復帰)。

これは私のredisのグラドルです:

implementation 'org.springframework.boot:spring-boot-starter-data-redis'

これは私のapplication.yamlです。

spring:
   redis:
      timeout: 5000
      Host: localhost

これは私のRedisConfig.Javaです

@Configuration
@EnableConfigurationProperties(RedisProperties.class)
public class RedisConfig {

  @Bean
  public LettuceConnectionFactory redisConnectionFactory(
      @Value("${spring.redis.Host:localhost}") String redisHost) {
    RedisStandaloneConfiguration redisStandaloneConfiguration =
        new RedisStandaloneConfiguration(redisHost);
    return new LettuceConnectionFactory(redisStandaloneConfiguration);
  }

  @Bean
  public StringRedisTemplate redisTemplate(RedisConnectionFactory jedisConnectionFactory) {
    final StringRedisTemplate template = new StringRedisTemplate();
    template.setConnectionFactory(jedisConnectionFactory);
    template.afterPropertiesSet();
    return template;
  }
}

Githubのこの問題によると、これは単なる設定の問題です: https://github.com/spring-projects/spring-boot/issues/15542

このjiraチケットによると、それは春のブーツ2.1.4で修正されるはずです(私は2.1.5にいます)。 https://jira.spring.io/browse/DATAREDIS-918

彼らは私が試した回避策について言及しています:

  @Bean
  public ClientOptions clientOptions() {
    return ClientOptions.builder()
        .timeoutOptions(TimeoutOptions.enabled())
        .build();
  }

それ自体は効果がありませんでした。どこかに注入する必要があります。グーグルはこれを与えました:

  @Bean
  LettucePoolingClientConfiguration lettucePoolConfig(ClientOptions options, ClientResources dcr){
    return LettucePoolingClientConfiguration.builder()
        .clientOptions(options)
        .clientResources(dcr)
        .build();
  }

それから私はこれを手に入れます:

Caused by: org.springframework.beans.BeanInstantiationException: Failed to instantiate [org.springframework.data.redis.connection.lettuce.LettucePoolingClientConfiguration]: Factory method 'lettucePoolConfig' threw exception; nested exception is Java.lang.NoClassDefFoundError: org/Apache/commons/pool2/impl/GenericObjectPoolConfig
    at org.springframework.beans.factory.support.SimpleInstantiationStrategy.instantiate(SimpleInstantiationStrategy.Java:185)
    at org.springframework.beans.factory.support.ConstructorResolver.instantiate(ConstructorResolver.Java:622)
    ... 50 more
Caused by: Java.lang.NoClassDefFoundError: org/Apache/commons/pool2/impl/GenericObjectPoolConfig
    at org.springframework.data.redis.connection.lettuce.LettucePoolingClientConfiguration$LettucePoolingClientConfigurationBuilder.<init>(LettucePoolingClientConfiguration.Java:91)
    at org.springframework.data.redis.connection.lettuce.LettucePoolingClientConfiguration.builder(LettucePoolingClientConfiguration.Java:50)
    at com.ikea.cps.mhs.config.RedisConfig.lettucePoolConfig(RedisConfig.Java:50)
    at com.ikea.cps.mhs.config.RedisConfig$$EnhancerBySpringCGLIB$$3804d114.CGLIB$lettucePoolConfig$3(<generated>)
    at com.ikea.cps.mhs.config.RedisConfig$$EnhancerBySpringCGLIB$$3804d114$$FastClassBySpringCGLIB$$ccabed80.invoke(<generated>)
    at org.springframework.cglib.proxy.MethodProxy.invokeSuper(MethodProxy.Java:244)
    at org.springframework.context.annotation.ConfigurationClassEnhancer$BeanMethodInterceptor.intercept(ConfigurationClassEnhancer.Java:363)
    at com.ikea.cps.mhs.config.RedisConfig$$EnhancerBySpringCGLIB$$3804d114.lettucePoolConfig(<generated>)
    at Java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at Java.base/jdk.internal.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.Java:62)
    at Java.base/jdk.internal.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.Java:43)
    at Java.base/Java.lang.reflect.Method.invoke(Method.Java:566)
    at org.springframework.beans.factory.support.SimpleInstantiationStrategy.instantiate(SimpleInstantiationStrategy.Java:154)
    ... 51 more
Caused by: Java.lang.ClassNotFoundException: org.Apache.commons.pool2.impl.GenericObjectPoolConfig
    at Java.base/jdk.internal.loader.BuiltinClassLoader.loadClass(BuiltinClassLoader.Java:583)
    at Java.base/jdk.internal.loader.ClassLoaders$AppClassLoader.loadClass(ClassLoaders.Java:178)
    at Java.base/Java.lang.ClassLoader.loadClass(ClassLoader.Java:521)
    ... 64 more

これを回避できるかもしれません。しかし、私は何か(基本的に)間違っていると思っています。すでに修正されているはずです。

編集:私はコモンズプールを追加し、エラーはなくなりましたが、ヘルスチェックはまだ永久にハングします。

以下も試してみましたが、効果はありませんでした。

@Component
public class RedisConfigurer implements LettuceClientConfigurationBuilderCustomizer {

  @Override
  public void customize(LettuceClientConfigurationBuilder builder) {
    builder.clientOptions(ClientOptions.builder()
        .timeoutOptions(TimeoutOptions.enabled(Duration.of(5, SECONDS))).build());
  }

}
3
Gunslinger

問題は手動の接続ファクトリ構成にあるようです。

その部分を削除すると、すべてが期待どおりにうまくいくはずです。

それ以外の場合は、LettuceClientConfigurationコンストラクタの2番目の引数にLettuceConnectionFactoryを指定する必要があり、ClientOptionsを有効にしてTimeoutOptionsを構成できます

1
madz