Merge pull request #46975 from dungdm93
Build and Deploy Snapshot / Build and Deploy Snapshot (push) Waiting to run
Details
Build and Deploy Snapshot / Trigger Docs Build (push) Blocked by required conditions
Details
Build and Deploy Snapshot / Verify (push) Blocked by required conditions
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:false version:24], map[id:${{ vars.UBUNTU_MEDIUM || 'ubuntu-latest' }} name:Linux]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:false version:24], map[id:windows-latest name:Windows]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:true version:17], map[id:${{ vars.UBUNTU_MEDIUM || 'ubuntu-latest' }} name:Linux]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:true version:17], map[id:windows-latest name:Windows]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:true version:21], map[id:${{ vars.UBUNTU_MEDIUM || 'ubuntu-latest' }} name:Linux]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:true version:21], map[id:windows-latest name:Windows]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:true version:25], map[id:${{ vars.UBUNTU_MEDIUM || 'ubuntu-latest' }} name:Linux]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:true version:25], map[id:windows-latest name:Windows]) (push) Waiting to run
Details
Run CodeQL Analysis / run-analysis (push) Waiting to run
Details
Run System Tests / Java ${{ matrix.java.version}} (map[toolchain:true version:17]) (push) Waiting to run
Details
Run System Tests / Java ${{ matrix.java.version}} (map[toolchain:true version:21]) (push) Waiting to run
Details
Build and Deploy Snapshot / Build and Deploy Snapshot (push) Waiting to run
Details
Build and Deploy Snapshot / Trigger Docs Build (push) Blocked by required conditions
Details
Build and Deploy Snapshot / Verify (push) Blocked by required conditions
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:false version:24], map[id:${{ vars.UBUNTU_MEDIUM || 'ubuntu-latest' }} name:Linux]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:false version:24], map[id:windows-latest name:Windows]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:true version:17], map[id:${{ vars.UBUNTU_MEDIUM || 'ubuntu-latest' }} name:Linux]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:true version:17], map[id:windows-latest name:Windows]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:true version:21], map[id:${{ vars.UBUNTU_MEDIUM || 'ubuntu-latest' }} name:Linux]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:true version:21], map[id:windows-latest name:Windows]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:true version:25], map[id:${{ vars.UBUNTU_MEDIUM || 'ubuntu-latest' }} name:Linux]) (push) Waiting to run
Details
CI / ${{ matrix.os.name}} | Java ${{ matrix.java.version}} (map[toolchain:true version:25], map[id:windows-latest name:Windows]) (push) Waiting to run
Details
Run CodeQL Analysis / run-analysis (push) Waiting to run
Details
Run System Tests / Java ${{ matrix.java.version}} (map[toolchain:true version:17]) (push) Waiting to run
Details
Run System Tests / Java ${{ matrix.java.version}} (map[toolchain:true version:21]) (push) Waiting to run
Details
* pr/46975: Polish "Auto-configure observation of Redis with Lettuce" Auto-configure observation of Redis with Lettuce Closes gh-46975
This commit is contained in:
commit
79ad5e6999
|
@ -1053,7 +1053,7 @@ Metrics for Jetty's javadoc:org.eclipse.jetty.server.Connector[] instances are b
|
||||||
[[actuator.metrics.supported.redis]]
|
[[actuator.metrics.supported.redis]]
|
||||||
=== Redis Metrics
|
=== Redis Metrics
|
||||||
|
|
||||||
Auto-configuration registers a javadoc:io.lettuce.core.metrics.MicrometerCommandLatencyRecorder[] for the auto-configured javadoc:org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory[].
|
Auto-configuration registers a javadoc:io.lettuce.core.tracing.MicrometerTracing[] for the auto-configured javadoc:org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory[].
|
||||||
For more detail, see the {url-lettuce-docs}/advanced-usage/#observability[Observability section] of the Lettuce documentation.
|
For more detail, see the {url-lettuce-docs}/advanced-usage/#observability[Observability section] of the Lettuce documentation.
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -56,6 +56,7 @@ dependencies {
|
||||||
|
|
||||||
testImplementation(project(":core:spring-boot-test"))
|
testImplementation(project(":core:spring-boot-test"))
|
||||||
testImplementation(project(":test-support:spring-boot-test-support"))
|
testImplementation(project(":test-support:spring-boot-test-support"))
|
||||||
|
testImplementation("io.micrometer:micrometer-observation-test")
|
||||||
testImplementation("io.projectreactor:reactor-test")
|
testImplementation("io.projectreactor:reactor-test")
|
||||||
|
|
||||||
testRuntimeOnly("ch.qos.logback:logback-classic")
|
testRuntimeOnly("ch.qos.logback:logback-classic")
|
||||||
|
|
|
@ -14,43 +14,36 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package org.springframework.boot.data.redis.autoconfigure.metrics;
|
package org.springframework.boot.data.redis.autoconfigure.observation;
|
||||||
|
|
||||||
import io.lettuce.core.RedisClient;
|
import io.lettuce.core.RedisClient;
|
||||||
import io.lettuce.core.metrics.MicrometerCommandLatencyRecorder;
|
import io.lettuce.core.tracing.MicrometerTracing;
|
||||||
import io.lettuce.core.metrics.MicrometerOptions;
|
import io.micrometer.observation.ObservationRegistry;
|
||||||
import io.micrometer.core.instrument.MeterRegistry;
|
|
||||||
|
|
||||||
import org.springframework.boot.autoconfigure.AutoConfiguration;
|
import org.springframework.boot.autoconfigure.AutoConfiguration;
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
|
|
||||||
import org.springframework.boot.data.redis.autoconfigure.ClientResourcesBuilderCustomizer;
|
import org.springframework.boot.data.redis.autoconfigure.ClientResourcesBuilderCustomizer;
|
||||||
import org.springframework.boot.data.redis.autoconfigure.DataRedisAutoConfiguration;
|
import org.springframework.boot.data.redis.autoconfigure.DataRedisAutoConfiguration;
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Auto-configuration for Lettuce metrics.
|
* Auto-configuration for Lettuce observability.
|
||||||
*
|
*
|
||||||
* @author Antonin Arquey
|
* @author Antonin Arquey
|
||||||
* @author Yanming Zhou
|
* @author Yanming Zhou
|
||||||
|
* @author Dũng Đăng Minh
|
||||||
* @since 4.0.0
|
* @since 4.0.0
|
||||||
*/
|
*/
|
||||||
@AutoConfiguration(before = DataRedisAutoConfiguration.class,
|
@AutoConfiguration(before = DataRedisAutoConfiguration.class,
|
||||||
afterName = "org.springframework.boot.micrometer.metrics.autoconfigure.CompositeMeterRegistryAutoConfiguration")
|
afterName = "org.springframework.boot.micrometer.observation.autoconfigure.ObservationAutoConfiguration")
|
||||||
@ConditionalOnClass({ RedisClient.class, MicrometerCommandLatencyRecorder.class, MeterRegistry.class })
|
@ConditionalOnClass({ RedisClient.class, MicrometerTracing.class, ObservationRegistry.class })
|
||||||
@ConditionalOnBean(MeterRegistry.class)
|
@ConditionalOnBean(ObservationRegistry.class)
|
||||||
public final class LettuceMetricsAutoConfiguration {
|
public final class LettuceObservationAutoConfiguration {
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
@ConditionalOnMissingBean
|
ClientResourcesBuilderCustomizer lettuceObservation(ObservationRegistry observationRegistry) {
|
||||||
MicrometerOptions micrometerOptions() {
|
return (client) -> client.tracing(new MicrometerTracing(observationRegistry, "Redis"));
|
||||||
return MicrometerOptions.create();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Bean
|
|
||||||
ClientResourcesBuilderCustomizer lettuceMetrics(MeterRegistry meterRegistry, MicrometerOptions options) {
|
|
||||||
return (client) -> client.commandLatencyRecorder(new MicrometerCommandLatencyRecorder(meterRegistry, options));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -15,9 +15,9 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Auto-configuration for Spring Data Redis metrics.
|
* Auto-configuration for Spring Data Redis observation.
|
||||||
*/
|
*/
|
||||||
@NullMarked
|
@NullMarked
|
||||||
package org.springframework.boot.data.redis.autoconfigure.metrics;
|
package org.springframework.boot.data.redis.autoconfigure.observation;
|
||||||
|
|
||||||
import org.jspecify.annotations.NullMarked;
|
import org.jspecify.annotations.NullMarked;
|
|
@ -3,4 +3,4 @@ org.springframework.boot.data.redis.autoconfigure.DataRedisReactiveAutoConfigura
|
||||||
org.springframework.boot.data.redis.autoconfigure.DataRedisRepositoriesAutoConfiguration
|
org.springframework.boot.data.redis.autoconfigure.DataRedisRepositoriesAutoConfiguration
|
||||||
org.springframework.boot.data.redis.autoconfigure.health.DataRedisHealthContributorAutoConfiguration
|
org.springframework.boot.data.redis.autoconfigure.health.DataRedisHealthContributorAutoConfiguration
|
||||||
org.springframework.boot.data.redis.autoconfigure.health.DataRedisReactiveHealthContributorAutoConfiguration
|
org.springframework.boot.data.redis.autoconfigure.health.DataRedisReactiveHealthContributorAutoConfiguration
|
||||||
org.springframework.boot.data.redis.autoconfigure.metrics.LettuceMetricsAutoConfiguration
|
org.springframework.boot.data.redis.autoconfigure.observation.LettuceObservationAutoConfiguration
|
||||||
|
|
|
@ -1,102 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright 2012-present the original author or authors.
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* https://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package org.springframework.boot.data.redis.autoconfigure.metrics;
|
|
||||||
|
|
||||||
import io.lettuce.core.metrics.MicrometerCommandLatencyRecorder;
|
|
||||||
import io.lettuce.core.metrics.MicrometerOptions;
|
|
||||||
import io.lettuce.core.resource.ClientResources;
|
|
||||||
import io.micrometer.core.instrument.simple.SimpleMeterRegistry;
|
|
||||||
import org.junit.jupiter.api.Test;
|
|
||||||
|
|
||||||
import org.springframework.boot.autoconfigure.AutoConfigurations;
|
|
||||||
import org.springframework.boot.data.redis.autoconfigure.DataRedisAutoConfiguration;
|
|
||||||
import org.springframework.boot.test.context.runner.ApplicationContextRunner;
|
|
||||||
import org.springframework.context.annotation.Bean;
|
|
||||||
import org.springframework.context.annotation.Configuration;
|
|
||||||
import org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory;
|
|
||||||
|
|
||||||
import static org.assertj.core.api.Assertions.assertThat;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Tests for {@link LettuceMetricsAutoConfiguration}.
|
|
||||||
*
|
|
||||||
* @author Antonin Arquey
|
|
||||||
*/
|
|
||||||
class LettuceMetricsAutoConfigurationTests {
|
|
||||||
|
|
||||||
private final ApplicationContextRunner contextRunner = new ApplicationContextRunner()
|
|
||||||
.withConfiguration(AutoConfigurations.of(LettuceMetricsAutoConfiguration.class));
|
|
||||||
|
|
||||||
@Test
|
|
||||||
void whenThereIsAMeterRegistryThenCommandLatencyRecorderIsAdded() {
|
|
||||||
this.contextRunner.withBean(SimpleMeterRegistry.class)
|
|
||||||
.withConfiguration(AutoConfigurations.of(DataRedisAutoConfiguration.class))
|
|
||||||
.run((context) -> {
|
|
||||||
ClientResources clientResources = context.getBean(LettuceConnectionFactory.class).getClientResources();
|
|
||||||
assertThat(clientResources.commandLatencyRecorder())
|
|
||||||
.isInstanceOf(MicrometerCommandLatencyRecorder.class);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
void autoConfiguredMicrometerOptionsUsesLettucesDefaults() {
|
|
||||||
this.contextRunner.withBean(SimpleMeterRegistry.class)
|
|
||||||
.withConfiguration(AutoConfigurations.of(DataRedisAutoConfiguration.class))
|
|
||||||
.run((context) -> {
|
|
||||||
MicrometerOptions micrometerOptions = context.getBean(MicrometerOptions.class);
|
|
||||||
assertThat(micrometerOptions.isEnabled()).isTrue();
|
|
||||||
assertThat(micrometerOptions.isHistogram()).isFalse();
|
|
||||||
assertThat(micrometerOptions.localDistinction()).isFalse();
|
|
||||||
assertThat(micrometerOptions.maxLatency()).isEqualTo(MicrometerOptions.DEFAULT_MAX_LATENCY);
|
|
||||||
assertThat(micrometerOptions.minLatency()).isEqualTo(MicrometerOptions.DEFAULT_MIN_LATENCY);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
void whenUserDefinesAMicrometerOptionsBeanThenCommandLatencyRecorderUsesIt() {
|
|
||||||
this.contextRunner.withBean(SimpleMeterRegistry.class)
|
|
||||||
.withConfiguration(AutoConfigurations.of(DataRedisAutoConfiguration.class))
|
|
||||||
.withUserConfiguration(CustomMicrometerOptionsConfiguration.class)
|
|
||||||
.run((context) -> {
|
|
||||||
ClientResources clientResources = context.getBean(LettuceConnectionFactory.class).getClientResources();
|
|
||||||
assertThat(clientResources.commandLatencyRecorder())
|
|
||||||
.isInstanceOf(MicrometerCommandLatencyRecorder.class);
|
|
||||||
assertThat(clientResources.commandLatencyRecorder()).hasFieldOrPropertyWithValue("options",
|
|
||||||
context.getBean("customMicrometerOptions"));
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
void whenThereIsNoMeterRegistryThenClientResourcesCustomizationBacksOff() {
|
|
||||||
this.contextRunner.withConfiguration(AutoConfigurations.of(DataRedisAutoConfiguration.class)).run((context) -> {
|
|
||||||
ClientResources clientResources = context.getBean(LettuceConnectionFactory.class).getClientResources();
|
|
||||||
assertThat(clientResources.commandLatencyRecorder())
|
|
||||||
.isNotInstanceOf(MicrometerCommandLatencyRecorder.class);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
@Configuration(proxyBeanMethods = false)
|
|
||||||
static class CustomMicrometerOptionsConfiguration {
|
|
||||||
|
|
||||||
@Bean
|
|
||||||
MicrometerOptions customMicrometerOptions() {
|
|
||||||
return MicrometerOptions.create();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -0,0 +1,61 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-present the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.springframework.boot.data.redis.autoconfigure.observation;
|
||||||
|
|
||||||
|
import io.lettuce.core.resource.ClientResources;
|
||||||
|
import io.lettuce.core.tracing.MicrometerTracing;
|
||||||
|
import io.micrometer.observation.tck.TestObservationRegistry;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import org.springframework.boot.autoconfigure.AutoConfigurations;
|
||||||
|
import org.springframework.boot.data.redis.autoconfigure.DataRedisAutoConfiguration;
|
||||||
|
import org.springframework.boot.test.context.runner.ApplicationContextRunner;
|
||||||
|
import org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link LettuceObservationAutoConfiguration}.
|
||||||
|
*
|
||||||
|
* @author Antonin Arquey
|
||||||
|
* @author Stephane Nicoll
|
||||||
|
*/
|
||||||
|
class LettuceObservationAutoConfigurationTests {
|
||||||
|
|
||||||
|
private final ApplicationContextRunner contextRunner = new ApplicationContextRunner()
|
||||||
|
.withConfiguration(AutoConfigurations.of(LettuceObservationAutoConfiguration.class));
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void whenThereIsAnObservationRegistryThenMicrometerTracingIsAdded() {
|
||||||
|
this.contextRunner.withBean(TestObservationRegistry.class, TestObservationRegistry::create)
|
||||||
|
.withConfiguration(AutoConfigurations.of(DataRedisAutoConfiguration.class))
|
||||||
|
.run((context) -> {
|
||||||
|
ClientResources clientResources = context.getBean(LettuceConnectionFactory.class).getClientResources();
|
||||||
|
assertThat(clientResources.tracing()).isInstanceOf(MicrometerTracing.class);
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void whenThereIsNoObservationRegistryThenClientResourcesCustomizationBacksOff() {
|
||||||
|
this.contextRunner.withConfiguration(AutoConfigurations.of(DataRedisAutoConfiguration.class)).run((context) -> {
|
||||||
|
ClientResources clientResources = context.getBean(LettuceConnectionFactory.class).getClientResources();
|
||||||
|
assertThat(clientResources.tracing()).isNotInstanceOf(MicrometerTracing.class);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue