| # Copyright 2026 gRPC authors. |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| """The Python AsyncIO example of the GRPC helloworld.Greeter server with observability enabled.""" |
| |
| import asyncio |
| import collections |
| import faulthandler |
| import logging |
| |
| import grpc |
| import grpc_observability |
| import helloworld_pb2 |
| import helloworld_pb2_grpc |
| import open_telemetry_exporter |
| from opentelemetry.sdk import metrics as otel_metrics |
| from opentelemetry.sdk.metrics import export as otel_metrics_export |
| |
| _OTEL_EXPORT_INTERVAL_S = 0.5 |
| _SERVER_PORT = "50051" |
| |
| |
| class Greeter(helloworld_pb2_grpc.GreeterServicer): |
| async def SayHello(self, request, context): |
| message = request.name |
| return helloworld_pb2.HelloReply(message=f"Hello {message}") |
| |
| |
| async def serve(): |
| all_metrics = collections.defaultdict(list) |
| otel_exporter = open_telemetry_exporter.OTelMetricExporter( |
| all_metrics, print_live=False |
| ) |
| reader = otel_metrics_export.PeriodicExportingMetricReader( |
| exporter=otel_exporter, |
| export_interval_millis=_OTEL_EXPORT_INTERVAL_S * 1000, |
| ) |
| provider = otel_metrics.MeterProvider(metric_readers=(reader,)) |
| |
| otel_plugin = grpc_observability.OpenTelemetryPlugin( |
| meter_provider=provider |
| ) |
| otel_plugin.register_global() |
| |
| server = grpc.aio.server() |
| helloworld_pb2_grpc.add_GreeterServicer_to_server(Greeter(), server) |
| server.add_insecure_port("[::]:" + _SERVER_PORT) |
| await server.start() |
| print("Server started, listening on " + _SERVER_PORT) |
| |
| # Wait for the client to send an RPC, and for metrics to export. |
| await asyncio.sleep(10) |
| print("Metrics exported on Server side:") |
| for metric in all_metrics: |
| print(metric) |
| |
| await server.stop(0) |
| otel_plugin.deregister_global() |
| provider.shutdown() |
| |
| |
| if __name__ == "__main__": |
| faulthandler.enable() |
| logging.basicConfig() |
| asyncio.run(serve()) |