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

com.palantir.atlasdb.util.SlidingWindowMetricsInvocationHandler Maven / Gradle / Ivy

/*
 * (c) Copyright 2019 Palantir Technologies Inc. All rights reserved.
 *
 * 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
 *
 *     http://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 com.palantir.atlasdb.util;

import com.codahale.metrics.MetricRegistry;
import com.codahale.metrics.SlidingTimeWindowArrayReservoir;
import com.codahale.metrics.Timer;
import com.palantir.logsafe.Preconditions;
import com.palantir.logsafe.logger.SafeLogger;
import com.palantir.logsafe.logger.SafeLoggerFactory;
import com.palantir.tritium.event.AbstractInvocationEventHandler;
import com.palantir.tritium.event.DefaultInvocationContext;
import com.palantir.tritium.event.InvocationContext;
import java.lang.reflect.Method;
import java.time.Duration;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import javax.annotation.Nonnull;
import javax.annotation.Nullable;

/**
 * A modified version of {@link com.palantir.tritium.event.metrics.MetricsInvocationEventHandler} that uses a
 * {@link SlidingTimeWindowArrayReservoir} for the timer's reservoir.
 */
public final class SlidingWindowMetricsInvocationHandler extends AbstractInvocationEventHandler {
    private static final SafeLogger log = SafeLoggerFactory.get(SlidingWindowMetricsInvocationHandler.class);

    private final MetricRegistry metricRegistry;
    private final Map timers = new ConcurrentHashMap<>();
    private final String serviceName;

    public SlidingWindowMetricsInvocationHandler(MetricRegistry metricRegistry, String serviceName) {
        super(InstrumentationUtils.getEnabledSupplier(serviceName));
        this.metricRegistry = Preconditions.checkNotNull(metricRegistry, "metricRegistry");
        this.serviceName = Preconditions.checkNotNull(serviceName, "serviceName");
    }

    @Override
    public InvocationContext preInvocation(Object instance, Method method, Object[] args) {
        return DefaultInvocationContext.of(instance, method, args);
    }

    @Override
    public void onSuccess(@Nullable InvocationContext context, @Nullable Object result) {
        if (context == null) {
            log.debug("Encountered null metric context likely due to exception in preInvocation");
            return;
        }

        long nanos = System.nanoTime() - context.getStartTimeNanos();
        Method method = context.getMethod();
        Timer timer = timers.computeIfAbsent(method, this::getTimer);
        timer.update(Duration.ofNanos(nanos));
    }

    @Override
    public void onFailure(@Nullable InvocationContext context, @Nonnull Throwable cause) {
        markGlobalFailure();
        if (context == null) {
            log.debug("Encountered null metric context likely due to exception in preInvocation", cause);
            return;
        }

        String failuresMetricName = InstrumentationUtils.getFailuresMetricName(context, serviceName);
        metricRegistry.meter(failuresMetricName).mark();
        metricRegistry
                .meter(MetricRegistry.name(failuresMetricName, cause.getClass().getName()))
                .mark();
    }

    private Timer getTimer(Method method) {
        return metricRegistry.timer(
                InstrumentationUtils.getBaseMetricName(method, serviceName), InstrumentationUtils::createNewTimer);
    }

    private void markGlobalFailure() {
        metricRegistry
                .meter(InstrumentationUtils.TAGGED_FAILURES_METRIC_NAME.safeName())
                .mark();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy