4
4
import logging
5
5
import uuid
6
6
import sys
7
+ import asyncio
7
8
9
+ from threading import Thread
8
10
from concurrent import futures
9
11
from .exceptions import (JsonRpcException , JsonRpcRequestCancelled ,
10
12
JsonRpcInternalError , JsonRpcMethodNotFound )
14
16
CANCEL_METHOD = '$/cancelRequest'
15
17
16
18
19
+ async def run_as_daemon (func , * args ):
20
+ future = futures .Future ()
21
+ future .set_running_or_notify_cancel ()
22
+
23
+ # A bug in python 3.7 makes it a bad idea to set a BaseException
24
+ # in a wrapped future (see except statement in asyncio.Task.__wakeup)
25
+ # Instead, we'll wrap base exceptions into exceptions and unwrap them
26
+ # on the other side of the call.
27
+ class BaseExceptionWrapper (Exception ):
28
+ pass
29
+
30
+ def daemon ():
31
+ try :
32
+ result = func (* args )
33
+ except Exception as e :
34
+ future .set_exception (e )
35
+ except BaseException as e :
36
+ future .set_exception (BaseExceptionWrapper (e ))
37
+ else :
38
+ future .set_result (result )
39
+
40
+ Thread (target = daemon , daemon = True ).start ()
41
+ try :
42
+ return await asyncio .wrap_future (future )
43
+ except BaseExceptionWrapper as exc :
44
+ raise exc .args [0 ]
45
+
46
+
17
47
class Endpoint :
18
48
19
49
def __init__ (self , dispatcher , consumer , id_generator = lambda : str (uuid .uuid4 ()), max_workers = 5 ):
@@ -35,6 +65,19 @@ def __init__(self, dispatcher, consumer, id_generator=lambda: str(uuid.uuid4()),
35
65
self ._client_request_futures = {}
36
66
self ._server_request_futures = {}
37
67
self ._executor_service = futures .ThreadPoolExecutor (max_workers = max_workers )
68
+ self ._cancelledRequests = set ()
69
+
70
+ def init_async (self ):
71
+ log .warning ("init async" )
72
+ self ._messageQueue = asyncio .Queue ()
73
+
74
+ async def consume_task (self ):
75
+ log .warning ("starting task" )
76
+ while True :
77
+ message = await self ._messageQueue .get ()
78
+ await run_as_daemon (self .consume , message )
79
+ log .warning ("got message in task" )
80
+ self ._messageQueue .task_done ()
38
81
39
82
def shutdown (self ):
40
83
self ._executor_service .shutdown ()
@@ -94,7 +137,15 @@ def callback(future):
94
137
future .set_exception (JsonRpcRequestCancelled ())
95
138
return callback
96
139
140
+ async def consume_async (self , message ):
141
+ log .warning ("got message put in queue" )
142
+ if message ['method' ] == CANCEL_METHOD :
143
+ self ._cancelledRequests .add (message .get ('params' )['id' ])
144
+ await self ._messageQueue .put (message )
145
+
146
+
97
147
def consume (self , message ):
148
+ log .warning ("consume message" )
98
149
"""Consume a JSON RPC message from the client.
99
150
100
151
Args:
@@ -182,6 +233,9 @@ def _handle_request(self, msg_id, method, params):
182
233
except KeyError as e :
183
234
raise JsonRpcMethodNotFound .of (method ) from e
184
235
236
+ if msg_id in self ._cancelledRequests :
237
+ raise JsonRpcRequestCancelled ()
238
+
185
239
handler_result = handler (params )
186
240
187
241
if callable (handler_result ):
0 commit comments