gRPC can use protocol buffers as both its Interface Definition Language (IDL)
- https://developers.google.com/protocol-buffers/docs/proto3
- On the server side, the server implements this interface and runs a gRPC server to handle client calls.
- On the client side, the client has a stub (referred to as just a client in some languages) that provides the same methods as the server.
By default, gRPC uses Protocol Buffers , Google’s mature open source mechanism for serializing structured data (although it can be used with other data formats such as JSON).
-
define the structure for the data you want to serialize in a .proto file
Protocol buffer data is structured as messages, where each message is a small logical record of information containing a series of name-value pairs called fields. Here’s a simple example:
message Person {
string name = 1;
int32 id = 2;
bool has_ponycopter = 3;
}
- use the protocol buffer compiler
protoc
to generate data access classes in your preferred language(s) from your proto definition. if your chosen language is Python, running the compiler on the example above will generate a class calledPerson
.
$ virtualenv venv
$ source ./venv/bin/activate
$ pip install -r requirements.txt
gRPC is based around the idea of defining a service, specifying the methods that can be called remotely with their parameters and return types.
그러니까 service에 gPRC server와 client가 request, response를 할 수 있는 method를 정의하는 것이다.
service HelloService {
rpc SayHello (HelloRequest)returns (HelloResponse);
}
message yesHelloRequest {
string greeting = 1;
}
message HelloResponse {
string reply = 1;
}
- client가 single request를 보내고, server가 그 요청을 받아서 single response를 보내는 식의 service method는
rpc SayHello(HelloRequest) returns (HelloResponse);
이런 식으로 작성된다. - 반면 A response-streaming RPC where the client sends a request to the server and gets a stream to read a sequence of messages back. The client reads from the returned stream until there are no more messages. As you can see in the example, you specify a response-streaming method by placing the stream keyword before the response type. stream일 때는
rpc LotsOfReplies(HelloRequest) returns (stream HelloResponse);
이런 식으로 작성한다.
$ python -m grpc_tools.protoc -I./protos --python_out=. --grpc_python_out=. ./protos/helloworld.proto
This regenerates
helloworld_pb2.py
: contains our generated request and response classeshelloworld_pb2_grpc.py
: contains our generated client and server classes.- classes for the messages defined in helloworld.proto
- classes for the service defined in helloworld.proto
class GreeterStub(object):
, which can be used by clients to invoke RouteGuide RPCsclass GreeterServicer(object):
, which defines the interface for implementations of the RouteGuide service
- a function for the service defined in helloworld.proto
def add_GreeterServicer_to_server(servicer, server):
, which adds a RouteGuideServicer to a grpc.Server
three primary elements are generated:
Stub
:class GreeterStub(object):
used by the client to connect to a gRPC service.stub = helloworld_pb2_grpc.GreeterStub(channel)
Servicer
:class GreeterServicer(object):
used by the server to implement a gRPC service.class Greeter(helloworld_pb2_grpc.GreeterServicer):
Registration
Function:add_GreeterServicer_to_server
function used to register a servicer with a grpc.Server object.
def serve():
server = grpc.server(futures.ThreadPoolExecutor(max_workers=10))
helloworld_pb2_grpc.add_GreeterServicer_to_server(Greeter(), server)
server.add_insecure_port('[::]:50051')
server.start()
server.wait_for_termination()
The Python implementation of the GRPC helloworld.Greeter server.
- https://developers.google.com/protocol-buffers/docs/reference/python-generated
- https://grpc.io/docs/languages/python/basics/
- https://github.com/betandr/grpcdemo
- https://www.vipmind.me/programing/python/write-unit-test-for-grpc-with-pytest-and-pytest-grpc.html
- https://grpc.github.io/grpc/python/grpc_testing.html#
- https://stackoverflow.com/questions/44718078/how-to-write-a-grpc-python-unittest
- https://kibua20.tistory.com/226
$ python <your_test_cases.py>
# or
$ python3 -m unittest <your_test_cases.py>
# or
# test code discovery
python -m unittest discover -s <project_directory> -p <"*_test.py" 파일명 Pattern>
Run the server:
$ python greeter_server.py
From another terminal, run the client:
$ python greeter_client.py
Result
Greeter client received: Hello, you!
Greeter client received: Hello again, you!
Length of `Name` cannot be more than 10 characters
INVALID_ARGUMENT
(3, 'invalid argument')
Looking for features between 40, -75 and 42, -73
Feature called Patriots Path, Mendham, NJ 07945, USA at latitude: 407838351
longitude: -746143763
Feature called 101 New Jersey 10, Whippany, NJ 07981, USA at latitude: 408122808
longitude: -743999179
Feature called U.S. 6, Shohola, PA 18458, USA at latitude: 413628156
longitude: -749015468
Feature called 5 Conners Road, Kingston, NY 12401, USA at latitude: 419999544
longitude: -740371136
Feature called Mid Hudson Psychiatric Center, New Hampton, NY 10958, USA at latitude: 414008389
longitude: -743951297
Feature called 287 Flugertown Road, Livingston Manor, NY 12758, USA at latitude: 419611318
longitude: -746524769
Feature called 4001 Tremley Point Road, Linden, NJ 07036, USA at latitude: 406109563
longitude: -742186778
.....
...