2022-05-31 17:56:37 +00:00
|
|
|
|
|
|
|
import asyncio
|
|
|
|
import threading
|
|
|
|
import time
|
|
|
|
|
2022-06-02 19:00:42 +00:00
|
|
|
from .native import UVLoop
|
|
|
|
|
2022-05-31 20:53:20 +00:00
|
|
|
|
|
|
|
def future_handler(future, loop, exception_handler, response):
|
|
|
|
try:
|
|
|
|
future.result()
|
|
|
|
return None
|
|
|
|
except Exception as error:
|
|
|
|
if hasattr(exception_handler, '__call__'):
|
|
|
|
exception_handler(loop, error, response)
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
#just log in console the error to call attention
|
|
|
|
print("Uncaught Exception: %s" % str(error))
|
|
|
|
if response != None:
|
|
|
|
response.write_status(500).end("Internal Error")
|
|
|
|
finally:
|
|
|
|
return
|
|
|
|
|
2022-05-31 17:56:37 +00:00
|
|
|
class Loop:
|
2022-05-31 20:53:20 +00:00
|
|
|
def __init__(self, exception_handler=None):
|
2022-05-31 17:56:37 +00:00
|
|
|
self.loop = asyncio.new_event_loop()
|
2022-06-02 19:00:42 +00:00
|
|
|
self.uv_loop = UVLoop()
|
2022-05-31 20:53:20 +00:00
|
|
|
if hasattr(exception_handler, '__call__'):
|
|
|
|
self.exception_handler = exception_handler
|
|
|
|
self.loop.set_exception_handler(lambda loop, context: exception_handler(loop, context, None))
|
|
|
|
else:
|
|
|
|
self.exception_handler = None
|
|
|
|
|
2022-05-31 17:56:37 +00:00
|
|
|
asyncio.set_event_loop(self.loop)
|
2022-06-02 19:00:42 +00:00
|
|
|
self.started = False
|
|
|
|
# self.loop_thread = None
|
2022-05-31 17:56:37 +00:00
|
|
|
|
|
|
|
def start(self):
|
2022-06-02 19:00:42 +00:00
|
|
|
self.started = True
|
|
|
|
self.timer = self.uv_loop.create_timer(0, 100, lambda loop: loop.run_once_asyncio(), self)
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
self.uv_loop.run()
|
|
|
|
|
|
|
|
def run_once(self):
|
|
|
|
self.uv_loop.run_once()
|
|
|
|
|
|
|
|
def run_once_asyncio(self):
|
|
|
|
#run only one step
|
|
|
|
self.loop.call_soon(self.loop.stop)
|
|
|
|
self.loop.run_forever()
|
2022-05-31 17:56:37 +00:00
|
|
|
def stop(self):
|
2022-06-02 19:00:42 +00:00
|
|
|
if(self.started):
|
|
|
|
self.timer.stop()
|
|
|
|
self.started = False
|
|
|
|
#unbind run_once
|
|
|
|
#if is still running stops
|
|
|
|
if self.loop.is_running():
|
|
|
|
self.loop.stop()
|
|
|
|
|
2022-05-31 17:56:37 +00:00
|
|
|
# Find all running tasks in main thread:
|
|
|
|
pending = asyncio.all_tasks(self.loop)
|
|
|
|
# Run loop until tasks done
|
|
|
|
self.loop.run_until_complete(asyncio.gather(*pending))
|
2022-06-02 19:00:42 +00:00
|
|
|
|
|
|
|
#Exposes native loop for uWS
|
|
|
|
def get_native_loop(self):
|
|
|
|
return self.uv_loop.get_native_loop()
|
2022-05-31 17:56:37 +00:00
|
|
|
|
2022-05-31 20:53:20 +00:00
|
|
|
def run_async(self, task, response=None):
|
2022-06-02 19:00:42 +00:00
|
|
|
#with run_once
|
|
|
|
future = asyncio.ensure_future(task, loop=self.loop)
|
|
|
|
|
|
|
|
#with threads
|
2022-05-31 20:53:20 +00:00
|
|
|
future.add_done_callback(lambda f: future_handler(f, self.loop, self.exception_handler, response))
|
|
|
|
return future
|
2022-06-01 23:00:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# if sys.version_info >= (3, 11)
|
|
|
|
# with asyncio.Runner(loop_factory=uvloop.new_event_loop) as runner:
|
|
|
|
# runner.run(main())
|
|
|
|
# else:
|
|
|
|
# uvloop.install()
|
|
|
|
# asyncio.run(main())
|