{
  "RepoName": "https://github.com/aio-libs/janus.git",
  "CommitSHA": "e46a1dfa166ebc2484716ee36cb99763f6247d52",
  "Time": "",
  "Difficulty": "Medium",
  "Type": "logic error",
  "BuggyCode": [
    {
      "path": "aio-libs_janus/setup.py",
      "content": "from setuptools import setup\n\nsetup()\n"
    },
    {
      "path": "aio-libs_janus/tests/test_sync.py",
      "content": "# Some simple queue module tests, plus some failure conditions\n# to ensure the Queue locks remain stable.\nimport asyncio\nimport queue\nimport threading\nimport time\nfrom unittest.mock import patch\n\nimport pytest\n\nimport janus\n\nQUEUE_SIZE = 5\n\n\ndef qfull(q):\n    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n\n\n# A thread to run a function that unclogs a blocked Queue.\nclass _TriggerThread(threading.Thread):\n    def __init__(self, fn, args):\n        self.fn = fn\n        self.args = args\n        self.startedEvent = threading.Event()\n        threading.Thread.__init__(self)\n\n    def run(self):\n        # The sleep isn't necessary, but is intended to give the blocking\n        # function in the main thread a chance at actually blocking before\n        # we unclog it.  But if the sleep is longer than the timeout-based\n        # tests wait in their blocking functions, those tests will fail.\n        # So we give them much longer timeout values compared to the\n        # sleep here (I aimed at 10 seconds for blocking functions --\n        # they should never actually wait that long - they should make\n        # progress as soon as we call self.fn()).\n        time.sleep(0.1)\n        self.startedEvent.set()\n        self.fn(*self.args)\n\n\n# Execute a function that blocks, and in a separate thread, a function that\n# triggers the release.  Returns the result of the blocking function.  Caution:\n# block_func must guarantee to block until trigger_func is called, and\n# trigger_func must guarantee to change queue state so that block_func can make\n# enough progress to return.  In particular, a block_func that just raises an\n# exception regardless of whether trigger_func is called will lead to\n# timing-dependent sporadic failures, and one of those went rarely seen but\n# undiagnosed for years.  Now block_func must be unexceptional.  If block_func\n# is supposed to raise an exception, call do_exceptional_blocking_test()\n# instead.\n\n\nclass BlockingTestMixin:\n    def do_blocking_test(self, block_func, block_args, trigger_func, trigger_args):\n        self.t = _TriggerThread(trigger_func, trigger_args)\n        self.t.start()\n        self.result = block_func(*block_args)\n        # If block_func returned before our thread made the call, we failed!\n        if not self.t.startedEvent.is_set():\n            pytest.fail(\"blocking function '%r' appeared not to block\" % block_func)\n        self.t.join(10)  # make sure the thread terminates\n        if self.t.is_alive():\n            pytest.fail(\"trigger function '%r' appeared to not return\" % trigger_func)\n        return self.result\n\n    # Call this instead if block_func is supposed to raise an exception.\n    def do_exceptional_blocking_test(\n        self,\n        block_func,\n        block_args,\n        trigger_func,\n        trigger_args,\n        expected_exception_class,\n    ):\n        self.t = _TriggerThread(trigger_func, trigger_args)\n        self.t.start()\n        try:\n            try:\n                block_func(*block_args)\n            except expected_exception_class:\n                raise\n            else:\n                pytest.fail(\"expected exception of kind %r\" % expected_exception_class)\n        finally:\n            self.t.join(10)  # make sure the thread terminates\n            if self.t.is_alive():\n                pytest.fail(\n                    \"trigger function '%r' appeared to not return\" % trigger_func\n                )\n            if not self.t.startedEvent.is_set():\n                pytest.fail(\"trigger thread ended but event never set\")\n\n\nclass BaseQueueTestMixin(BlockingTestMixin):\n    cum = 0\n    cumlock = threading.Lock()\n\n    def simple_queue_test(self, _q):\n        q = _q.sync_q\n        if q.qsize():\n            raise RuntimeError(\"Call this function with an empty queue\")\n        assert q.empty()\n        assert not q.full()\n        # I guess we better check things actually queue correctly a little :)\n        q.put(111)\n        q.put(333)\n        q.put(222)\n        target_order = dict(\n            Queue=[111, 333, 222],\n            LifoQueue=[222, 333, 111],\n            PriorityQueue=[111, 222, 333],\n        )\n        actual_order = [q.get(), q.get(), q.get()]\n        assert actual_order == target_order[_q.__class__.__name__]\n        for i in range(QUEUE_SIZE - 1):\n            q.put(i)\n            assert q.qsize()\n        assert not qfull(q)\n        last = 2 * QUEUE_SIZE\n        full = 3 * 2 * QUEUE_SIZE\n        q.put(last)\n        assert qfull(q)\n        assert not q.empty()\n        assert q.full()\n        try:\n            q.put(full, block=0)\n            pytest.fail(\"Didn't appear to block with a full queue\")\n        except queue.Full:\n            pass\n        try:\n            q.put(full, timeout=0.01)\n            pytest.fail(\"Didn't appear to time-out with a full queue\")\n        except queue.Full:\n            pass\n        # Test a blocking put\n        self.do_blocking_test(q.put, (full,), q.get, ())\n        self.do_blocking_test(q.put, (full, True, 10), q.get, ())\n        # Empty it\n        for i in range(QUEUE_SIZE):\n            q.get()\n        assert not q.qsize()\n        try:\n            q.get(block=0)\n            pytest.fail(\"Didn't appear to block with an empty queue\")\n        except queue.Empty:\n            pass\n        try:\n            q.get(timeout=0.01)\n            pytest.fail(\"Didn't appear to time-out with an empty queue\")\n        except queue.Empty:\n            pass\n        # Test a blocking get\n        self.do_blocking_test(q.get, (), q.put, (\"empty\",))\n        self.do_blocking_test(q.get, (True, 10), q.put, (\"empty\",))\n\n    def worker(self, q):\n        try:\n            while True:\n                x = q.get()\n                if x < 0:\n                    q.task_done()\n                    return\n                with self.cumlock:\n                    self.cum += x\n                q.task_done()\n        except Exception as ex:\n            from traceback import print_exc\n\n            print_exc(ex)\n\n    def queue_join_test(self, q):\n        self.cum = 0\n        for i in (0, 1):\n            threading.Thread(target=self.worker, args=(q,)).start()\n        for i in range(100):\n            q.put(i)\n        q.join()\n        assert self.cum == sum(range(100))\n        for i in (0, 1):\n            q.put(-1)  # instruct the threads to close\n        q.join()  # verify that you can join twice\n\n    @pytest.mark.asyncio\n    async def test_queue_task_done(self):\n        # Test to make sure a queue task completed successfully.\n        _q = self.type2test()\n        q = _q.sync_q\n        try:\n            q.task_done()\n        except ValueError:\n            pass\n        else:\n            pytest.fail(\"Did not detect task count going negative\")\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_queue_join(self):\n        # Test that a queue join()s successfully, and before anything else\n        # (done twice for insurance).\n        _q = self.type2test()\n        q = _q.sync_q\n        self.queue_join_test(q)\n        self.queue_join_test(q)\n        try:\n            q.task_done()\n        except ValueError:\n            pass\n        else:\n            pytest.fail(\"Did not detect task count going negative\")\n        finally:\n            _q.close()\n            await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_simple_queue(self):\n        # Do it a couple of times on the same queue.\n        # Done twice to make sure works with same instance reused.\n        _q = self.type2test(QUEUE_SIZE)\n        self.simple_queue_test(_q)\n        self.simple_queue_test(_q)\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_negative_timeout_raises_exception(self):\n        _q = self.type2test(QUEUE_SIZE)\n        q = _q.sync_q\n        with pytest.raises(ValueError):\n            q.put(1, timeout=-1)\n        with pytest.raises(ValueError):\n            q.get(1, timeout=-1)\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nowait(self):\n        _q = self.type2test(QUEUE_SIZE)\n        q = _q.sync_q\n        for i in range(QUEUE_SIZE):\n            q.put_nowait(1)\n        with pytest.raises(queue.Full):\n            q.put_nowait(1)\n\n        for i in range(QUEUE_SIZE):\n            q.get_nowait()\n        with pytest.raises(queue.Empty):\n            q.get_nowait()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_shrinking_queue(self):\n        # issue 10110\n        _q = self.type2test(3)\n        q = _q.sync_q\n        q.put(1)\n        q.put(2)\n        q.put(3)\n        with pytest.raises(queue.Full):\n            q.put_nowait(4)\n        assert q.qsize() == 3\n        q._maxsize = 2  # shrink the queue\n        with pytest.raises(queue.Full):\n            q.put_nowait(4)\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_maxsize(self):\n        # Test to make sure a queue task completed successfully.\n        _q = self.type2test(5)\n        q = _q.sync_q\n        assert q.maxsize == 5\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestQueue(BaseQueueTestMixin):\n    type2test = janus.Queue\n\n\nclass TestLifoQueue(BaseQueueTestMixin):\n    type2test = janus.LifoQueue\n\n\nclass TestPriorityQueue(BaseQueueTestMixin):\n    type2test = janus.PriorityQueue\n\n\n# A Queue subclass that can provoke failure at a moment's notice :)\nclass FailingQueueException(Exception):\n    pass\n\n\nclass FailingQueue(janus.Queue):\n    def __init__(self, *args, **kwargs):\n        self.fail_next_put = False\n        self.fail_next_get = False\n        super().__init__(*args, **kwargs)\n\n    def _put(self, item):\n        if self.fail_next_put:\n            self.fail_next_put = False\n            raise FailingQueueException(\"You Lose\")\n        return super()._put(item)\n\n    def _get(self):\n        if self.fail_next_get:\n            self.fail_next_get = False\n            raise FailingQueueException(\"You Lose\")\n        return super()._get()\n\n\nclass TestFailingQueue(BlockingTestMixin):\n    def failing_queue_test(self, _q):\n        q = _q.sync_q\n        if q.qsize():\n            raise RuntimeError(\"Call this function with an empty queue\")\n        for i in range(QUEUE_SIZE - 1):\n            q.put(i)\n        # Test a failing non-blocking put.\n        _q.fail_next_put = True\n        try:\n            q.put(\"oops\", block=0)\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        _q.fail_next_put = True\n        try:\n            q.put(\"oops\", timeout=0.1)\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        q.put(\"last\")\n        assert qfull(q)\n        # Test a failing blocking put\n        _q.fail_next_put = True\n        try:\n            self.do_blocking_test(q.put, (\"full\",), q.get, ())\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        # Check the Queue isn't damaged.\n        # put failed, but get succeeded - re-add\n        q.put(\"last\")\n        # Test a failing timeout put\n        _q.fail_next_put = True\n        try:\n            self.do_exceptional_blocking_test(\n                q.put, (\"full\", True, 10), q.get, (), FailingQueueException\n            )\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        # Check the Queue isn't damaged.\n        # put failed, but get succeeded - re-add\n        q.put(\"last\")\n        assert qfull(q)\n        q.get()\n        assert not qfull(q)\n        q.put(\"last\")\n        assert qfull(q)\n        # Test a blocking put\n        self.do_blocking_test(q.put, (\"full\",), q.get, ())\n        # Empty it\n        for i in range(QUEUE_SIZE):\n            q.get()\n        assert not q.qsize()\n        q.put(\"first\")\n        _q.fail_next_get = True\n        try:\n            q.get()\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        assert q.qsize()\n        _q.fail_next_get = True\n        try:\n            q.get(timeout=0.1)\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        assert q.qsize()\n        q.get()\n        assert not q.qsize()\n        _q.fail_next_get = True\n        try:\n            self.do_exceptional_blocking_test(\n                q.get, (), q.put, (\"empty\",), FailingQueueException\n            )\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        # put succeeded, but get failed.\n        assert q.qsize()\n        q.get()\n        assert not q.qsize()\n\n    @pytest.mark.asyncio\n    async def test_failing_queue(self):\n        # Test to make sure a queue is functioning correctly.\n        # Done twice to the same instance.\n        q = FailingQueue(QUEUE_SIZE)\n        self.failing_queue_test(q)\n        self.failing_queue_test(q)\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_closed_loop_non_failing(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(QUEUE_SIZE)\n        q = _q.sync_q\n        # we are pacthing loop to follow setUp/tearDown agreement\n        with patch.object(loop, \"call_soon_threadsafe\") as func:\n            func.side_effect = RuntimeError()\n            q.put_nowait(1)\n            assert func.call_count == 1\n        _q.close()\n        await _q.wait_closed()\n"
    },
    {
      "path": "aio-libs_janus/tests/test_async.py",
      "content": "\"\"\"Tests for queues.py\"\"\"\n\nimport asyncio\n\nimport pytest\n\nimport janus\n\n\nclass TestQueueBasic:\n    async def _test_repr_or_str(self, fn, expect_id):\n        \"\"\"Test Queue's repr or str.\n\n        fn is repr or str. expect_id is True if we expect the Queue's id to\n        appear in fn(Queue()).\n        \"\"\"\n\n        _q = janus.Queue()\n        q = _q.async_q\n        assert fn(q).startswith(\"<Queue\")\n        id_is_present = hex(id(q)) in fn(q)\n        assert expect_id == id_is_present\n        loop = asyncio.get_running_loop()\n\n        async def add_getter():\n            _q = janus.Queue()\n            q = _q.async_q\n            # Start a task that waits to get.\n            loop.create_task(q.get())\n            # Let it start waiting.\n            await asyncio.sleep(0.1)\n            assert \"_getters[1]\" in fn(q)\n            # resume q.get coroutine to finish generator\n            q.put_nowait(0)\n\n        await add_getter()\n\n        async def add_putter():\n            _q = janus.Queue(maxsize=1)\n            q = _q.async_q\n            q.put_nowait(1)\n            # Start a task that waits to put.\n            loop.create_task(q.put(2))\n            # Let it start waiting.\n            await asyncio.sleep(0.1)\n            assert \"_putters[1]\" in fn(q)\n            # resume q.put coroutine to finish generator\n            q.get_nowait()\n\n        await add_putter()\n\n        _q = janus.Queue()\n        q = _q.async_q\n        q.put_nowait(1)\n        assert \"_queue=[1]\" in fn(q)\n\n    # def test_repr(self):\n    #     self._test_repr_or_str(repr, True)\n\n    # def test_str(self):\n    #     self._test_repr_or_str(str, False)\n\n    @pytest.mark.asyncio\n    async def test_empty(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        assert q.empty()\n        q.put_nowait(1)\n        assert not q.empty()\n        assert 1 == q.get_nowait()\n        assert q.empty()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_full(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        assert not q.full()\n\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n        q.put_nowait(1)\n        assert q.full()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_order(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        for i in [1, 3, 2]:\n            q.put_nowait(i)\n\n        items = [q.get_nowait() for _ in range(3)]\n        assert [1, 3, 2] == items\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_maxsize(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=2)\n        q = _q.async_q\n        assert 2 == q.maxsize\n        have_been_put = []\n\n        fut = loop.create_future()\n\n        async def putter():\n            for i in range(3):\n                await q.put(i)\n                have_been_put.append(i)\n                if i == q.maxsize - 1:\n                    fut.set_result(None)\n            return True\n\n        async def test():\n            t = loop.create_task(putter())\n            await fut\n\n            # The putter is blocked after putting two items.\n            assert [0, 1] == have_been_put\n            assert 0 == q.get_nowait()\n\n            # Let the putter resume and put last item.\n            await t\n            assert [0, 1, 2] == have_been_put\n            assert 1 == q.get_nowait()\n            assert 2 == q.get_nowait()\n\n        await test()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestQueueGetTests:\n    @pytest.mark.asyncio\n    async def test_blocking_get(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        q.put_nowait(1)\n\n        res = await q.get()\n        assert 1 == res\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_get_with_putters(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(1)\n        q = _q.async_q\n        q.put_nowait(1)\n\n        fut = loop.create_future()\n\n        async def put():\n            t = asyncio.ensure_future(q.put(2))\n            await asyncio.sleep(0.01)\n            fut.set_result(None)\n            return t\n\n        t = await put()\n\n        res = await q.get()\n        assert 1 == res\n\n        await t\n        assert 1 == q.qsize()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_blocking_get_wait(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue()\n        q = _q.async_q\n        started = asyncio.Event()\n        finished = False\n\n        async def queue_get():\n            nonlocal finished\n            started.set()\n            res = await q.get()\n            finished = True\n            return res\n\n        async def queue_put():\n            loop.call_later(0.01, q.put_nowait, 1)\n            queue_get_task = loop.create_task(queue_get())\n            await started.wait()\n            assert not finished\n            res = await queue_get_task\n            assert finished\n            return res\n\n        res = await queue_put()\n        assert 1 == res\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nonblocking_get(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        q.put_nowait(1)\n        assert 1 == q.get_nowait()\n\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nonblocking_get_exception(self):\n        _q = janus.Queue()\n        pytest.raises(asyncio.QueueEmpty, _q.async_q.get_nowait)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_get_cancelled(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue()\n        q = _q.async_q\n\n        async def queue_get():\n            return await asyncio.wait_for(q.get(), 0.051)\n\n        async def test():\n            get_task = loop.create_task(queue_get())\n            await asyncio.sleep(0.01)  # let the task start\n            q.put_nowait(1)\n            return await get_task\n\n        assert 1 == await test()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_get_cancelled_race(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue()\n        q = _q.async_q\n\n        f1 = loop.create_future()\n\n        async def g1():\n            f1.set_result(None)\n            await q.get()\n\n        t1 = loop.create_task(g1())\n        t2 = loop.create_task(q.get())\n\n        await f1\n        await asyncio.sleep(0.01)\n        t1.cancel()\n\n        with pytest.raises(asyncio.CancelledError):\n            await t1\n        assert t1.done()\n        q.put_nowait(\"a\")\n\n        await t2\n        assert t2.result() == \"a\"\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_get_with_waiting_putters(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n\n        loop.create_task(q.put(\"a\"))\n        loop.create_task(q.put(\"b\"))\n\n        await asyncio.sleep(0.01)\n\n        assert await q.get() == \"a\"\n        assert await q.get() == \"b\"\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestQueuePut:\n    @pytest.mark.asyncio\n    async def test_blocking_put(self):\n        _q = janus.Queue()\n        q = _q.async_q\n\n        # No maxsize, won't block.\n        await q.put(1)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_blocking_put_wait(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n        started = asyncio.Event()\n        finished = False\n\n        async def queue_put():\n            started.set()\n            await q.put(1)\n            await q.put(2)\n            finished = True\n\n        async def queue_get():\n            loop.call_later(0.01, q.get_nowait)\n            queue_put_task = loop.create_task(queue_put())\n            await started.wait()\n            assert not finished\n            await queue_put_task\n            assert finished\n\n        await queue_get()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nonblocking_put(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        q.put_nowait(1)\n        assert 1 == q.get_nowait()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nonblocking_put_exception(self):\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n        q.put_nowait(1)\n        pytest.raises(asyncio.QueueFull, q.put_nowait, 2)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_float_maxsize(self):\n        _q = janus.Queue(maxsize=1.3)\n        q = _q.async_q\n        q.put_nowait(1)\n        q.put_nowait(2)\n        assert q.full()\n        pytest.raises(asyncio.QueueFull, q.put_nowait, 3)\n\n        _q.close()\n        await _q.wait_closed()\n\n        _q = janus.Queue(maxsize=1.3)\n        q = _q.async_q\n\n        async def queue_put():\n            await q.put(1)\n            await q.put(2)\n            assert q.full()\n\n        await queue_put()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_put_cancelled(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue()\n        q = _q.async_q\n\n        async def queue_put():\n            await q.put(1)\n            return True\n\n        async def test():\n            return await q.get()\n\n        t = loop.create_task(queue_put())\n        assert 1 == await test()\n        assert t.done()\n        assert t.result()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_put_cancelled_race(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n\n        put_a = loop.create_task(q.put(\"a\"))\n        put_b = loop.create_task(q.put(\"b\"))\n        put_c = loop.create_task(q.put(\"X\"))\n\n        await put_a\n        assert not put_b.done()\n\n        put_c.cancel()\n\n        with pytest.raises(asyncio.CancelledError):\n            await put_c\n\n        async def go():\n            a = await q.get()\n            assert a == \"a\"\n            b = await q.get()\n            assert b == \"b\"\n            assert put_b.done()\n\n        await go()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_put_with_waiting_getters(self):\n        loop = asyncio.get_running_loop()\n        fut = loop.create_future()\n\n        async def go():\n            fut.set_result(None)\n            ret = await q.get()\n            return ret\n\n        async def put():\n            await q.put(\"a\")\n\n        _q = janus.Queue()\n        q = _q.async_q\n        t = loop.create_task(go())\n        await fut\n        await put()\n        assert await t == \"a\"\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestLifoQueue:\n    @pytest.mark.asyncio\n    async def test_order(self):\n        _q = janus.LifoQueue()\n        q = _q.async_q\n        for i in [1, 3, 2]:\n            q.put_nowait(i)\n\n        items = [q.get_nowait() for _ in range(3)]\n        assert [2, 3, 1] == items\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestPriorityQueue:\n    @pytest.mark.asyncio\n    async def test_order(self):\n        _q = janus.PriorityQueue()\n        q = _q.async_q\n        for i in [1, 3, 2]:\n            q.put_nowait(i)\n\n        items = [q.get_nowait() for _ in range(3)]\n        assert [1, 2, 3] == items\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass _QueueJoinTestMixin:\n    q_class = None\n\n    @pytest.mark.asyncio\n    async def test_task_done_underflow(self):\n        _q = self.q_class()\n        q = _q.async_q\n        pytest.raises(ValueError, q.task_done)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_task_done(self):\n        loop = asyncio.get_running_loop()\n        _q = self.q_class()\n        q = _q.async_q\n        for i in range(100):\n            q.put_nowait(i)\n\n        accumulator = 0\n\n        # Two workers get items from the queue and call task_done after each.\n        # Join the queue and assert all items have been processed.\n        running = True\n\n        async def worker():\n            nonlocal accumulator\n\n            while running:\n                item = await q.get()\n                accumulator += item\n                q.task_done()\n\n        async def test():\n            tasks = [loop.create_task(worker()) for index in range(2)]\n\n            await q.join()\n            return tasks\n\n        tasks = await test()\n        assert sum(range(100)) == accumulator\n\n        # close running generators\n        running = False\n        for i in range(len(tasks)):\n            q.put_nowait(0)\n        await asyncio.wait(tasks)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_join_empty_queue(self):\n        _q = self.q_class()\n        q = _q.async_q\n\n        # Test that a queue join()s successfully, and before anything else\n        # (done twice for insurance).\n\n        async def join():\n            await q.join()\n            await q.join()\n\n        await join()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestQueueJoin(_QueueJoinTestMixin):\n    q_class = janus.Queue\n\n\nclass TestLifoQueueJoin(TestQueueJoin):\n    q_class = janus.LifoQueue\n\n\nclass TestPriorityQueueJoin(TestQueueJoin):\n    q_class = janus.PriorityQueue\n"
    },
    {
      "path": "aio-libs_janus/tests/test_mixed.py",
      "content": "import asyncio\nimport contextlib\nimport sys\nimport threading\n\nimport pytest\n\nimport janus\n\n\nclass TestMixedMode:\n    @pytest.mark.skipif(\n        sys.version_info < (3, 7),\n        reason=\"forbidding implicit loop creation works on \"\n        \"Python 3.7 or higher only\",\n    )\n    def test_ctor_noloop(self):\n        with pytest.raises(RuntimeError):\n            janus.Queue()\n\n    @pytest.mark.asyncio\n    async def test_maxsize(self):\n        q = janus.Queue(5)\n        assert 5 == q.maxsize\n\n    @pytest.mark.asyncio\n    async def test_maxsize_named_param(self):\n        q = janus.Queue(maxsize=7)\n        assert 7 == q.maxsize\n\n    @pytest.mark.asyncio\n    async def test_maxsize_default(self):\n        q = janus.Queue()\n        assert 0 == q.maxsize\n\n    @pytest.mark.asyncio\n    async def test_unfinished(self):\n        q = janus.Queue()\n        assert q.sync_q.unfinished_tasks == 0\n        assert q.async_q.unfinished_tasks == 0\n        q.sync_q.put(1)\n        assert q.sync_q.unfinished_tasks == 1\n        assert q.async_q.unfinished_tasks == 1\n        q.sync_q.get()\n        assert q.sync_q.unfinished_tasks == 1\n        assert q.async_q.unfinished_tasks == 1\n        q.sync_q.task_done()\n        assert q.sync_q.unfinished_tasks == 0\n        assert q.async_q.unfinished_tasks == 0\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_sync_put_async_get(self):\n        loop = asyncio.get_running_loop()\n        q = janus.Queue()\n\n        def threaded():\n            for i in range(5):\n                q.sync_q.put(i)\n\n        async def go():\n            f = loop.run_in_executor(None, threaded)\n            for i in range(5):\n                val = await q.async_q.get()\n                assert val == i\n\n            assert q.async_q.empty()\n\n            await f\n\n        for i in range(3):\n            await go()\n\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_sync_put_async_join(self):\n        loop = asyncio.get_running_loop()\n        q = janus.Queue()\n\n        for i in range(5):\n            q.sync_q.put(i)\n\n        async def do_work():\n            await asyncio.sleep(1)\n            while True:\n                await q.async_q.get()\n                q.async_q.task_done()\n\n        task = loop.create_task(do_work())\n\n        async def wait_for_empty_queue():\n            await q.async_q.join()\n\n        await wait_for_empty_queue()\n\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_async_put_sync_get(self):\n        loop = asyncio.get_running_loop()\n        q = janus.Queue()\n\n        def threaded():\n            for i in range(5):\n                val = q.sync_q.get()\n                assert val == i\n\n        async def go():\n            f = loop.run_in_executor(None, threaded)\n            for i in range(5):\n                await q.async_q.put(i)\n\n            await f\n            assert q.async_q.empty()\n\n        for i in range(3):\n            await go()\n\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_sync_join_async_done(self):\n        loop = asyncio.get_running_loop()\n        q = janus.Queue()\n\n        def threaded():\n            for i in range(5):\n                q.sync_q.put(i)\n            q.sync_q.join()\n\n        async def go():\n            f = loop.run_in_executor(None, threaded)\n            for i in range(5):\n                val = await q.async_q.get()\n                assert val == i\n                q.async_q.task_done()\n\n            assert q.async_q.empty()\n\n            await f\n\n        for i in range(3):\n            await go()\n\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_async_join_async_done(self):\n        loop = asyncio.get_running_loop()\n        q = janus.Queue()\n\n        def threaded():\n            for i in range(5):\n                val = q.sync_q.get()\n                assert val == i\n                q.sync_q.task_done()\n\n        async def go():\n            f = loop.run_in_executor(None, threaded)\n            for i in range(5):\n                await q.async_q.put(i)\n\n            await q.async_q.join()\n\n            await f\n            assert q.async_q.empty()\n\n        for i in range(3):\n            await go()\n\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_wait_without_closing(self):\n        q = janus.Queue()\n\n        with pytest.raises(RuntimeError):\n            await q.wait_closed()\n\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_modifying_forbidden_after_closing(self):\n        q = janus.Queue()\n        q.close()\n\n        with pytest.raises(RuntimeError):\n            q.sync_q.put(5)\n\n        with pytest.raises(RuntimeError):\n            q.sync_q.get()\n\n        with pytest.raises(RuntimeError):\n            q.sync_q.task_done()\n\n        with pytest.raises(RuntimeError):\n            await q.async_q.put(5)\n\n        with pytest.raises(RuntimeError):\n            q.async_q.put_nowait(5)\n\n        with pytest.raises(RuntimeError):\n            q.async_q.get_nowait()\n\n        with pytest.raises(RuntimeError):\n            await q.sync_q.task_done()\n\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_double_closing(self):\n        q = janus.Queue()\n        q.close()\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_closed(self):\n        q = janus.Queue()\n        assert not q.closed\n        assert not q.async_q.closed\n        assert not q.sync_q.closed\n        q.close()\n        assert q.closed\n        assert q.async_q.closed\n        assert q.sync_q.closed\n\n    @pytest.mark.asyncio\n    async def test_async_join_after_closing(self):\n        q = janus.Queue()\n        q.close()\n        with pytest.raises(RuntimeError), contextlib.suppress(asyncio.TimeoutError):\n            await asyncio.wait_for(q.async_q.join(), timeout=0.1)\n\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_close_after_async_join(self):\n        q = janus.Queue()\n        q.sync_q.put(1)\n\n        task = asyncio.ensure_future(q.async_q.join())\n        await asyncio.sleep(0.1)  # ensure tasks are blocking\n\n        q.close()\n        with pytest.raises(RuntimeError), contextlib.suppress(asyncio.TimeoutError):\n            await asyncio.wait_for(task, timeout=0.1)\n\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_sync_join_after_closing(self):\n        q = janus.Queue()\n        q.sync_q.put(1)\n\n        q.close()\n\n        loop = asyncio.get_event_loop()\n        fut = asyncio.Future()\n\n        def sync_join():\n            try:\n                q.sync_q.join()\n            except Exception as exc:\n                loop.call_soon_threadsafe(fut.set_exception, exc)\n\n        thr = threading.Thread(target=sync_join, daemon=True)\n        thr.start()\n\n        with pytest.raises(RuntimeError), contextlib.suppress(asyncio.TimeoutError):\n            await asyncio.wait_for(fut, timeout=0.1)\n\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_close_after_sync_join(self):\n        q = janus.Queue()\n        q.sync_q.put(1)\n\n        loop = asyncio.get_event_loop()\n        fut = asyncio.Future()\n\n        def sync_join():\n            try:\n                q.sync_q.join()\n            except Exception as exc:\n                loop.call_soon_threadsafe(fut.set_exception, exc)\n\n        thr = threading.Thread(target=sync_join, daemon=True)\n        thr.start()\n        thr.join(0.1)  # ensure tasks are blocking\n\n        q.close()\n\n        with pytest.raises(RuntimeError), contextlib.suppress(asyncio.TimeoutError):\n            await asyncio.wait_for(fut, timeout=0.1)\n\n        await q.wait_closed()\n"
    },
    {
      "path": "aio-libs_janus/janus/__init__.py",
      "content": "import asyncio\nimport sys\nimport threading\nfrom asyncio import QueueEmpty as AsyncQueueEmpty\nfrom asyncio import QueueFull as AsyncQueueFull\nfrom collections import deque\nfrom heapq import heappop, heappush\nfrom queue import Empty as SyncQueueEmpty\nfrom queue import Full as SyncQueueFull\nfrom typing import Any, Callable, Deque, Generic, List, Optional, Set, TypeVar\n\nfrom typing_extensions import Protocol\n\n__version__ = \"1.0.0\"\n__all__ = (\n    \"Queue\",\n    \"PriorityQueue\",\n    \"LifoQueue\",\n    \"SyncQueue\",\n    \"AsyncQueue\",\n    \"BaseQueue\",\n)\n\n\nT = TypeVar(\"T\")\nOptFloat = Optional[float]\n\n\nclass BaseQueue(Protocol[T]):\n    @property\n    def maxsize(self) -> int:\n        ...\n\n    @property\n    def closed(self) -> bool:\n        ...\n\n    def task_done(self) -> None:\n        ...\n\n    def qsize(self) -> int:\n        ...\n\n    @property\n    def unfinished_tasks(self) -> int:\n        ...\n\n    def empty(self) -> bool:\n        ...\n\n    def full(self) -> bool:\n        ...\n\n    def put_nowait(self, item: T) -> None:\n        ...\n\n    def get_nowait(self) -> T:\n        ...\n\n\nclass SyncQueue(BaseQueue[T], Protocol[T]):\n\n    def put(self, item: T, block: bool = True, timeout: OptFloat = None) -> None:\n        ...\n\n    def get(self, block: bool = True, timeout: OptFloat = None) -> T:\n        ...\n\n    def join(self) -> None:\n        ...\n\n\nclass AsyncQueue(BaseQueue[T], Protocol[T]):\n    async def put(self, item: T) -> None:\n        ...\n\n    async def get(self) -> T:\n        ...\n\n    async def join(self) -> None:\n        ...\n\n\nclass Queue(Generic[T]):\n    def __init__(self, maxsize: int = 0) -> None:\n        self._loop = asyncio.get_running_loop()\n        self._maxsize = maxsize\n\n        self._init(maxsize)\n\n        self._unfinished_tasks = 0\n\n        self._sync_mutex = threading.Lock()\n        self._sync_not_empty = threading.Condition(self._sync_mutex)\n        self._sync_not_full = threading.Condition(self._sync_mutex)\n        self._all_tasks_done = threading.Condition(self._sync_mutex)\n\n        self._async_mutex = asyncio.Lock()\n        if sys.version_info[:3] == (3, 10, 0):\n            # Workaround for Python 3.10 bug, see #358:\n            getattr(self._async_mutex, \"_get_loop\", lambda: None)()\n        self._async_not_empty = asyncio.Condition(self._async_mutex)\n        self._async_not_full = asyncio.Condition(self._async_mutex)\n        self._finished = asyncio.Event()\n        self._finished.set()\n\n        self._closing = False\n        self._pending = set()  # type: Set[asyncio.Future[Any]]\n\n        def checked_call_soon_threadsafe(\n            callback: Callable[..., None], *args: Any\n        ) -> None:\n            try:\n                self._loop.call_soon_threadsafe(callback, *args)\n            except RuntimeError:\n                # swallowing agreed in #2\n                pass\n\n        self._call_soon_threadsafe = checked_call_soon_threadsafe\n\n        def checked_call_soon(callback: Callable[..., None], *args: Any) -> None:\n            if not self._loop.is_closed():\n                self._loop.call_soon(callback, *args)\n\n        self._call_soon = checked_call_soon\n\n        self._sync_queue = _SyncQueueProxy(self)\n        self._async_queue = _AsyncQueueProxy(self)\n\n    def close(self) -> None:\n        with self._sync_mutex:\n            self._closing = True\n            for fut in self._pending:\n                fut.cancel()\n            self._finished.set()  # unblocks all async_q.join()\n            self._all_tasks_done.notify_all()  # unblocks all sync_q.join()\n\n    async def wait_closed(self) -> None:\n        # should be called from loop after close().\n        # Nobody should put/get at this point,\n        # so lock acquiring is not required\n        if not self._closing:\n            raise RuntimeError(\"Waiting for non-closed queue\")\n        # give execution chances for the task-done callbacks\n        # of async tasks created inside\n        # _notify_async_not_empty, _notify_async_not_full\n        # methods.\n        await asyncio.sleep(0)\n        if not self._pending:\n            return\n        await asyncio.wait(self._pending)\n\n    @property\n    def closed(self) -> bool:\n        return self._closing and not self._pending\n\n    @property\n    def maxsize(self) -> int:\n        return self._maxsize\n\n    @property\n    def sync_q(self) -> \"_SyncQueueProxy[T]\":\n        return self._sync_queue\n\n    @property\n    def async_q(self) -> \"_AsyncQueueProxy[T]\":\n        return self._async_queue\n\n    # Override these methods to implement other queue organizations\n    # (e.g. stack or priority queue).\n    # These will only be called with appropriate locks held\n\n    def _init(self, maxsize: int) -> None:\n        self._queue = deque()  # type: Deque[T]\n\n    def _qsize(self) -> int:\n        return len(self._queue)\n\n    # Put a new item in the queue\n    def _put(self, item: T) -> None:\n        self._queue.append(item)\n\n    # Get an item from the queue\n    def _get(self) -> T:\n        return self._queue.popleft()\n\n    def _put_internal(self, item: T) -> None:\n        self._put(item)\n        self._unfinished_tasks += 1\n        self._finished.clear()\n\n    def _notify_sync_not_empty(self) -> None:\n        def f() -> None:\n            with self._sync_mutex:\n                self._sync_not_empty.notify()\n\n        self._loop.run_in_executor(None, f)\n\n    def _notify_sync_not_full(self) -> None:\n        def f() -> None:\n            with self._sync_mutex:\n                self._sync_not_full.notify()\n\n        fut = asyncio.ensure_future(self._loop.run_in_executor(None, f))\n        fut.add_done_callback(self._pending.discard)\n        self._pending.add(fut)\n\n    def _notify_async_not_empty(self, *, threadsafe: bool) -> None:\n        async def f() -> None:\n            async with self._async_mutex:\n                self._async_not_empty.notify()\n\n        def task_maker() -> None:\n            task = self._loop.create_task(f())\n            task.add_done_callback(self._pending.discard)\n            self._pending.add(task)\n\n        if threadsafe:\n            self._call_soon_threadsafe(task_maker)\n        else:\n            self._call_soon(task_maker)\n\n    def _notify_async_not_full(self, *, threadsafe: bool) -> None:\n        async def f() -> None:\n            async with self._async_mutex:\n                self._async_not_full.notify()\n\n        def task_maker() -> None:\n            task = self._loop.create_task(f())\n            task.add_done_callback(self._pending.discard)\n            self._pending.add(task)\n\n        if threadsafe:\n            self._call_soon_threadsafe(task_maker)\n        else:\n            self._call_soon(task_maker)\n\n    def _check_closing(self) -> None:\n        if self._closing:\n            raise RuntimeError(\"Operation on the closed queue is forbidden\")\n\n\nclass _SyncQueueProxy(SyncQueue[T]):\n    \"\"\"Create a queue object with a given maximum size.\n\n    If maxsize is <= 0, the queue size is infinite.\n    \"\"\"\n\n    def __init__(self, parent: Queue[T]):\n        self._parent = parent\n\n    @property\n    def maxsize(self) -> int:\n        return self._parent._maxsize\n\n    @property\n    def closed(self) -> bool:\n        return self._parent.closed\n\n    def task_done(self) -> None:\n        \"\"\"Indicate that a formerly enqueued task is complete.\n\n        Used by Queue consumer threads.  For each get() used to fetch a task,\n        a subsequent call to task_done() tells the queue that the processing\n        on the task is complete.\n\n        If a join() is currently blocking, it will resume when all items\n        have been processed (meaning that a task_done() call was received\n        for every item that had been put() into the queue).\n\n        Raises a ValueError if called more times than there were items\n        placed in the queue.\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._all_tasks_done:\n            unfinished = self._parent._unfinished_tasks - 1\n            if unfinished <= 0:\n                if unfinished < 0:\n                    raise ValueError(\"task_done() called too many times\")\n                self._parent._all_tasks_done.notify_all()\n                self._parent._loop.call_soon_threadsafe(self._parent._finished.set)\n            self._parent._unfinished_tasks = unfinished\n\n    def join(self) -> None:\n        \"\"\"Blocks until all items in the Queue have been gotten and processed.\n\n        The count of unfinished tasks goes up whenever an item is added to the\n        queue. The count goes down whenever a consumer thread calls task_done()\n        to indicate the item was retrieved and all work on it is complete.\n\n        When the count of unfinished tasks drops to zero, join() unblocks.\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._all_tasks_done:\n            while self._parent._unfinished_tasks:\n                self._parent._all_tasks_done.wait()\n                self._parent._check_closing()\n\n    def qsize(self) -> int:\n        \"\"\"Return the approximate size of the queue (not reliable!).\"\"\"\n        return self._parent._qsize()\n\n    @property\n    def unfinished_tasks(self) -> int:\n        \"\"\"Return the number of unfinished tasks.\"\"\"\n        return self._parent._unfinished_tasks\n\n    def empty(self) -> bool:\n        \"\"\"Return True if the queue is empty, False otherwise (not reliable!).\n\n        This method is likely to be removed at some point.  Use qsize() == 0\n        as a direct substitute, but be aware that either approach risks a race\n        condition where a queue can grow before the result of empty() or\n        qsize() can be used.\n\n        To create code that needs to wait for all queued tasks to be\n        completed, the preferred technique is to use the join() method.\n        \"\"\"\n        return not self._parent._qsize()\n\n    def full(self) -> bool:\n        \"\"\"Return True if the queue is full, False otherwise (not reliable!).\n\n        This method is likely to be removed at some point.  Use qsize() >= n\n        as a direct substitute, but be aware that either approach risks a race\n        condition where a queue can shrink before the result of full() or\n        qsize() can be used.\n        \"\"\"\n        return 0 < self._parent._maxsize <= self._parent._qsize()\n\n    def put(self, item: T, block: bool = True, timeout: OptFloat = None) -> None:\n        \"\"\"Put an item into the queue.\n\n        If optional args 'block' is true and 'timeout' is None (the default),\n        block if necessary until a free slot is available. If 'timeout' is\n        a non-negative number, it blocks at most 'timeout' seconds and raises\n        the Full exception if no free slot was available within that time.\n        Otherwise ('block' is false), put an item on the queue if a free slot\n        is immediately available, else raise the Full exception ('timeout'\n        is ignored in that case).\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._sync_not_full:\n            if self._parent._maxsize > 0:\n                if not block:\n                    if self._parent._qsize() >= self._parent._maxsize:\n                        raise SyncQueueFull\n                elif timeout is None:\n                    while self._parent._qsize() >= self._parent._maxsize:\n                        self._parent._sync_not_full.wait()\n                elif timeout < 0:\n                    raise ValueError(\"'timeout' must be a non-negative number\")\n                else:\n                    time = self._parent._loop.time\n                    endtime = time() + timeout\n                    while self._parent._qsize() >= self._parent._maxsize:\n                        remaining = endtime - time()\n                        if remaining <= 0.0:\n                            raise SyncQueueFull\n                        self._parent._sync_not_full.wait(remaining)\n            self._parent._put_internal(item)\n            self._parent._sync_not_empty.notify()\n            self._parent._notify_async_not_empty(threadsafe=True)\n\n    def get(self, block: bool = True, timeout: OptFloat = None) -> T:\n        \"\"\"Remove and return an item from the queue.\n\n        If optional args 'block' is true and 'timeout' is None (the default),\n        block if necessary until an item is available. If 'timeout' is\n        a non-negative number, it blocks at most 'timeout' seconds and raises\n        the Empty exception if no item was available within that time.\n        Otherwise ('block' is false), return an item if one is immediately\n        available, else raise the Empty exception ('timeout' is ignored\n        in that case).\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._sync_not_empty:\n            if not block:\n                if not self._parent._qsize():\n                    raise SyncQueueEmpty\n            elif timeout is None:\n                while not self._parent._qsize():\n                    self._parent._sync_not_empty.wait()\n            elif timeout < 0:\n                raise ValueError(\"'timeout' must be a non-negative number\")\n            else:\n                time = self._parent._loop.time\n                endtime = time() + timeout\n                while not self._parent._qsize():\n                    remaining = endtime - time()\n                    if remaining <= 0.0:\n                        raise SyncQueueEmpty\n                    self._parent._sync_not_empty.wait(remaining)\n            item = self._parent._get()\n            self._parent._sync_not_full.notify()\n            self._parent._notify_async_not_full(threadsafe=True)\n            return item\n\n    def put_nowait(self, item: T) -> None:\n        \"\"\"Put an item into the queue without blocking.\n\n        Only enqueue the item if a free slot is immediately available.\n        Otherwise raise the Full exception.\n        \"\"\"\n        return self.put(item, block=False)\n\n    def get_nowait(self) -> T:\n        \"\"\"Remove and return an item from the queue without blocking.\n\n        Only get an item if one is immediately available. Otherwise\n        raise the Empty exception.\n        \"\"\"\n        return self.get(block=False)\n\n\nclass _AsyncQueueProxy(AsyncQueue[T]):\n    \"\"\"Create a queue object with a given maximum size.\n\n    If maxsize is <= 0, the queue size is infinite.\n    \"\"\"\n\n    def __init__(self, parent: Queue[T]):\n        self._parent = parent\n\n    @property\n    def closed(self) -> bool:\n        return self._parent.closed\n\n    def qsize(self) -> int:\n        \"\"\"Number of items in the queue.\"\"\"\n        return self._parent._qsize()\n\n    @property\n    def unfinished_tasks(self) -> int:\n        \"\"\"Return the number of unfinished tasks.\"\"\"\n        return self._parent._unfinished_tasks\n\n    @property\n    def maxsize(self) -> int:\n        \"\"\"Number of items allowed in the queue.\"\"\"\n        return self._parent._maxsize\n\n    def empty(self) -> bool:\n        \"\"\"Return True if the queue is empty, False otherwise.\"\"\"\n        return self.qsize() == 0\n\n    def full(self) -> bool:\n        \"\"\"Return True if there are maxsize items in the queue.\n\n        Note: if the Queue was initialized with maxsize=0 (the default),\n        then full() is never True.\n        \"\"\"\n        if self._parent._maxsize <= 0:\n            return False\n        else:\n            return self.qsize() >= self._parent._maxsize\n\n    async def put(self, item: T) -> None:\n        \"\"\"Put an item into the queue.\n\n        Put an item into the queue. If the queue is full, wait until a free\n        slot is available before adding item.\n\n        This method is a coroutine.\n        \"\"\"\n        self._parent._check_closing()\n        async with self._parent._async_not_full:\n            self._parent._sync_mutex.acquire()\n            locked = True\n            try:\n                if self._parent._maxsize > 0:\n                    do_wait = True\n                    while do_wait:\n                        do_wait = self._parent._qsize() >= self._parent._maxsize\n                        if do_wait:\n                            locked = False\n                            self._parent._sync_mutex.release()\n                            await self._parent._async_not_full.wait()\n                            self._parent._sync_mutex.acquire()\n                            locked = True\n\n                self._parent._put_internal(item)\n                self._parent._async_not_empty.notify()\n                self._parent._notify_sync_not_empty()\n            finally:\n                if locked:\n                    self._parent._sync_mutex.release()\n\n    def put_nowait(self, item: T) -> None:\n        \"\"\"Put an item into the queue without blocking.\n\n        If no free slot is immediately available, raise QueueFull.\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._sync_mutex:\n            if self._parent._maxsize > 0:\n                if self._parent._qsize() >= self._parent._maxsize:\n                    raise AsyncQueueFull\n\n            self._parent._put_internal(item)\n            self._parent._notify_async_not_empty(threadsafe=False)\n            self._parent._notify_sync_not_empty()\n\n    async def get(self) -> T:\n        \"\"\"Remove and return an item from the queue.\n\n        If queue is empty, wait until an item is available.\n\n        This method is a coroutine.\n        \"\"\"\n        self._parent._check_closing()\n        async with self._parent._async_not_empty:\n            self._parent._sync_mutex.acquire()\n            locked = True\n            try:\n                do_wait = True\n                while do_wait:\n                    do_wait = self._parent._qsize() == 0\n\n                    if do_wait:\n                        locked = False\n                        self._parent._sync_mutex.release()\n                        await self._parent._async_not_empty.wait()\n                        self._parent._sync_mutex.acquire()\n                        locked = True\n\n                item = self._parent._get()\n                self._parent._async_not_full.notify()\n                self._parent._notify_sync_not_full()\n                return item\n            finally:\n                if locked:\n                    self._parent._sync_mutex.release()\n\n    def get_nowait(self) -> T:\n        \"\"\"Remove and return an item from the queue.\n\n        Return an item if one is immediately available, else raise QueueEmpty.\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._sync_mutex:\n            if self._parent._qsize() == 0:\n                raise AsyncQueueEmpty\n\n            item = self._parent._get()\n            self._parent._notify_async_not_full(threadsafe=False)\n            self._parent._notify_sync_not_full()\n            return item\n\n    def task_done(self) -> None:\n        \"\"\"Indicate that a formerly enqueued task is complete.\n\n        Used by queue consumers. For each get() used to fetch a task,\n        a subsequent call to task_done() tells the queue that the processing\n        on the task is complete.\n\n        If a join() is currently blocking, it will resume when all items have\n        been processed (meaning that a task_done() call was received for every\n        item that had been put() into the queue).\n\n        Raises ValueError if called more times than there were items placed in\n        the queue.\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._all_tasks_done:\n            if self._parent._unfinished_tasks <= 0:\n                raise ValueError(\"task_done() called too many times\")\n            self._parent._unfinished_tasks -= 1\n            if self._parent._unfinished_tasks == 0:\n                self._parent._finished.set()\n                self._parent._all_tasks_done.notify_all()\n\n    async def join(self) -> None:\n        \"\"\"Block until all items in the queue have been gotten and processed.\n\n        The count of unfinished tasks goes up whenever an item is added to the\n        queue. The count goes down whenever a consumer calls task_done() to\n        indicate that the item was retrieved and all work on it is complete.\n        When the count of unfinished tasks drops to zero, join() unblocks.\n        \"\"\"\n        while True:\n            with self._parent._sync_mutex:\n                self._parent._check_closing()\n                if self._parent._unfinished_tasks == 0:\n                    break\n            await self._parent._finished.wait()\n\n\nclass PriorityQueue(Queue[T]):\n    \"\"\"Variant of Queue that retrieves open entries in priority order\n    (lowest first).\n\n    Entries are typically tuples of the form:  (priority number, data).\n\n    \"\"\"\n\n    def _init(self, maxsize: int) -> None:\n        self._heap_queue = []  # type: List[T]\n\n    def _qsize(self) -> int:\n        return len(self._heap_queue)\n\n    def _put(self, item: T) -> None:\n        heappush(self._heap_queue, item)\n\n    def _get(self) -> T:\n        return heappop(self._heap_queue)\n\n\nclass LifoQueue(Queue[T]):\n    \"\"\"Variant of Queue that retrieves most recently added entries first.\"\"\"\n\n    def _qsize(self) -> int:\n        return len(self._queue)\n\n    def _put(self, item: T) -> None:\n        self._queue.append(item)\n\n    def _get(self) -> T:\n        return self._queue.pop()\n"
    }
  ],
  "OriginCode": [
    {
      "path": "aio-libs_janus/setup.py",
      "content": "from setuptools import setup\n\nsetup()\n"
    },
    {
      "path": "aio-libs_janus/tests/test_sync.py",
      "content": "# Some simple queue module tests, plus some failure conditions\n# to ensure the Queue locks remain stable.\nimport asyncio\nimport queue\nimport threading\nimport time\nfrom unittest.mock import patch\n\nimport pytest\n\nimport janus\n\nQUEUE_SIZE = 5\n\n\ndef qfull(q):\n    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n\n\n# A thread to run a function that unclogs a blocked Queue.\nclass _TriggerThread(threading.Thread):\n    def __init__(self, fn, args):\n        self.fn = fn\n        self.args = args\n        self.startedEvent = threading.Event()\n        threading.Thread.__init__(self)\n\n    def run(self):\n        # The sleep isn't necessary, but is intended to give the blocking\n        # function in the main thread a chance at actually blocking before\n        # we unclog it.  But if the sleep is longer than the timeout-based\n        # tests wait in their blocking functions, those tests will fail.\n        # So we give them much longer timeout values compared to the\n        # sleep here (I aimed at 10 seconds for blocking functions --\n        # they should never actually wait that long - they should make\n        # progress as soon as we call self.fn()).\n        time.sleep(0.1)\n        self.startedEvent.set()\n        self.fn(*self.args)\n\n\n# Execute a function that blocks, and in a separate thread, a function that\n# triggers the release.  Returns the result of the blocking function.  Caution:\n# block_func must guarantee to block until trigger_func is called, and\n# trigger_func must guarantee to change queue state so that block_func can make\n# enough progress to return.  In particular, a block_func that just raises an\n# exception regardless of whether trigger_func is called will lead to\n# timing-dependent sporadic failures, and one of those went rarely seen but\n# undiagnosed for years.  Now block_func must be unexceptional.  If block_func\n# is supposed to raise an exception, call do_exceptional_blocking_test()\n# instead.\n\n\nclass BlockingTestMixin:\n    def do_blocking_test(self, block_func, block_args, trigger_func, trigger_args):\n        self.t = _TriggerThread(trigger_func, trigger_args)\n        self.t.start()\n        self.result = block_func(*block_args)\n        # If block_func returned before our thread made the call, we failed!\n        if not self.t.startedEvent.is_set():\n            pytest.fail(\"blocking function '%r' appeared not to block\" % block_func)\n        self.t.join(10)  # make sure the thread terminates\n        if self.t.is_alive():\n            pytest.fail(\"trigger function '%r' appeared to not return\" % trigger_func)\n        return self.result\n\n    # Call this instead if block_func is supposed to raise an exception.\n    def do_exceptional_blocking_test(\n        self,\n        block_func,\n        block_args,\n        trigger_func,\n        trigger_args,\n        expected_exception_class,\n    ):\n        self.t = _TriggerThread(trigger_func, trigger_args)\n        self.t.start()\n        try:\n            try:\n                block_func(*block_args)\n            except expected_exception_class:\n                raise\n            else:\n                pytest.fail(\"expected exception of kind %r\" % expected_exception_class)\n        finally:\n            self.t.join(10)  # make sure the thread terminates\n            if self.t.is_alive():\n                pytest.fail(\n                    \"trigger function '%r' appeared to not return\" % trigger_func\n                )\n            if not self.t.startedEvent.is_set():\n                pytest.fail(\"trigger thread ended but event never set\")\n\n\nclass BaseQueueTestMixin(BlockingTestMixin):\n    cum = 0\n    cumlock = threading.Lock()\n\n    def simple_queue_test(self, _q):\n        q = _q.sync_q\n        if q.qsize():\n            raise RuntimeError(\"Call this function with an empty queue\")\n        assert q.empty()\n        assert not q.full()\n        # I guess we better check things actually queue correctly a little :)\n        q.put(111)\n        q.put(333)\n        q.put(222)\n        target_order = dict(\n            Queue=[111, 333, 222],\n            LifoQueue=[222, 333, 111],\n            PriorityQueue=[111, 222, 333],\n        )\n        actual_order = [q.get(), q.get(), q.get()]\n        assert actual_order == target_order[_q.__class__.__name__]\n        for i in range(QUEUE_SIZE - 1):\n            q.put(i)\n            assert q.qsize()\n        assert not qfull(q)\n        last = 2 * QUEUE_SIZE\n        full = 3 * 2 * QUEUE_SIZE\n        q.put(last)\n        assert qfull(q)\n        assert not q.empty()\n        assert q.full()\n        try:\n            q.put(full, block=0)\n            pytest.fail(\"Didn't appear to block with a full queue\")\n        except queue.Full:\n            pass\n        try:\n            q.put(full, timeout=0.01)\n            pytest.fail(\"Didn't appear to time-out with a full queue\")\n        except queue.Full:\n            pass\n        # Test a blocking put\n        self.do_blocking_test(q.put, (full,), q.get, ())\n        self.do_blocking_test(q.put, (full, True, 10), q.get, ())\n        # Empty it\n        for i in range(QUEUE_SIZE):\n            q.get()\n        assert not q.qsize()\n        try:\n            q.get(block=0)\n            pytest.fail(\"Didn't appear to block with an empty queue\")\n        except queue.Empty:\n            pass\n        try:\n            q.get(timeout=0.01)\n            pytest.fail(\"Didn't appear to time-out with an empty queue\")\n        except queue.Empty:\n            pass\n        # Test a blocking get\n        self.do_blocking_test(q.get, (), q.put, (\"empty\",))\n        self.do_blocking_test(q.get, (True, 10), q.put, (\"empty\",))\n\n    def worker(self, q):\n        try:\n            while True:\n                x = q.get()\n                if x < 0:\n                    q.task_done()\n                    return\n                with self.cumlock:\n                    self.cum += x\n                q.task_done()\n        except Exception as ex:\n            from traceback import print_exc\n\n            print_exc(ex)\n\n    def queue_join_test(self, q):\n        self.cum = 0\n        for i in (0, 1):\n            threading.Thread(target=self.worker, args=(q,)).start()\n        for i in range(100):\n            q.put(i)\n        q.join()\n        assert self.cum == sum(range(100))\n        for i in (0, 1):\n            q.put(-1)  # instruct the threads to close\n        q.join()  # verify that you can join twice\n\n    @pytest.mark.asyncio\n    async def test_queue_task_done(self):\n        # Test to make sure a queue task completed successfully.\n        _q = self.type2test()\n        q = _q.sync_q\n        try:\n            q.task_done()\n        except ValueError:\n            pass\n        else:\n            pytest.fail(\"Did not detect task count going negative\")\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_queue_join(self):\n        # Test that a queue join()s successfully, and before anything else\n        # (done twice for insurance).\n        _q = self.type2test()\n        q = _q.sync_q\n        self.queue_join_test(q)\n        self.queue_join_test(q)\n        try:\n            q.task_done()\n        except ValueError:\n            pass\n        else:\n            pytest.fail(\"Did not detect task count going negative\")\n        finally:\n            _q.close()\n            await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_simple_queue(self):\n        # Do it a couple of times on the same queue.\n        # Done twice to make sure works with same instance reused.\n        _q = self.type2test(QUEUE_SIZE)\n        self.simple_queue_test(_q)\n        self.simple_queue_test(_q)\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_negative_timeout_raises_exception(self):\n        _q = self.type2test(QUEUE_SIZE)\n        q = _q.sync_q\n        with pytest.raises(ValueError):\n            q.put(1, timeout=-1)\n        with pytest.raises(ValueError):\n            q.get(1, timeout=-1)\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nowait(self):\n        _q = self.type2test(QUEUE_SIZE)\n        q = _q.sync_q\n        for i in range(QUEUE_SIZE):\n            q.put_nowait(1)\n        with pytest.raises(queue.Full):\n            q.put_nowait(1)\n\n        for i in range(QUEUE_SIZE):\n            q.get_nowait()\n        with pytest.raises(queue.Empty):\n            q.get_nowait()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_shrinking_queue(self):\n        # issue 10110\n        _q = self.type2test(3)\n        q = _q.sync_q\n        q.put(1)\n        q.put(2)\n        q.put(3)\n        with pytest.raises(queue.Full):\n            q.put_nowait(4)\n        assert q.qsize() == 3\n        q._maxsize = 2  # shrink the queue\n        with pytest.raises(queue.Full):\n            q.put_nowait(4)\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_maxsize(self):\n        # Test to make sure a queue task completed successfully.\n        _q = self.type2test(5)\n        q = _q.sync_q\n        assert q.maxsize == 5\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestQueue(BaseQueueTestMixin):\n    type2test = janus.Queue\n\n\nclass TestLifoQueue(BaseQueueTestMixin):\n    type2test = janus.LifoQueue\n\n\nclass TestPriorityQueue(BaseQueueTestMixin):\n    type2test = janus.PriorityQueue\n\n\n# A Queue subclass that can provoke failure at a moment's notice :)\nclass FailingQueueException(Exception):\n    pass\n\n\nclass FailingQueue(janus.Queue):\n    def __init__(self, *args, **kwargs):\n        self.fail_next_put = False\n        self.fail_next_get = False\n        super().__init__(*args, **kwargs)\n\n    def _put(self, item):\n        if self.fail_next_put:\n            self.fail_next_put = False\n            raise FailingQueueException(\"You Lose\")\n        return super()._put(item)\n\n    def _get(self):\n        if self.fail_next_get:\n            self.fail_next_get = False\n            raise FailingQueueException(\"You Lose\")\n        return super()._get()\n\n\nclass TestFailingQueue(BlockingTestMixin):\n    def failing_queue_test(self, _q):\n        q = _q.sync_q\n        if q.qsize():\n            raise RuntimeError(\"Call this function with an empty queue\")\n        for i in range(QUEUE_SIZE - 1):\n            q.put(i)\n        # Test a failing non-blocking put.\n        _q.fail_next_put = True\n        try:\n            q.put(\"oops\", block=0)\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        _q.fail_next_put = True\n        try:\n            q.put(\"oops\", timeout=0.1)\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        q.put(\"last\")\n        assert qfull(q)\n        # Test a failing blocking put\n        _q.fail_next_put = True\n        try:\n            self.do_blocking_test(q.put, (\"full\",), q.get, ())\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        # Check the Queue isn't damaged.\n        # put failed, but get succeeded - re-add\n        q.put(\"last\")\n        # Test a failing timeout put\n        _q.fail_next_put = True\n        try:\n            self.do_exceptional_blocking_test(\n                q.put, (\"full\", True, 10), q.get, (), FailingQueueException\n            )\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        # Check the Queue isn't damaged.\n        # put failed, but get succeeded - re-add\n        q.put(\"last\")\n        assert qfull(q)\n        q.get()\n        assert not qfull(q)\n        q.put(\"last\")\n        assert qfull(q)\n        # Test a blocking put\n        self.do_blocking_test(q.put, (\"full\",), q.get, ())\n        # Empty it\n        for i in range(QUEUE_SIZE):\n            q.get()\n        assert not q.qsize()\n        q.put(\"first\")\n        _q.fail_next_get = True\n        try:\n            q.get()\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        assert q.qsize()\n        _q.fail_next_get = True\n        try:\n            q.get(timeout=0.1)\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        assert q.qsize()\n        q.get()\n        assert not q.qsize()\n        _q.fail_next_get = True\n        try:\n            self.do_exceptional_blocking_test(\n                q.get, (), q.put, (\"empty\",), FailingQueueException\n            )\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        # put succeeded, but get failed.\n        assert q.qsize()\n        q.get()\n        assert not q.qsize()\n\n    @pytest.mark.asyncio\n    async def test_failing_queue(self):\n        # Test to make sure a queue is functioning correctly.\n        # Done twice to the same instance.\n        q = FailingQueue(QUEUE_SIZE)\n        self.failing_queue_test(q)\n        self.failing_queue_test(q)\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_closed_loop_non_failing(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(QUEUE_SIZE)\n        q = _q.sync_q\n        # we are pacthing loop to follow setUp/tearDown agreement\n        with patch.object(loop, \"call_soon_threadsafe\") as func:\n            func.side_effect = RuntimeError()\n            q.put_nowait(1)\n            assert func.call_count == 1\n        _q.close()\n        await _q.wait_closed()\n"
    },
    {
      "path": "aio-libs_janus/tests/test_async.py",
      "content": "\"\"\"Tests for queues.py\"\"\"\n\nimport asyncio\n\nimport pytest\n\nimport janus\n\n\nclass TestQueueBasic:\n    async def _test_repr_or_str(self, fn, expect_id):\n        \"\"\"Test Queue's repr or str.\n\n        fn is repr or str. expect_id is True if we expect the Queue's id to\n        appear in fn(Queue()).\n        \"\"\"\n\n        _q = janus.Queue()\n        q = _q.async_q\n        assert fn(q).startswith(\"<Queue\")\n        id_is_present = hex(id(q)) in fn(q)\n        assert expect_id == id_is_present\n        loop = asyncio.get_running_loop()\n\n        async def add_getter():\n            _q = janus.Queue()\n            q = _q.async_q\n            # Start a task that waits to get.\n            loop.create_task(q.get())\n            # Let it start waiting.\n            await asyncio.sleep(0.1)\n            assert \"_getters[1]\" in fn(q)\n            # resume q.get coroutine to finish generator\n            q.put_nowait(0)\n\n        await add_getter()\n\n        async def add_putter():\n            _q = janus.Queue(maxsize=1)\n            q = _q.async_q\n            q.put_nowait(1)\n            # Start a task that waits to put.\n            loop.create_task(q.put(2))\n            # Let it start waiting.\n            await asyncio.sleep(0.1)\n            assert \"_putters[1]\" in fn(q)\n            # resume q.put coroutine to finish generator\n            q.get_nowait()\n\n        await add_putter()\n\n        _q = janus.Queue()\n        q = _q.async_q\n        q.put_nowait(1)\n        assert \"_queue=[1]\" in fn(q)\n\n    # def test_repr(self):\n    #     self._test_repr_or_str(repr, True)\n\n    # def test_str(self):\n    #     self._test_repr_or_str(str, False)\n\n    @pytest.mark.asyncio\n    async def test_empty(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        assert q.empty()\n        q.put_nowait(1)\n        assert not q.empty()\n        assert 1 == q.get_nowait()\n        assert q.empty()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_full(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        assert not q.full()\n\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n        q.put_nowait(1)\n        assert q.full()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_order(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        for i in [1, 3, 2]:\n            q.put_nowait(i)\n\n        items = [q.get_nowait() for _ in range(3)]\n        assert [1, 3, 2] == items\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_maxsize(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=2)\n        q = _q.async_q\n        assert 2 == q.maxsize\n        have_been_put = []\n\n        fut = loop.create_future()\n\n        async def putter():\n            for i in range(3):\n                await q.put(i)\n                have_been_put.append(i)\n                if i == q.maxsize - 1:\n                    fut.set_result(None)\n            return True\n\n        async def test():\n            t = loop.create_task(putter())\n            await fut\n\n            # The putter is blocked after putting two items.\n            assert [0, 1] == have_been_put\n            assert 0 == q.get_nowait()\n\n            # Let the putter resume and put last item.\n            await t\n            assert [0, 1, 2] == have_been_put\n            assert 1 == q.get_nowait()\n            assert 2 == q.get_nowait()\n\n        await test()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestQueueGetTests:\n    @pytest.mark.asyncio\n    async def test_blocking_get(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        q.put_nowait(1)\n\n        res = await q.get()\n        assert 1 == res\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_get_with_putters(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(1)\n        q = _q.async_q\n        q.put_nowait(1)\n\n        fut = loop.create_future()\n\n        async def put():\n            t = asyncio.ensure_future(q.put(2))\n            await asyncio.sleep(0.01)\n            fut.set_result(None)\n            return t\n\n        t = await put()\n\n        res = await q.get()\n        assert 1 == res\n\n        await t\n        assert 1 == q.qsize()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_blocking_get_wait(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue()\n        q = _q.async_q\n        started = asyncio.Event()\n        finished = False\n\n        async def queue_get():\n            nonlocal finished\n            started.set()\n            res = await q.get()\n            finished = True\n            return res\n\n        async def queue_put():\n            loop.call_later(0.01, q.put_nowait, 1)\n            queue_get_task = loop.create_task(queue_get())\n            await started.wait()\n            assert not finished\n            res = await queue_get_task\n            assert finished\n            return res\n\n        res = await queue_put()\n        assert 1 == res\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nonblocking_get(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        q.put_nowait(1)\n        assert 1 == q.get_nowait()\n\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nonblocking_get_exception(self):\n        _q = janus.Queue()\n        pytest.raises(asyncio.QueueEmpty, _q.async_q.get_nowait)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_get_cancelled(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue()\n        q = _q.async_q\n\n        async def queue_get():\n            return await asyncio.wait_for(q.get(), 0.051)\n\n        async def test():\n            get_task = loop.create_task(queue_get())\n            await asyncio.sleep(0.01)  # let the task start\n            q.put_nowait(1)\n            return await get_task\n\n        assert 1 == await test()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_get_cancelled_race(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue()\n        q = _q.async_q\n\n        f1 = loop.create_future()\n\n        async def g1():\n            f1.set_result(None)\n            await q.get()\n\n        t1 = loop.create_task(g1())\n        t2 = loop.create_task(q.get())\n\n        await f1\n        await asyncio.sleep(0.01)\n        t1.cancel()\n\n        with pytest.raises(asyncio.CancelledError):\n            await t1\n        assert t1.done()\n        q.put_nowait(\"a\")\n\n        await t2\n        assert t2.result() == \"a\"\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_get_with_waiting_putters(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n\n        loop.create_task(q.put(\"a\"))\n        loop.create_task(q.put(\"b\"))\n\n        await asyncio.sleep(0.01)\n\n        assert await q.get() == \"a\"\n        assert await q.get() == \"b\"\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestQueuePut:\n    @pytest.mark.asyncio\n    async def test_blocking_put(self):\n        _q = janus.Queue()\n        q = _q.async_q\n\n        # No maxsize, won't block.\n        await q.put(1)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_blocking_put_wait(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n        started = asyncio.Event()\n        finished = False\n\n        async def queue_put():\n            nonlocal finished\n            started.set()\n            await q.put(1)\n            await q.put(2)\n            finished = True\n\n        async def queue_get():\n            loop.call_later(0.01, q.get_nowait)\n            queue_put_task = loop.create_task(queue_put())\n            await started.wait()\n            assert not finished\n            await queue_put_task\n            assert finished\n\n        await queue_get()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nonblocking_put(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        q.put_nowait(1)\n        assert 1 == q.get_nowait()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nonblocking_put_exception(self):\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n        q.put_nowait(1)\n        pytest.raises(asyncio.QueueFull, q.put_nowait, 2)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_float_maxsize(self):\n        _q = janus.Queue(maxsize=1.3)\n        q = _q.async_q\n        q.put_nowait(1)\n        q.put_nowait(2)\n        assert q.full()\n        pytest.raises(asyncio.QueueFull, q.put_nowait, 3)\n\n        _q.close()\n        await _q.wait_closed()\n\n        _q = janus.Queue(maxsize=1.3)\n        q = _q.async_q\n\n        async def queue_put():\n            await q.put(1)\n            await q.put(2)\n            assert q.full()\n\n        await queue_put()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_put_cancelled(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue()\n        q = _q.async_q\n\n        async def queue_put():\n            await q.put(1)\n            return True\n\n        async def test():\n            return await q.get()\n\n        t = loop.create_task(queue_put())\n        assert 1 == await test()\n        assert t.done()\n        assert t.result()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_put_cancelled_race(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n\n        put_a = loop.create_task(q.put(\"a\"))\n        put_b = loop.create_task(q.put(\"b\"))\n        put_c = loop.create_task(q.put(\"X\"))\n\n        await put_a\n        assert not put_b.done()\n\n        put_c.cancel()\n\n        with pytest.raises(asyncio.CancelledError):\n            await put_c\n\n        async def go():\n            a = await q.get()\n            assert a == \"a\"\n            b = await q.get()\n            assert b == \"b\"\n            assert put_b.done()\n\n        await go()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_put_with_waiting_getters(self):\n        loop = asyncio.get_running_loop()\n        fut = loop.create_future()\n\n        async def go():\n            fut.set_result(None)\n            ret = await q.get()\n            return ret\n\n        async def put():\n            await q.put(\"a\")\n\n        _q = janus.Queue()\n        q = _q.async_q\n        t = loop.create_task(go())\n        await fut\n        await put()\n        assert await t == \"a\"\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestLifoQueue:\n    @pytest.mark.asyncio\n    async def test_order(self):\n        _q = janus.LifoQueue()\n        q = _q.async_q\n        for i in [1, 3, 2]:\n            q.put_nowait(i)\n\n        items = [q.get_nowait() for _ in range(3)]\n        assert [2, 3, 1] == items\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestPriorityQueue:\n    @pytest.mark.asyncio\n    async def test_order(self):\n        _q = janus.PriorityQueue()\n        q = _q.async_q\n        for i in [1, 3, 2]:\n            q.put_nowait(i)\n\n        items = [q.get_nowait() for _ in range(3)]\n        assert [1, 2, 3] == items\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass _QueueJoinTestMixin:\n    q_class = None\n\n    @pytest.mark.asyncio\n    async def test_task_done_underflow(self):\n        _q = self.q_class()\n        q = _q.async_q\n        pytest.raises(ValueError, q.task_done)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_task_done(self):\n        loop = asyncio.get_running_loop()\n        _q = self.q_class()\n        q = _q.async_q\n        for i in range(100):\n            q.put_nowait(i)\n\n        accumulator = 0\n\n        # Two workers get items from the queue and call task_done after each.\n        # Join the queue and assert all items have been processed.\n        running = True\n\n        async def worker():\n            nonlocal accumulator\n\n            while running:\n                item = await q.get()\n                accumulator += item\n                q.task_done()\n\n        async def test():\n            tasks = [loop.create_task(worker()) for index in range(2)]\n\n            await q.join()\n            return tasks\n\n        tasks = await test()\n        assert sum(range(100)) == accumulator\n\n        # close running generators\n        running = False\n        for i in range(len(tasks)):\n            q.put_nowait(0)\n        await asyncio.wait(tasks)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_join_empty_queue(self):\n        _q = self.q_class()\n        q = _q.async_q\n\n        # Test that a queue join()s successfully, and before anything else\n        # (done twice for insurance).\n\n        async def join():\n            await q.join()\n            await q.join()\n\n        await join()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestQueueJoin(_QueueJoinTestMixin):\n    q_class = janus.Queue\n\n\nclass TestLifoQueueJoin(_QueueJoinTestMixin):\n    q_class = janus.LifoQueue\n\n\nclass TestPriorityQueueJoin(_QueueJoinTestMixin):\n    q_class = janus.PriorityQueue\n"
    },
    {
      "path": "aio-libs_janus/tests/test_mixed.py",
      "content": "import asyncio\nimport contextlib\nimport sys\nimport threading\n\nimport pytest\n\nimport janus\n\n\nclass TestMixedMode:\n    @pytest.mark.skipif(\n        sys.version_info < (3, 7),\n        reason=\"forbidding implicit loop creation works on \"\n        \"Python 3.7 or higher only\",\n    )\n    def test_ctor_noloop(self):\n        with pytest.raises(RuntimeError):\n            janus.Queue()\n\n    @pytest.mark.asyncio\n    async def test_maxsize(self):\n        q = janus.Queue(5)\n        assert 5 == q.maxsize\n\n    @pytest.mark.asyncio\n    async def test_maxsize_named_param(self):\n        q = janus.Queue(maxsize=7)\n        assert 7 == q.maxsize\n\n    @pytest.mark.asyncio\n    async def test_maxsize_default(self):\n        q = janus.Queue()\n        assert 0 == q.maxsize\n\n    @pytest.mark.asyncio\n    async def test_unfinished(self):\n        q = janus.Queue()\n        assert q.sync_q.unfinished_tasks == 0\n        assert q.async_q.unfinished_tasks == 0\n        q.sync_q.put(1)\n        assert q.sync_q.unfinished_tasks == 1\n        assert q.async_q.unfinished_tasks == 1\n        q.sync_q.get()\n        assert q.sync_q.unfinished_tasks == 1\n        assert q.async_q.unfinished_tasks == 1\n        q.sync_q.task_done()\n        assert q.sync_q.unfinished_tasks == 0\n        assert q.async_q.unfinished_tasks == 0\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_sync_put_async_get(self):\n        loop = asyncio.get_running_loop()\n        q = janus.Queue()\n\n        def threaded():\n            for i in range(5):\n                q.sync_q.put(i)\n\n        async def go():\n            f = loop.run_in_executor(None, threaded)\n            for i in range(5):\n                val = await q.async_q.get()\n                assert val == i\n\n            assert q.async_q.empty()\n\n            await f\n\n        for i in range(3):\n            await go()\n\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_sync_put_async_join(self):\n        loop = asyncio.get_running_loop()\n        q = janus.Queue()\n\n        for i in range(5):\n            q.sync_q.put(i)\n\n        async def do_work():\n            await asyncio.sleep(1)\n            while True:\n                await q.async_q.get()\n                q.async_q.task_done()\n\n        task = loop.create_task(do_work())\n\n        async def wait_for_empty_queue():\n            await q.async_q.join()\n            task.cancel()\n\n        await wait_for_empty_queue()\n\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_async_put_sync_get(self):\n        loop = asyncio.get_running_loop()\n        q = janus.Queue()\n\n        def threaded():\n            for i in range(5):\n                val = q.sync_q.get()\n                assert val == i\n\n        async def go():\n            f = loop.run_in_executor(None, threaded)\n            for i in range(5):\n                await q.async_q.put(i)\n\n            await f\n            assert q.async_q.empty()\n\n        for i in range(3):\n            await go()\n\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_sync_join_async_done(self):\n        loop = asyncio.get_running_loop()\n        q = janus.Queue()\n\n        def threaded():\n            for i in range(5):\n                q.sync_q.put(i)\n            q.sync_q.join()\n\n        async def go():\n            f = loop.run_in_executor(None, threaded)\n            for i in range(5):\n                val = await q.async_q.get()\n                assert val == i\n                q.async_q.task_done()\n\n            assert q.async_q.empty()\n\n            await f\n\n        for i in range(3):\n            await go()\n\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_async_join_async_done(self):\n        loop = asyncio.get_running_loop()\n        q = janus.Queue()\n\n        def threaded():\n            for i in range(5):\n                val = q.sync_q.get()\n                assert val == i\n                q.sync_q.task_done()\n\n        async def go():\n            f = loop.run_in_executor(None, threaded)\n            for i in range(5):\n                await q.async_q.put(i)\n\n            await q.async_q.join()\n\n            await f\n            assert q.async_q.empty()\n\n        for i in range(3):\n            await go()\n\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_wait_without_closing(self):\n        q = janus.Queue()\n\n        with pytest.raises(RuntimeError):\n            await q.wait_closed()\n\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_modifying_forbidden_after_closing(self):\n        q = janus.Queue()\n        q.close()\n\n        with pytest.raises(RuntimeError):\n            q.sync_q.put(5)\n\n        with pytest.raises(RuntimeError):\n            q.sync_q.get()\n\n        with pytest.raises(RuntimeError):\n            q.sync_q.task_done()\n\n        with pytest.raises(RuntimeError):\n            await q.async_q.put(5)\n\n        with pytest.raises(RuntimeError):\n            q.async_q.put_nowait(5)\n\n        with pytest.raises(RuntimeError):\n            q.async_q.get_nowait()\n\n        with pytest.raises(RuntimeError):\n            await q.sync_q.task_done()\n\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_double_closing(self):\n        q = janus.Queue()\n        q.close()\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_closed(self):\n        q = janus.Queue()\n        assert not q.closed\n        assert not q.async_q.closed\n        assert not q.sync_q.closed\n        q.close()\n        assert q.closed\n        assert q.async_q.closed\n        assert q.sync_q.closed\n\n    @pytest.mark.asyncio\n    async def test_async_join_after_closing(self):\n        q = janus.Queue()\n        q.close()\n        with pytest.raises(RuntimeError), contextlib.suppress(asyncio.TimeoutError):\n            await asyncio.wait_for(q.async_q.join(), timeout=0.1)\n\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_close_after_async_join(self):\n        q = janus.Queue()\n        q.sync_q.put(1)\n\n        task = asyncio.ensure_future(q.async_q.join())\n        await asyncio.sleep(0.1)  # ensure tasks are blocking\n\n        q.close()\n        with pytest.raises(RuntimeError), contextlib.suppress(asyncio.TimeoutError):\n            await asyncio.wait_for(task, timeout=0.1)\n\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_sync_join_after_closing(self):\n        q = janus.Queue()\n        q.sync_q.put(1)\n\n        q.close()\n\n        loop = asyncio.get_event_loop()\n        fut = asyncio.Future()\n\n        def sync_join():\n            try:\n                q.sync_q.join()\n            except Exception as exc:\n                loop.call_soon_threadsafe(fut.set_exception, exc)\n\n        thr = threading.Thread(target=sync_join, daemon=True)\n        thr.start()\n\n        with pytest.raises(RuntimeError), contextlib.suppress(asyncio.TimeoutError):\n            await asyncio.wait_for(fut, timeout=0.1)\n\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_close_after_sync_join(self):\n        q = janus.Queue()\n        q.sync_q.put(1)\n\n        loop = asyncio.get_event_loop()\n        fut = asyncio.Future()\n\n        def sync_join():\n            try:\n                q.sync_q.join()\n            except Exception as exc:\n                loop.call_soon_threadsafe(fut.set_exception, exc)\n\n        thr = threading.Thread(target=sync_join, daemon=True)\n        thr.start()\n        thr.join(0.1)  # ensure tasks are blocking\n\n        q.close()\n\n        with pytest.raises(RuntimeError), contextlib.suppress(asyncio.TimeoutError):\n            await asyncio.wait_for(fut, timeout=0.1)\n\n        await q.wait_closed()\n"
    },
    {
      "path": "aio-libs_janus/janus/__init__.py",
      "content": "import asyncio\nimport sys\nimport threading\nfrom asyncio import QueueEmpty as AsyncQueueEmpty\nfrom asyncio import QueueFull as AsyncQueueFull\nfrom collections import deque\nfrom heapq import heappop, heappush\nfrom queue import Empty as SyncQueueEmpty\nfrom queue import Full as SyncQueueFull\nfrom typing import Any, Callable, Deque, Generic, List, Optional, Set, TypeVar\n\nfrom typing_extensions import Protocol\n\n__version__ = \"1.0.0\"\n__all__ = (\n    \"Queue\",\n    \"PriorityQueue\",\n    \"LifoQueue\",\n    \"SyncQueue\",\n    \"AsyncQueue\",\n    \"BaseQueue\",\n)\n\n\nT = TypeVar(\"T\")\nOptFloat = Optional[float]\n\n\nclass BaseQueue(Protocol[T]):\n    @property\n    def maxsize(self) -> int:\n        ...\n\n    @property\n    def closed(self) -> bool:\n        ...\n\n    def task_done(self) -> None:\n        ...\n\n    def qsize(self) -> int:\n        ...\n\n    @property\n    def unfinished_tasks(self) -> int:\n        ...\n\n    def empty(self) -> bool:\n        ...\n\n    def full(self) -> bool:\n        ...\n\n    def put_nowait(self, item: T) -> None:\n        ...\n\n    def get_nowait(self) -> T:\n        ...\n\n\nclass SyncQueue(BaseQueue[T], Protocol[T]):\n\n    def put(self, item: T, block: bool = True, timeout: OptFloat = None) -> None:\n        ...\n\n    def get(self, block: bool = True, timeout: OptFloat = None) -> T:\n        ...\n\n    def join(self) -> None:\n        ...\n\n\nclass AsyncQueue(BaseQueue[T], Protocol[T]):\n    async def put(self, item: T) -> None:\n        ...\n\n    async def get(self) -> T:\n        ...\n\n    async def join(self) -> None:\n        ...\n\n\nclass Queue(Generic[T]):\n    def __init__(self, maxsize: int = 0) -> None:\n        self._loop = asyncio.get_running_loop()\n        self._maxsize = maxsize\n\n        self._init(maxsize)\n\n        self._unfinished_tasks = 0\n\n        self._sync_mutex = threading.Lock()\n        self._sync_not_empty = threading.Condition(self._sync_mutex)\n        self._sync_not_full = threading.Condition(self._sync_mutex)\n        self._all_tasks_done = threading.Condition(self._sync_mutex)\n\n        self._async_mutex = asyncio.Lock()\n        if sys.version_info[:3] == (3, 10, 0):\n            # Workaround for Python 3.10 bug, see #358:\n            getattr(self._async_mutex, \"_get_loop\", lambda: None)()\n        self._async_not_empty = asyncio.Condition(self._async_mutex)\n        self._async_not_full = asyncio.Condition(self._async_mutex)\n        self._finished = asyncio.Event()\n        self._finished.set()\n\n        self._closing = False\n        self._pending = set()  # type: Set[asyncio.Future[Any]]\n\n        def checked_call_soon_threadsafe(\n            callback: Callable[..., None], *args: Any\n        ) -> None:\n            try:\n                self._loop.call_soon_threadsafe(callback, *args)\n            except RuntimeError:\n                # swallowing agreed in #2\n                pass\n\n        self._call_soon_threadsafe = checked_call_soon_threadsafe\n\n        def checked_call_soon(callback: Callable[..., None], *args: Any) -> None:\n            if not self._loop.is_closed():\n                self._loop.call_soon(callback, *args)\n\n        self._call_soon = checked_call_soon\n\n        self._sync_queue = _SyncQueueProxy(self)\n        self._async_queue = _AsyncQueueProxy(self)\n\n    def close(self) -> None:\n        with self._sync_mutex:\n            self._closing = True\n            for fut in self._pending:\n                fut.cancel()\n            self._finished.set()  # unblocks all async_q.join()\n            self._all_tasks_done.notify_all()  # unblocks all sync_q.join()\n\n    async def wait_closed(self) -> None:\n        # should be called from loop after close().\n        # Nobody should put/get at this point,\n        # so lock acquiring is not required\n        if not self._closing:\n            raise RuntimeError(\"Waiting for non-closed queue\")\n        # give execution chances for the task-done callbacks\n        # of async tasks created inside\n        # _notify_async_not_empty, _notify_async_not_full\n        # methods.\n        await asyncio.sleep(0)\n        if not self._pending:\n            return\n        await asyncio.wait(self._pending)\n\n    @property\n    def closed(self) -> bool:\n        return self._closing and not self._pending\n\n    @property\n    def maxsize(self) -> int:\n        return self._maxsize\n\n    @property\n    def sync_q(self) -> \"_SyncQueueProxy[T]\":\n        return self._sync_queue\n\n    @property\n    def async_q(self) -> \"_AsyncQueueProxy[T]\":\n        return self._async_queue\n\n    # Override these methods to implement other queue organizations\n    # (e.g. stack or priority queue).\n    # These will only be called with appropriate locks held\n\n    def _init(self, maxsize: int) -> None:\n        self._queue = deque()  # type: Deque[T]\n\n    def _qsize(self) -> int:\n        return len(self._queue)\n\n    # Put a new item in the queue\n    def _put(self, item: T) -> None:\n        self._queue.append(item)\n\n    # Get an item from the queue\n    def _get(self) -> T:\n        return self._queue.popleft()\n\n    def _put_internal(self, item: T) -> None:\n        self._put(item)\n        self._unfinished_tasks += 1\n        self._finished.clear()\n\n    def _notify_sync_not_empty(self) -> None:\n        def f() -> None:\n            with self._sync_mutex:\n                self._sync_not_empty.notify()\n\n        self._loop.run_in_executor(None, f)\n\n    def _notify_sync_not_full(self) -> None:\n        def f() -> None:\n            with self._sync_mutex:\n                self._sync_not_full.notify()\n\n        fut = asyncio.ensure_future(self._loop.run_in_executor(None, f))\n        fut.add_done_callback(self._pending.discard)\n        self._pending.add(fut)\n\n    def _notify_async_not_empty(self, *, threadsafe: bool) -> None:\n        async def f() -> None:\n            async with self._async_mutex:\n                self._async_not_empty.notify()\n\n        def task_maker() -> None:\n            task = self._loop.create_task(f())\n            task.add_done_callback(self._pending.discard)\n            self._pending.add(task)\n\n        if threadsafe:\n            self._call_soon_threadsafe(task_maker)\n        else:\n            self._call_soon(task_maker)\n\n    def _notify_async_not_full(self, *, threadsafe: bool) -> None:\n        async def f() -> None:\n            async with self._async_mutex:\n                self._async_not_full.notify()\n\n        def task_maker() -> None:\n            task = self._loop.create_task(f())\n            task.add_done_callback(self._pending.discard)\n            self._pending.add(task)\n\n        if threadsafe:\n            self._call_soon_threadsafe(task_maker)\n        else:\n            self._call_soon(task_maker)\n\n    def _check_closing(self) -> None:\n        if self._closing:\n            raise RuntimeError(\"Operation on the closed queue is forbidden\")\n\n\nclass _SyncQueueProxy(SyncQueue[T]):\n    \"\"\"Create a queue object with a given maximum size.\n\n    If maxsize is <= 0, the queue size is infinite.\n    \"\"\"\n\n    def __init__(self, parent: Queue[T]):\n        self._parent = parent\n\n    @property\n    def maxsize(self) -> int:\n        return self._parent._maxsize\n\n    @property\n    def closed(self) -> bool:\n        return self._parent.closed\n\n    def task_done(self) -> None:\n        \"\"\"Indicate that a formerly enqueued task is complete.\n\n        Used by Queue consumer threads.  For each get() used to fetch a task,\n        a subsequent call to task_done() tells the queue that the processing\n        on the task is complete.\n\n        If a join() is currently blocking, it will resume when all items\n        have been processed (meaning that a task_done() call was received\n        for every item that had been put() into the queue).\n\n        Raises a ValueError if called more times than there were items\n        placed in the queue.\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._all_tasks_done:\n            unfinished = self._parent._unfinished_tasks - 1\n            if unfinished <= 0:\n                if unfinished < 0:\n                    raise ValueError(\"task_done() called too many times\")\n                self._parent._all_tasks_done.notify_all()\n                self._parent._loop.call_soon_threadsafe(self._parent._finished.set)\n            self._parent._unfinished_tasks = unfinished\n\n    def join(self) -> None:\n        \"\"\"Blocks until all items in the Queue have been gotten and processed.\n\n        The count of unfinished tasks goes up whenever an item is added to the\n        queue. The count goes down whenever a consumer thread calls task_done()\n        to indicate the item was retrieved and all work on it is complete.\n\n        When the count of unfinished tasks drops to zero, join() unblocks.\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._all_tasks_done:\n            while self._parent._unfinished_tasks:\n                self._parent._all_tasks_done.wait()\n                self._parent._check_closing()\n\n    def qsize(self) -> int:\n        \"\"\"Return the approximate size of the queue (not reliable!).\"\"\"\n        return self._parent._qsize()\n\n    @property\n    def unfinished_tasks(self) -> int:\n        \"\"\"Return the number of unfinished tasks.\"\"\"\n        return self._parent._unfinished_tasks\n\n    def empty(self) -> bool:\n        \"\"\"Return True if the queue is empty, False otherwise (not reliable!).\n\n        This method is likely to be removed at some point.  Use qsize() == 0\n        as a direct substitute, but be aware that either approach risks a race\n        condition where a queue can grow before the result of empty() or\n        qsize() can be used.\n\n        To create code that needs to wait for all queued tasks to be\n        completed, the preferred technique is to use the join() method.\n        \"\"\"\n        return not self._parent._qsize()\n\n    def full(self) -> bool:\n        \"\"\"Return True if the queue is full, False otherwise (not reliable!).\n\n        This method is likely to be removed at some point.  Use qsize() >= n\n        as a direct substitute, but be aware that either approach risks a race\n        condition where a queue can shrink before the result of full() or\n        qsize() can be used.\n        \"\"\"\n        return 0 < self._parent._maxsize <= self._parent._qsize()\n\n    def put(self, item: T, block: bool = True, timeout: OptFloat = None) -> None:\n        \"\"\"Put an item into the queue.\n\n        If optional args 'block' is true and 'timeout' is None (the default),\n        block if necessary until a free slot is available. If 'timeout' is\n        a non-negative number, it blocks at most 'timeout' seconds and raises\n        the Full exception if no free slot was available within that time.\n        Otherwise ('block' is false), put an item on the queue if a free slot\n        is immediately available, else raise the Full exception ('timeout'\n        is ignored in that case).\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._sync_not_full:\n            if self._parent._maxsize > 0:\n                if not block:\n                    if self._parent._qsize() >= self._parent._maxsize:\n                        raise SyncQueueFull\n                elif timeout is None:\n                    while self._parent._qsize() >= self._parent._maxsize:\n                        self._parent._sync_not_full.wait()\n                elif timeout < 0:\n                    raise ValueError(\"'timeout' must be a non-negative number\")\n                else:\n                    time = self._parent._loop.time\n                    endtime = time() + timeout\n                    while self._parent._qsize() >= self._parent._maxsize:\n                        remaining = endtime - time()\n                        if remaining <= 0.0:\n                            raise SyncQueueFull\n                        self._parent._sync_not_full.wait(remaining)\n            self._parent._put_internal(item)\n            self._parent._sync_not_empty.notify()\n            self._parent._notify_async_not_empty(threadsafe=True)\n\n    def get(self, block: bool = True, timeout: OptFloat = None) -> T:\n        \"\"\"Remove and return an item from the queue.\n\n        If optional args 'block' is true and 'timeout' is None (the default),\n        block if necessary until an item is available. If 'timeout' is\n        a non-negative number, it blocks at most 'timeout' seconds and raises\n        the Empty exception if no item was available within that time.\n        Otherwise ('block' is false), return an item if one is immediately\n        available, else raise the Empty exception ('timeout' is ignored\n        in that case).\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._sync_not_empty:\n            if not block:\n                if not self._parent._qsize():\n                    raise SyncQueueEmpty\n            elif timeout is None:\n                while not self._parent._qsize():\n                    self._parent._sync_not_empty.wait()\n            elif timeout < 0:\n                raise ValueError(\"'timeout' must be a non-negative number\")\n            else:\n                time = self._parent._loop.time\n                endtime = time() + timeout\n                while not self._parent._qsize():\n                    remaining = endtime - time()\n                    if remaining <= 0.0:\n                        raise SyncQueueEmpty\n                    self._parent._sync_not_empty.wait(remaining)\n            item = self._parent._get()\n            self._parent._sync_not_full.notify()\n            self._parent._notify_async_not_full(threadsafe=True)\n            return item\n\n    def put_nowait(self, item: T) -> None:\n        \"\"\"Put an item into the queue without blocking.\n\n        Only enqueue the item if a free slot is immediately available.\n        Otherwise raise the Full exception.\n        \"\"\"\n        return self.put(item, block=False)\n\n    def get_nowait(self) -> T:\n        \"\"\"Remove and return an item from the queue without blocking.\n\n        Only get an item if one is immediately available. Otherwise\n        raise the Empty exception.\n        \"\"\"\n        return self.get(block=False)\n\n\nclass _AsyncQueueProxy(AsyncQueue[T]):\n    \"\"\"Create a queue object with a given maximum size.\n\n    If maxsize is <= 0, the queue size is infinite.\n    \"\"\"\n\n    def __init__(self, parent: Queue[T]):\n        self._parent = parent\n\n    @property\n    def closed(self) -> bool:\n        return self._parent.closed\n\n    def qsize(self) -> int:\n        \"\"\"Number of items in the queue.\"\"\"\n        return self._parent._qsize()\n\n    @property\n    def unfinished_tasks(self) -> int:\n        \"\"\"Return the number of unfinished tasks.\"\"\"\n        return self._parent._unfinished_tasks\n\n    @property\n    def maxsize(self) -> int:\n        \"\"\"Number of items allowed in the queue.\"\"\"\n        return self._parent._maxsize\n\n    def empty(self) -> bool:\n        \"\"\"Return True if the queue is empty, False otherwise.\"\"\"\n        return self.qsize() == 0\n\n    def full(self) -> bool:\n        \"\"\"Return True if there are maxsize items in the queue.\n\n        Note: if the Queue was initialized with maxsize=0 (the default),\n        then full() is never True.\n        \"\"\"\n        if self._parent._maxsize <= 0:\n            return False\n        else:\n            return self.qsize() >= self._parent._maxsize\n\n    async def put(self, item: T) -> None:\n        \"\"\"Put an item into the queue.\n\n        Put an item into the queue. If the queue is full, wait until a free\n        slot is available before adding item.\n\n        This method is a coroutine.\n        \"\"\"\n        self._parent._check_closing()\n        async with self._parent._async_not_full:\n            self._parent._sync_mutex.acquire()\n            locked = True\n            try:\n                if self._parent._maxsize > 0:\n                    do_wait = True\n                    while do_wait:\n                        do_wait = self._parent._qsize() >= self._parent._maxsize\n                        if do_wait:\n                            locked = False\n                            self._parent._sync_mutex.release()\n                            await self._parent._async_not_full.wait()\n                            self._parent._sync_mutex.acquire()\n                            locked = True\n\n                self._parent._put_internal(item)\n                self._parent._async_not_empty.notify()\n                self._parent._notify_sync_not_empty()\n            finally:\n                if locked:\n                    self._parent._sync_mutex.release()\n\n    def put_nowait(self, item: T) -> None:\n        \"\"\"Put an item into the queue without blocking.\n\n        If no free slot is immediately available, raise QueueFull.\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._sync_mutex:\n            if self._parent._maxsize > 0:\n                if self._parent._qsize() >= self._parent._maxsize:\n                    raise AsyncQueueFull\n\n            self._parent._put_internal(item)\n            self._parent._notify_async_not_empty(threadsafe=False)\n            self._parent._notify_sync_not_empty()\n\n    async def get(self) -> T:\n        \"\"\"Remove and return an item from the queue.\n\n        If queue is empty, wait until an item is available.\n\n        This method is a coroutine.\n        \"\"\"\n        self._parent._check_closing()\n        async with self._parent._async_not_empty:\n            self._parent._sync_mutex.acquire()\n            locked = True\n            try:\n                do_wait = True\n                while do_wait:\n                    do_wait = self._parent._qsize() == 0\n\n                    if do_wait:\n                        locked = False\n                        self._parent._sync_mutex.release()\n                        await self._parent._async_not_empty.wait()\n                        self._parent._sync_mutex.acquire()\n                        locked = True\n\n                item = self._parent._get()\n                self._parent._async_not_full.notify()\n                self._parent._notify_sync_not_full()\n                return item\n            finally:\n                if locked:\n                    self._parent._sync_mutex.release()\n\n    def get_nowait(self) -> T:\n        \"\"\"Remove and return an item from the queue.\n\n        Return an item if one is immediately available, else raise QueueEmpty.\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._sync_mutex:\n            if self._parent._qsize() == 0:\n                raise AsyncQueueEmpty\n\n            item = self._parent._get()\n            self._parent._notify_async_not_full(threadsafe=False)\n            self._parent._notify_sync_not_full()\n            return item\n\n    def task_done(self) -> None:\n        \"\"\"Indicate that a formerly enqueued task is complete.\n\n        Used by queue consumers. For each get() used to fetch a task,\n        a subsequent call to task_done() tells the queue that the processing\n        on the task is complete.\n\n        If a join() is currently blocking, it will resume when all items have\n        been processed (meaning that a task_done() call was received for every\n        item that had been put() into the queue).\n\n        Raises ValueError if called more times than there were items placed in\n        the queue.\n        \"\"\"\n        self._parent._check_closing()\n        with self._parent._all_tasks_done:\n            if self._parent._unfinished_tasks <= 0:\n                raise ValueError(\"task_done() called too many times\")\n            self._parent._unfinished_tasks -= 1\n            if self._parent._unfinished_tasks == 0:\n                self._parent._finished.set()\n                self._parent._all_tasks_done.notify_all()\n\n    async def join(self) -> None:\n        \"\"\"Block until all items in the queue have been gotten and processed.\n\n        The count of unfinished tasks goes up whenever an item is added to the\n        queue. The count goes down whenever a consumer calls task_done() to\n        indicate that the item was retrieved and all work on it is complete.\n        When the count of unfinished tasks drops to zero, join() unblocks.\n        \"\"\"\n        while True:\n            with self._parent._sync_mutex:\n                self._parent._check_closing()\n                if self._parent._unfinished_tasks == 0:\n                    break\n            await self._parent._finished.wait()\n\n\nclass PriorityQueue(Queue[T]):\n    \"\"\"Variant of Queue that retrieves open entries in priority order\n    (lowest first).\n\n    Entries are typically tuples of the form:  (priority number, data).\n\n    \"\"\"\n\n    def _init(self, maxsize: int) -> None:\n        self._heap_queue = []  # type: List[T]\n\n    def _qsize(self) -> int:\n        return len(self._heap_queue)\n\n    def _put(self, item: T) -> None:\n        heappush(self._heap_queue, item)\n\n    def _get(self) -> T:\n        return heappop(self._heap_queue)\n\n\nclass LifoQueue(Queue[T]):\n    \"\"\"Variant of Queue that retrieves most recently added entries first.\"\"\"\n\n    def _qsize(self) -> int:\n        return len(self._queue)\n\n    def _put(self, item: T) -> None:\n        self._queue.append(item)\n\n    def _get(self) -> T:\n        return self._queue.pop()\n"
    }
  ],
  "ErrorMessage": "========================================================================================= FAILURES =========================================================================================\n___________________________________________________________________________ TestQueuePut.test_blocking_put_wait ____________________________________________________________________________\n\nself = <test_async.TestQueuePut object at 0x71e22c2229d0>\n\n    @pytest.mark.asyncio\n    async def test_blocking_put_wait(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n        started = asyncio.Event()\n        finished = False\n    \n        async def queue_put():\n            started.set()\n            await q.put(1)\n            await q.put(2)\n            finished = True\n    \n        async def queue_get():\n            loop.call_later(0.01, q.get_nowait)\n            queue_put_task = loop.create_task(queue_put())\n            await started.wait()\n            assert not finished\n            await queue_put_task\n            assert finished\n    \n>       await queue_get()\n\ntests/test_async.py:342: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\n    async def queue_get():\n        loop.call_later(0.01, q.get_nowait)\n        queue_put_task = loop.create_task(queue_put())\n        await started.wait()\n        assert not finished\n        await queue_put_task\n>       assert finished\nE       assert False\n\ntests/test_async.py:340: AssertionError\n================================================================================= short test summary info ==================================================================================\nFAILED tests/test_async.py::TestQueuePut::test_blocking_put_wait - assert False\n=============================================================================== 1 failed, 71 passed in 4.78s ===============================================================================",
  "Patch": "--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -326,6 +326,7 @@\n         finished = False\n \n         async def queue_put():\n+            nonlocal finished\n             started.set()\n             await q.put(1)\n             await q.put(2)\n@@ -580,9 +581,9 @@\n     q_class = janus.Queue\n \n \n-class TestLifoQueueJoin(TestQueueJoin):\n+class TestLifoQueueJoin(_QueueJoinTestMixin):\n     q_class = janus.LifoQueue\n \n \n-class TestPriorityQueueJoin(TestQueueJoin):\n+class TestPriorityQueueJoin(_QueueJoinTestMixin):\n     q_class = janus.PriorityQueue\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -93,6 +93,7 @@\n \n         async def wait_for_empty_queue():\n             await q.async_q.join()\n+            task.cancel()\n \n         await wait_for_empty_queue()\n \n",
  "BuggyCodeLocation": [
    {
      "file": "aio-libs_janus/tests/test_async.py",
      "function": null,
      "content_all": {
        "580": "    q_class = janus.Queue\n",
        "581": "\n",
        "582": "\n",
        "583": "class TestLifoQueueJoin(TestQueueJoin):\n",
        "584": "    q_class = janus.LifoQueue\n",
        "585": "\n",
        "586": "\n",
        "587": "class TestPriorityQueueJoin(TestQueueJoin):\n",
        "588": "    q_class = janus.PriorityQueue\n"
      },
      "content_change": {
        "583": "class TestLifoQueueJoin(TestQueueJoin):\n",
        "587": "class TestPriorityQueueJoin(TestQueueJoin):\n"
      }
    },
    {
      "file": "aio-libs_janus/tests/test_mixed.py",
      "function": null,
      "content_all": {},
      "content_change": {}
    }
  ],
  "Issue": {
    "title": "Fixing Async Queue Execution Context and Missing nonlocal Declaration",
    "description": "There are two primary issues in the current implementation:\n\n1. **Async Queue Execution Context for Lifos and Priority Queues:**\n   In the tests for `LifoQueue` and `PriorityQueue`, the tests are incorrectly inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`. This oversight could lead to improper handling and testing of task joins specific to Lifo and Priority queues.\n   - **Impact:** This can cause incorrect test behaviors, especially under edge cases, and might not thoroughly validate the functionality specific to Lifo and Priority queue implementations.\n\n2. **Missing `nonlocal` Declaration in Async Function:**\n   In the `queue_put` async function, the variable `finished` is being used without the `nonlocal` declaration, which means any modification of `finished` within `queue_put` does not affect the `finished` variable in the enclosing scope.\n   - **Impact:** This can result in inaccurate tracking of the `finished` state, leading to possible inconsistencies during async operations where the state needs to be accurately reflected/updated across multiple scopes.\n\nThe combination of these two issues can lead to unreliable tests for the async queue operations. Addressing them is essential to ensure accurate and consistent behavior in the queue operations and their corresponding tests.",
    "explanation": "### Summary of the Issue\n\nThe issue at hand concerns two primary problems in the current implementation of tests for asynchronous queues in the `janus` library. These issues are:\n\n1. **Async Queue Execution Context for LifoQueue and PriorityQueue**: The test classes for `LifoQueue` and `PriorityQueue` are incorrectly inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`. This could lead to improper handling and testing of task joins specific to Lifo and Priority queues.\n\n2. **Missing `nonlocal` Declaration in Async Function**: In the `queue_put` async function, the `finished` variable is used without the `nonlocal` declaration, causing any modification within `queue_put` not to affect the `finished` variable in the enclosing scope.\n\n\n### Commit Details\n\nTo address these issues, a commit has been made with the following changes:\n\n1. **Class Inheritance Adjustment**:\n    - The test classes `TestLifoQueueJoin` and `TestPriorityQueueJoin` have been modified to inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This change ensures that the tests correctly handle the specialized behavior of LifoQueue and PriorityQueue when joining tasks.\n\n2. **Add `nonlocal` Declaration**:\n    - In the `queue_put` async function within the test code, a `nonlocal` declaration for the `finished` variable has been added. This ensures that modifications to `finished` within the `queue_put` function are correctly reflected in the enclosing scope.\n\n\n### Explanation of the Issue and Solution\n\n#### Issue 1: Async Queue Execution Context for LifoQueue and PriorityQueue\n\n**Cause**:\nThe inheritance chain in the test classes is crucial for ensuring that the tests have the correct setup and assertions tailored to the specific queue types. By inheriting from `TestQueueJoin`, which is more generic, the LifoQueue and PriorityQueue tests might miss out on specific behaviors and checks that are pertinent only to these queue types.\n\n**Solution**:\nBy modifying the inheritance to `_QueueJoinTestMixin`, the tests now utilize a mixin that is likely designed to handle the idiosyncrasies of various queue join operations, particularly for Lifo and Priority queues. This correction ensures that the tests are more accurately reflecting the behaviors and edge cases specific to these queue types.\n\n#### Issue 2: Missing `nonlocal` Declaration in Async Function\n\n**Cause**:\nIn Python, the `nonlocal` keyword is used to indicate that a variable refers to a previously bound variable in the nearest enclosing scope that is not global. Without this keyword, any modification of the variable within the inner scope does not affect the variable in the outer scope. \n\nIn this case, without `nonlocal`, the `finished` variable inside the `queue_put` function would not update the `finished` status in the enclosing function or test, leading to incorrect behavior, particularly in cases where async operations depend on the accurate tracking of this state.\n\n**Solution**:\nBy adding the `nonlocal` keyword, the `finished` variable inside the `queue_put` function is correctly recognized as the same variable from the enclosing scope. This ensures that any changes to `finished` within `queue_put` are correctly propagated back to the outer function or test, maintaining accurate state tracking.\n\n### Conclusion\n\nThe commit addresses both issues by:\n1. Correcting the inheritance of test classes to ensure precise behavior checks specific to LifoQueue and PriorityQueue.\n2. Adding `nonlocal` declarations to ensure proper state management within async functions.\n\nThis leads to more reliable and accurate tests, ultimately enhancing the robustness of the library's async queue implementations."
  },
  "Explain": "### Summary of the Issue\n\nThe issue at hand concerns two primary problems in the current implementation of tests for asynchronous queues in the `janus` library. These issues are:\n\n1. **Async Queue Execution Context for LifoQueue and PriorityQueue**: The test classes for `LifoQueue` and `PriorityQueue` are incorrectly inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`. This could lead to improper handling and testing of task joins specific to Lifo and Priority queues.\n\n2. **Missing `nonlocal` Declaration in Async Function**: In the `queue_put` async function, the `finished` variable is used without the `nonlocal` declaration, causing any modification within `queue_put` not to affect the `finished` variable in the enclosing scope.\n\n\n### Commit Details\n\nTo address these issues, a commit has been made with the following changes:\n\n1. **Class Inheritance Adjustment**:\n    - The test classes `TestLifoQueueJoin` and `TestPriorityQueueJoin` have been modified to inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This change ensures that the tests correctly handle the specialized behavior of LifoQueue and PriorityQueue when joining tasks.\n\n2. **Add `nonlocal` Declaration**:\n    - In the `queue_put` async function within the test code, a `nonlocal` declaration for the `finished` variable has been added. This ensures that modifications to `finished` within the `queue_put` function are correctly reflected in the enclosing scope.\n\n\n### Explanation of the Issue and Solution\n\n#### Issue 1: Async Queue Execution Context for LifoQueue and PriorityQueue\n\n**Cause**:\nThe inheritance chain in the test classes is crucial for ensuring that the tests have the correct setup and assertions tailored to the specific queue types. By inheriting from `TestQueueJoin`, which is more generic, the LifoQueue and PriorityQueue tests might miss out on specific behaviors and checks that are pertinent only to these queue types.\n\n**Solution**:\nBy modifying the inheritance to `_QueueJoinTestMixin`, the tests now utilize a mixin that is likely designed to handle the idiosyncrasies of various queue join operations, particularly for Lifo and Priority queues. This correction ensures that the tests are more accurately reflecting the behaviors and edge cases specific to these queue types.\n\n#### Issue 2: Missing `nonlocal` Declaration in Async Function\n\n**Cause**:\nIn Python, the `nonlocal` keyword is used to indicate that a variable refers to a previously bound variable in the nearest enclosing scope that is not global. Without this keyword, any modification of the variable within the inner scope does not affect the variable in the outer scope. \n\nIn this case, without `nonlocal`, the `finished` variable inside the `queue_put` function would not update the `finished` status in the enclosing function or test, leading to incorrect behavior, particularly in cases where async operations depend on the accurate tracking of this state.\n\n**Solution**:\nBy adding the `nonlocal` keyword, the `finished` variable inside the `queue_put` function is correctly recognized as the same variable from the enclosing scope. This ensures that any changes to `finished` within `queue_put` are correctly propagated back to the outer function or test, maintaining accurate state tracking.\n\n### Conclusion\n\nThe commit addresses both issues by:\n1. Correcting the inheritance of test classes to ensure precise behavior checks specific to LifoQueue and PriorityQueue.\n2. Adding `nonlocal` declarations to ensure proper state management within async functions.\n\nThis leads to more reliable and accurate tests, ultimately enhancing the robustness of the library's async queue implementations.",
  "Source": "Human",
  "Token": 1142,
  "Command": [
    "pytest -v tests"
  ],
  "FilteredCode": [
    {
      "path": "aio-libs_janus/tests/test_async.py",
      "content": "1 \"\"\"Tests for queues.py\"\"\"\n2 \n3 import asyncio\n4 \n5 import pytest\n6 \n7 import janus\n8 \n9 \n10 class TestQueueBasic:\n11     async def _test_repr_or_str(self, fn, expect_id):\n12         \"\"\"Test Queue's repr or str.\n13 \n14         fn is repr or str. expect_id is True if we expect the Queue's id to\n15         appear in fn(Queue()).\n16         \"\"\"\n17 \n18         _q = janus.Queue()\n19         q = _q.async_q\n20         assert fn(q).startswith(\"<Queue\")\n21         id_is_present = hex(id(q)) in fn(q)\n22         assert expect_id == id_is_present\n23         loop = asyncio.get_running_loop()\n24 \n25         async def add_getter():\n26             _q = janus.Queue()\n27             q = _q.async_q\n28             # Start a task that waits to get.\n29             loop.create_task(q.get())\n30             # Let it start waiting.\n31             await asyncio.sleep(0.1)\n32             assert \"_getters[1]\" in fn(q)\n33             # resume q.get coroutine to finish generator\n34             q.put_nowait(0)\n35 \n36         await add_getter()\n37 \n38         async def add_putter():\n39             _q = janus.Queue(maxsize=1)\n40             q = _q.async_q\n41             q.put_nowait(1)\n42             # Start a task that waits to put.\n43             loop.create_task(q.put(2))\n44             # Let it start waiting.\n45             await asyncio.sleep(0.1)\n46             assert \"_putters[1]\" in fn(q)\n47             # resume q.put coroutine to finish generator\n48             q.get_nowait()\n49 \n50         await add_putter()\n51 \n52         _q = janus.Queue()\n53         q = _q.async_q\n54         q.put_nowait(1)\n55         assert \"_queue=[1]\" in fn(q(...truncated)"
    },
    {
      "path": "aio-libs_janus/tests/test_sync.py",
      "content": "1 # Some simple queue module tests, plus some failure conditions\n2 # to ensure the Queue locks remain stable.\n3 import asyncio\n4 import queue\n5 import threading\n6 import time\n7 from unittest.mock import patch\n8 \n9 import pytest\n10 \n11 import janus\n12 \n13 QUEUE_SIZE = 5\n14 \n15 \n16 def qfull(q):\n17     return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n18 \n19 \n20 # A thread to run a function that unclogs a blocked Queue.\n21 class _TriggerThread(threading.Thread):\n22     def __init__(self, fn, args):\n23         self.fn = fn\n24         self.args = args\n25         self.startedEvent = threading.Event()\n26         threading.Thread.__init__(self)\n27 \n28     def run(self):\n29         # The sleep isn't necessary, but is intended to give the blocking\n30         # function in the main thread a chance at actually blocking before\n31         # we unclog it.  But if the sleep is longer than the timeout-based\n32         # tests wait in their blocking functions, those tests will fail.\n33         # So we give them much longer timeout values compared to the\n34         # sleep here (I aimed at 10 seconds for blockin(...truncated)"
    },
    {
      "path": "aio-libs_janus/janus/__init__.py",
      "content": "1 import asyncio\n2 import sys\n3 import threading\n4 from asyncio import QueueEmpty as AsyncQueueEmpty\n5 from asyncio import QueueFull as AsyncQueueFull\n6 from collections import deque\n7 from heapq import heappop, heappush\n8 from queue import Empty as SyncQueueEmpty\n9 from queue import Full as SyncQueueFull\n10 from typing import Any, Callable, Deque, Generic, List, Optional, Set, TypeVar\n11 \n12 from typing_extensions import Protocol\n13 \n14 __version__ = \"1.0.0\"\n15 __all__ = (\n16     \"Queue\",\n17     \"PriorityQueue\",\n18     \"LifoQueue\",\n19     \"SyncQueue\",\n20     \"AsyncQueue\",\n21     \"BaseQueue\",\n22 )\n23 \n24 \n25 T = TypeVar(\"T\")\n26 OptFloat = Optional[float]\n27 \n28 \n29 class BaseQueue(Protocol[T]):\n30     @property\n31     def maxsize(self) -> int:\n32         ...\n33 \n34     @property\n35     def closed(self) -> bool:\n36         ...\n37 \n38     def task_d(...truncated)"
    },
    {
      "path": "aio-libs_janus/tests/test_mixed.py",
      "content": "1 import asyncio\n2 import contextlib\n3 import sys\n4 import threading\n5 \n6 import pytest\n7 \n8 import janus\n9 \n10 \n11 class TestMixedMode:\n12     @pytest.mark.skipif(\n13         sys.version_info < (3, 7),\n14         reason=\"forbidding implicit loop creation works on \"\n15         \"Python 3.7 or higher only\",\n16     )\n17     def test_ctor_noloop(self):\n18         with pytest.raises(RuntimeError):\n19             janus.Queue()\n20 \n21     @pytest.mark.asyncio\n22     async def test_maxsize(self):\n23         q = janus.Queue(5)\n24         assert 5 == q.maxsize\n25 \n26     @pytest.ma(...truncated)"
    }
  ],
  "TokenAll": 15097,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Improper Use of create_task",
      "description": "In the file aio-libs_janus/tests/test_async.py, line 29 and line 43 use loop.create_task without handling the created task. This can cause unhandled exceptions if the created tasks fail. It is recommended to keep a reference to the created task and handle any potential exceptions within it.",
      "explanation": "### Issue Summary\n\n**Title: \"Improper Use of `create_task`\"**\n\nThe issue at hand is related to the improper usage of the `create_task` method within an asyncio event loop in the `test_async.py` file. Specifically, the lines in question (29 and 43) launch tasks without maintaining any reference to them or handling any possible exceptions that might be raised if those tasks fail. This can lead to unhandled exceptions, which could undermine the reliability of the test suite and cause unpredictable behavior.\n\n### Detailed Code Analysis\n\nHere are the relevant lines from the specified file:\n\n```python\n24         loop = asyncio.get_running_loop()\n25 \n26         async def add_getter():\n27             _q = janus.Queue()\n28             q = _q.async_q\n29             loop.create_task(q.get())  # Problematic Line\n30             await asyncio.sleep(0.1)\n31             assert \"_getters[1]\" in fn(q)\n32             q.put_nowait(0)\n33 \n34         await add_getter()\n35 \n36         async def add_putter():\n37             _q = janus.Queue(maxsize=1)\n38             q = _q.async_q\n39             q.put_nowait(1)\n40             loop.create_task(q.put(2))  # Problematic Line\n41             await asyncio.sleep(0.1)\n42             assert \"_putters[1]\" in fn(q)\n43             q.get_nowait()\n44 \n45         await add_putter()\n```\n\nIn both `add_getter` and `add_putter`, a `create_task` call is made that starts a coroutine in the background (`q.get()` for `add_getter` and `q.put(2)` for `add_putter`). However, these tasks are not referenced or managed, leading to potential unhandled exceptions if those tasks fail.\n\n### Proposed Commit\nHere's a possible commit fixing the issue:\n\n#### Commit Message\n```\nfix(tests): Handle tasks created with create_task in test_async.py\n\n- Added task references for background tasks created using loop.create_task.\n- Included proper exception handling to avoid unhandled exceptions.\n```\n\n#### Updated Code:\n\n```python\n24         loop = asyncio.get_running_loop()\n25 \n26         async def add_getter():\n27             _q = janus.Queue()\n28             q = _q.async_q\n29             getter_task = loop.create_task(q.get())  # Reference and handle the task\n30             await asyncio.sleep(0.1)\n31             assert \"_getters[1]\" in fn(q)\n32             q.put_nowait(0)\n33             await getter_task  # Properly await the task to catch exceptions\n34 \n35         await add_getter()\n36 \n37         async def add_putter():\n38             _q = janus.Queue(maxsize=1)\n39             q = _q.async_q\n40             q.put_nowait(1)\n41             putter_task = loop.create_task(q.put(2))  # Reference and handle the task\n42             await asyncio.sleep(0.1)\n43             assert \"_putters[1]\" in fn(q)\n44             q.get_nowait()\n45             await putter_task  # Properly await the task to catch exceptions\n46 \n47         await add_putter()\n```\n\n### Explanation of the Solution\n\n1. **Create Task References**: \n   - `getter_task = loop.create_task(q.get())`\n   - `putter_task = loop.create_task(q.put(2))`\n   \n   Here, we create variables (`getter_task` and `putter_task`) to keep references to the tasks created by `loop.create_task`.\n\n2. **Await Tasks Properly**:\n   - `await getter_task`\n   - `await putter_task`\n   \n   Instead of leaving the tasks running unattended, we await their completion. This ensures that the code waits for these tasks to finish and can catch and handle any exceptions they may raise.\n\n### How the Commit Solves the Issue\n\nBy referencing and awaiting the tasks, we ensure:\n- **Exception Handling**: Any exceptions raised within the `q.get()` or `q.put()` coroutines will be caught at the point of awaiting them (`await getter_task` and `await putter_task`). This prevents unhandled exceptions from leaking and causing instability in the event loop.\n- **Predictable Behavior**: Awaiting the tasks means the test will ensure those operations complete as expected, maintaining the correct order of operations and predictable behavior during testing.\n\nThis small change makes the code much more robust by ensuring all started tasks are properly managed, thus addressing the core issue of unhandled asynchronous tasks."
    },
    "issue_message": {
      "title": "Variable 'finished' Assignment inside Coroutine",
      "description": "The variable 'finished' in the test_blocking_put_wait function is being modified inside the coroutine queue_put, but it is not declared as nonlocal. This leads to the 'finished' variable inside the coroutine being different from the one in the outer scope, causing the assert finished to fail in the queue_get coroutine. Please declare 'finished' as nonlocal inside queue_put.",
      "explanation": "### Summary of the Issue\nThe issue revolves around the variable `finished` in the `test_blocking_put_wait` function being incorrectly referenced within a coroutine. Inside the coroutine `queue_put`, `finished` is being modified, but because it is not declared as `nonlocal`, it does not refer to the `finished` variable in the outer scope. This results in the `assert finished` check inside the `queue_get` coroutine to fail, as it checks the `finished` variable from the outer scope, which remains unchanged.\n\n### Analysis of the Commit\nThe proposed solution would involve modifying the `queue_put` coroutine to declare the `finished` variable as `nonlocal`. This ensures that modifications made to `finished` within the coroutine affect the `finished` variable in the outer scope. \n\n### Detailed Content of the Commit\nThe commit would change the `queue_put` coroutine to:\n\n```python\nasync def queue_put():\n    nonlocal finished\n    started.set()\n    await q.put(1)\n    await q.put(2)\n    finished = True\n```\n\n### Explanation of How the Commit Solves the Issue\nLet's break down what would happen after the change:\n\n1. **Declaration of `nonlocal`**:\n    - The keyword `nonlocal` is used inside a nested function to declare that the variable `finished` refers to the nearest parent scope, which is the enclosing `test_blocking_put_wait` function in this case.\n    - Without `nonlocal`, Python creates a new local variable `finished` within the `queue_put` coroutine, leaving the `finished` variable in the outer scope unchanged.\n\n2. **Modification of `finished`**:\n    - With `nonlocal` declared, any assignment to `finished` inside `queue_put` updates the `finished` variable declared in `test_blocking_put_wait`.\n\n3. **Assertion Check in `queue_get`**:\n    - The `queue_get` coroutine will first assert `not finished`, which checks the outer scope `finished` variable and should be `False`.\n    - After `await queue_put_task`, it expects `finished` to be `True`. With `nonlocal` ensures this is correctly updated, the assertion will pass.\n\n### Explanation with Example\nBefore the fix:\n```python\nfinished = False  # This is the outer scope variable\n\nasync def queue_put():\n    finished = True  # This creates a new local `finished` variable\n```\nAfter the fix:\n```python\nfinished = False  # This is the outer scope variable\n\nasync def queue_put():\n    nonlocal finished  # Now we use the outer scope variable\n    finished = True  # This correctly updates the outer scope variable\n```\n\nWith this change, when `queue_put` sets `finished = True`, it actually updates the `finished` variable in the `test_blocking_put_wait` function's scope. Hence, the final assertion `assert finished` inside the `queue_get` coroutine will pass because the `finished` variable will correctly be `True`.\n\n### Conclusion\nDeclaring `finished` as `nonlocal` inside the `queue_put` coroutine ensures that the `finished` variable in the outer scope (`test_blocking_put_wait` function) is correctly modified. This aligns with the intended functionality and fixes the error where the `assert finished` would fail because the `finished` variable was not properly updated."
    },
    "issue_ground": {
      "title": "Fixing Async Queue Execution Context and Missing nonlocal Declaration",
      "description": "There are two primary issues in the current implementation:\n\n1. **Async Queue Execution Context for Lifos and Priority Queues:**\n   In the tests for `LifoQueue` and `PriorityQueue`, the tests are incorrectly inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`. This oversight could lead to improper handling and testing of task joins specific to Lifo and Priority queues.\n   - **Impact:** This can cause incorrect test behaviors, especially under edge cases, and might not thoroughly validate the functionality specific to Lifo and Priority queue implementations.\n\n2. **Missing `nonlocal` Declaration in Async Function:**\n   In the `queue_put` async function, the variable `finished` is being used without the `nonlocal` declaration, which means any modification of `finished` within `queue_put` does not affect the `finished` variable in the enclosing scope.\n   - **Impact:** This can result in inaccurate tracking of the `finished` state, leading to possible inconsistencies during async operations where the state needs to be accurately reflected/updated across multiple scopes.\n\nThe combination of these two issues can lead to unreliable tests for the async queue operations. Addressing them is essential to ensure accurate and consistent behavior in the queue operations and their corresponding tests.",
      "explanation": "### Issue Summary\n\nThe reported problem has two primary components:\n\n1. **Async Queue Execution Context for `LifoQueue` and `PriorityQueue`**:\n   - In the test implementations for `LifoQueue` and `PriorityQueue`, there is a problem with inheritance. The tests inherit from `TestQueueJoin` instead of `_QueueJoinTestMixin`, which is designated for testing task joins specific to Lifo and Priority queues.\n   - **Impact**: Misaligned inheritance can lead to inadequate testing and incorrect behaviors in edge cases, undermining the validation of `LifoQueue` and `PriorityQueue`.\n\n2. **Missing `nonlocal` Declaration in Async Function**:\n   - The `queue_put` async function modifies the `finished` variable without declaring it as `nonlocal`, meaning changes do not affect the `finished` variable in the enclosing scope.\n   - **Impact**: Incorrect tracking of the `finished` state due to this scope issue can result in inconsistencies in async operations where the accurate state needs to be shared across multiple scopes.\n\n### Detailed Explanation of the Commit\n\n**Patch for `aio-libs_janus/tests/test_async.py`:**\n```diff\n@@ -228,7 +228,7 @@ class TestQueuePut:\n \n class TestQueueBasic(unittest.TestCase):\n     ...\n \n- class TestQueueJoin(BaseQueue):\n+ class TestQueueJoin(_QueueJoinTestMixin):\n \n     ...\n \n- class TestLifoQueueJoin(TestQueueJoin, unittest.TestCase):\n+ class TestLifoQueueJoin(unittest.TestCase, _QueueJoinTestMixin):\n \n     ...\n \n- class TestPriorityQueueJoin(TestQueueJoin, unittest.TestCase):\n+ class TestPriorityQueueJoin(unittest.TestCase, _QueueJoinTestMixin):\n \n     ...\n \n@@ -338,9 +338,13 @@ class TestQueuePut:\n         finished = False\n \n         async def queue_put():\n             started.set()\n             await q.put(1)\n             await q.put(2)\n+            nonlocal finished\n             finished = True\n \n         ...\n \n         await queue_get()\n```\n\n### Solution & Impact Analysis\n\n1. **Fixing Async Queue Execution Context for `LifoQueue` and `PriorityQueue`**:\n   - The commit changes the superclass of `TestLifoQueueJoin` and `TestPriorityQueueJoin` to ensure they now inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This ensures the tests align with the specific behaviors intended for Lifo and Priority queues rather than generic queue join behaviors.\n   \n   **Why This Solves the Issue**:\n   - **Changed Inheritance**: This ensures that the Lifo and Priority specific implementations of task joins are properly tested, thereby validating their unique behaviors. The correct inheritance hierarchy ensures that any edge cases in LifoQueue and PriorityQueue-specific implementations are thoroughly tested and do not inherit incorrect testing behaviors from a generic base class.\n\n2. **Adding `nonlocal` Declaration**:\n   - The commit adds the `nonlocal` declaration for the `finished` variable within the `queue_put` async function. This ensures that modifications within `queue_put` reflect the state of `finished` in the enclosing `test_blocking_put_wait` function scope.\n   \n   **Why This Solves the Issue**:\n   - **Scope Correction**: Declaring `finished` as `nonlocal` properly links the inner scope's usage of `finished` with the outer scope's variable. This ensures that the true state of `finished` is accurately tracked and modified, thus correcting the assert statement's logic in `queue_get`.\n\n### Conclusion\n\nThis commit fixes crucial issues by:\n- Correcting inheritance for specific queue test classes (`LifoQueue`, `PriorityQueue`).\n- Ensuring accurate state tracking across async functions via the `nonlocal` keyword.\n\nBy making these changes:\n- The tests now accurately verify the behaviors of Lifo and Priority Queues.\n- The state used in async operations is correctly shared and updated, preventing false negatives in tests due to scope misalignment.\n\nAs a result, the applied patch straightens the testing framework and resolves the assertion failure observed in the `test_blocking_put_wait` test case."
    },
    "issue_ground_truth": {
      "title": "Fixing Async Queue Execution Context and Missing nonlocal Declaration",
      "description": "There are two primary issues in the current implementation:\n\n1. **Async Queue Execution Context for Lifos and Priority Queues:**\n   In the tests for `LifoQueue` and `PriorityQueue`, the tests are incorrectly inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`. This oversight could lead to improper handling and testing of task joins specific to Lifo and Priority queues.\n   - **Impact:** This can cause incorrect test behaviors, especially under edge cases, and might not thoroughly validate the functionality specific to Lifo and Priority queue implementations.\n\n2. **Missing `nonlocal` Declaration in Async Function:**\n   In the `queue_put` async function, the variable `finished` is being used without the `nonlocal` declaration, which means any modification of `finished` within `queue_put` does not affect the `finished` variable in the enclosing scope.\n   - **Impact:** This can result in inaccurate tracking of the `finished` state, leading to possible inconsistencies during async operations where the state needs to be accurately reflected/updated across multiple scopes.\n\nThe combination of these two issues can lead to unreliable tests for the async queue operations. Addressing them is essential to ensure accurate and consistent behavior in the queue operations and their corresponding tests.",
      "explanation": "### Summary of the Issue\n\nThe issue at hand concerns two primary problems in the current implementation of tests for asynchronous queues in the `janus` library. These issues are:\n\n1. **Async Queue Execution Context for LifoQueue and PriorityQueue**: The test classes for `LifoQueue` and `PriorityQueue` are incorrectly inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`. This could lead to improper handling and testing of task joins specific to Lifo and Priority queues.\n\n2. **Missing `nonlocal` Declaration in Async Function**: In the `queue_put` async function, the `finished` variable is used without the `nonlocal` declaration, causing any modification within `queue_put` not to affect the `finished` variable in the enclosing scope.\n\n\n### Commit Details\n\nTo address these issues, a commit has been made with the following changes:\n\n1. **Class Inheritance Adjustment**:\n    - The test classes `TestLifoQueueJoin` and `TestPriorityQueueJoin` have been modified to inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This change ensures that the tests correctly handle the specialized behavior of LifoQueue and PriorityQueue when joining tasks.\n\n2. **Add `nonlocal` Declaration**:\n    - In the `queue_put` async function within the test code, a `nonlocal` declaration for the `finished` variable has been added. This ensures that modifications to `finished` within the `queue_put` function are correctly reflected in the enclosing scope.\n\n\n### Explanation of the Issue and Solution\n\n#### Issue 1: Async Queue Execution Context for LifoQueue and PriorityQueue\n\n**Cause**:\nThe inheritance chain in the test classes is crucial for ensuring that the tests have the correct setup and assertions tailored to the specific queue types. By inheriting from `TestQueueJoin`, which is more generic, the LifoQueue and PriorityQueue tests might miss out on specific behaviors and checks that are pertinent only to these queue types.\n\n**Solution**:\nBy modifying the inheritance to `_QueueJoinTestMixin`, the tests now utilize a mixin that is likely designed to handle the idiosyncrasies of various queue join operations, particularly for Lifo and Priority queues. This correction ensures that the tests are more accurately reflecting the behaviors and edge cases specific to these queue types.\n\n#### Issue 2: Missing `nonlocal` Declaration in Async Function\n\n**Cause**:\nIn Python, the `nonlocal` keyword is used to indicate that a variable refers to a previously bound variable in the nearest enclosing scope that is not global. Without this keyword, any modification of the variable within the inner scope does not affect the variable in the outer scope. \n\nIn this case, without `nonlocal`, the `finished` variable inside the `queue_put` function would not update the `finished` status in the enclosing function or test, leading to incorrect behavior, particularly in cases where async operations depend on the accurate tracking of this state.\n\n**Solution**:\nBy adding the `nonlocal` keyword, the `finished` variable inside the `queue_put` function is correctly recognized as the same variable from the enclosing scope. This ensures that any changes to `finished` within `queue_put` are correctly propagated back to the outer function or test, maintaining accurate state tracking.\n\n### Conclusion\n\nThe commit addresses both issues by:\n1. Correcting the inheritance of test classes to ensure precise behavior checks specific to LifoQueue and PriorityQueue.\n2. Adding `nonlocal` declarations to ensure proper state management within async functions.\n\nThis leads to more reliable and accurate tests, ultimately enhancing the robustness of the library's async queue implementations."
    },
    "location_origin": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "25": "add_getter"
        },
        "content_all": {
          "22": "        loop = asyncio.get_running_loop()\n",
          "23": "\n",
          "24": "        async def add_getter():\n",
          "25": "            _q = janus.Queue()\n",
          "26": "            q = _q.async_q\n",
          "27": "            # Start a task that waits to get.\n",
          "28": "            loop.create_task(q.get())\n",
          "29": "            # Let it start waiting.\n",
          "30": "            await asyncio.sleep(0.1)\n",
          "31": "            assert \"_getters[1]\" in fn(q)\n",
          "32": "            # resume q.get coroutine to finish generator\n",
          "33": "            q.put_nowait(0)\n",
          "34": "\n",
          "35": "        await add_getter()\n"
        },
        "content_change": {
          "28": "            getter_task = loop.create_task(q.get())\n",
          "33": "            await getter_task\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "38": "add_putter"
        },
        "content_all": {
          "35": "        await add_getter()\n",
          "36": "\n",
          "37": "        async def add_putter():\n",
          "38": "            _q = janus.Queue(maxsize=1)\n",
          "39": "            q = _q.async_q\n",
          "40": "            q.put_nowait(1)\n",
          "41": "            # Start a task that waits to put.\n",
          "42": "            loop.create_task(q.put(2))\n",
          "43": "            # Let it start waiting.\n",
          "44": "            await asyncio.sleep(0.1)\n",
          "45": "            assert \"_putters[1]\" in fn(q)\n",
          "46": "            # resume q.put coroutine to finish generator\n",
          "47": "            q.get_nowait()\n",
          "48": "\n",
          "49": "        await add_putter()\n"
        },
        "content_change": {
          "42": "            putter_task = loop.create_task(q.put(2))\n",
          "47": "            await putter_task\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "326": "test_blocking_put_wait"
        },
        "content_all": {
          "323": "    @pytest.mark.asyncio\n",
          "324": "    async def test_blocking_put_wait(self):\n",
          "325": "        loop = asyncio.get_running_loop()\n",
          "326": "        _q = janus.Queue(maxsize=1)\n",
          "327": "        q = _q.async_q\n",
          "328": "        started = asyncio.Event()\n",
          "329": "        finished = False\n",
          "330": "\n",
          "331": "        async def queue_put():\n",
          "332": "            started.set()\n",
          "333": "            await q.put(1)\n",
          "334": "            await q.put(2)\n",
          "335": "            finished = True\n",
          "336": "\n",
          "337": "        async def queue_get():\n",
          "338": "            loop.call_later(0.01, q.get_nowait)\n",
          "339": "            queue_put_task = loop.create_task(queue_put())\n",
          "340": "            await started.wait()\n",
          "341": "            assert not finished\n",
          "342": "            await queue_put_task\n",
          "343": "            assert finished\n",
          "344": "\n",
          "345": "        await queue_get()\n"
        },
        "content_change": {
          "331": "        async def queue_put():\n",
          "329": "        finished = False\n",
          "332": "            nonlocal finished\n"
        }
      }
    ],
    "location_ground": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "230": "TestQueuePut"
        },
        "content_all": {
          "227": "class TestQueuePut:\n",
          "228": "    ...\n",
          "229": "    async def test_blocking_put_wait(self):\n",
          "230": "        finished = False\n",
          "231": "        async def queue_put():\n",
          "232": "            started.set()\n",
          "233": "            await q.put(1)\n",
          "234": "            await q.put(2)\n",
          "235": "            finished = True\n",
          "236": "        ...\n",
          "237": "        await queue_get()\n"
        },
        "content_change": {
          "230": "        nonlocal finished\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "259": "TestLifoQueueJoin"
        },
        "content_all": {
          "256": "class TestQueueJoin(BaseQueue):\n",
          "257": "    ...\n",
          "258": "# Correct inheritance\n",
          "259": "class TestLifoQueueJoin(unittest.TestCase, _QueueJoinTestMixin):\n",
          "260": "    ...\n"
        },
        "content_change": {
          "259": "class TestLifoQueueJoin(unittest.TestCase, _QueueJoinTestMixin):\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "262": "TestPriorityQueueJoin"
        },
        "content_all": {
          "259": "class TestLifoQueueJoin(unittest.TestCase, _QueueJoinTestMixin):\n",
          "260": "    ...\n",
          "261": "# Correct inheritance\n",
          "262": "class TestPriorityQueueJoin(unittest.TestCase, _QueueJoinTestMixin):\n",
          "263": "    ...\n"
        },
        "content_change": {
          "262": "class TestPriorityQueueJoin(unittest.TestCase, _QueueJoinTestMixin):\n"
        }
      }
    ],
    "location_ground_exp": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "105": "TestLifoQueueJoin",
          "145": "TestPriorityQueueJoin"
        },
        "content_all": {
          "102": "",
          "103": "import asyncio",
          "104": "",
          "105": "class TestLifoQueueJoin(TestQueueJoin):",
          "106": "    pass",
          "107": "",
          "143": "",
          "144": "import asyncio",
          "145": "",
          "146": "class TestPriorityQueueJoin(TestQueueJoin):",
          "147": "    pass",
          "148": "",
          "174": "    async def queue_put(self):",
          "175": "        finished = False",
          "176": "        def inner():",
          "177": "            finished = True",
          "178": "        await inner()"
        },
        "content_change": {
          "105": "class TestLifoQueueJoin(_QueueJoinTestMixin):",
          "146": "class TestPriorityQueueJoin(_QueueJoinTestMixin):",
          "176": "            nonlocal finished"
        }
      }
    ],
    "location_ground_truth": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": null,
        "content_all": {
          "580": "    q_class = janus.Queue\n",
          "581": "\n",
          "582": "\n",
          "583": "class TestLifoQueueJoin(TestQueueJoin):\n",
          "584": "    q_class = janus.LifoQueue\n",
          "585": "\n",
          "586": "\n",
          "587": "class TestPriorityQueueJoin(TestQueueJoin):\n",
          "588": "    q_class = janus.PriorityQueue\n"
        },
        "content_change": {
          "583": "class TestLifoQueueJoin(TestQueueJoin):\n",
          "587": "class TestPriorityQueueJoin(TestQueueJoin):\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": null,
        "content_all": {},
        "content_change": {}
      }
    ],
    "patch_i": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -24,17 +24,20 @@\n \n         async def add_getter():\n             _q = janus.Queue()\n             q = _q.async_q\n             # Start a task that waits to get.\n-            loop.create_task(q.get())\n+            getter_task = loop.create_task(q.get())  # Reference and handle the task\n             # Let it start waiting.\n             await asyncio.sleep(0.1)\n             assert \"_getters[1]\" in fn(q)\n             # resume q.get coroutine to finish generator\n             q.put_nowait(0)\n+            await getter_task  # Properly await the task to catch exceptions\n \n         await add_getter()\n \n         async def add_putter():\n             _q = janus.Queue(maxsize=1)\n             q = _q.async_q\n             q.put_nowait(1)\n             # Start a task that waits to put.\n-            loop.create_task(q.put(2))\n+            putter_task = loop.create_task(q.put(2))  # Reference and handle the task\n             # Let it start waiting.\n             await asyncio.sleep(0.1)\n             assert \"_putters[1]\" in fn(q)\n             # resume q.put coroutine to finish generator\n             q.get_nowait()\n+            await putter_task  # Properly await the task to catch exceptions\n \n         await add_putter()\n \n         _q = janus.Queue()\n         q = _q.async_q\n         q.put_nowait(1)\n         assert \"_queue=[1]\" in fn(q)\n",
    "patch_im": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -60,10 +60,11 @@\n     async def test_blocking_put_wait(self):\n         q = janus.Queue(maxsize=1)\n         started = asyncio.Event()\n+        finished = False\n \n         async def queue_put():\n+            nonlocal finished\n             started.set()\n             await q.async_q.put(1)\n             await q.async_q.put(2)\n             finished = True\n \n         async def queue_get():\n@@ -72,11 +73,10 @@\n             await q.async_q.get()\n             assert finished\n\n         queue_put_task = asyncio.create_task(queue_put())\n         await started.wait()\n         await asyncio.sleep(0.1)\n         assert not finished\n         await queue_get()\n         await queue_put_task\n\n",
    "patch_il": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -26,7 +26,8 @@\n             _q = janus.Queue()\n             q = _q.async_q\n             # Start a task that waits to get.\n-            loop.create_task(q.get())\n+            getter_task = loop.create_task(q.get())  # Reference and handle the task\n+            await getter_task  # Properly await the task to catch exceptions\n             # Let it start waiting.\n             await asyncio.sleep(0.1)\n             assert \"_getters[1]\" in fn(q)\n@@ -41,7 +42,8 @@\n             q.put_nowait(1)\n             # Start a task that waits to put.\n-            loop.create_task(q.put(2))\n+            putter_task = loop.create_task(q.put(2))  # Reference and handle the task\n             # Let it start waiting.\n             await asyncio.sleep(0.1)\n             assert \"_putters[1]\" in fn(q)\n             # resume q.put coroutine to finish generator\n             q.get_nowait()\n+            await putter_task  # Properly await the task to catch exceptions\n",
    "patch_iml": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -328,335 +328,337 @@ class TestQueueBasic:\n     @pytest.mark.asyncio\n     async def test_blocking_put_wait(self):\n         loop = asyncio.get_running_loop()\n         _q = janus.Queue(maxsize=1)\n         q = _q.async_q\n         started = asyncio.Event()\n         finished = False\n \n         async def queue_put():\n+            nonlocal finished\n             started.set()\n             await q.put(1)\n             await q.put(2)\n             finished = True\n \n         async def queue_get():\n             loop.call_later(0.01, q.get_nowait)\n             queue_put_task = loop.create_task(queue_put())\n             await started.wait()\n             assert not finished\n             await queue_put_task\n             assert finished\n \n         await queue_get()\n",
    "patch_ground": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -228,7 +228,7 @@ class TestQueuePut:\n \n class TestQueueBasic(unittest.TestCase):\n     ...\n \n- class TestQueueJoin(BaseQueue):\n+ class TestQueueJoin(_QueueJoinTestMixin):\n \n     ...\n \n- class TestLifoQueueJoin(TestQueueJoin, unittest.TestCase):\n+ class TestLifoQueueJoin(unittest.TestCase, _QueueJoinTestMixin):\n \n     ...\n \n- class TestPriorityQueueJoin(TestQueueJoin, unittest.TestCase):\n+ class TestPriorityQueueJoin(unittest.TestCase, _QueueJoinTestMixin):\n \n     ...\n \n@@ -338,9 +338,13 @@ class TestQueuePut:\n         finished = False\n \n         async def queue_put():\n             started.set()\n             await q.put(1)\n             await q.put(2)\n+            nonlocal finished\n             finished = True\n \n         ...\n \n         await queue_get()\n",
    "patch_ground_location": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -228,7 +228,8 @@ class TestQueuePut:\n         finished = False\n\n         async def queue_put():\n+            nonlocal finished\n             started.set()\n             await q.put(1)\n             await q.put(2)\n@@ -256,8 +257,8 @@ class TestQueueJoin(BaseQueue):\n class TestLifoQueueJoin(TestQueueJoin, unittest.TestCase):\n     ...\n\n-class TestLifoQueueJoin(TestQueueJoin, unittest.TestCase):\n+class TestLifoQueueJoin(unittest.TestCase, _QueueJoinTestMixin):\n     ...\n\n-class TestPriorityQueueJoin(TestQueueJoin, unittest.TestCase):\n+class TestPriorityQueueJoin(unittest.TestCase, _QueueJoinTestMixin):\n     ...\n",
    "patch_ground_exp": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -105,7 +105,7 @@\n import asyncio\n \n \n-class TestLifoQueueJoin(TestQueueJoin):\n+class TestLifoQueueJoin(_QueueJoinTestMixin):\n     pass\n \n \n@@ -145,7 +145,7 @@\n import asyncio\n \n \n-class TestPriorityQueueJoin(TestQueueJoin):\n+class TestPriorityQueueJoin(_QueueJoinTestMixin):\n     pass\n \n \n@@ -176,6 +176,7 @@\n     async def queue_put(self):\n         finished = False\n         def inner():\n+            nonlocal finished\n             finished = True\n         await inner()\n",
    "patch_ground_all": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -580,8 +580,8 @@\n     q_class = janus.Queue\n\n class TestLifoQueueJoin(_QueueJoinTestMixin):\n     q_class = janus.LifoQueue\n\n class TestPriorityQueueJoin(_QueueJoinTestMixin):\n     q_class = janus.PriorityQueue\n",
    "patch_ground_truth": "--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -326,6 +326,7 @@\n         finished = False\n \n         async def queue_put():\n+            nonlocal finished\n             started.set()\n             await q.put(1)\n             await q.put(2)\n@@ -580,9 +581,9 @@\n     q_class = janus.Queue\n \n \n-class TestLifoQueueJoin(TestQueueJoin):\n+class TestLifoQueueJoin(_QueueJoinTestMixin):\n     q_class = janus.LifoQueue\n \n \n-class TestPriorityQueueJoin(TestQueueJoin):\n+class TestPriorityQueueJoin(_QueueJoinTestMixin):\n     q_class = janus.PriorityQueue\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -93,6 +93,7 @@\n \n         async def wait_for_empty_queue():\n             await q.async_q.join()\n+            task.cancel()\n \n         await wait_for_empty_queue()\n \n",
    "message": "========================================================================================= FAILURES =========================================================================================\n___________________________________________________________________________ TestQueuePut.test_blocking_put_wait ____________________________________________________________________________\n\nself = <test_async.TestQueuePut object at 0x71e22c2229d0>\n\n    @pytest.mark.asyncio\n    async def test_blocking_put_wait(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n        started = asyncio.Event()\n        finished = False\n    \n        async def queue_put():\n            started.set()\n            await q.put(1)\n            await q.put(2)\n            finished = True\n    \n        async def queue_get():\n            loop.call_later(0.01, q.get_nowait)\n            queue_put_task = loop.create_task(queue_put())\n            await started.wait()\n            assert not finished\n            await queue_put_task\n            assert finished\n    \n>       await queue_get()\n\ntests/test_async.py:342: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\n    async def queue_get():\n        loop.call_later(0.01, q.get_nowait)\n        queue_put_task = loop.create_task(queue_put())\n        await started.wait()\n        assert not finished\n        await queue_put_task\n>       assert finished\nE       assert False\n\ntests/test_async.py:340: AssertionError\n================================================================================= short test summary info ==================================================================================\nFAILED tests/test_async.py::TestQueuePut::test_blocking_put_wait - assert False\n=============================================================================== 1 failed, 71 passed in 4.78s ===============================================================================",
    "CodeBase": [
      {
        "path": "aio-libs_janus/tests/test_async.py",
        "content": "1 \"\"\"Tests for queues.py\"\"\"\n2 \n3 import asyncio\n4 \n5 import pytest\n6 \n7 import janus\n8 \n9 \n10 class TestQueueBasic:\n11     async def _test_repr_or_str(self, fn, expect_id):\n12         \"\"\"Test Queue's repr or str.\n13 \n14         fn is repr or str. expect_id is True if we expect the Queue's id to\n15         appear in fn(Queue()).\n16         \"\"\"\n17 \n18         _q = janus.Queue()\n19         q = _q.async_q\n20         assert fn(q).startswith(\"<Queue\")\n21         id_is_present = hex(id(q)) in fn(q)\n22         assert expect_id == id_is_present\n23         loop = asyncio.get_running_loop()\n24 \n25         async def add_getter():\n26             _q = janus.Queue()\n27             q = _q.async_q\n28             # Start a task that waits to get.\n29             loop.create_task(q.get())\n30             # Let it start waiting.\n31             await asyncio.sleep(0.1)\n32             assert \"_getters[1]\" in fn(q)\n33             # resume q.get coroutine to finish generator\n34             q.put_nowait(0)\n35 \n36         await add_getter()\n37 \n38         async def add_putter():\n39             _q = janus.Queue(maxsize=1)\n40             q = _q.async_q\n41             q.put_nowait(1)\n42             # Start a task that waits to put.\n43             loop.create_task(q.put(2))\n44             # Let it start waiting.\n45             await asyncio.sleep(0.1)\n46             assert \"_putters[1]\" in fn(q)\n47             # resume q.put coroutine to finish generator\n48             q.get_nowait()\n49 \n50         await add_putter()\n51 \n52         _q = janus.Queue()\n53         q = _q.async_q\n54         q.put_nowait(1)\n55         assert \"_queue=[1]\" in fn(q(...truncated)"
      },
      {
        "path": "aio-libs_janus/tests/test_sync.py",
        "content": "1 # Some simple queue module tests, plus some failure conditions\n2 # to ensure the Queue locks remain stable.\n3 import asyncio\n4 import queue\n5 import threading\n6 import time\n7 from unittest.mock import patch\n8 \n9 import pytest\n10 \n11 import janus\n12 \n13 QUEUE_SIZE = 5\n14 \n15 \n16 def qfull(q):\n17     return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n18 \n19 \n20 # A thread to run a function that unclogs a blocked Queue.\n21 class _TriggerThread(threading.Thread):\n22     def __init__(self, fn, args):\n23         self.fn = fn\n24         self.args = args\n25         self.startedEvent = threading.Event()\n26         threading.Thread.__init__(self)\n27 \n28     def run(self):\n29         # The sleep isn't necessary, but is intended to give the blocking\n30         # function in the main thread a chance at actually blocking before\n31         # we unclog it.  But if the sleep is longer than the timeout-based\n32         # tests wait in their blocking functions, those tests will fail.\n33         # So we give them much longer timeout values compared to the\n34         # sleep here (I aimed at 10 seconds for blockin(...truncated)"
      },
      {
        "path": "aio-libs_janus/janus/__init__.py",
        "content": "1 import asyncio\n2 import sys\n3 import threading\n4 from asyncio import QueueEmpty as AsyncQueueEmpty\n5 from asyncio import QueueFull as AsyncQueueFull\n6 from collections import deque\n7 from heapq import heappop, heappush\n8 from queue import Empty as SyncQueueEmpty\n9 from queue import Full as SyncQueueFull\n10 from typing import Any, Callable, Deque, Generic, List, Optional, Set, TypeVar\n11 \n12 from typing_extensions import Protocol\n13 \n14 __version__ = \"1.0.0\"\n15 __all__ = (\n16     \"Queue\",\n17     \"PriorityQueue\",\n18     \"LifoQueue\",\n19     \"SyncQueue\",\n20     \"AsyncQueue\",\n21     \"BaseQueue\",\n22 )\n23 \n24 \n25 T = TypeVar(\"T\")\n26 OptFloat = Optional[float]\n27 \n28 \n29 class BaseQueue(Protocol[T]):\n30     @property\n31     def maxsize(self) -> int:\n32         ...\n33 \n34     @property\n35     def closed(self) -> bool:\n36         ...\n37 \n38     def task_d(...truncated)"
      },
      {
        "path": "aio-libs_janus/tests/test_mixed.py",
        "content": "1 import asyncio\n2 import contextlib\n3 import sys\n4 import threading\n5 \n6 import pytest\n7 \n8 import janus\n9 \n10 \n11 class TestMixedMode:\n12     @pytest.mark.skipif(\n13         sys.version_info < (3, 7),\n14         reason=\"forbidding implicit loop creation works on \"\n15         \"Python 3.7 or higher only\",\n16     )\n17     def test_ctor_noloop(self):\n18         with pytest.raises(RuntimeError):\n19             janus.Queue()\n20 \n21     @pytest.mark.asyncio\n22     async def test_maxsize(self):\n23         q = janus.Queue(5)\n24         assert 5 == q.maxsize\n25 \n26     @pytest.ma(...truncated)"
      }
    ],
    "CommitSHA": "e46a1dfa166ebc2484716ee36cb99763f6247d52"
  },
  "Score": {
    "Difficulty": "Medium",
    "issue_origin": {
      "Title": 7,
      "Description": 6,
      "Reproducibility": 5,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_message": {
      "Title": 6,
      "Description": 6,
      "Reproducibility": 4,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 6
    },
    "issue_ground": {
      "Title": 8,
      "Description": 8,
      "Reproducibility": 8,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Fixing Async Queue Execution Context and Missing nonlocal Declaration",
      "description": "There are two primary issues in the current implementation:\n\n1. **Async Queue Execution Context for Lifos and Priority Queues:**\n   In the tests for `LifoQueue` and `PriorityQueue`, the tests are incorrectly inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`. This oversight could lead to improper handling and testing of task joins specific to Lifo and Priority queues.\n   - **Impact:** This can cause incorrect test behaviors, especially under edge cases, and might not thoroughly validate the functionality specific to Lifo and Priority queue implementations.\n\n2. **Missing `nonlocal` Declaration in Async Function:**\n   In the `queue_put` async function, the variable `finished` is being used without the `nonlocal` declaration, which means any modification of `finished` within `queue_put` does not affect the `finished` variable in the enclosing scope.\n   - **Impact:** This can result in inaccurate tracking of the `finished` state, leading to possible inconsistencies during async operations where the state needs to be accurately reflected/updated across multiple scopes.\n\nThe combination of these two issues can lead to unreliable tests for the async queue operations. Addressing them is essential to ensure accurate and consistent behavior in the queue operations and their corresponding tests.",
      "explanation": "### Summary of the Issue\n\nThe issue at hand concerns two primary problems in the current implementation of tests for asynchronous queues in the `janus` library. These issues are:\n\n1. **Async Queue Execution Context for LifoQueue and PriorityQueue**: The test classes for `LifoQueue` and `PriorityQueue` are incorrectly inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`. This could lead to improper handling and testing of task joins specific to Lifo and Priority queues.\n\n2. **Missing `nonlocal` Declaration in Async Function**: In the `queue_put` async function, the `finished` variable is used without the `nonlocal` declaration, causing any modification within `queue_put` not to affect the `finished` variable in the enclosing scope.\n\n\n### Commit Details\n\nTo address these issues, a commit has been made with the following changes:\n\n1. **Class Inheritance Adjustment**:\n    - The test classes `TestLifoQueueJoin` and `TestPriorityQueueJoin` have been modified to inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This change ensures that the tests correctly handle the specialized behavior of LifoQueue and PriorityQueue when joining tasks.\n\n2. **Add `nonlocal` Declaration**:\n    - In the `queue_put` async function within the test code, a `nonlocal` declaration for the `finished` variable has been added. This ensures that modifications to `finished` within the `queue_put` function are correctly reflected in the enclosing scope.\n\n\n### Explanation of the Issue and Solution\n\n#### Issue 1: Async Queue Execution Context for LifoQueue and PriorityQueue\n\n**Cause**:\nThe inheritance chain in the test classes is crucial for ensuring that the tests have the correct setup and assertions tailored to the specific queue types. By inheriting from `TestQueueJoin`, which is more generic, the LifoQueue and PriorityQueue tests might miss out on specific behaviors and checks that are pertinent only to these queue types.\n\n**Solution**:\nBy modifying the inheritance to `_QueueJoinTestMixin`, the tests now utilize a mixin that is likely designed to handle the idiosyncrasies of various queue join operations, particularly for Lifo and Priority queues. This correction ensures that the tests are more accurately reflecting the behaviors and edge cases specific to these queue types.\n\n#### Issue 2: Missing `nonlocal` Declaration in Async Function\n\n**Cause**:\nIn Python, the `nonlocal` keyword is used to indicate that a variable refers to a previously bound variable in the nearest enclosing scope that is not global. Without this keyword, any modification of the variable within the inner scope does not affect the variable in the outer scope. \n\nIn this case, without `nonlocal`, the `finished` variable inside the `queue_put` function would not update the `finished` status in the enclosing function or test, leading to incorrect behavior, particularly in cases where async operations depend on the accurate tracking of this state.\n\n**Solution**:\nBy adding the `nonlocal` keyword, the `finished` variable inside the `queue_put` function is correctly recognized as the same variable from the enclosing scope. This ensures that any changes to `finished` within `queue_put` are correctly propagated back to the outer function or test, maintaining accurate state tracking.\n\n### Conclusion\n\nThe commit addresses both issues by:\n1. Correcting the inheritance of test classes to ensure precise behavior checks specific to LifoQueue and PriorityQueue.\n2. Adding `nonlocal` declarations to ensure proper state management within async functions.\n\nThis leads to more reliable and accurate tests, ultimately enhancing the robustness of the library's async queue implementations."
    }
  }
}