Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

python gRPC requiring a context argument

Tags:

python

grpc

I am just getting started with gRPC and have tried to implement a simple python service after going through the starter guides. But when I invoke my client call python is asking for a context argument to be provided. Why does my code need to supply a context object when it was not needed in the examples?

P.S. I started trying to create my own concrete context subclass but wasn't sure how it should be implemented. I've added my start to it but if possible would really appreciate an example to work from

Thanks!

Protofile

syntax = "proto2";

package parsefile;

service ParseFile {
  rpc SendFile (File) returns (Empty) {}
}

message File {

  message MetaData {
    optional string file_name = 1;
    optional string file_path = 2 [default = '.'];
    optional string mime_type = 3 [default = 'application/pdf'];
  }

  message Source {
    optional string title = 1;
    optional int32 id = 2;
  }

  optional MetaData document = 1;
  optional Source supplier = 2;
}

message Empty {
}

Server

from concurrent import futures
import time

import grpc

import parsefile_pb2_grpc
import parsefile_pb2

_ONE_DAY_IN_SECONDS = 60 * 60 * 24

class ParseFileServicer(parsefile_pb2_grpc.ParseFileServicer):

    def SendFile(self, request, context):
        supplier = request.supplier.title
        file_name = request.document.file_name
        print('Received {} from {}'.format(file_name, supplier))
        return parsefile_pb2.Empty()

def serve():
    server = grpc.server(futures.ThreadPoolExecutor(max_workers=2))
    parsefile_pb2_grpc.add_ParseFileServicer_to_server(
        ParseFileServicer, server)
    server.add_insecure_port('[::]:50051')
    server.start()
    try:
        while True:
            time.sleep(_ONE_DAY_IN_SECONDS)
    except KeyboardInterrupt:
        server.stop(0)

if __name__ == '__main__':
    serve()

Client

import grpc

import parsefile_pb2_grpc
import parsefile_pb2

def get_file_info():
    return parsefile_pb2.File(
        document = parsefile_pb2.File.MetaData(
            file_name = 'example.txt'
        ),
        supplier = parsefile_pb2.File.Source(
            title = 'Example Supplier'
        )
    )

def run():
    channel = grpc.insecure_channel('localhost:50051')
    stub = parsefile_pb2_grpc.ParseFileStub(channel)
    context = RequestContext()
    print('object created')
    response = stub.SendFile(get_file_info())
    print('File info sent to server')

if __name__ == '__main__':
    run()

Error Trace

Traceback (most recent call last):   File "parse_client.py", line 60, in <module>
    run()   File "parse_client.py", line 56, in run
    response = stub.SendFile(get_file_info(), 2)   File "/Users/davidbowe/.virtualenvs/post/lib/python3.6/site-packages/grpc/_channel.py", line 507, in __call__
    return _end_unary_response_blocking(state, call, False, deadline)   File "/Users/davidbowe/.virtualenvs/post/lib/python3.6/site-packages/grpc/_channel.py", line 455, in _end_unary_response_bl ocking
    raise _Rendezvous(state, None, None, deadline) grpc._channel._Rendezvous: <_Rendezvous of RPC that terminated with (StatusCode.UNKNOWN, Exception calling application: SendF ile() missing 1 required positional argument: 'context')>
like image 420
user7815551 Avatar asked Apr 04 '17 18:04

user7815551


People also ask

What is gRPC context?

A context propagation mechanism which can carry scoped-values across API boundaries and between threads.

Is Python gRPC multithreaded?

gRPC Python wraps gRPC core, which uses multithreading for performance, and hence doesn't support fork() .


1 Answers

In your server code you are missing the brackets to instatiate the servicer object. You've written...

parsefile_pb2_grpc.add_ParseFileServicer_to_server(ParseFileServicer, server)

It should be:

parsefile_pb2_grpc.add_ParseFileServicer_to_server(ParseFileServicer(), server)
like image 177
Tiki Avatar answered Sep 26 '22 02:09

Tiki