Monitoring and Metrics

The ability to monitor and measure what is going on is very important. Especially in a location transparent environment like an Axon application it is very important to be able to trace your message and check the ingestion rate of it.

Monitoring

Monitoring a message centric application will require you to be able to see where your messages are at a given point in time. This translates to being able to track your commands, events and queries from one component to another in an Axon application.

Correlation Data

One import aspect in regards to this is tracing a given message. To that end the framework provides the CorrelationDataProvider, as described briefly here. This interface and its implementations provide you the means to populate the meta-data of your messages with specific fields, like a 'trace-id', 'correlation-id' or any other field you might be interested in.

For configuring the MessageOriginProvider you can do the following:

public class MonitoringConfiguration {

    public Configurer buildConfigurer() {
        return DefaultConfigurer.defaultConfiguration();
    }

    public void configureMessageOriginProvider(Configurer configurer) {
        configurer.configureCorrelationDataProviders(configuration -> {
            List<CorrelationDataProvider> correlationDataProviders = new ArrayList<>();
            correlationDataProviders.add(new MessageOriginProvider());
            return correlationDataProviders;
        });
    }

}

Interceptor Logging

Another good approach to track the flow of messages throughout an Axon application is by setting up the right interceptors in your application. There are two flavors of interceptors, the Dispatch and Handler Interceptors (like discussed here), which intercept a message prior to publishing (Dispatch Interceptor) or whilst it is being handled (Handler Interceptor). The interceptor mechanism lends itself quite nicely to introduce a way to consistently log when a message is being dispatched/handled. The LoggingInterceptor is an out of the box solution to log any type of message to SLF4J, but also provides a simple overridable template to set up your own desired logging format. We refer to the command, event and query sections for the specifics on how to configure message interceptors.

Metrics

Interesting metrics in a message centric system come in several forms and flavors, like count, capacity and latency for example. The Axon Framework allows you to retrieve such measurements through the use of the axon-metrics or axon-micrometer module. With this module(s) you can register a number of MessageMonitor implementations to your messaging components, like the CommandBus, EventBus, QueryBus and EventProcessors.

axon-metrics module uses Dropwizard Metrics for registering the measurements correctly. That thus means that MessageMonitors are registered against the Dropwizard MetricRegistry.

axon-micrometer module uses Micrometer which is a dimensional-first metrics collection facade whose aim is to allow you to time, count, and gauge your code with a vendor neutral API. That thus means that MessageMonitors are registered against the Micrometer MeterRegistry.

The following monitor implementations are currently provided:

  1. CapacityMonitor - Measure the message capacity by keeping track of the total time spent on message handling compared to total time it is active.

    This returns a number between 0 and n number of threads, thus if there are 4 threads working, the maximum capacity is 4 if every thread is active for 100%

  2. EventProcessorLatencyMonitor - Measures the difference in message timestamps between the last ingested and the last processed event message.

  3. MessageCountingMonitor - Counts the number of ingested, successful, failed, ignored and processed messages.

  4. MessageTimerMonitor - Keeps a timer for all successful, failed and ignored messages, as well as an overall timer for all three combined.

  5. PayloadTypeMessageMonitorWrapper - A special MessageMonitor implementation which allows setting a monitor per message type instead of per message publishing/handling component.

You are free to configure any combination of MessageMonitors through constructors on your messaging components, and even simpler by using the Configuration API. The GlobalMetricRegistry contained in the axon-metrics and axon-micrometer modules provides a set of sensible defaults per type of messaging component. The following example shows you how to configure default metrics for your message handling components:

Dropwizard

public class MetricsConfiguration {

    public Configurer buildConfigurer() {
        return DefaultConfigurer.defaultConfiguration();
    }

    // The MetricRegistry is a class from the Dropwizard Metrics framework
    public void configureDefaultMetrics(Configurer configurer, MetricRegistry metricRegistry) {
        GlobalMetricRegistry globalMetricRegistry = new GlobalMetricRegistry(metricRegistry);
        // We register the default monitors to our messaging components by doing the following
        globalMetricRegistry.registerWithConfigurer(configurer);
    }
}

Micrometer

public class MetricsConfiguration {

    public Configurer buildConfigurer() {
        return DefaultConfigurer.defaultConfiguration();
    }

    // The MeterRegistry is a class from the Micrometer library
    public void configureDefaultMetrics(Configurer configurer, MeterRegistry meterRegistry) {
        GlobalMetricRegistry globalMetricRegistry = new GlobalMetricRegistry(meterRegistry);
        // We register the default monitors to our messaging components by doing the following
        globalMetricRegistry.registerWithConfigurer(configurer);
    }
}

If you want to have more specific metrics on a message handling component like the Command Bus, EventBus (or more specifically TrackingEventProcessor), you can do the following:

// Java (Spring Boot Configuration) - Micrometer example
@Configuration
public class MetricConfig {

    @Bean
    public ConfigurerModule metricConfigurer(MeterRegistry meterRegistry){
        return configurer -> {
            instrumentEventProcessors(meterRegistry, configurer);
            instrumentCommandBus(meterRegistry, configurer);
        };
    }

    private void instrumentEventProcessors(MeterRegistry meterRegistry, Configurer configurer) {
        MessageMonitorFactory messageMonitorFactory = (configuration, componentType, componentName) -> {
            // We want to count the messages per type of event being published.
            PayloadTypeMessageMonitorWrapper<MessageCountingMonitor> messageCounterPerType =
                    new PayloadTypeMessageMonitorWrapper<>(monitorName -> MessageCountingMonitor.buildMonitor(monitorName, meterRegistry),
                                                           clazz ->  componentName + "_" + clazz.getSimpleName());
            // And we also want to set a message timer per payload type
            PayloadTypeMessageMonitorWrapper<MessageTimerMonitor> messageTimerPerType =
                    new PayloadTypeMessageMonitorWrapper<>(monitorName -> MessageTimerMonitor.buildMonitor(monitorName, meterRegistry),
                                                           clazz ->  componentName + "_" + clazz.getSimpleName());
            //Which we group in a MultiMessageMonitor
            return new MultiMessageMonitor<>(messageCounterPerType, messageTimerPerType);
        };
        configurer.configureMessageMonitor(TrackingEventProcessor.class, messageMonitorFactory);
    }

    private void instrumentCommandBus(MeterRegistry meterRegistry, Configurer configurer) {
            MessageMonitorFactory messageMonitorFactory = (configuration, componentType, componentName) -> {
                PayloadTypeMessageMonitorWrapper<MessageCountingMonitor> messageCounterPerType =
                        new PayloadTypeMessageMonitorWrapper<>(monitorName -> MessageCountingMonitor.buildMonitor(monitorName, meterRegistry),
                                                               clazz ->  componentName + "_" + clazz.getSimpleName());

                PayloadTypeMessageMonitorWrapper<MessageTimerMonitor> messageTimerPerType =
                        new PayloadTypeMessageMonitorWrapper<>(monitorName -> MessageTimerMonitor.buildMonitor(monitorName, meterRegistry),
                                                               clazz -> componentName + "_" + clazz.getSimpleName());

                PayloadTypeMessageMonitorWrapper<CapacityMonitor> capacityMonitor =
                        new PayloadTypeMessageMonitorWrapper<>(monitorName -> CapacityMonitor.buildMonitor(monitorName, meterRegistry),
                                                               clazz -> componentName + "_" + clazz.getSimpleName());

                return new MultiMessageMonitor<>(messageCounterPerType, messageTimerPerType, capacityMonitor);
            };
            configurer.configureMessageMonitor(CommandBus.class, messageMonitorFactory);
        }

}

Last updated