InvokeAI/invokeai/app/services
psychedelicious 29325a7214 fix(app): use asyncio queue and existing event loop for events
Around the time we (I) implemented pydantic events, I noticed a short pause between progress images every 4 or 5 steps when generating with SDXL. It didn't happen with SD1.5, but I did notice that with SD1.5, we'd get 4 or 5 progress events simultaneously. I'd expect one event every ~25ms, matching my it/s with SD1.5. Mysterious!

Digging in, I found an issue is related to our use of a synchronous queue for events. When the event queue is empty, we must call `asyncio.sleep` before checking again. We were sleeping for 100ms.

Said another way, every time we clear the event queue, we have to wait 100ms before another event can be dispatched, even if it is put on the queue immediately after we start waiting. In practice, this means our events get buffered into batches, dispatched once every 100ms.

This explains why I was getting batches of 4 or 5 SD1.5 progress events at once, but not the intermittent SDXL delay.

But this 100ms wait has another effect when the events are put on the queue in intervals that don't perfectly line up with the 100ms wait. This is most noticeable when the time between events is >100ms, and can add up to 100ms delay before the event is dispatched.

For example, say the queue is empty and we start a 100ms wait. Then, immediately after - like 0.01ms later - we push an event on to the queue. We still need to wait another 99.9ms before that event will be dispatched. That's the SDXL delay.

The easy fix is to reduce the sleep to something like 0.01 seconds, but this feels kinda dirty. Can't we just wait on the queue and dispatch every event immediately? Not with the normal synchronous queue - but we can with `asyncio.Queue`.

I switched the events queue to use `asyncio.Queue` (as seen in this commit), which lets us asynchronous wait on the queue in a loop.

Unfortunately, I ran into another issue - events now felt like their timing was inconsistent, but in a different way than with the 100ms sleep. The time between pushing events on the queue and dispatching them was not consistently ~0ms as I'd expect - it was highly variable from ~0ms up to ~100ms.

This is resolved by passing the asyncio loop directly into the events service and using its methods to create the task and interact with the queue. I don't fully understand why this resolved the issue, because either way we are interacting with the same event loop (as shown by `asyncio.get_running_loop()`). I suppose there's some scheduling magic happening.
2024-08-12 07:49:58 +10:00
..
board_image_records Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
board_images Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
board_records Merge branch 'main' into boards-ui-update 2024-07-08 22:06:26 -04:00
boards update BoardRecord 2024-07-08 14:55:04 -04:00
bulk_download Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
config Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
download Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
events fix(app): use asyncio queue and existing event loop for events 2024-08-12 07:49:58 +10:00
image_files feat(app): clean up DiskImageStorage types 2024-08-04 09:43:20 +10:00
image_records Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
images Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
invocation_cache fix(nodes): correctly serialize outputs 2024-03-06 08:14:12 -05:00
invocation_stats Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
item_storage revert(nodes): revert making tensors/conditioning use item storage 2024-03-01 10:42:33 +11:00
model_images feat(app): delete model_images instead of using send2trash 2024-08-04 09:43:20 +10:00
model_install [MM2] Use typed ModelRecordChanges for model_install() rather than untyped dict (#6645) 2024-07-23 21:41:00 +00:00
model_load Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
model_manager Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
model_records [MM2] Use typed ModelRecordChanges for model_install() rather than untyped dict (#6645) 2024-07-23 21:41:00 +00:00
names Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
object_serializer tidy(app): remove unused class 2024-04-23 17:12:14 +10:00
session_processor Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
session_queue fix(queue): add clear_queue_on_startup config to clear problematic queues 2024-06-19 11:39:25 +10:00
shared fix commented out migration 2024-07-01 20:06:28 +10:00
urls Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
workflow_records feat(app): remove beta from multidiffusion workflows 2024-07-26 13:47:51 +10:00
__init__.py all files migrated; tweaks needed 2023-03-03 00:02:15 -05:00
invocation_services.py Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00
invoker.py Apply ruff rule to disallow all relative imports. 2024-07-04 09:35:37 -04:00