-
Notifications
You must be signed in to change notification settings - Fork 22
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
add async interface #12
base: develop
Are you sure you want to change the base?
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -4,6 +4,7 @@ | |
import logging | ||
import uuid | ||
import sys | ||
import asyncio | ||
|
||
from concurrent import futures | ||
from .exceptions import (JsonRpcException, JsonRpcRequestCancelled, | ||
|
@@ -12,6 +13,7 @@ | |
log = logging.getLogger(__name__) | ||
JSONRPC_VERSION = '2.0' | ||
CANCEL_METHOD = '$/cancelRequest' | ||
EXIT_METHOD = 'exit' | ||
|
||
|
||
class Endpoint: | ||
|
@@ -35,9 +37,24 @@ def __init__(self, dispatcher, consumer, id_generator=lambda: str(uuid.uuid4()), | |
self._client_request_futures = {} | ||
self._server_request_futures = {} | ||
self._executor_service = futures.ThreadPoolExecutor(max_workers=max_workers) | ||
self._cancelledRequests = set() | ||
self._messageQueue = None | ||
self._consume_task = None | ||
|
||
def init_async(self): | ||
self._messageQueue = asyncio.Queue() | ||
self._consume_task = asyncio.create_task(self.consume_task()) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Will this task run on the executor? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This task will not run in an executor from https://docs.python.org/3/library/asyncio-eventloop.html#asyncio.loop.run_in_executor, but needs to be started after the event loop was started. I pushed an example implementation to davschul/python-lsp-server@9f0f03d There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The infinite loop has potential to block the main event loop, which could lead to performance degradation/unexpected blockage. I think in this case it would be better to run this on a separate event loop There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The asyncio queues used to send the messages cannot be used from different event loops. What might cause the block here in your opinion? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Sorry for the long break, I was side tracked a bit in the python lsp support in Qt Creator and some urgent bugs before our release. I added a restriction to the loop, but I guess that not really what you've expected right? |
||
|
||
async def consume_task(self): | ||
while self._consume_task is not None and not self._consume_task.cancelled(): | ||
message = await self._messageQueue.get() | ||
await asyncio.to_thread(self.consume, message) | ||
self._messageQueue.task_done() | ||
|
||
def shutdown(self): | ||
self._executor_service.shutdown() | ||
if self._consume_task is not None: | ||
self._consume_task.cancel() | ||
|
||
def notify(self, method, params=None): | ||
"""Send a JSON RPC notification to the client. | ||
|
@@ -94,6 +111,21 @@ def callback(future): | |
future.set_exception(JsonRpcRequestCancelled()) | ||
return callback | ||
|
||
async def consume_async(self, message): | ||
"""Consume a JSON RPC message from the client and put it into a queue. | ||
|
||
Args: | ||
message (dict): The JSON RPC message sent by the client | ||
""" | ||
if message['method'] == CANCEL_METHOD: | ||
self._cancelledRequests.add(message.get('params')['id']) | ||
|
||
# The exit message needs to be handled directly since the stream cannot be closed asynchronously | ||
if message['method'] == EXIT_METHOD: | ||
self.consume(message) | ||
else: | ||
await self._messageQueue.put(message) | ||
|
||
def consume(self, message): | ||
"""Consume a JSON RPC message from the client. | ||
|
||
|
@@ -182,6 +214,9 @@ def _handle_request(self, msg_id, method, params): | |
except KeyError as e: | ||
raise JsonRpcMethodNotFound.of(method) from e | ||
|
||
if msg_id in self._cancelledRequests: | ||
raise JsonRpcRequestCancelled() | ||
|
||
handler_result = handler(params) | ||
|
||
if callable(handler_result): | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Shouldn't we have a way to set a
asyncio
event loop executor by parameter? https://docs.python.org/3/library/asyncio-eventloop.html#asyncio.set_event_loopThere was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This init_async function was more a relict of a previous attempt... I think it is save to initialize the queue in the default initializer...
Regarding set_event_loop I don't how I achieve something similar to davschul/python-lsp-server@215f79d in a more ellegant way, And I'm open to suggestions :)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No worries, this can be looked in a future iteration.