gRPC
[gRPC] Python gRPC Server/Client 구현
범범조조
2023. 3. 4. 05:35
참조
소개
- 안녕하세요. 오늘은 Python 에서 gRPC Server/Client 코드 구현하는 방법에 대해서 설명 드리려고 합니다.
- 간단하게 calc.proto 버퍼 하나를 생성하고, Python 코드로 gRPC Server/Cleint 를 구현하여 통신하는 예제 코드를 작성하도록 하겠습니다.
calc.proto 버퍼 작성
- 제일 먼저, calc.proto 프로토콜 버퍼 하나를 생성하고 아래와 같이 내용을 작성합니다.
- 간단히 calc.proto 버퍼 파일 안에는 덧셈, 뺄셈, 곱셈, 나눗셈의 Input, Output 결과값을 Server, Client 에서 알 수 있는 Service와 Message 들을 작성하였습니다.
syntax = "proto3";
package calc;
service Calculator {
rpc Add (AddRequest) returns (AddReply) {}
rpc Substract (SubstractRequest) returns (SubstractReply) {}
rpc Multiply (MultiplyRequest) returns (MeuliplyReply) {}
rpc Divide (DivideRequest) returns (DivideReply) {}
}
message AddRequest {
int32 n1=1;
int32 n2=2;
}
message AddReply {
int32 n1=1;
}
message SubstractRequest {
int32 n1=1;
int32 n2=2;
}
message SubstractReply {
int32 n1=1;
}
message MultiplyRequest {
int32 n1=1;
int32 n2=2;
}
message MeuliplyReply {
int32 n1=1;
}
message DivideRequest {
int32 n1=1;
int32 n2=2;
}
message DivideReply {
float f1=1;
}
proto 컴파일
- 다음으로는 앞서 생성한 calc.proto 버퍼를 python 에서 직접 grpc.proto 를 이용하여 컴파일 할 수 있도록 해야 합니다.
- 해당 작업을 통해 Python 코드로 calc.proto 버퍼 내용의 함수들이 자동으로 생성됩니다.
- calc.proto 컴파일 하는 명령어는 아래와 같습니다.
- 참고로 저는 파이썬\protos\calc.proto 경로에 calc.proto 파일을 생성해 놓았습니다.
- 다음 명령어를 실행하게 되면 calc_pb.py, calc_pb2.grpc.py 2개의 파일이 자동 생성됩니다.
python -m grpc_tools.protoc -I./protos --python_out=. --grpc_python_out=. ./protos/calc.proto
calc_pb2.py
# -*- coding: utf-8 -*-
# Generated by the protocol buffer compiler. DO NOT EDIT!
# source: calc.proto
"""Generated protocol buffer code."""
from google.protobuf import descriptor as _descriptor
from google.protobuf import descriptor_pool as _descriptor_pool
from google.protobuf import message as _message
from google.protobuf import reflection as _reflection
from google.protobuf import symbol_database as _symbol_database
# @@protoc_insertion_point(imports)
_sym_db = _symbol_database.Default()
DESCRIPTOR = _descriptor_pool.Default().AddSerializedFile(b'\n\ncalc.proto\x12\x04\x63\x61lc\"$\n\nAddRequest\x12\n\n\x02n1\x18\x01 \x01(\x05\x12\n\n\x02n2\x18\x02 \x01(\x05\"\x16\n\x08\x41\x64\x64Reply\x12\n\n\x02n1\x18\x01 \x01(\x05\"*\n\x10SubstractRequest\x12\n\n\x02n1\x18\x01 \x01(\x05\x12\n\n\x02n2\x18\x02 \x01(\x05\"\x1c\n\x0eSubstractReply\x12\n\n\x02n1\x18\x01 \x01(\x05\")\n\x0fMultiplyRequest\x12\n\n\x02n1\x18\x01 \x01(\x05\x12\n\n\x02n2\x18\x02 \x01(\x05\"\x1b\n\rMeuliplyReply\x12\n\n\x02n1\x18\x01 \x01(\x05\"\'\n\rDivideRequest\x12\n\n\x02n1\x18\x01 \x01(\x05\x12\n\n\x02n2\x18\x02 \x01(\x05\"\x19\n\x0b\x44ivideReply\x12\n\n\x02n1\x18\x01 \x01(\x05\x32\xe2\x01\n\nCalculator\x12)\n\x03\x41\x64\x64\x12\x10.calc.AddRequest\x1a\x0e.calc.AddReply\"\x00\x12;\n\tSubstract\x12\x16.calc.SubstractRequest\x1a\x14.calc.SubstractReply\"\x00\x12\x38\n\x08Multiply\x12\x15.calc.MultiplyRequest\x1a\x13.calc.MeuliplyReply\"\x00\x12\x32\n\x06\x44ivide\x12\x13.calc.DivideRequest\x1a\x11.calc.DivideReply\"\x00\x62\x06proto3')
_ADDREQUEST = DESCRIPTOR.message_types_by_name['AddRequest']
_ADDREPLY = DESCRIPTOR.message_types_by_name['AddReply']
_SUBSTRACTREQUEST = DESCRIPTOR.message_types_by_name['SubstractRequest']
_SUBSTRACTREPLY = DESCRIPTOR.message_types_by_name['SubstractReply']
_MULTIPLYREQUEST = DESCRIPTOR.message_types_by_name['MultiplyRequest']
_MEULIPLYREPLY = DESCRIPTOR.message_types_by_name['MeuliplyReply']
_DIVIDEREQUEST = DESCRIPTOR.message_types_by_name['DivideRequest']
_DIVIDEREPLY = DESCRIPTOR.message_types_by_name['DivideReply']
AddRequest = _reflection.GeneratedProtocolMessageType('AddRequest', (_message.Message,), {
'DESCRIPTOR' : _ADDREQUEST,
'__module__' : 'calc_pb2'
# @@protoc_insertion_point(class_scope:calc.AddRequest)
})
_sym_db.RegisterMessage(AddRequest)
AddReply = _reflection.GeneratedProtocolMessageType('AddReply', (_message.Message,), {
'DESCRIPTOR' : _ADDREPLY,
'__module__' : 'calc_pb2'
# @@protoc_insertion_point(class_scope:calc.AddReply)
})
_sym_db.RegisterMessage(AddReply)
SubstractRequest = _reflection.GeneratedProtocolMessageType('SubstractRequest', (_message.Message,), {
'DESCRIPTOR' : _SUBSTRACTREQUEST,
'__module__' : 'calc_pb2'
# @@protoc_insertion_point(class_scope:calc.SubstractRequest)
})
_sym_db.RegisterMessage(SubstractRequest)
SubstractReply = _reflection.GeneratedProtocolMessageType('SubstractReply', (_message.Message,), {
'DESCRIPTOR' : _SUBSTRACTREPLY,
'__module__' : 'calc_pb2'
# @@protoc_insertion_point(class_scope:calc.SubstractReply)
})
_sym_db.RegisterMessage(SubstractReply)
MultiplyRequest = _reflection.GeneratedProtocolMessageType('MultiplyRequest', (_message.Message,), {
'DESCRIPTOR' : _MULTIPLYREQUEST,
'__module__' : 'calc_pb2'
# @@protoc_insertion_point(class_scope:calc.MultiplyRequest)
})
_sym_db.RegisterMessage(MultiplyRequest)
MeuliplyReply = _reflection.GeneratedProtocolMessageType('MeuliplyReply', (_message.Message,), {
'DESCRIPTOR' : _MEULIPLYREPLY,
'__module__' : 'calc_pb2'
# @@protoc_insertion_point(class_scope:calc.MeuliplyReply)
})
_sym_db.RegisterMessage(MeuliplyReply)
DivideRequest = _reflection.GeneratedProtocolMessageType('DivideRequest', (_message.Message,), {
'DESCRIPTOR' : _DIVIDEREQUEST,
'__module__' : 'calc_pb2'
# @@protoc_insertion_point(class_scope:calc.DivideRequest)
})
_sym_db.RegisterMessage(DivideRequest)
DivideReply = _reflection.GeneratedProtocolMessageType('DivideReply', (_message.Message,), {
'DESCRIPTOR' : _DIVIDEREPLY,
'__module__' : 'calc_pb2'
# @@protoc_insertion_point(class_scope:calc.DivideReply)
})
_sym_db.RegisterMessage(DivideReply)
_CALCULATOR = DESCRIPTOR.services_by_name['Calculator']
if _descriptor._USE_C_DESCRIPTORS == False:
DESCRIPTOR._options = None
_ADDREQUEST._serialized_start=20
_ADDREQUEST._serialized_end=56
_ADDREPLY._serialized_start=58
_ADDREPLY._serialized_end=80
_SUBSTRACTREQUEST._serialized_start=82
_SUBSTRACTREQUEST._serialized_end=124
_SUBSTRACTREPLY._serialized_start=126
_SUBSTRACTREPLY._serialized_end=154
_MULTIPLYREQUEST._serialized_start=156
_MULTIPLYREQUEST._serialized_end=197
_MEULIPLYREPLY._serialized_start=199
_MEULIPLYREPLY._serialized_end=226
_DIVIDEREQUEST._serialized_start=228
_DIVIDEREQUEST._serialized_end=267
_DIVIDEREPLY._serialized_start=269
_DIVIDEREPLY._serialized_end=294
_CALCULATOR._serialized_start=297
_CALCULATOR._serialized_end=523
# @@protoc_insertion_point(module_scope)
calc_pb2_grpc.py
# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT!
"""Client and server classes corresponding to protobuf-defined services."""
import grpc
import calc_pb2 as calc__pb2
class CalculatorStub(object):
"""Missing associated documentation comment in .proto file."""
def __init__(self, channel):
"""Constructor.
Args:
channel: A grpc.Channel.
"""
self.Add = channel.unary_unary(
'/calc.Calculator/Add',
request_serializer=calc__pb2.AddRequest.SerializeToString,
response_deserializer=calc__pb2.AddReply.FromString,
)
self.Substract = channel.unary_unary(
'/calc.Calculator/Substract',
request_serializer=calc__pb2.SubstractRequest.SerializeToString,
response_deserializer=calc__pb2.SubstractReply.FromString,
)
self.Multiply = channel.unary_unary(
'/calc.Calculator/Multiply',
request_serializer=calc__pb2.MultiplyRequest.SerializeToString,
response_deserializer=calc__pb2.MeuliplyReply.FromString,
)
self.Divide = channel.unary_unary(
'/calc.Calculator/Divide',
request_serializer=calc__pb2.DivideRequest.SerializeToString,
response_deserializer=calc__pb2.DivideReply.FromString,
)
class CalculatorServicer(object):
"""Missing associated documentation comment in .proto file."""
def Add(self, request, context):
"""Missing associated documentation comment in .proto file."""
context.set_code(grpc.StatusCode.UNIMPLEMENTED)
context.set_details('Method not implemented!')
raise NotImplementedError('Method not implemented!')
def Substract(self, request, context):
"""Missing associated documentation comment in .proto file."""
context.set_code(grpc.StatusCode.UNIMPLEMENTED)
context.set_details('Method not implemented!')
raise NotImplementedError('Method not implemented!')
def Multiply(self, request, context):
"""Missing associated documentation comment in .proto file."""
context.set_code(grpc.StatusCode.UNIMPLEMENTED)
context.set_details('Method not implemented!')
raise NotImplementedError('Method not implemented!')
def Divide(self, request, context):
"""Missing associated documentation comment in .proto file."""
context.set_code(grpc.StatusCode.UNIMPLEMENTED)
context.set_details('Method not implemented!')
raise NotImplementedError('Method not implemented!')
def add_CalculatorServicer_to_server(servicer, server):
rpc_method_handlers = {
'Add': grpc.unary_unary_rpc_method_handler(
servicer.Add,
request_deserializer=calc__pb2.AddRequest.FromString,
response_serializer=calc__pb2.AddReply.SerializeToString,
),
'Substract': grpc.unary_unary_rpc_method_handler(
servicer.Substract,
request_deserializer=calc__pb2.SubstractRequest.FromString,
response_serializer=calc__pb2.SubstractReply.SerializeToString,
),
'Multiply': grpc.unary_unary_rpc_method_handler(
servicer.Multiply,
request_deserializer=calc__pb2.MultiplyRequest.FromString,
response_serializer=calc__pb2.MeuliplyReply.SerializeToString,
),
'Divide': grpc.unary_unary_rpc_method_handler(
servicer.Divide,
request_deserializer=calc__pb2.DivideRequest.FromString,
response_serializer=calc__pb2.DivideReply.SerializeToString,
),
}
generic_handler = grpc.method_handlers_generic_handler(
'calc.Calculator', rpc_method_handlers)
server.add_generic_rpc_handlers((generic_handler,))
# This class is part of an EXPERIMENTAL API.
class Calculator(object):
"""Missing associated documentation comment in .proto file."""
@staticmethod
def Add(request,
target,
options=(),
channel_credentials=None,
call_credentials=None,
insecure=False,
compression=None,
wait_for_ready=None,
timeout=None,
metadata=None):
return grpc.experimental.unary_unary(request, target, '/calc.Calculator/Add',
calc__pb2.AddRequest.SerializeToString,
calc__pb2.AddReply.FromString,
options, channel_credentials,
insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
@staticmethod
def Substract(request,
target,
options=(),
channel_credentials=None,
call_credentials=None,
insecure=False,
compression=None,
wait_for_ready=None,
timeout=None,
metadata=None):
return grpc.experimental.unary_unary(request, target, '/calc.Calculator/Substract',
calc__pb2.SubstractRequest.SerializeToString,
calc__pb2.SubstractReply.FromString,
options, channel_credentials,
insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
@staticmethod
def Multiply(request,
target,
options=(),
channel_credentials=None,
call_credentials=None,
insecure=False,
compression=None,
wait_for_ready=None,
timeout=None,
metadata=None):
return grpc.experimental.unary_unary(request, target, '/calc.Calculator/Multiply',
calc__pb2.MultiplyRequest.SerializeToString,
calc__pb2.MeuliplyReply.FromString,
options, channel_credentials,
insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
@staticmethod
def Divide(request,
target,
options=(),
channel_credentials=None,
call_credentials=None,
insecure=False,
compression=None,
wait_for_ready=None,
timeout=None,
metadata=None):
return grpc.experimental.unary_unary(request, target, '/calc.Calculator/Divide',
calc__pb2.DivideRequest.SerializeToString,
calc__pb2.DivideReply.FromString,
options, channel_credentials,
insecure, call_credentials, compression, wait_for_ready, timeout, metadata)
- 이제 Python 으로 gRPC Server/Client 코드를 작성해 보도록 하겠습니다.
Server
from concurrent import futures
import time
import grpc
import calc_pb2
import calc_pb2_grpc
class Calculator(calc_pb2_grpc.CalculatorServicer):
def Add(self, request, context):
return calc_pb2.AddReply(n1=request.n1 + request.n2)
def Substract(self, request, context):
return calc_pb2.SubstractReply(n1=request.n1 - request.n2)
def Multiply(self, request, context):
return calc_pb2.MeuliplyReply(n1=request.n1 * request.n2)
def Divide(self, request, context):
return calc_pb2.DivideReply(f1=request.n1 / request.n2)
def serve():
print("Server Start..")
server = grpc.server(futures.ThreadPoolExecutor(max_workers=10))
calc_pb2_grpc.add_CalculatorServicer_to_server(Calculator(), server)
server.add_insecure_port('[::]:50050')
server.start()
server.wait_for_termination()
if __name__ == '__main__':
serve()
Client
import grpc
import calc_pb2
import calc_pb2_grpc
def run():
channel = grpc.insecure_channel('localhost:50050')
stub = calc_pb2_grpc.CalculatorStub(channel)
respones = stub.Add(calc_pb2.AddRequest(n1=20, n2=10))
print(respones.n1)
respones = stub.Substract(calc_pb2.SubstractRequest(n1=20, n2=10))
print(respones.n1)
respones = stub.Multiply(calc_pb2.MultiplyRequest(n1=20, n2=10))
print(respones.n1)
respones = stub.Divide(calc_pb2.DivideRequest(n1=10, n2=2))
print(respones.f1)
if __name__ == '__main__':
run()
서버 실행
- 서버는 cmd 로 실행하도록 하겠습니다.
- calc_server.py 가 있는 경로로 cmd 실행하고,
python calc_server.py
명령어를 입력하여 Server 를 실행합니다.
C:\Users\Desktop\파이썬>python calc_server.py
Server Start..
클라이언트 실행
- 다음과 같이 덧셈, 뺄셈, 곱셈, 나눗셈의 값이 서버를 통해 계산되어 결과값이 반환되어 출려된 것을 확인할 수 있습니다.
30
10
200
5.0
728x90