| # 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. |
| """gRPC Python AsyncIO helloworld.Greeter client with observability enabled.""" |
| |
| import asyncio |
| import collections |
| 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 |
| |
| |
| async def run() -> None: |
| all_metrics = collections.defaultdict(list) |
| otel_exporter = open_telemetry_exporter.OTelMetricExporter(all_metrics) |
| 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() |
| |
| async with grpc.aio.insecure_channel(target="localhost:50051") as channel: |
| stub = helloworld_pb2_grpc.GreeterStub(channel) |
| try: |
| response = await stub.SayHello( |
| helloworld_pb2.HelloRequest(name="AsyncWorld") |
| ) |
| print(f"Greeter client received: {response.message}") |
| except grpc.RpcError as rpc_error: |
| print("Call failed with code: ", rpc_error.code()) |
| otel_plugin.deregister_global() |
| |
| # Shutdown the provider to force a flush of all metrics. |
| provider.shutdown() |
| |
| print("Metrics exported on client side:") |
| for metric in all_metrics: |
| print(metric) |
| |
| |
| if __name__ == "__main__": |
| logging.basicConfig() |
| asyncio.run(run()) |