-
Notifications
You must be signed in to change notification settings - Fork 4
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
access to the event loop is needed #2
Comments
@dBeau It seems to me that the idea of using one websocket client for different tasks is not very correct. Maybe it makes sense to use one websocket client for the Stream Deck application, and the other for your tasks? import asyncio
import random
from streamdeck_sdk import in_separate_thread, StreamDeck, logger
async def async_printer(i: int):
sleep_time = random.random() * 100
await asyncio.sleep(sleep_time)
logger.info(f"async_printer: {i=}, {sleep_time=}")
async def asynchronous():
tasks = [asyncio.ensure_future(async_printer(i=i)) for i in range(100)]
await asyncio.wait(tasks)
@in_separate_thread(daemon=True)
def run_async_printer():
event_loop = asyncio.new_event_loop()
event_loop.run_until_complete(asynchronous())
run_async_printer()
if __name__ == '__main__':
import os
from pathlib import Path
PLUGIN_LOGS_DIR_PATH: Path = Path(os.environ["PLUGIN_LOGS_DIR_PATH"])
PLUGIN_NAME: str = os.environ["PLUGIN_NAME"]
LOG_FILE_PATH: Path = PLUGIN_LOGS_DIR_PATH / Path(f"{PLUGIN_NAME}.log")
StreamDeck(
log_file=LOG_FILE_PATH,
).run() This code will write messages to the log from The from streamdeck_sdk import in_separate_thread, StreamDeck, Action
class MyAction(Action):
UUID = "myaction"
@in_separate_thread(daemon=True)
def run_monitoring(self):
...
my_action = MyAction()
my_action.run_monitoring()
if __name__ == '__main__':
StreamDeck(
actions=[
my_action,
]
).run() Maybe this will help you more in solving your problem? |
@gri-gus , thanks much for your well thought out answer. You are correct that multiple threads can be used to solve my problem. However, I like to avoid threads as much as possible so as to not worry about interactions between them. ...suddenly, this turns into a discussion of design philosophy. To be clearer though (perhaps?) I wasnt suggesting using a single websocket client for different tasks. I'm not sure what that would even mean. However, multiple websocket clients in a single program would, however, make perfectly good sense to me. In my case though, there is just one websocket client and that is the one used used by streamdeck-sdk to talk to the Stream Deck application. The other sockets I was refering to were good ole TCP streams. My goal is to have a single event loop service all of the I/O for the plug-in. The rel library is one way of doing this. The asyncio library, currently very popular, also has facilities for doing this. I might go so far as to suggest that this is why the asyncio library exists. In effect, it provides an alternative to using threads in I/O heavy applications. To succeed in this goal it's necessary for libraries, like streamdeck-sdk, to either work directly with asyncio or at the very least expose their I/O in a generic way for use by other async libraries (curio, trio, AnyIO, pyevent, rel, even twisted are all examples). The websocket library does this by allowing the user to provide a 'dispatcher' to its run_forever() method. In effect, it's happy to not worry about checking its TCP sockets for read/writes and hands that responsibilty off to the dispatcher. Again, the suggestion you provided will work. Thanks much for that and for streamdeck-sdk too. |
*Edit: Actually, I can't seem to demonstrate that the monitoring thread is actually running. To test this I attempted to open and write to a hardcoded path from within Original message:I'm having trouble with logging when performing asyncio as described. I thought this may have been because logging was not yet set up during the init of StreamDeck so I first constructed the StreamDeck instance: my_action = MyAction()
#Intializing StreamDeck here so that logging is configured before calling run_monitoring
stream_deck = StreamDeck(
actions=[
my_action,
],
log_file=settings.LOG_FILE_PATH,
log_level=settings.LOG_LEVEL,
log_backup_count=1,
)
my_action.run_monitoring()
if __name__ == '__main__':
stream_deck.run() @in_separate_thread(daemon=True)
async def run_monitoring(self):
logger = logging.getLogger(__name__)
logger.debug('run_monitoring logging test') My logging from elsewhere within MyAction works. Any ideas? |
#2 (comment) I'm not sure about the correct operation of the asynchronous function in the in_separate_thread decorator. And in general, so far everything has been done for synchronous code. An asynchronous version will come later. |
I believe I've gotten run_monitoring to work by wrapping the async portion in loop.run_until_complete(...): async def async_run_monitoring(self):
logger = logging.getLogger(__name__)
logger.debug('run_monitoring logging test')
@in_separate_thread(daemon=True)
def run_monitoring(self):
loop = asyncio.new_event_loop()
asyncio.set_event_loop(loop)
loop.run_until_complete(self.async_run_monitoring())
loop.close() I'm now struggling with figuring out how to make I stepped away from this project for a while, so I'm not sure if it's due to Elgato's efforts to move to the new Stream Deck (node) API, but I can't seem to find any documentation on these contexts. |
#2 (comment) They did update the documentation so it's more focused on nodejs, but the API section remains. According to the documentation, the Let's figure out what context is and how it works. I have a project that shows quite well how to change the Also pay attention to the |
The plug-in I am writing makes long running socket connections to other services. These connections need to be read from and written to inorder to maintain the state needed by the plug-in. The existing StreamDeck.run() method requies the use of the websocket run_forever() method and does not provide access to its event loop or for selecting from other socket connections. It would be helpful if either the event loop was exposed or if the StreamDeck class could use an externally provided event loop.
A quick fix can be employed by the StreamDeck user with a simple modification. The idea is to import the rel module and pass its dispatcher to the StreamDeck.run() method. The websocket examples show how this works.
Here's some plug-in code assuming the modifications have been made.
then in sdk.py two changes are needed, the first is for the run() method to accept the dispatcher parameter and the second is for it to make use of it
self.ws.run_forever(dispatcher=dispatcher)
This is a very simple change based on some older technology that has largely been replaced by asyncio. An asyncio solution could be "better", but just two lines and no breakage is hard to beat.
The text was updated successfully, but these errors were encountered: