|
1 | 1 | """Utilities for collecting and reporting performance metrics in the SDK.""" |
2 | 2 |
|
3 | 3 | import logging |
| 4 | +import os |
4 | 5 | import time |
5 | 6 | import uuid |
6 | 7 | from dataclasses import dataclass, field |
7 | | -from typing import Any, Dict, Iterable, List, Optional, Set, Tuple |
| 8 | +from importlib.metadata import version |
| 9 | +from typing import Any, Dict, Iterable, List, Optional, Sequence, Set, Tuple |
| 10 | + |
| 11 | +import opentelemetry.metrics as metrics_api |
| 12 | +import opentelemetry.sdk.metrics as metrics_sdk |
| 13 | +from opentelemetry.exporter.otlp.proto.http.metric_exporter import OTLPMetricExporter |
| 14 | +from opentelemetry.sdk.metrics.export import ( |
| 15 | + AggregationTemporality, |
| 16 | + MetricReader, |
| 17 | + PeriodicExportingMetricReader, |
| 18 | +) |
| 19 | +from opentelemetry.sdk.resources import Resource |
8 | 20 |
|
9 | 21 | from ..types.content import Message |
10 | 22 | from ..types.streaming import Metrics, Usage |
@@ -355,3 +367,80 @@ def metrics_to_string(event_loop_metrics: EventLoopMetrics, allowed_names: Optio |
355 | 367 | A formatted string representation of the metrics. |
356 | 368 | """ |
357 | 369 | return "\n".join(_metrics_summary_to_lines(event_loop_metrics, allowed_names or set())) |
| 370 | + |
| 371 | + |
| 372 | +class Meter: |
| 373 | + """Handles OpenTelemetry metrics. |
| 374 | +
|
| 375 | + This class provides a simple interface for creating and managing metrics, |
| 376 | + with support for sending to OTLP endpoints. |
| 377 | +
|
| 378 | + When the OTEL_EXPORTER_OTLP_ENDPOINT environment variable is set, metrics |
| 379 | + are sent to the OTLP endpoint. |
| 380 | + """ |
| 381 | + |
| 382 | + def __init__( |
| 383 | + self, |
| 384 | + service_name: str = "strands-agents", |
| 385 | + ) -> None: |
| 386 | + """Initialize the meter with the given service name. |
| 387 | +
|
| 388 | + Args: |
| 389 | + service_name: The name of the service for which metrics are being collected. |
| 390 | + """ |
| 391 | + self.service_name = service_name |
| 392 | + self.meter_provider: Optional[metrics_api.MeterProvider] = None |
| 393 | + self.meter: Optional[metrics_api.Meter] = None |
| 394 | + env_metrics_exporter = os.environ.get("OTEL_METRICS_EXPORTER") |
| 395 | + if env_metrics_exporter and "none" == env_metrics_exporter: |
| 396 | + logger.info("OTEL_METRICS_EXPORTER set to 'none', using NoOpMeterProvider") |
| 397 | + self.meter_provider = metrics_api.NoOpMeterProvider() |
| 398 | + self.meter = self.meter_provider.get_meter(self.service_name) |
| 399 | + return |
| 400 | + if self._is_meter_initialized(): |
| 401 | + self.meter_provider = metrics_api.get_meter_provider() |
| 402 | + self.meter = self.meter_provider.get_meter(self.service_name) |
| 403 | + return |
| 404 | + self._initialize_meter() |
| 405 | + |
| 406 | + def _initialize_meter(self) -> None: |
| 407 | + """Initialize the OpenTelemetry meter.""" |
| 408 | + logger.info("initializing meter") |
| 409 | + |
| 410 | + # Create resource with service information |
| 411 | + resource = Resource.create( |
| 412 | + { |
| 413 | + "service.name": self.service_name, |
| 414 | + "service.version": version("strands-agents"), |
| 415 | + "telemetry.sdk.name": "opentelemetry", |
| 416 | + "telemetry.sdk.language": "python", |
| 417 | + } |
| 418 | + ) |
| 419 | + |
| 420 | + # note: it is a concrete implementation from `opentelemetry.sdk` |
| 421 | + metric_readers = self._create_metric_readers() |
| 422 | + self.meter_provider = metrics_sdk.MeterProvider(resource=resource, metric_readers=metric_readers) |
| 423 | + |
| 424 | + # initialize global meter provider |
| 425 | + metrics_api.set_meter_provider(self.meter_provider) |
| 426 | + self.meter = metrics_api.get_meter(self.service_name) |
| 427 | + |
| 428 | + def _is_meter_initialized(self) -> bool: |
| 429 | + meter_provider = metrics_api.get_meter_provider() |
| 430 | + return isinstance(meter_provider, metrics_sdk.MeterProvider) |
| 431 | + |
| 432 | + def _create_metric_readers(self) -> Sequence[MetricReader]: |
| 433 | + """MetricReaders define how metrics are emitted by OTEL.""" |
| 434 | + # export metrics in batches at designated intervals |
| 435 | + periodic_metric_reader = PeriodicExportingMetricReader( |
| 436 | + exporter=OTLPMetricExporter( |
| 437 | + preferred_temporality={ |
| 438 | + # Prefer 'DELTA' temporality to avoid storing previous measurements |
| 439 | + # https://opentelemetry.io/docs/specs/otel/metrics/data-model/#temporality |
| 440 | + metrics_sdk.Counter: AggregationTemporality.DELTA, |
| 441 | + metrics_sdk.UpDownCounter: AggregationTemporality.DELTA, |
| 442 | + metrics_sdk.Histogram: AggregationTemporality.DELTA, |
| 443 | + } |
| 444 | + ), |
| 445 | + ) |
| 446 | + return [periodic_metric_reader] |
0 commit comments