Rework @Timer annotation support
Rework existing `@Timer` annotation support to remove duplicate code and offer general purpose utilities that can be used in future metrics support. See gh-23112 See gh-22217
This commit is contained in:
parent
3d7e5e3abd
commit
9f16491535
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2012-2020 the original author or authors.
|
* Copyright 2012-2021 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
|
@ -16,12 +16,16 @@
|
||||||
|
|
||||||
package org.springframework.boot.actuate.metrics;
|
package org.springframework.boot.actuate.metrics;
|
||||||
|
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.function.Consumer;
|
||||||
import java.util.function.Supplier;
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
import io.micrometer.core.annotation.Timed;
|
import io.micrometer.core.annotation.Timed;
|
||||||
import io.micrometer.core.instrument.Timer;
|
import io.micrometer.core.instrument.Timer;
|
||||||
import io.micrometer.core.instrument.Timer.Builder;
|
import io.micrometer.core.instrument.Timer.Builder;
|
||||||
|
|
||||||
|
import org.springframework.util.CollectionUtils;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Strategy that can be used to apply {@link Timer Timers} automatically instead of using
|
* Strategy that can be used to apply {@link Timer Timers} automatically instead of using
|
||||||
* {@link Timed @Timed}.
|
* {@link Timed @Timed}.
|
||||||
|
|
@ -94,4 +98,17 @@ public interface AutoTimer {
|
||||||
*/
|
*/
|
||||||
void apply(Timer.Builder builder);
|
void apply(Timer.Builder builder);
|
||||||
|
|
||||||
|
static void apply(AutoTimer autoTimer, String metricName, Set<Timed> annotations, Consumer<Timer.Builder> action) {
|
||||||
|
if (!CollectionUtils.isEmpty(annotations)) {
|
||||||
|
for (Timed annotation : annotations) {
|
||||||
|
action.accept(Timer.builder(annotation, metricName));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if (autoTimer != null && autoTimer.isEnabled()) {
|
||||||
|
action.accept(autoTimer.builder(metricName));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -14,9 +14,10 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package org.springframework.boot.actuate.metrics.web.method;
|
package org.springframework.boot.actuate.metrics.annotation;
|
||||||
|
|
||||||
import java.lang.reflect.AnnotatedElement;
|
import java.lang.reflect.AnnotatedElement;
|
||||||
|
import java.lang.reflect.Method;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
@ -26,30 +27,39 @@ import io.micrometer.core.annotation.Timed;
|
||||||
import org.springframework.core.annotation.MergedAnnotationCollectors;
|
import org.springframework.core.annotation.MergedAnnotationCollectors;
|
||||||
import org.springframework.core.annotation.MergedAnnotations;
|
import org.springframework.core.annotation.MergedAnnotations;
|
||||||
import org.springframework.util.ConcurrentReferenceHashMap;
|
import org.springframework.util.ConcurrentReferenceHashMap;
|
||||||
import org.springframework.web.method.HandlerMethod;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility used to obtain {@link Timed @Timed} annotations from a {@link HandlerMethod}.
|
* Utility used to obtain {@link Timed @Timed} annotations from bean methods.
|
||||||
*
|
*
|
||||||
* @author Phillip Webb
|
* @author Phillip Webb
|
||||||
* @since 2.5.0
|
* @since 2.5.0
|
||||||
*/
|
*/
|
||||||
public final class HandlerMethodTimedAnnotations {
|
public final class TimedAnnotations {
|
||||||
|
|
||||||
private static Map<AnnotatedElement, Set<Timed>> cache = new ConcurrentReferenceHashMap<>();
|
private static Map<AnnotatedElement, Set<Timed>> cache = new ConcurrentReferenceHashMap<>();
|
||||||
|
|
||||||
private HandlerMethodTimedAnnotations() {
|
private TimedAnnotations() {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Set<Timed> get(HandlerMethod handler) {
|
/**
|
||||||
Set<Timed> methodAnnotations = findTimedAnnotations(handler.getMethod());
|
* Return {@link Timed} annotation that should be used for the given {@code method}
|
||||||
|
* and {@code type}.
|
||||||
|
* @param method the source method
|
||||||
|
* @param type the source type
|
||||||
|
* @return the {@link Timed} annotations to use or an empty set
|
||||||
|
*/
|
||||||
|
public static Set<Timed> get(Method method, Class<?> type) {
|
||||||
|
Set<Timed> methodAnnotations = findTimedAnnotations(method);
|
||||||
if (!methodAnnotations.isEmpty()) {
|
if (!methodAnnotations.isEmpty()) {
|
||||||
return methodAnnotations;
|
return methodAnnotations;
|
||||||
}
|
}
|
||||||
return findTimedAnnotations(handler.getBeanType());
|
return findTimedAnnotations(type);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static Set<Timed> findTimedAnnotations(AnnotatedElement element) {
|
private static Set<Timed> findTimedAnnotations(AnnotatedElement element) {
|
||||||
|
if (element == null) {
|
||||||
|
return Collections.emptySet();
|
||||||
|
}
|
||||||
Set<Timed> result = cache.get(element);
|
Set<Timed> result = cache.get(element);
|
||||||
if (result != null) {
|
if (result != null) {
|
||||||
return result;
|
return result;
|
||||||
|
|
@ -0,0 +1,20 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2021 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Support classes for handler method metrics.
|
||||||
|
*/
|
||||||
|
package org.springframework.boot.actuate.metrics.annotation;
|
||||||
|
|
@ -23,13 +23,11 @@ import java.util.concurrent.TimeUnit;
|
||||||
import io.micrometer.core.annotation.Timed;
|
import io.micrometer.core.annotation.Timed;
|
||||||
import io.micrometer.core.instrument.MeterRegistry;
|
import io.micrometer.core.instrument.MeterRegistry;
|
||||||
import io.micrometer.core.instrument.Tag;
|
import io.micrometer.core.instrument.Tag;
|
||||||
import io.micrometer.core.instrument.Timer;
|
|
||||||
import io.micrometer.core.instrument.Timer.Builder;
|
|
||||||
import org.reactivestreams.Publisher;
|
import org.reactivestreams.Publisher;
|
||||||
import reactor.core.publisher.Mono;
|
import reactor.core.publisher.Mono;
|
||||||
|
|
||||||
import org.springframework.boot.actuate.metrics.AutoTimer;
|
import org.springframework.boot.actuate.metrics.AutoTimer;
|
||||||
import org.springframework.boot.actuate.metrics.web.method.HandlerMethodTimedAnnotations;
|
import org.springframework.boot.actuate.metrics.annotation.TimedAnnotations;
|
||||||
import org.springframework.boot.web.reactive.error.ErrorAttributes;
|
import org.springframework.boot.web.reactive.error.ErrorAttributes;
|
||||||
import org.springframework.core.Ordered;
|
import org.springframework.core.Ordered;
|
||||||
import org.springframework.core.annotation.Order;
|
import org.springframework.core.annotation.Order;
|
||||||
|
|
@ -102,22 +100,19 @@ public class MetricsWebFilter implements WebFilter {
|
||||||
private void record(ServerWebExchange exchange, Throwable cause, long start) {
|
private void record(ServerWebExchange exchange, Throwable cause, long start) {
|
||||||
cause = (cause != null) ? cause : exchange.getAttribute(ErrorAttributes.ERROR_ATTRIBUTE);
|
cause = (cause != null) ? cause : exchange.getAttribute(ErrorAttributes.ERROR_ATTRIBUTE);
|
||||||
Object handler = exchange.getAttribute(HandlerMapping.BEST_MATCHING_HANDLER_ATTRIBUTE);
|
Object handler = exchange.getAttribute(HandlerMapping.BEST_MATCHING_HANDLER_ATTRIBUTE);
|
||||||
Set<Timed> annotations = (handler instanceof HandlerMethod)
|
Set<Timed> annotations = getTimedAnnotations(handler);
|
||||||
? HandlerMethodTimedAnnotations.get((HandlerMethod) handler) : Collections.emptySet();
|
|
||||||
Iterable<Tag> tags = this.tagsProvider.httpRequestTags(exchange, cause);
|
Iterable<Tag> tags = this.tagsProvider.httpRequestTags(exchange, cause);
|
||||||
long duration = System.nanoTime() - start;
|
long duration = System.nanoTime() - start;
|
||||||
if (annotations.isEmpty()) {
|
AutoTimer.apply(this.autoTimer, this.metricName, annotations,
|
||||||
if (this.autoTimer.isEnabled()) {
|
(builder) -> builder.tags(tags).register(this.registry).record(duration, TimeUnit.NANOSECONDS));
|
||||||
Builder builder = this.autoTimer.builder(this.metricName);
|
|
||||||
builder.tags(tags).register(this.registry).record(duration, TimeUnit.NANOSECONDS);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
for (Timed annotation : annotations) {
|
|
||||||
Builder builder = Timer.builder(annotation, this.metricName);
|
|
||||||
builder.tags(tags).register(this.registry).record(duration, TimeUnit.NANOSECONDS);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Set<Timed> getTimedAnnotations(Object handler) {
|
||||||
|
if (handler instanceof HandlerMethod) {
|
||||||
|
HandlerMethod handlerMethod = (HandlerMethod) handler;
|
||||||
|
return TimedAnnotations.get(handlerMethod.getMethod(), handlerMethod.getBeanType());
|
||||||
}
|
}
|
||||||
|
return Collections.emptySet();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -32,7 +32,7 @@ import io.micrometer.core.instrument.Timer.Builder;
|
||||||
import io.micrometer.core.instrument.Timer.Sample;
|
import io.micrometer.core.instrument.Timer.Sample;
|
||||||
|
|
||||||
import org.springframework.boot.actuate.metrics.AutoTimer;
|
import org.springframework.boot.actuate.metrics.AutoTimer;
|
||||||
import org.springframework.boot.actuate.metrics.web.method.HandlerMethodTimedAnnotations;
|
import org.springframework.boot.actuate.metrics.annotation.TimedAnnotations;
|
||||||
import org.springframework.boot.web.servlet.error.ErrorAttributes;
|
import org.springframework.boot.web.servlet.error.ErrorAttributes;
|
||||||
import org.springframework.http.HttpStatus;
|
import org.springframework.http.HttpStatus;
|
||||||
import org.springframework.web.filter.OncePerRequestFilter;
|
import org.springframework.web.filter.OncePerRequestFilter;
|
||||||
|
|
@ -42,8 +42,8 @@ import org.springframework.web.servlet.HandlerMapping;
|
||||||
import org.springframework.web.util.NestedServletException;
|
import org.springframework.web.util.NestedServletException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Intercepts incoming HTTP requests and records metrics about Spring MVC execution time
|
* Intercepts incoming HTTP requests handled by Spring MVC handlers and records metrics
|
||||||
* and results.
|
* about execution time and results.
|
||||||
*
|
*
|
||||||
* @author Jon Schneider
|
* @author Jon Schneider
|
||||||
* @author Phillip Webb
|
* @author Phillip Webb
|
||||||
|
|
@ -128,27 +128,24 @@ public class WebMvcMetricsFilter extends OncePerRequestFilter {
|
||||||
private void record(TimingContext timingContext, HttpServletRequest request, HttpServletResponse response,
|
private void record(TimingContext timingContext, HttpServletRequest request, HttpServletResponse response,
|
||||||
Throwable exception) {
|
Throwable exception) {
|
||||||
Object handler = getHandler(request);
|
Object handler = getHandler(request);
|
||||||
Set<Timed> annotations = (handler instanceof HandlerMethod)
|
Set<Timed> annotations = getTimedAnnotations(handler);
|
||||||
? HandlerMethodTimedAnnotations.get((HandlerMethod) handler) : Collections.emptySet();
|
|
||||||
Timer.Sample timerSample = timingContext.getTimerSample();
|
Timer.Sample timerSample = timingContext.getTimerSample();
|
||||||
if (annotations.isEmpty()) {
|
AutoTimer.apply(this.autoTimer, this.metricName, annotations,
|
||||||
if (this.autoTimer.isEnabled()) {
|
(builder) -> timerSample.stop(getTimer(builder, handler, request, response, exception)));
|
||||||
Builder builder = this.autoTimer.builder(this.metricName);
|
|
||||||
timerSample.stop(getTimer(builder, handler, request, response, exception));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
for (Timed annotation : annotations) {
|
|
||||||
Builder builder = Timer.builder(annotation, this.metricName);
|
|
||||||
timerSample.stop(getTimer(builder, handler, request, response, exception));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private Object getHandler(HttpServletRequest request) {
|
private Object getHandler(HttpServletRequest request) {
|
||||||
return request.getAttribute(HandlerMapping.BEST_MATCHING_HANDLER_ATTRIBUTE);
|
return request.getAttribute(HandlerMapping.BEST_MATCHING_HANDLER_ATTRIBUTE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Set<Timed> getTimedAnnotations(Object handler) {
|
||||||
|
if (handler instanceof HandlerMethod) {
|
||||||
|
HandlerMethod handlerMethod = (HandlerMethod) handler;
|
||||||
|
return TimedAnnotations.get(handlerMethod.getMethod(), handlerMethod.getBeanType());
|
||||||
|
}
|
||||||
|
return Collections.emptySet();
|
||||||
|
}
|
||||||
|
|
||||||
private Timer getTimer(Builder builder, Object handler, HttpServletRequest request, HttpServletResponse response,
|
private Timer getTimer(Builder builder, Object handler, HttpServletRequest request, HttpServletResponse response,
|
||||||
Throwable exception) {
|
Throwable exception) {
|
||||||
return builder.tags(this.tagsProvider.getTags(request, response, handler, exception)).register(this.registry);
|
return builder.tags(this.tagsProvider.getTags(request, response, handler, exception)).register(this.registry);
|
||||||
|
|
|
||||||
|
|
@ -14,7 +14,7 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package org.springframework.boot.actuate.metrics.web.method;
|
package org.springframework.boot.actuate.metrics.annotation;
|
||||||
|
|
||||||
import java.lang.reflect.Method;
|
import java.lang.reflect.Method;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
@ -23,22 +23,21 @@ import io.micrometer.core.annotation.Timed;
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
import org.springframework.util.ReflectionUtils;
|
import org.springframework.util.ReflectionUtils;
|
||||||
import org.springframework.web.method.HandlerMethod;
|
|
||||||
|
|
||||||
import static org.assertj.core.api.Assertions.assertThat;
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tests for {@link HandlerMethodTimedAnnotations}.
|
* Tests for {@link TimedAnnotations}.
|
||||||
*
|
*
|
||||||
* @author Phillip Webb
|
* @author Phillip Webb
|
||||||
*/
|
*/
|
||||||
class HandlerMethodTimedAnnotationsTests {
|
class TimedAnnotationsTests {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void getWhenNoneReturnsEmptySet() {
|
void getWhenNoneReturnsEmptySet() {
|
||||||
Object bean = new None();
|
Object bean = new None();
|
||||||
Method method = ReflectionUtils.findMethod(bean.getClass(), "handle");
|
Method method = ReflectionUtils.findMethod(bean.getClass(), "handle");
|
||||||
Set<Timed> annotations = HandlerMethodTimedAnnotations.get(new HandlerMethod(bean, method));
|
Set<Timed> annotations = TimedAnnotations.get(method, bean.getClass());
|
||||||
assertThat(annotations).isEmpty();
|
assertThat(annotations).isEmpty();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -46,7 +45,7 @@ class HandlerMethodTimedAnnotationsTests {
|
||||||
void getWhenOnMethodReturnsMethodAnnotations() {
|
void getWhenOnMethodReturnsMethodAnnotations() {
|
||||||
Object bean = new OnMethod();
|
Object bean = new OnMethod();
|
||||||
Method method = ReflectionUtils.findMethod(bean.getClass(), "handle");
|
Method method = ReflectionUtils.findMethod(bean.getClass(), "handle");
|
||||||
Set<Timed> annotations = HandlerMethodTimedAnnotations.get(new HandlerMethod(bean, method));
|
Set<Timed> annotations = TimedAnnotations.get(method, bean.getClass());
|
||||||
assertThat(annotations).extracting(Timed::value).containsOnly("y", "z");
|
assertThat(annotations).extracting(Timed::value).containsOnly("y", "z");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -54,7 +53,7 @@ class HandlerMethodTimedAnnotationsTests {
|
||||||
void getWhenNonOnMethodReturnsBeanAnnotations() {
|
void getWhenNonOnMethodReturnsBeanAnnotations() {
|
||||||
Object bean = new OnBean();
|
Object bean = new OnBean();
|
||||||
Method method = ReflectionUtils.findMethod(bean.getClass(), "handle");
|
Method method = ReflectionUtils.findMethod(bean.getClass(), "handle");
|
||||||
Set<Timed> annotations = HandlerMethodTimedAnnotations.get(new HandlerMethod(bean, method));
|
Set<Timed> annotations = TimedAnnotations.get(method, bean.getClass());
|
||||||
assertThat(annotations).extracting(Timed::value).containsOnly("y", "z");
|
assertThat(annotations).extracting(Timed::value).containsOnly("y", "z");
|
||||||
}
|
}
|
||||||
|
|
||||||
Loading…
Reference in New Issue