[gnome-music/wip/mschraal/async-queue: 1/8] asyncqueue: Add generic AsyncQueue class
- From: Marinus Schraal <mschraal src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gnome-music/wip/mschraal/async-queue: 1/8] asyncqueue: Add generic AsyncQueue class
- Date: Wed, 11 Aug 2021 14:31:18 +0000 (UTC)
commit 3e852823bbcf3b5ffb8dcddafe37e425651fbe30
Author: Marinus Schraal <mschraal gnome org>
Date: Wed Aug 11 14:09:10 2021 +0200
asyncqueue: Add generic AsyncQueue class
A generic queue for async tasks, to prevent too many tasks at once from
hammering the system.
gnomemusic/asyncqueue.py | 84 ++++++++++++++++++++++++++++++++++++++++++++++++
1 file changed, 84 insertions(+)
---
diff --git a/gnomemusic/asyncqueue.py b/gnomemusic/asyncqueue.py
new file mode 100644
index 000000000..4439cf87a
--- /dev/null
+++ b/gnomemusic/asyncqueue.py
@@ -0,0 +1,84 @@
+# Copyright 2021 The GNOME Music developers
+#
+# GNOME Music is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+#
+# GNOME Music is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License along
+# with GNOME Music; if not, write to the Free Software Foundation, Inc.,
+# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+#
+# The GNOME Music authors hereby grant permission for non-GPL compatible
+# GStreamer plugins to be used and distributed together with GStreamer
+# and GNOME Music. This permission is above and beyond the permissions
+# granted by the GPL license by which GNOME Music is covered. If you
+# modify this code, you may extend this exception to your version of the
+# code, but you are not obligated to do so. If you do not wish to do so,
+# delete this exception statement from your version.
+
+from typing import Dict, Tuple
+
+from gi.repository import GObject
+
+
+class AsyncQueue(GObject.GObject):
+ """Queue async classes
+
+ Allows for queueing async class calls and limit the amount of
+ concurrent async operations ongoing. This to alleviate the
+ pressure of having numerous ongoing async tasks that do IO or
+ networking.
+
+ A queued class should have a `query` function which starts the
+ async task and a `result` signal, which indicates it is done.
+ The signal may be used by the caller and have an arbitrary
+ number and type of arguments.
+ The query function' first argument should be the async class and
+ may have an arbitrary number of arguments following.
+ """
+
+ def __init__(self) -> None:
+ """Initialize AsyncQueue
+ """
+ super().__init__()
+
+ self._async_pool: Dict[int, Tuple] = {}
+ self._async_active_pool: Dict[int, Tuple] = {}
+ self._max_async = 4
+
+ def queue(self, *args) -> None:
+ """Queue an async call
+ """
+ async_obj = args[0]
+ async_obj_id = id(async_obj)
+ result_id = 0
+
+ if (async_obj_id not in self._async_pool
+ and async_obj_id not in self._async_active_pool):
+ self._async_pool[async_obj_id] = (args)
+ else:
+ return
+
+ def on_async_finished(*args):
+ async_obj = args[0]
+ async_obj.disconnect(result_id)
+ self._async_active_pool.pop(id(async_obj))
+
+ if len(self._async_pool) > 0:
+ key = list(self._async_pool.keys())[0]
+ args = self._async_pool.pop(key)
+ self.queue(*args)
+
+ if len(self._async_active_pool) < self._max_async:
+ async_task = self._async_pool.pop(async_obj_id)
+ async_obj = async_task[0]
+ self._async_active_pool[async_obj_id] = async_obj
+
+ result_id = async_obj.connect("result", on_async_finished)
+ async_obj.query(*args[1:])
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]