1*cc02d7e2SAndroid Build Coastguard Worker# Copyright 2021 The gRPC Authors 2*cc02d7e2SAndroid Build Coastguard Worker# 3*cc02d7e2SAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License"); 4*cc02d7e2SAndroid Build Coastguard Worker# you may not use this file except in compliance with the License. 5*cc02d7e2SAndroid Build Coastguard Worker# You may obtain a copy of the License at 6*cc02d7e2SAndroid Build Coastguard Worker# 7*cc02d7e2SAndroid Build Coastguard Worker# http://www.apache.org/licenses/LICENSE-2.0 8*cc02d7e2SAndroid Build Coastguard Worker# 9*cc02d7e2SAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software 10*cc02d7e2SAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS, 11*cc02d7e2SAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12*cc02d7e2SAndroid Build Coastguard Worker# See the License for the specific language governing permissions and 13*cc02d7e2SAndroid Build Coastguard Worker# limitations under the License. 14*cc02d7e2SAndroid Build Coastguard Worker"""A flaky backend for the gRPC Python retry example.""" 15*cc02d7e2SAndroid Build Coastguard Worker 16*cc02d7e2SAndroid Build Coastguard Workerimport asyncio 17*cc02d7e2SAndroid Build Coastguard Workerimport collections 18*cc02d7e2SAndroid Build Coastguard Workerimport logging 19*cc02d7e2SAndroid Build Coastguard Workerimport random 20*cc02d7e2SAndroid Build Coastguard Worker 21*cc02d7e2SAndroid Build Coastguard Workerimport grpc 22*cc02d7e2SAndroid Build Coastguard Worker 23*cc02d7e2SAndroid Build Coastguard Workerhelloworld_pb2, helloworld_pb2_grpc = grpc.protos_and_services( 24*cc02d7e2SAndroid Build Coastguard Worker "helloworld.proto" 25*cc02d7e2SAndroid Build Coastguard Worker) 26*cc02d7e2SAndroid Build Coastguard Worker 27*cc02d7e2SAndroid Build Coastguard Worker 28*cc02d7e2SAndroid Build Coastguard Workerclass ErrorInjectingGreeter(helloworld_pb2_grpc.GreeterServicer): 29*cc02d7e2SAndroid Build Coastguard Worker def __init__(self): 30*cc02d7e2SAndroid Build Coastguard Worker self._counter = collections.defaultdict(int) 31*cc02d7e2SAndroid Build Coastguard Worker 32*cc02d7e2SAndroid Build Coastguard Worker async def SayHello( 33*cc02d7e2SAndroid Build Coastguard Worker self, 34*cc02d7e2SAndroid Build Coastguard Worker request: helloworld_pb2.HelloRequest, 35*cc02d7e2SAndroid Build Coastguard Worker context: grpc.aio.ServicerContext, 36*cc02d7e2SAndroid Build Coastguard Worker ) -> helloworld_pb2.HelloReply: 37*cc02d7e2SAndroid Build Coastguard Worker self._counter[context.peer()] += 1 38*cc02d7e2SAndroid Build Coastguard Worker if self._counter[context.peer()] < 5: 39*cc02d7e2SAndroid Build Coastguard Worker if random.random() < 0.75: 40*cc02d7e2SAndroid Build Coastguard Worker logging.info("Injecting error to RPC from %s", context.peer()) 41*cc02d7e2SAndroid Build Coastguard Worker await context.abort( 42*cc02d7e2SAndroid Build Coastguard Worker grpc.StatusCode.UNAVAILABLE, "injected error" 43*cc02d7e2SAndroid Build Coastguard Worker ) 44*cc02d7e2SAndroid Build Coastguard Worker logging.info("Successfully responding to RPC from %s", context.peer()) 45*cc02d7e2SAndroid Build Coastguard Worker return helloworld_pb2.HelloReply(message="Hello, %s!" % request.name) 46*cc02d7e2SAndroid Build Coastguard Worker 47*cc02d7e2SAndroid Build Coastguard Worker 48*cc02d7e2SAndroid Build Coastguard Workerasync def serve() -> None: 49*cc02d7e2SAndroid Build Coastguard Worker server = grpc.aio.server() 50*cc02d7e2SAndroid Build Coastguard Worker helloworld_pb2_grpc.add_GreeterServicer_to_server( 51*cc02d7e2SAndroid Build Coastguard Worker ErrorInjectingGreeter(), server 52*cc02d7e2SAndroid Build Coastguard Worker ) 53*cc02d7e2SAndroid Build Coastguard Worker listen_addr = "[::]:50051" 54*cc02d7e2SAndroid Build Coastguard Worker server.add_insecure_port(listen_addr) 55*cc02d7e2SAndroid Build Coastguard Worker logging.info("Starting flaky server on %s", listen_addr) 56*cc02d7e2SAndroid Build Coastguard Worker await server.start() 57*cc02d7e2SAndroid Build Coastguard Worker await server.wait_for_termination() 58*cc02d7e2SAndroid Build Coastguard Worker 59*cc02d7e2SAndroid Build Coastguard Worker 60*cc02d7e2SAndroid Build Coastguard Workerif __name__ == "__main__": 61*cc02d7e2SAndroid Build Coastguard Worker logging.basicConfig(level=logging.INFO) 62*cc02d7e2SAndroid Build Coastguard Worker asyncio.run(serve()) 63