On 4/14/21 1:44 AM, Stefan Hajnoczi wrote:
On Tue, Apr 13, 2021 at 11:55:52AM -0400, John Snow wrote:
+    async def _execute(self, msg: Message) -> object:
+        """
+        The same as `execute_msg()`, but without safety mechanisms.
+
+        Does not assign an execution ID and does not check that the form
+        of the message being sent is valid.
+
+        This method *Requires* an 'id' parameter to be set on the
+        message, it will not set one for you like `execute()` or
+        `execute_msg()`.
+
+        Do not use "__aqmp#00000" style IDs, use something else to avoid
+        potential clashes. If this ID clashes with an ID presently
+        in-use or otherwise clashes with the auto-generated IDs, the
+        response routing mechanisms in _on_message may very well fail
+        loudly enough to cause the entire loop to crash.
+
+        The ID should be a str; or at least something JSON
+        serializable. It *must* be hashable.
+        """
+        exec_id = cast(str, msg['id'])
+        self.logger.debug("Execute(%s): '%s'", exec_id,
+                          msg.get('execute', msg.get('exec-oob')))
+
+        queue: asyncio.Queue[Message] = asyncio.Queue(maxsize=1)
+        task = create_task(self._bh_execute(msg, queue))

We're already in a coroutine, can we await queue.get() ourselves instead
of creating a new task?

I guess this is done in order to use Task.cancel() in _bh_disconnect()
but it seems simpler to use queue both for success and cancellation.
Fewer tasks are easier to reason about.


...queues do not have a cancellation signal :( :( :( :(

There's no way to "cancel" a queue:
https://docs.python.org/3/library/asyncio-queue.html#queue

You *could* craft a special message and inject an exception into the queue to notify the reader that the message will never arrive, but it feels like working against the intended mechanism of that primitive. It really feels like it wants to be wrapped in a *task*.

An earlier draft used an approach where it crafted a special "mailbox" object, comprised of message, event, and error fields. The waiter sets up a mailbox and then blocks on the event. Upon being notified of an event, the caller checks to see if the message OR the error field was filled.

I wound up removing it, because I felt it added too much custom machinery/terminology and instead went with Tasks and a queue with a depth of one.

Both feel like they are working against the intended mechanisms to a degree. I am open to suggestions here!

(It's also worth noting that iotests will want the ability to separate the queueing of a message and the waiting for that message. The current design only allows for send-and-wait, and not separate send-then-wait semantics. Tasks do provide a rather convenient handle if I want to split that mechanism out.)

All of the above options are a little hacky to me. Any thoughts or preferences?

--js


Reply via email to