All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.micrometer.core.aop.TimedAspect Maven / Gradle / Ivy

There is a newer version: 1.13.0
Show newest version
/**
 * Copyright 2017 VMware, Inc.
 * 

* 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 io.micrometer.core.aop; import io.micrometer.core.annotation.Incubating; import io.micrometer.core.annotation.Timed; import io.micrometer.core.instrument.LongTaskTimer; import io.micrometer.core.instrument.MeterRegistry; import io.micrometer.core.instrument.Tag; import io.micrometer.core.instrument.Tags; import io.micrometer.core.instrument.Timer; import io.micrometer.core.instrument.Metrics; import io.micrometer.core.lang.NonNullApi; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.reflect.MethodSignature; import java.lang.reflect.Method; import java.util.Optional; import java.util.concurrent.CompletionStage; import java.util.function.Function; import java.util.function.Predicate; /** *

* AspectJ aspect for intercepting types or methods annotated with {@link Timed @Timed}.
* The aspect supports programmatic customizations through constructor-injectable custom logic. *

*

* You might want to add tags programmatically to the {@link Timer}.
* In this case, the tags provider function (Function<ProceedingJoinPoint, Iterable<Tag>>) can help. * It receives a {@link ProceedingJoinPoint} and returns the {@link Tag}s that will be attached to the {@link Timer}. *

*

* You might also want to skip the {@link Timer} creation programmatically.
* One use-case can be having another component in your application that already processes the {@link Timed @Timed} annotation * in some cases so that {@code TimedAspect} should not intercept these methods. E.g.: Spring Boot does this for its controllers. * By using the skip predicate (Predicate<ProceedingJoinPoint>) * you can tell the {@code TimedAspect} when not to create a {@link Timer}. * * Here's an example to disable {@link Timer} creation for Spring controllers: *

*
 * @Bean
 * public TimedAspect timedAspect(MeterRegistry meterRegistry) {
 *     return new TimedAspect(meterRegistry, this::skipControllers);
 * }
 *
 * private boolean skipControllers(ProceedingJoinPoint pjp) {
 *     Class<?> targetClass = pjp.getTarget().getClass();
 *     return targetClass.isAnnotationPresent(RestController.class) || targetClass.isAnnotationPresent(Controller.class);
 * }
 * 
* * @author David J. M. Karlsen * @author Jon Schneider * @author Johnny Lim * @author Nejc Korasa * @author Jonatan Ivanov * @since 1.0.0 */ @Aspect @NonNullApi @Incubating(since = "1.0.0") public class TimedAspect { private static final Predicate DONT_SKIP_ANYTHING = pjp -> false; public static final String DEFAULT_METRIC_NAME = "method.timed"; public static final String DEFAULT_EXCEPTION_TAG_VALUE = "none"; /** * Tag key for an exception. * * @since 1.1.0 */ public static final String EXCEPTION_TAG = "exception"; private final MeterRegistry registry; private final Function> tagsBasedOnJoinPoint; private final Predicate shouldSkip; /** * Creates a {@code TimedAspect} instance with {@link Metrics#globalRegistry}. * * @since 1.2.0 */ public TimedAspect() { this(Metrics.globalRegistry); } /** * Creates a {@code TimedAspect} instance with the given {@code registry}. * * @param registry Where we're going to register metrics. */ public TimedAspect(MeterRegistry registry) { this(registry, DONT_SKIP_ANYTHING); } /** * Creates a {@code TimedAspect} instance with the given {@code registry} and tags provider function. * * @param registry Where we're going to register metrics. * @param tagsBasedOnJoinPoint A function to generate tags given a join point. */ public TimedAspect(MeterRegistry registry, Function> tagsBasedOnJoinPoint) { this(registry, tagsBasedOnJoinPoint, DONT_SKIP_ANYTHING); } /** * Creates a {@code TimedAspect} instance with the given {@code registry} and skip predicate. * * @param registry Where we're going to register metrics. * @param shouldSkip A predicate to decide if creating the timer should be skipped or not. * @since 1.7.0 */ public TimedAspect(MeterRegistry registry, Predicate shouldSkip) { this( registry, pjp -> Tags.of("class", pjp.getStaticPart().getSignature().getDeclaringTypeName(), "method", pjp.getStaticPart().getSignature().getName()), shouldSkip ); } /** * Creates a {@code TimedAspect} instance with the given {@code registry}, tags provider function and skip predicate. * * @param registry Where we're going to register metrics. * @param tagsBasedOnJoinPoint A function to generate tags given a join point. * @param shouldSkip A predicate to decide if creating the timer should be skipped or not. * @since 1.7.0 */ public TimedAspect(MeterRegistry registry, Function> tagsBasedOnJoinPoint, Predicate shouldSkip) { this.registry = registry; this.tagsBasedOnJoinPoint = tagsBasedOnJoinPoint; this.shouldSkip = shouldSkip; } @Around("execution (@io.micrometer.core.annotation.Timed * *.*(..))") public Object timedMethod(ProceedingJoinPoint pjp) throws Throwable { if (shouldSkip.test(pjp)) { return pjp.proceed(); } Method method = ((MethodSignature) pjp.getSignature()).getMethod(); Timed timed = method.getAnnotation(Timed.class); if (timed == null) { method = pjp.getTarget().getClass().getMethod(method.getName(), method.getParameterTypes()); timed = method.getAnnotation(Timed.class); } final String metricName = timed.value().isEmpty() ? DEFAULT_METRIC_NAME : timed.value(); final boolean stopWhenCompleted = CompletionStage.class.isAssignableFrom(method.getReturnType()); if (!timed.longTask()) { return processWithTimer(pjp, timed, metricName, stopWhenCompleted); } else { return processWithLongTaskTimer(pjp, timed, metricName, stopWhenCompleted); } } private Object processWithTimer(ProceedingJoinPoint pjp, Timed timed, String metricName, boolean stopWhenCompleted) throws Throwable { Timer.Sample sample = Timer.start(registry); if (stopWhenCompleted) { try { return ((CompletionStage) pjp.proceed()).whenComplete((result, throwable) -> record(pjp, timed, metricName, sample, getExceptionTag(throwable))); } catch (Exception ex) { record(pjp, timed, metricName, sample, ex.getClass().getSimpleName()); throw ex; } } String exceptionClass = DEFAULT_EXCEPTION_TAG_VALUE; try { return pjp.proceed(); } catch (Exception ex) { exceptionClass = ex.getClass().getSimpleName(); throw ex; } finally { record(pjp, timed, metricName, sample, exceptionClass); } } private void record(ProceedingJoinPoint pjp, Timed timed, String metricName, Timer.Sample sample, String exceptionClass) { try { sample.stop(Timer.builder(metricName) .description(timed.description().isEmpty() ? null : timed.description()) .tags(timed.extraTags()) .tags(EXCEPTION_TAG, exceptionClass) .tags(tagsBasedOnJoinPoint.apply(pjp)) .publishPercentileHistogram(timed.histogram()) .publishPercentiles(timed.percentiles().length == 0 ? null : timed.percentiles()) .register(registry)); } catch (Exception e) { // ignoring on purpose } } private String getExceptionTag(Throwable throwable) { if (throwable == null) { return DEFAULT_EXCEPTION_TAG_VALUE; } if (throwable.getCause() == null) { return throwable.getClass().getSimpleName(); } return throwable.getCause().getClass().getSimpleName(); } private Object processWithLongTaskTimer(ProceedingJoinPoint pjp, Timed timed, String metricName, boolean stopWhenCompleted) throws Throwable { Optional sample = buildLongTaskTimer(pjp, timed, metricName).map(LongTaskTimer::start); if (stopWhenCompleted) { try { return ((CompletionStage) pjp.proceed()).whenComplete((result, throwable) -> sample.ifPresent(this::stopTimer)); } catch (Exception ex) { sample.ifPresent(this::stopTimer); throw ex; } } try { return pjp.proceed(); } finally { sample.ifPresent(this::stopTimer); } } private void stopTimer(LongTaskTimer.Sample sample) { try { sample.stop(); } catch (Exception e) { // ignoring on purpose } } /** * Secure long task timer creation - it should not disrupt the application flow in case of exception */ private Optional buildLongTaskTimer(ProceedingJoinPoint pjp, Timed timed, String metricName) { try { return Optional.of(LongTaskTimer.builder(metricName) .description(timed.description().isEmpty() ? null : timed.description()) .tags(timed.extraTags()) .tags(tagsBasedOnJoinPoint.apply(pjp)) .register(registry)); } catch (Exception e) { return Optional.empty(); } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy