1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186
|
The protocol layer
==================
Any protocol is implemented by deriving from :py:class:`~tinyrpc.RPCProtocol`
and implementing all of its members:
.. autoclass:: tinyrpc.RPCProtocol
:members:
These require implementations of the following classes as well:
.. autoclass:: tinyrpc.RPCRequest
:members:
.. autoclass:: tinyrpc.RPCResponse
:members:
.. autoclass:: tinyrpc.BadRequestError
:members:
Every protocol deals with multiple kinds of structures: ``data`` arguments are
always byte strings, either messages or replies, that are sent via or received
from a transport.
There are two protocol-specific subclasses of
:py:class:`~tinyrpc.RPCRequest` and :py:class:`~tinyrpc.RPCResponse`, these
represent well-formed requests and responses.
Finally, if an error occurs during parsing of a request, a
:py:class:`~tinyrpc.BadRequestError` instance must be thrown. These need to be
subclassed for each protocol as well, since they generate error replies.
Batch protocols
---------------
Some protocols may support batch requests. In this case, they need to derive
from :py:class:`~tinyrpc.RPCBatchProtocol`.
Batch protocols differ in that their
:py:func:`~tinyrpc.RPCProtocol.parse_request` method may return an instance of
:py:class:`~tinyrpc.RPCBatchRequest`. They also possess an addional method in
:py:func:`~tinyrpc.RPCBatchProtocol.create_batch_request`.
Handling a batch request is slightly different, while it supports
:py:func:`~tinyrpc.RPCBatchRequest.error_respond`, to make actual responses,
:py:func:`~tinyrpc.RPCBatchRequest.create_batch_response` needs to be used.
No assumptions are made whether or not it is okay for batch requests to be
handled in parallel. This is up to the server/dispatch implementation, which
must be chosen appropriately.
.. autoclass:: tinyrpc.RPCBatchProtocol
:members:
.. autoclass:: tinyrpc.RPCBatchRequest
:members:
.. autoclass:: tinyrpc.RPCBatchResponse
:members:
Supported protocols
-------------------
Any supported protocol is used by instantiating its class and calling the
interface of :py:class:`~tinyrpc.RPCProtocol`. Note that constructors are not
part of the interface, any protocol may have specific arguments for its
instances.
Protocols usually live in their own module because they may need to import
optional modules that needn't be a dependency for all of ``tinyrpc``.
Example
-------
The following example shows how to use the
:py:class:`~tinyrpc.protcols.jsonrpc.JSONRPCProtocol` class in a custom
application, without using any other components:
Server
~~~~~~
.. code-block:: python
from tinyrpc.protocols.jsonrpc import JSONRPCProtocol
from tinyrpc import BadRequestError, RPCBatchRequest
rpc = JSONRPCProtocol()
# the code below is valid for all protocols, not just JSONRPC:
def handle_incoming_message(self, data):
try:
request = rpc.parse_request(data)
except BadRequestError as e:
# request was invalid, directly create response
response = e.error_respond(e)
else:
# we got a valid request
# the handle_request function is user-defined
# and returns some form of response
if hasattr(request, create_batch_response):
response = request.create_batch_response(
handle_request(req) for req in request
)
else:
response = handle_request(request)
# now send the response to the client
if response != None:
send_to_client(response.serialize())
def handle_request(request):
try:
# do magic with method, args, kwargs...
return request.respond(result)
except Exception as e:
# for example, a method wasn't found
return request.error_respond(e)
Client
~~~~~~
.. code-block:: python
from tinyrpc.protocols.jsonrpc import JSONRPCProtocol
rpc = JSONRPCProtocol()
# again, code below is protocol-independent
# assuming you want to call method(*args, **kwargs)
request = rpc.create_request(method, args, kwargs)
reply = send_to_server_and_get_reply(request)
response = rpc.parse_reply(reply)
if hasattr(response, 'error'):
# error handling...
else:
# the return value is found in response.result
do_something_with(response.result)
Another example, this time using batch requests:
.. code-block:: python
# or using batch requests:
requests = rpc.create_batch_request([
rpc.create_request(method_1, args_1, kwargs_1)
rpc.create_request(method_2, args_2, kwargs_2)
# ...
])
reply = send_to_server_and_get_reply(request)
responses = rpc.parse_reply(reply)
for responses in response:
if hasattr(reponse, 'error'):
# ...
Finally, one-way requests are requests where the client does not expect an
answer:
.. code-block:: python
request = rpc.create_request(method, args, kwargs, one_way=True)
send_to_server(request)
# done
JSON-RPC
~~~~~~~~
.. autoclass:: tinyrpc.protocols.jsonrpc.JSONRPCProtocol
:members:
.. _jsonrpc: http://jsonrpc.org
|