aboutsummaryrefslogtreecommitdiffstats
path: root/container-core/src/test/java/com/yahoo/container/jdisc/state/MetricsPacketsHandlerTest.java
blob: ead26a0a89887be70d816408447cbac3752b0244 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
// Copyright Yahoo. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
package com.yahoo.container.jdisc.state;

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.yahoo.container.jdisc.RequestHandlerTestDriver;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;

import java.util.ArrayList;
import java.util.List;
import java.util.Map;

import static com.yahoo.container.jdisc.state.MetricsPacketsHandler.APPLICATION_KEY;
import static com.yahoo.container.jdisc.state.MetricsPacketsHandler.DIMENSIONS_KEY;
import static com.yahoo.container.jdisc.state.MetricsPacketsHandler.METRICS_KEY;
import static com.yahoo.container.jdisc.state.MetricsPacketsHandler.PACKET_SEPARATOR;
import static com.yahoo.container.jdisc.state.MetricsPacketsHandler.TIMESTAMP_KEY;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertTrue;

/**
 * @author gjoranv
 */
public class MetricsPacketsHandlerTest extends StateHandlerTestBase {

    private static final String APPLICATION_NAME = "state-handler-test-base";
    private static final String HOST_DIMENSION = "some-hostname";

    private static MetricsPacketsHandler metricsPacketsHandler;

    @BeforeEach
    public void setupHandler() {
        metricsPacketsHandlerConfig = new MetricsPacketsHandlerConfig(new MetricsPacketsHandlerConfig.Builder()
                                                                              .application(APPLICATION_NAME).hostname(HOST_DIMENSION));
        metricsPacketsHandler = new MetricsPacketsHandler(timer, snapshotProviderRegistry, metricsPacketsHandlerConfig);
        testDriver = new RequestHandlerTestDriver(metricsPacketsHandler);
    }

    @Test
    void metrics_are_included_after_snapshot() throws Exception {
        createSnapshotWithCountMetric("counter", 1, null);
        List<JsonNode> packets = incrementTimeAndGetJsonPackets();
        assertEquals(1, packets.size());

        JsonNode counterPacket = packets.get(0);
        assertCountMetric(counterPacket, "counter.count", 1);
    }

    @Test
    void metadata_is_included_in_each_metrics_packet() throws Exception {
        createSnapshotWithCountMetric("counter", 1, null);
        List<JsonNode> packets = incrementTimeAndGetJsonPackets();
        JsonNode counterPacket = packets.get(0);

        assertTrue(counterPacket.has(TIMESTAMP_KEY));
        assertTrue(counterPacket.has(APPLICATION_KEY));
        assertEquals(APPLICATION_NAME, counterPacket.get(APPLICATION_KEY).asText());
    }

    @Test
    void timestamp_resolution_is_in_seconds() throws Exception {
        createSnapshotWithCountMetric("counter", 1, null);
        List<JsonNode> packets = incrementTimeAndGetJsonPackets();
        JsonNode counterPacket = packets.get(0);

        assertEquals(SNAPSHOT_INTERVAL / 1000L, counterPacket.get(TIMESTAMP_KEY).asLong());
    }

    @Test
    void expected_aggregators_are_output_for_gauge_metrics() throws Exception {
        var context = StateMetricContext.newInstance(Map.of("dim1", "value1"));
        var snapshot = new MetricSnapshot();
        snapshot.set(context, "gauge", 0.2);
        snapshotProvider.setSnapshot(snapshot);

        List<JsonNode> packets = incrementTimeAndGetJsonPackets();
        JsonNode gaugeMetric = packets.get(0).get(METRICS_KEY);

        assertEquals(0.2, gaugeMetric.get("gauge.last").asDouble(), 0.1);
        assertEquals(0.2, gaugeMetric.get("gauge.average").asDouble(), 0.1);
        assertEquals(0.2, gaugeMetric.get("gauge.max").asDouble(), 0.1);
    }

    @Test
    void dimensions_from_context_are_included() throws Exception {
        var context = StateMetricContext.newInstance(Map.of("dim1", "value1"));
        createSnapshotWithCountMetric("counter", 1, context);

        List<JsonNode> packets = incrementTimeAndGetJsonPackets();
        JsonNode counterPacket = packets.get(0);

        assertTrue(counterPacket.has(DIMENSIONS_KEY));
        JsonNode dimensions = counterPacket.get(DIMENSIONS_KEY);
        assertEquals("value1", dimensions.get("dim1").asText());
    }

    @Test
    void metrics_with_identical_dimensions_are_contained_in_the_same_packet() throws Exception {
        var context = StateMetricContext.newInstance(Map.of("dim1", "value1"));
        var snapshot = new MetricSnapshot();
        snapshot.add(context, "counter1", 1);
        snapshot.add(context, "counter2", 2);
        snapshotProvider.setSnapshot(snapshot);

        List<JsonNode> packets = incrementTimeAndGetJsonPackets();
        assertEquals(1, packets.size());
        JsonNode countersPacket = packets.get(0);

        assertEquals("value1", countersPacket.get(DIMENSIONS_KEY).get("dim1").asText());
        assertCountMetric(countersPacket, "counter1.count", 1);
        assertCountMetric(countersPacket, "counter2.count", 2);
    }

    @Test
    void metrics_with_different_dimensions_get_separate_packets() throws Exception {
        var context1 = StateMetricContext.newInstance(Map.of("dim1", "value1"));
        var context2 = StateMetricContext.newInstance(Map.of("dim2", "value2"));
        var snapshot = new MetricSnapshot();
        snapshot.add(context1, "counter1", 1);
        snapshot.add(context2, "counter2", 2);
        snapshotProvider.setSnapshot(snapshot);

        List<JsonNode> packets = incrementTimeAndGetJsonPackets();
        assertEquals(2, packets.size());
    }

    @Test
    void host_dimension_only_created_if_absent() throws Exception {
        var context1 = StateMetricContext.newInstance(Map.of("dim1", "value1", "host", "foo.bar"));
        var context2 = StateMetricContext.newInstance(Map.of("dim2", "value2"));
        var snapshot = new MetricSnapshot();
        snapshot.add(context1, "counter1", 1);
        snapshot.add(context2, "counter2", 2);
        snapshotProvider.setSnapshot(snapshot);

        var packets = incrementTimeAndGetJsonPackets();
        assertEquals(2, packets.size());

        packets.forEach(packet -> {
            if (!packet.has(DIMENSIONS_KEY)) return;
            var dimensions = packet.get(DIMENSIONS_KEY);
            if (dimensions.has("dim1")) assertDimension(packet, "host", "foo.bar");
            if (dimensions.has("dim2")) assertDimension(packet, "host", HOST_DIMENSION);
        });
    }

    @Test
    public void prometheus_metrics() {
        var context = StateMetricContext.newInstance(Map.of("dim-1", "value1"));
        var snapshot = new MetricSnapshot();
        snapshot.set(context, "gauge.metric", 0.2);
        snapshot.add(context, "counter.metric", 5);
        snapshotProvider.setSnapshot(snapshot);
        var response = requestAsString("http://localhost/metrics-packets?format=prometheus");
        var expectedResponse = """
                # HELP gauge_metric_last\s
                # TYPE gauge_metric_last untyped
                gauge_metric_last{dim_1="value1",vespa_service="state-handler-test-base",} 0.2 0
                # HELP counter_metric_count\s
                # TYPE counter_metric_count untyped
                counter_metric_count{dim_1="value1",vespa_service="state-handler-test-base",} 5 0
                """;
        assertEquals(expectedResponse, response);
    }

    @Test
    public void test_metric_filtering() {
        var context = StateMetricContext.newInstance(Map.of("dim-1", "value1"));
        var snapshot = new MetricSnapshot();
        snapshot.set(context, "gauge.metric", 0.2);
        snapshot.add(context, "counter.metric", 5);
        snapshot.add(context, "configserver.requests", 120);
        // Infrastructure set only contains max and average
        snapshot.set(context, "lockAttempt.lockedLoad", 500);

        // Without filtering
        snapshotProvider.setSnapshot(snapshot);
        var response = requestAsString("http://localhost/metrics-packets");
        var expectedResponse = """
                {
                  "application" : "state-handler-test-base",
                  "timestamp" : 0,
                  "dimensions" : {
                    "dim-1" : "value1",
                    "host" : "some-hostname"
                  },
                  "metrics" : {
                    "gauge.metric.average" : 0.2,
                    "gauge.metric.last" : 0.2,
                    "gauge.metric.max" : 0.2,
                    "gauge.metric.min" : 0.2,
                    "gauge.metric.sum" : 0.2,
                    "gauge.metric.count" : 1,
                    "configserver.requests.count" : 120,
                    "lockAttempt.lockedLoad.average" : 500.0,
                    "lockAttempt.lockedLoad.last" : 500.0,
                    "lockAttempt.lockedLoad.max" : 500.0,
                    "lockAttempt.lockedLoad.min" : 500.0,
                    "lockAttempt.lockedLoad.sum" : 500.0,
                    "lockAttempt.lockedLoad.count" : 1,
                    "counter.metric.count" : 5
                  }
                }
                """;
        assertTrue(response.startsWith(expectedResponse));

        // With filtering
        response = requestAsString("http://localhost/metrics-packets?metric-set=infrastructure");
        expectedResponse = """
                {
                  "application" : "state-handler-test-base",
                  "timestamp" : 0,
                  "dimensions" : {
                    "dim-1" : "value1",
                    "host" : "some-hostname"
                  },
                  "metrics" : {
                    "configserver.requests.count" : 120,
                    "lockAttempt.lockedLoad.average" : 500.0,
                    "lockAttempt.lockedLoad.max" : 500.0
                  }
                }
                """;
        assertTrue(response.startsWith(expectedResponse));
    }
    
    private List<JsonNode> incrementTimeAndGetJsonPackets() throws Exception {
        advanceToNextSnapshot();
        String response = requestAsString("http://localhost/metrics-packets");

        return toJsonPackets(response);
    }

    private List<JsonNode> toJsonPackets(String response) throws Exception {
        List<JsonNode> jsonPackets = new ArrayList<>();
        String[] packets = response.split(PACKET_SEPARATOR);
        ObjectMapper mapper = new ObjectMapper();
        for (String packet : packets) {
            jsonPackets.add(mapper.readTree(mapper.getFactory().createParser(packet)));
        }
        return jsonPackets;
    }

    private void assertCountMetric(JsonNode metricsPacket, String metricName, long expected) {
        assertTrue(metricsPacket.has(METRICS_KEY));
        JsonNode counterMetrics = metricsPacket.get(METRICS_KEY);
        assertTrue(counterMetrics.has(metricName));
        assertEquals(expected, counterMetrics.get(metricName).asLong());
    }

    private void assertDimension(JsonNode metricsPacket, String dimensionName, String expectedDimensionValue) {
        assertTrue(metricsPacket.has(DIMENSIONS_KEY));
        var dimensions = metricsPacket.get(DIMENSIONS_KEY);
        assertTrue(dimensions.has(dimensionName));
        assertEquals(expectedDimensionValue, dimensions.get(dimensionName).asText());
    }

    private void createSnapshotWithCountMetric(String name, Number value, MetricDimensions context) {
        var snapshot = new MetricSnapshot();
        snapshot.add(context, name, value);
        snapshotProvider.setSnapshot(snapshot);
    }

}