From 9e0d511162dd9c3689f26300c67a8ba6c7b5af10 Mon Sep 17 00:00:00 2001 From: Emmanuel Evbuomwan Date: Thu, 19 Dec 2024 10:44:33 +0100 Subject: [PATCH] feat: OTel metrics metering --- container/compose.yml | 9 +++- container/opentelemetry/collector-config.yaml | 2 +- container/prometheus/prometheus.yml | 4 ++ src/karapace/config.py | 1 + src/schema_registry/telemetry/meter.py | 32 +++++++++++++ tests/conftest.py | 1 + .../schema_registry/telemetry/test_meter.py | 45 +++++++++++++++++++ 7 files changed, 91 insertions(+), 3 deletions(-) create mode 100644 src/schema_registry/telemetry/meter.py create mode 100644 tests/unit/schema_registry/telemetry/test_meter.py diff --git a/container/compose.yml b/container/compose.yml index f50268096..39fb14393 100644 --- a/container/compose.yml +++ b/container/compose.yml @@ -78,7 +78,7 @@ services: KARAPACE_GROUP_ID: karapace-schema-registry KARAPACE_MASTER_ELIGIBILITY: true KARAPACE_TOPIC_NAME: _schemas - KARAPACE_LOG_LEVEL: DEBUG + KARAPACE_LOG_LEVEL: INFO KARAPACE_COMPATIBILITY: FULL KARAPACE_STATSD_HOST: statsd-exporter KARAPACE_STATSD_PORT: 8125 @@ -118,7 +118,7 @@ services: KARAPACE_REGISTRY_HOST: karapace-schema-registry KARAPACE_REGISTRY_PORT: 8081 KARAPACE_ADMIN_METADATA_MAX_AGE: 0 - KARAPACE_LOG_LEVEL: DEBUG + KARAPACE_LOG_LEVEL: INFO KARAPACE_STATSD_HOST: statsd-exporter KARAPACE_STATSD_PORT: 8125 KARAPACE_KAFKA_SCHEMA_READER_STRICT_MODE: false @@ -154,6 +154,11 @@ services: prometheus: image: prom/prometheus + command: + - --storage.tsdb.path=/prometheus + - --storage.tsdb.retention.time=1d + - --enable-feature=otlp-write-receiver + - --config.file=/etc/prometheus/prometheus.yml volumes: - ./prometheus/prometheus.yml:/etc/prometheus/prometheus.yml - ./prometheus/rules.yml:/etc/prometheus/rules.yml diff --git a/container/opentelemetry/collector-config.yaml b/container/opentelemetry/collector-config.yaml index 3fcd0df0c..a7b67746d 100644 --- a/container/opentelemetry/collector-config.yaml +++ b/container/opentelemetry/collector-config.yaml @@ -15,7 +15,7 @@ exporters: tls: insecure: true otlphttp/prometheus: - endpoint: prometheus:9090/api/v1/otlp + endpoint: http://prometheus:9090/api/v1/otlp tls: insecure: true diff --git a/container/prometheus/prometheus.yml b/container/prometheus/prometheus.yml index 68b64109f..745041b82 100644 --- a/container/prometheus/prometheus.yml +++ b/container/prometheus/prometheus.yml @@ -3,6 +3,10 @@ global: scrape_timeout: 5s # How long until a scrape request times out. evaluation_interval: 10s # How frequently to evaluate rules. +storage: + tsdb: + out_of_order_time_window: 30m + rule_files: - /etc/prometheus/rules.yml diff --git a/src/karapace/config.py b/src/karapace/config.py index ccc8a9ad3..89a188b82 100644 --- a/src/karapace/config.py +++ b/src/karapace/config.py @@ -47,6 +47,7 @@ class KarapaceTelemetryOTelExporter(str, enum.Enum): class KarapaceTelemetry(BaseModel): otel_endpoint_url: str | None = None otel_exporter: KarapaceTelemetryOTelExporter = KarapaceTelemetryOTelExporter.NOOP + metrics_export_interval_milliseconds: int = 10000 resource_service_name: str = "karapace" resource_service_instance_id: str = "karapace" resource_telemetry_sdk_name: str = "opentelemetry" diff --git a/src/schema_registry/telemetry/meter.py b/src/schema_registry/telemetry/meter.py new file mode 100644 index 000000000..83eaf7046 --- /dev/null +++ b/src/schema_registry/telemetry/meter.py @@ -0,0 +1,32 @@ +""" +Copyright (c) 2024 Aiven Ltd +See LICENSE for details +""" + +from dependency_injector.wiring import inject, Provide +from karapace.config import Config +from karapace.container import KarapaceContainer +from opentelemetry import metrics +from opentelemetry.exporter.otlp.proto.grpc.metric_exporter import OTLPMetricExporter +from opentelemetry.sdk.metrics.export import ConsoleMetricExporter, MetricReader, PeriodicExportingMetricReader +from typing import Final + + +class Meter: + START_TIME_KEY: Final = "start_time" + + @staticmethod + @inject + def get_meter(config: Config = Provide[KarapaceContainer.config]) -> metrics.Meter: + return metrics.get_meter_provider().get_meter(f"{config.tags.app}.meter") + + @staticmethod + @inject + def get_metric_reader(config: Config = Provide[KarapaceContainer.config]) -> MetricReader: + exporter = ConsoleMetricExporter() + if config.telemetry.otel_endpoint_url: + exporter = OTLPMetricExporter(endpoint=config.telemetry.otel_endpoint_url) + return PeriodicExportingMetricReader( + exporter=exporter, + export_interval_millis=config.telemetry.metrics_export_interval_milliseconds, + ) diff --git a/tests/conftest.py b/tests/conftest.py index 28f565b30..3f7b6ad9e 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -195,6 +195,7 @@ def fixture_karapace_container() -> KarapaceContainer: modules=[ schema_registry.controller, schema_registry.telemetry.tracer, + schema_registry.telemetry.meter, ] ) return karapace_container diff --git a/tests/unit/schema_registry/telemetry/test_meter.py b/tests/unit/schema_registry/telemetry/test_meter.py new file mode 100644 index 000000000..3a60d62fe --- /dev/null +++ b/tests/unit/schema_registry/telemetry/test_meter.py @@ -0,0 +1,45 @@ +""" +schema_registry - telemetry meter tests + +Copyright (c) 2024 Aiven Ltd +See LICENSE for details +""" + +from karapace.container import KarapaceContainer +from schema_registry.telemetry.meter import Meter +from unittest.mock import patch + + +def test_meter(karapace_container: KarapaceContainer): + with patch("schema_registry.telemetry.meter.metrics") as mock_metrics: + Meter.get_meter(config=karapace_container.config()) + mock_metrics.get_meter_provider.return_value.get_meter.assert_called_once_with("Karapace.meter") + + +def test_get_metric_reader_with_otel_endpoint(karapace_container: KarapaceContainer) -> None: + with ( + patch("schema_registry.telemetry.meter.OTLPMetricExporter") as mock_otlp_exporter, + patch("schema_registry.telemetry.meter.PeriodicExportingMetricReader") as mock_periodic_exporting_metric_reader, + ): + karapace_container.config().telemetry.otel_endpoint_url = "http://otel:4317" + reader = Meter.get_metric_reader(config=karapace_container.config()) + mock_otlp_exporter.assert_called_once_with(endpoint="http://otel:4317") + mock_periodic_exporting_metric_reader.assert_called_once_with( + exporter=mock_otlp_exporter.return_value, + export_interval_millis=10000, + ) + assert reader is mock_periodic_exporting_metric_reader.return_value + + +def test_get_metric_reader_without_otel_endpoint(karapace_container: KarapaceContainer) -> None: + with ( + patch("schema_registry.telemetry.meter.ConsoleMetricExporter") as mock_console_exporter, + patch("schema_registry.telemetry.meter.PeriodicExportingMetricReader") as mock_periodic_exporting_metric_reader, + ): + reader = Meter.get_metric_reader(config=karapace_container.config()) + mock_console_exporter.assert_called_once() + mock_periodic_exporting_metric_reader.assert_called_once_with( + exporter=mock_console_exporter.return_value, + export_interval_millis=10000, + ) + assert reader is mock_periodic_exporting_metric_reader.return_value