|
| 1 | +import logging |
| 2 | +import os |
| 3 | + |
1 | 4 | from fastmcp.server.middleware import Middleware, MiddlewareContext |
2 | | -from opentelemetry import trace |
| 5 | +from opentelemetry import metrics, trace |
| 6 | +from opentelemetry._logs import set_logger_provider |
| 7 | +from opentelemetry.exporter.otlp.proto.grpc._log_exporter import OTLPLogExporter |
| 8 | +from opentelemetry.exporter.otlp.proto.grpc.metric_exporter import OTLPMetricExporter |
| 9 | +from opentelemetry.exporter.otlp.proto.grpc.trace_exporter import OTLPSpanExporter |
| 10 | +from opentelemetry.sdk._logs import LoggerProvider, LoggingHandler # _logs is "experimental", not "private" |
| 11 | +from opentelemetry.sdk._logs.export import BatchLogRecordProcessor |
| 12 | +from opentelemetry.sdk.metrics import MeterProvider |
| 13 | +from opentelemetry.sdk.metrics.export import PeriodicExportingMetricReader |
| 14 | +from opentelemetry.sdk.resources import Resource |
| 15 | +from opentelemetry.sdk.trace import TracerProvider |
| 16 | +from opentelemetry.sdk.trace.export import BatchSpanProcessor |
3 | 17 | from opentelemetry.trace import Status, StatusCode |
4 | 18 |
|
5 | 19 |
|
| 20 | +def configure_aspire_dashboard(service_name: str = "expenses-mcp"): |
| 21 | + """Configure OpenTelemetry to send telemetry to the Aspire standalone dashboard. |
| 22 | +
|
| 23 | + Requires the OTEL_EXPORTER_OTLP_ENDPOINT environment variable to be set. |
| 24 | + """ |
| 25 | + otlp_endpoint = os.getenv("OTEL_EXPORTER_OTLP_ENDPOINT") |
| 26 | + if not otlp_endpoint: |
| 27 | + raise ValueError("OTEL_EXPORTER_OTLP_ENDPOINT environment variable must be set to configure telemetry export.") |
| 28 | + |
| 29 | + # Create resource with service name |
| 30 | + resource = Resource.create({"service.name": service_name}) |
| 31 | + |
| 32 | + # Configure Tracing |
| 33 | + tracer_provider = TracerProvider(resource=resource) |
| 34 | + tracer_provider.add_span_processor(BatchSpanProcessor(OTLPSpanExporter(endpoint=otlp_endpoint))) |
| 35 | + trace.set_tracer_provider(tracer_provider) |
| 36 | + |
| 37 | + # Configure Metrics |
| 38 | + metric_reader = PeriodicExportingMetricReader(OTLPMetricExporter(endpoint=otlp_endpoint)) |
| 39 | + meter_provider = MeterProvider(resource=resource, metric_readers=[metric_reader]) |
| 40 | + metrics.set_meter_provider(meter_provider) |
| 41 | + |
| 42 | + # Configure Logging |
| 43 | + logger_provider = LoggerProvider(resource=resource) |
| 44 | + logger_provider.add_log_record_processor(BatchLogRecordProcessor(OTLPLogExporter(endpoint=otlp_endpoint))) |
| 45 | + set_logger_provider(logger_provider) |
| 46 | + |
| 47 | + # Add logging handler to send Python logs to OTLP |
| 48 | + root_logger = logging.getLogger() |
| 49 | + handler_exists = any( |
| 50 | + isinstance(existing, LoggingHandler) and getattr(existing, "logger_provider", None) is logger_provider |
| 51 | + for existing in root_logger.handlers |
| 52 | + ) |
| 53 | + |
| 54 | + if not handler_exists: |
| 55 | + handler = LoggingHandler(level=logging.NOTSET, logger_provider=logger_provider) |
| 56 | + root_logger.addHandler(handler) |
| 57 | + |
| 58 | + |
6 | 59 | class OpenTelemetryMiddleware(Middleware): |
7 | 60 | """Middleware that creates OpenTelemetry spans for MCP operations.""" |
8 | 61 |
|
|
0 commit comments