{
  "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            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_b.cancel()\n\n        with pytest.raises(asyncio.CancelledError):\n            await put_b\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_put_cancelled_race ___________________________________________________________________________\n\nself = <test_async.TestQueuePut object at 0x7acab0cbf580>\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_b.cancel()\n    \n        with pytest.raises(asyncio.CancelledError):\n            await put_b\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\ntests/test_async.py:444: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\n    async def go():\n        a = await q.get()\n        assert a == \"a\"\n        b = await q.get()\n>       assert b == \"b\"\nE       AssertionError: assert 'X' == 'b'\nE         \nE         - b\nE         + X\n\ntests/test_async.py:441: AssertionError\n================================================================================= short test summary info ==================================================================================\nFAILED tests/test_async.py::TestQueuePut::test_put_cancelled_race - AssertionError: assert 'X' == 'b'\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@@ -429,10 +429,10 @@\n         await put_a\n         assert not put_b.done()\n \n-        put_b.cancel()\n+        put_c.cancel()\n \n         with pytest.raises(asyncio.CancelledError):\n-            await put_b\n+            await put_c\n \n         async def go():\n             a = await q.get()\n@@ -581,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": {
        "420": "test_put_cancelled_race"
      },
      "content_all": {
        "429": "        await put_a\n",
        "430": "        assert not put_b.done()\n",
        "431": "\n",
        "432": "        put_b.cancel()\n",
        "433": "\n",
        "434": "        with pytest.raises(asyncio.CancelledError):\n",
        "435": "            await put_b\n",
        "436": "\n",
        "437": "        async def go():\n",
        "438": "            a = await q.get()\n"
      },
      "content_change": {
        "432": "        put_b.cancel()\n",
        "435": "            await put_b\n"
      }
    },
    {
      "file": "aio-libs_janus/tests/test_async.py",
      "function": null,
      "content_all": {
        "581": "    q_class = janus.Queue\n",
        "582": "\n",
        "583": "\n",
        "584": "class TestLifoQueueJoin(TestQueueJoin):\n",
        "585": "    q_class = janus.LifoQueue\n",
        "586": "\n",
        "587": "\n",
        "588": "class TestPriorityQueueJoin(TestQueueJoin):\n",
        "589": "    q_class = janus.PriorityQueue\n"
      },
      "content_change": {
        "584": "class TestLifoQueueJoin(TestQueueJoin):\n",
        "588": "class TestPriorityQueueJoin(TestQueueJoin):\n"
      }
    },
    {
      "file": "aio-libs_janus/tests/test_mixed.py",
      "function": null,
      "content_all": {},
      "content_change": {}
    }
  ],
  "Issue": {
    "title": "Incorrect Test Case References and Class Inheritance in Async Testing",
    "description": "There are issues in the asynchronous tests which include incorrect task references and class inheritance problems. Specifically:\n\n1. In the test case where multiple put tasks are created (test_put_cancelled_race), the cancellation and assertion checks are being applied to `put_b` instead of `put_c`. Initially, `put_b` is created as a put task, but the subsequent cancellation and `await` operation incorrectly reference `put_c`, which causes failures and incorrect test behavior.\n\n2. The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes do not inherit from a mixin that provides essential queue testing functionality. Instead of inheriting from `TestQueueJoin`, these classes should inherit from `_QueueJoinTestMixin` to properly leverage the mixin's functionality and ensure comprehensive testing of LIFO and priority queue join operations.\n\n3. In the mixed mode tests, when waiting for an empty queue, there is an issue where the task is not cancelled after joining (wait_for_empty_queue). This could potentially result in the task hanging indefinitely, thereby affecting the reliability and robustness of the test suite.\n\nAddressing these issues will ensure that the tests accurately reflect the intended scenarios and provide reliable coverage for the queue functionalities.",
    "explanation": "### Summary of the Issue\n\nThe issue involves several problems with the asynchronous tests in a Python project, specifically in the `aio-libs_janus` repository. The key points are:\n\n1. **Incorrect task reference in tests:** In some test cases, cancellation and assertion checks are applied to the wrong tasks, leading to test failures and incorrect behaviors.\n2. **Incorrect class inheritance:** Some test classes (`TestLifoQueueJoin` and `TestPriorityQueueJoin`) do not inherit from the correct mixin class that provides essential queue functionalities.\n3. **Hanging tasks in the test suite:** There is an issue in mixed mode tests where a task might not be canceled properly after joining, potentially causing the test suite to hang indefinitely.\n\n### Commit Changes\n\nThe commit addresses these issues as follows:\n\n1. **Fixed Incorrect Task References:**\n   - The commit updates the asynchronous test case to ensure that cancellation and assertion checks are applied to the correct task (`put_c` instead of `put_b`). This corrects the logic and ensures that the assertions correctly reflect the intended behavior of the tasks, preventing false test failures.\n\n2. **Corrected Class Inheritance:**\n   - The commit changes the inheritance of `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes to inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This ensures that these test classes leverage the proper functionality provided by the mixin, leading to comprehensive and accurate testing of LIFO and priority queue join operations.\n\n3. **Resolved Hanging Tasks:**\n   - In the mixed mode tests, particularly when waiting for an empty queue, the commit introduces an additional step to cancel the task explicitly after joining. This prevents the task from hanging indefinitely, thereby improving the reliability and robustness of the test suite.\n\n### Cause and Solution Explained\n\n- **Cause of Incorrect Task References:**\n  The issue in the test was caused by referencing `put_b` for cancellation and assertion checks when it should have been `put_c`. This mismatch led to incorrect behaviors in the tests as they were not checking the proper tasks.\n\n  **Solution:**\n  The commit correctly identifies the task that should be referred to (`put_c`) and updates the test accordingly. This small but critical correction ensures that the intended task is canceled and awaited, leading to accurate test outcomes.\n\n- **Cause of Incorrect Class Inheritance:**\n  The test classes `TestLifoQueueJoin` and `TestPriorityQueueJoin` were inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`. This incorrect inheritance caused the tests to miss essential functionalities provided by the mixin, consequently affecting the comprehensiveness and accuracy of queue join tests.\n\n  **Solution:**\n  The fix involves modifying the inheritance so that these classes inherit from `_QueueJoinTestMixin`. This ensures that they include the necessary queue operations and validations, leading to thorough testing.\n\n- **Cause of Hanging Tasks:**\n  In the mixed mode tests, there wasn't a proper mechanism to cancel tasks after the join operation, leading to scenarios where tasks could hang indefinitely. Such issues affect the robustness and reliability of the test suite.\n\n  **Solution:**\n  The commit adds a step to cancel the task explicitly after the join operation. This ensures that all tasks are properly concluded, preventing any potential indefinite hanging and enhancing the test suite's robustness.\n\n### Conclusion\n\nThe commit effectively addresses the issues by:\n1. Correcting task references in the test cases to ensure proper cancellation and assertions.\n2. Adjusting class inheritance to include the correct mixin, thereby enabling comprehensive queue join tests.\n3. Explicitly canceling tasks in mixed mode tests to prevent indefinite hanging.\n\nThese changes collectively ensure that the asynchronous tests behave as intended, reliable, and cover the necessary functionalities thoroughly. This improvement enhances the overall test accuracy and robustness, contributing to the project's reliability and maintainability."
  },
  "Explain": "### Summary of the Issue\n\nThe issue involves several problems with the asynchronous tests in a Python project, specifically in the `aio-libs_janus` repository. The key points are:\n\n1. **Incorrect task reference in tests:** In some test cases, cancellation and assertion checks are applied to the wrong tasks, leading to test failures and incorrect behaviors.\n2. **Incorrect class inheritance:** Some test classes (`TestLifoQueueJoin` and `TestPriorityQueueJoin`) do not inherit from the correct mixin class that provides essential queue functionalities.\n3. **Hanging tasks in the test suite:** There is an issue in mixed mode tests where a task might not be canceled properly after joining, potentially causing the test suite to hang indefinitely.\n\n### Commit Changes\n\nThe commit addresses these issues as follows:\n\n1. **Fixed Incorrect Task References:**\n   - The commit updates the asynchronous test case to ensure that cancellation and assertion checks are applied to the correct task (`put_c` instead of `put_b`). This corrects the logic and ensures that the assertions correctly reflect the intended behavior of the tasks, preventing false test failures.\n\n2. **Corrected Class Inheritance:**\n   - The commit changes the inheritance of `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes to inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This ensures that these test classes leverage the proper functionality provided by the mixin, leading to comprehensive and accurate testing of LIFO and priority queue join operations.\n\n3. **Resolved Hanging Tasks:**\n   - In the mixed mode tests, particularly when waiting for an empty queue, the commit introduces an additional step to cancel the task explicitly after joining. This prevents the task from hanging indefinitely, thereby improving the reliability and robustness of the test suite.\n\n### Cause and Solution Explained\n\n- **Cause of Incorrect Task References:**\n  The issue in the test was caused by referencing `put_b` for cancellation and assertion checks when it should have been `put_c`. This mismatch led to incorrect behaviors in the tests as they were not checking the proper tasks.\n\n  **Solution:**\n  The commit correctly identifies the task that should be referred to (`put_c`) and updates the test accordingly. This small but critical correction ensures that the intended task is canceled and awaited, leading to accurate test outcomes.\n\n- **Cause of Incorrect Class Inheritance:**\n  The test classes `TestLifoQueueJoin` and `TestPriorityQueueJoin` were inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`. This incorrect inheritance caused the tests to miss essential functionalities provided by the mixin, consequently affecting the comprehensiveness and accuracy of queue join tests.\n\n  **Solution:**\n  The fix involves modifying the inheritance so that these classes inherit from `_QueueJoinTestMixin`. This ensures that they include the necessary queue operations and validations, leading to thorough testing.\n\n- **Cause of Hanging Tasks:**\n  In the mixed mode tests, there wasn't a proper mechanism to cancel tasks after the join operation, leading to scenarios where tasks could hang indefinitely. Such issues affect the robustness and reliability of the test suite.\n\n  **Solution:**\n  The commit adds a step to cancel the task explicitly after the join operation. This ensures that all tasks are properly concluded, preventing any potential indefinite hanging and enhancing the test suite's robustness.\n\n### Conclusion\n\nThe commit effectively addresses the issues by:\n1. Correcting task references in the test cases to ensure proper cancellation and assertions.\n2. Adjusting class inheritance to include the correct mixin, thereby enabling comprehensive queue join tests.\n3. Explicitly canceling tasks in mixed mode tests to prevent indefinite hanging.\n\nThese changes collectively ensure that the asynchronous tests behave as intended, reliable, and cover the necessary functionalities thoroughly. This improvement enhances the overall test accuracy and robustness, contributing to the project's reliability and maintainability.",
  "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": 15102,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Unclosed Queue instances leading to resource leakage",
      "description": "In aio-libs_janus/tests/test_async.py, the Queue instances created within the test functions (e.g., _test_repr_or_str, add_getter, and add_putter) are not explicitly closed. This can lead to resource leakage as the queues are not properly cleaned up. It is recommended to ensure that each created Queue instance is properly closed after use.",
      "explanation": "### Summary of the Issue\nThe issue reported is about potential resource leakage in the `aio-libs_janus` project. Specifically, in the file `tests/test_async.py`, instances of `janus.Queue` created within several test functions are not being explicitly closed, leading to the risk of resource leakage.\n\n### Content of the Commit\nThe commit (not provided in the request, but we can assume the steps would be similar) should include changes that ensure all created `Queue` instances within the test functions are properly closed after use. This could typically involve calling a `.close()` or similar method on the `Queue` instances and ensuring that the `Queue` is awaited until it is fully closed.\n\nLet's assume the commit modifies the following parts of the code for illustration purposes:\n\n#### Before Commit\n```python\nclass TestQueueBasic:\n    async def _test_repr_or_str(self, fn, expect_id):\n        # other parts of the method...\n\n        _q = janus.Queue()\n        q = _q.async_q\n\n        # other parts of the method...\n\n        async def add_getter():\n            _q = janus.Queue()\n            q = _q.async_q\n            loop.create_task(q.get())\n            await asyncio.sleep(0.1)\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            loop.create_task(q.put(2))\n            await asyncio.sleep(0.1)\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```\n\n#### After Commit\n```python\nclass TestQueueBasic:\n    async def _test_repr_or_str(self, fn, expect_id):\n        # other parts of the method...\n\n        _q = janus.Queue()\n        q = _q.async_q\n\n        # other parts of the method...\n\n        try:\n            async def add_getter():\n                _q = janus.Queue()\n                q = _q.async_q\n                loop.create_task(q.get())\n                await asyncio.sleep(0.1)\n                q.put_nowait(0)\n                await _q.close()\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                loop.create_task(q.put(2))\n                await asyncio.sleep(0.1)\n                q.get_nowait()\n                await _q.close()\n\n            await add_putter()\n\n            _q = janus.Queue()\n            q = _q.async_q\n            q.put_nowait(1)\n        finally:\n            await _q.close()\n```\n\n### Explanation of the Solution\n\n1. **Context Creation:**\n   In each of the test functions (e.g., `_test_repr_or_str`, `add_getter`, `add_putter`), `janus.Queue` instances were created and used. However, they were not being properly closed.\n\n2. **Resource Management:**\n   With the modifications:\n\n   - Ensure that after using the queue (`_q`), the `.close()` method is called.\n   - Use `try...finally` constructs to guarantee that the queue is closed even if an error occurs during the function execution.\n\n3. **Execution Flow Changes:**\n   - **_test_repr_or_str**:\n     - After the main test logic, `_q.close()` is awaited to ensure closure of the queue.\n   - **add_getter** and **add_putter**:\n     - Similar changes ensure that the queues created inside these nested async functions are properly closed after their work is done.\n\n### How the Commit Solves the Issue\n\nCalling the `.close()` method ensures that any underlying resources tied to the `janus.Queue` instances are properly released. This prevents resource leakage, which can have detrimental effects on long-running applications or extensive test suites. The use of `await` ensures that the close operation completes, thus avoiding dangling resources.\n\n- The `finally` block guarantees that the closure operation is attempted irrespective of whether the preceding code executed successfully or encountered an exception.\n- The nested functions `add_getter` and `add_putter` also implement resource management to avoid any leaks from their respective `Queue` instances.\n\nThis approach ensures that each `janus.Queue` instance created within the tests is properly cleaned up, aligning with best practices for resource management in asynchronous programming in Python."
    },
    "issue_message": {
      "title": "Assertion Error in test_put_cancelled_race",
      "description": "The test case TestQueuePut.test_put_cancelled_race fails with an AssertionError. The issue is that the expected value 'b' is not matching the actual value 'X'. This indicates that the put_b task, which was cancelled, somehow still put 'X' into the queue. Investigate the handling of cancelled tasks and ensure proper synchronization and state management in the queue operations.",
      "explanation": "## Summary of the Issue\n\nThe issue revolves around the `test_put_cancelled_race` test case, which fails with an `AssertionError`. The problem is due to a mismatch between the expected output value 'b' and the actual output value 'X' from a queue operation. The test failure indicates that despite the `put_b` task being cancelled, it seems to still have inserted 'X' into the queue, conflicting with expected behavior.\n\n## Detailed Analysis of the Commit\n\nThe commit details provided are incomplete, but I'll analyze the provided code snippets and broader context to pinpoint the likely solution approach:\n\n```python\n========================================================================================= FAILURES =========================================================================================\n___________________________________________________________________________ TestQueuePut.test_put_cancelled_race ___________________________________________________________________________\n\nself = <test_async.TestQueuePut object at 0x7acab0cbf580>\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_b.cancel()\n    \n        with pytest.raises(asyncio.CancelledError):\n            await put_b\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\ntests/test_async.py:444: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\n    async def go():\n        a = await q.get()\n        assert a == \"a\"\n        b = await q.get()\n>       assert b == \"b\"\nE       AssertionError: assert 'X' == 'b'\nE         \nE         - b\nE         + X\n\ntests/test_async.py:441: AssertionError\n\n```\n\n### Cause of the Issue\n\n1. **Cancellation Handling**: The critical part here is whether the `put_b.cancel()` call prevents any value from being inserted into the queue. If the task `put_b` is correctly cancelled, it should not have any effect.\n2. **Synchronization and State Management**: The queue operations must handle race conditions properly. If `put_c` is getting executed after `put_b`, it might be interfering.\n\n### Potential Fix\n\nBased on the details, the fix likely involves:\n\n1. Ensuring that `q.put()` recognizes and respects task cancellations immediately and doesn't attempt to put any value into the queue post-cancellation.\n2. Synchronizing the tasks properly to prevent race conditions that allow cancelled operations to have side effects.\n\nLet's assume the commit will change the way task cancellations are handled within the queue system, ensuring that cancelled tasks do not accidentally commit their values into the queue.\n\n### Hypothetical Commit and Solution\n\n```python\ndef _put(self, item: T) -> None:\n    with self._putters_condition:\n        if self.maxsize > 0 and self.qsize() >= self.maxsize:\n            self._putters.append((self._loop, item))\n            return\n        self._queue.append(item)\n\ndef cancel_pending_putters(self):\n    with self._putters_condition:\n        for waiter, item in self._putters:\n            waiter.cancel()\n\n# Changes in put_nowait() and similar methods to ensure cancellation is respected:\nasync def put(self, item: T) -> None:\n    with self._putters_condition:\n        if self.maxsize > 0 and self.qsize() >= self.maxsize:\n            self._putters.append((self._loop, item))\n            #...\n        else:\n            self._queue.append(item)\n            self._putters_condition.notify()\n```\n\n### Explanation of How the Commit Solves the Issue\n\n1. **Immediate Recognition of Cancellation**:\n    - By integrating a cancellation check and handling mechanism within queue put operations, any ongoing `put()` operations that get cancelled do not proceed to modify the queue state.\n    - The potential fix ensures that if a `put` task is cancelled, the system will not attempt to add the item to the queue.\n\n2. **Synchronization and State Management**:\n    - The commit adds a condition to hold and manage `putters` in a way that respects cancellation.\n    - The queue state becomes consistent as tasks that register as cancelled will not be allowed to interfere further.\n\n3. **Additional Safeguards**:\n   "
    },
    "issue_ground": {
      "title": "Incorrect Test Case References and Class Inheritance in Async Testing",
      "description": "There are issues in the asynchronous tests which include incorrect task references and class inheritance problems. Specifically:\n\n1. In the test case where multiple put tasks are created (test_put_cancelled_race), the cancellation and assertion checks are being applied to `put_b` instead of `put_c`. Initially, `put_b` is created as a put task, but the subsequent cancellation and `await` operation incorrectly reference `put_c`, which causes failures and incorrect test behavior.\n\n2. The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes do not inherit from a mixin that provides essential queue testing functionality. Instead of inheriting from `TestQueueJoin`, these classes should inherit from `_QueueJoinTestMixin` to properly leverage the mixin's functionality and ensure comprehensive testing of LIFO and priority queue join operations.\n\n3. In the mixed mode tests, when waiting for an empty queue, there is an issue where the task is not cancelled after joining (wait_for_empty_queue). This could potentially result in the task hanging indefinitely, thereby affecting the reliability and robustness of the test suite.\n\nAddressing these issues will ensure that the tests accurately reflect the intended scenarios and provide reliable coverage for the queue functionalities.",
      "explanation": "### Summary of the Issue\n\nThe issue primarily revolves around three main problems identified within the asynchronous test suite:\n1. **Incorrect Task References:**\n   - In the `test_put_cancelled_race` test case, there's a mix-up where cancellation and assertions are incorrectly referencing `put_c` instead of `put_b`.\n   \n2. **Class Inheritance Problems:**\n   - The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes are incorrectly inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`, leading to incomplete testing of queue functionalities.\n\n3. **Task Not Canceling Properly:**\n   - The test for waiting on an empty queue (`wait_for_empty_queue`) does not cancel the task after joining, possibly causing it to hang indefinitely.\n\n### Content of the Commit\n\nThe commit would address the aforementioned issues by making specific changes to the code as follows:\n\n#### 1. Correcting the Task References in `test_put_cancelled_race`\n```python\n@pytest.mark.asyncio\nasync 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_b.cancel()\n\n    with pytest.raises(asyncio.CancelledError):\n        await put_b\n\n    async def go():\n        a = await q.get()\n        assert a == \"a\"\n        b = await q.get()  # This part remains, no changes needed.\n        assert b == \"b\"     # The rest of the code remains unchanged, since the correct put task is being referenced.\n        assert put_b.done()\n\n    await go()\n```\nThe correction here ensures that cancellation and assertion operations are properly referencing the `put_b` task, aligning the behavior with the correct task.\n\n#### 2. Adjusting Class Inheritance for Queue Join Tests\n```python\n# Original code had\nclass TestLifoQueueJoin(TestQueueJoin):\n    # ...\n\nclass TestPriorityQueueJoin(TestQueueJoin):\n    # ...\n\n# It should be changed to\nclass TestLifoQueueJoin(_QueueJoinTestMixin):\n    # ...\n\nclass TestPriorityQueueJoin(_QueueJoinTestMixin):\n    # ...\n```\nThis adjustment ensures the classes inherit from `_QueueJoinTestMixin` to fully leverage its comprehensive queue testing functionalities.\n\n#### 3. Proper Task Cancellation in Mixed Mode Tests\n```python\n@pytest.mark.asyncio\nasync def wait_for_empty_queue(self):\n    q = janus.Queue(maxsize=1)\n    ...\n    # After the join operation, we should ensure the task is cancelled properly.\n    some_task.cancel()\n    with pytest.raises(asyncio.CancelledError):\n        await some_task\n```\nThe fix ensures that tasks are properly cancelled after the join operation, and the use of `pytest.raises(asyncio.CancelledError)` confirms the cancellation, thus preventing indefinite hangs.\n\n### Explanation of How the Commit Solves the Issue\n\n1. **Correct Task References in `test_put_cancelled_race`:**\n   - By ensuring the tests reference the correct tasks (`put_b`), the commit rectifies the unintended behavior where the wrong task (`put_c`) was being checked for cancellation status. This guarantees that the assertions accurately reflect the intended operations.\n\n2. **Proper Class Inheritance:**\n   - Adjusting the inheritance allows `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes to utilize the full spectrum of functionalities provided by `_QueueJoinTestMixin`. This change ensures comprehensive and rigorous testing by incorporating methods and behaviors defined in the mixin.\n\n3. **Proper Cancellation in Mixed Mode Tests:**\n   - Amending the `wait_for_empty_queue` ensures that tasks are definitively cancelled after their respective operations, mitigating risks of tasks hanging indefinitely. This adjustment improves the reliability and robustness of the asynchronous test suite.\n\nOverall, these changes collectively enhance the accuracy, reliability, and completeness of the test cases, ensuring they align with the intended asynchronous queue behaviors and scenarios."
    },
    "issue_ground_truth": {
      "title": "Incorrect Test Case References and Class Inheritance in Async Testing",
      "description": "There are issues in the asynchronous tests which include incorrect task references and class inheritance problems. Specifically:\n\n1. In the test case where multiple put tasks are created (test_put_cancelled_race), the cancellation and assertion checks are being applied to `put_b` instead of `put_c`. Initially, `put_b` is created as a put task, but the subsequent cancellation and `await` operation incorrectly reference `put_c`, which causes failures and incorrect test behavior.\n\n2. The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes do not inherit from a mixin that provides essential queue testing functionality. Instead of inheriting from `TestQueueJoin`, these classes should inherit from `_QueueJoinTestMixin` to properly leverage the mixin's functionality and ensure comprehensive testing of LIFO and priority queue join operations.\n\n3. In the mixed mode tests, when waiting for an empty queue, there is an issue where the task is not cancelled after joining (wait_for_empty_queue). This could potentially result in the task hanging indefinitely, thereby affecting the reliability and robustness of the test suite.\n\nAddressing these issues will ensure that the tests accurately reflect the intended scenarios and provide reliable coverage for the queue functionalities.",
      "explanation": "### Summary of the Issue\n\nThe issue involves several problems with the asynchronous tests in a Python project, specifically in the `aio-libs_janus` repository. The key points are:\n\n1. **Incorrect task reference in tests:** In some test cases, cancellation and assertion checks are applied to the wrong tasks, leading to test failures and incorrect behaviors.\n2. **Incorrect class inheritance:** Some test classes (`TestLifoQueueJoin` and `TestPriorityQueueJoin`) do not inherit from the correct mixin class that provides essential queue functionalities.\n3. **Hanging tasks in the test suite:** There is an issue in mixed mode tests where a task might not be canceled properly after joining, potentially causing the test suite to hang indefinitely.\n\n### Commit Changes\n\nThe commit addresses these issues as follows:\n\n1. **Fixed Incorrect Task References:**\n   - The commit updates the asynchronous test case to ensure that cancellation and assertion checks are applied to the correct task (`put_c` instead of `put_b`). This corrects the logic and ensures that the assertions correctly reflect the intended behavior of the tasks, preventing false test failures.\n\n2. **Corrected Class Inheritance:**\n   - The commit changes the inheritance of `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes to inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This ensures that these test classes leverage the proper functionality provided by the mixin, leading to comprehensive and accurate testing of LIFO and priority queue join operations.\n\n3. **Resolved Hanging Tasks:**\n   - In the mixed mode tests, particularly when waiting for an empty queue, the commit introduces an additional step to cancel the task explicitly after joining. This prevents the task from hanging indefinitely, thereby improving the reliability and robustness of the test suite.\n\n### Cause and Solution Explained\n\n- **Cause of Incorrect Task References:**\n  The issue in the test was caused by referencing `put_b` for cancellation and assertion checks when it should have been `put_c`. This mismatch led to incorrect behaviors in the tests as they were not checking the proper tasks.\n\n  **Solution:**\n  The commit correctly identifies the task that should be referred to (`put_c`) and updates the test accordingly. This small but critical correction ensures that the intended task is canceled and awaited, leading to accurate test outcomes.\n\n- **Cause of Incorrect Class Inheritance:**\n  The test classes `TestLifoQueueJoin` and `TestPriorityQueueJoin` were inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`. This incorrect inheritance caused the tests to miss essential functionalities provided by the mixin, consequently affecting the comprehensiveness and accuracy of queue join tests.\n\n  **Solution:**\n  The fix involves modifying the inheritance so that these classes inherit from `_QueueJoinTestMixin`. This ensures that they include the necessary queue operations and validations, leading to thorough testing.\n\n- **Cause of Hanging Tasks:**\n  In the mixed mode tests, there wasn't a proper mechanism to cancel tasks after the join operation, leading to scenarios where tasks could hang indefinitely. Such issues affect the robustness and reliability of the test suite.\n\n  **Solution:**\n  The commit adds a step to cancel the task explicitly after the join operation. This ensures that all tasks are properly concluded, preventing any potential indefinite hanging and enhancing the test suite's robustness.\n\n### Conclusion\n\nThe commit effectively addresses the issues by:\n1. Correcting task references in the test cases to ensure proper cancellation and assertions.\n2. Adjusting class inheritance to include the correct mixin, thereby enabling comprehensive queue join tests.\n3. Explicitly canceling tasks in mixed mode tests to prevent indefinite hanging.\n\nThese changes collectively ensure that the asynchronous tests behave as intended, reliable, and cover the necessary functionalities thoroughly. This improvement enhances the overall test accuracy and robustness, contributing to the project's reliability and maintainability."
    },
    "location_origin": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "11": "_test_repr_or_str",
          "25": "add_getter",
          "38": "add_putter"
        },
        "content_all": {
          "8": "9\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)\n"
        },
        "content_change": {
          "18": "\n18         _q = janus.Queue()\n19         q = _q.async_q\n20         try:\n21             assert fn(q).startswith(\"<Queue\")\n22             id_is_present = hex(id(q)) in fn(q)\n23             assert expect_id == id_is_present\n24             loop = asyncio.get_running_loop()\n25 \n26             async def add_getter():\n27                 _q = janus.Queue()\n28                 q = _q.async_q\n29                 # Start a task that waits to get.\n30                 loop.create_task(q.get())\n31                 # Let it start waiting.\n32                 await asyncio.sleep(0.1)\n33                 assert \"_getters[1]\" in fn(q)\n34                 # resume q.get coroutine to finish generator\n35                 q.put_nowait(0)\n36                 await _q.close()\n37 \n38             await add_getter()\n39 \n40             async def add_putter():\n41                 _q = janus.Queue(maxsize=1)\n42                 q = _q.async_q\n43                 q.put_nowait(1)\n44                 # Start a task that waits to put.\n45                 loop.create_task(q.put(2))\n46                 # Let it start waiting.\n47                 await asyncio.sleep(0.1)\n48                 assert \"_putters[1]\" in fn(q)\n49                 # resume q.put coroutine to finish generator\n50                 q.get_nowait()\n51                 await _q.close()\n52 \n53             await add_putter()\n54 \n55             _q = janus.Queue()\n56             q = _q.async_q\n57             q.put_nowait(1)\n58         finally:\n59             await _q.close()\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "aio-libs_janus/janus/__init__.py",
        "function": {
          "327": "put"
        },
        "content_all": {
          "326": "            self._putters.append((self._loop, item))",
          "327": "            return",
          "328": "        self._queue.append(item)",
          "329": "        self._putters_condition.notify()"
        },
        "content_change": {
          "327": "            return",
          "328": "        if waiter.cancelled():",
          "329": "            continue",
          "330": "        self._queue.append(item)",
          "331": "        self._putters_condition.notify()"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "429": "test_put_cancelled_race"
        },
        "content_all": {
          "426": "        _q = janus.Queue(maxsize=1)",
          "427": "        q = _q.async_q",
          "428": "        put_a = loop.create_task(q.put(\"a\"))",
          "429": "        put_b = loop.create_task(q.put(\"b\"))",
          "430": "        put_c = loop.create_task(q.put(\"X\"))",
          "431": "        await put_a",
          "432": "        assert not put_b.done()"
        },
        "content_change": {
          "429": "        put_b = loop.create_task(q.put(\"b\"))",
          "430": "        put_c = loop.create_task(q.put(\"X\"))"
        }
      }
    ],
    "location_ground": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "80": "test_put_cancelled_race"
        },
        "content_all": {
          "77": "\n",
          "78": "    put_a = loop.create_task(q.put(\"a\"))\n",
          "79": "    put_b = loop.create_task(q.put(\"b\"))\n",
          "80": "    put_c = loop.create_task(q.put(\"X\"))\n",
          "81": "\n",
          "82": "    await put_a\n",
          "83": "    assert not put_b.done()\n",
          "84": "\n",
          "85": "    put_b.cancel()\n",
          "86": "\n",
          "87": "    with pytest.raises(asyncio.CancelledError):\n",
          "88": "        await put_c\n"
        },
        "content_change": {
          "88": "        await put_b\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_sync.py",
        "content_all": {
          "328": " \n",
          "329": "class TestLifoQueueJoin(TestQueueJoin):",
          "330": "    pass",
          "331": "\n",
          "332": "class TestPriorityQueueJoin(TestQueueJoin):",
          "333": "    pass"
        },
        "content_change": {
          "329": "class TestLifoQueueJoin(_QueueJoinTestMixin):",
          "332": "class TestPriorityQueueJoin(_QueueJoinTestMixin):"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": {
          "30": "wait_for_empty_queue"
        },
        "content_all": {
          "27": "\n",
          "28": "        # After the join\n",
          "29": "        await q.join()\n",
          "30": "        # Ensure task is cancelled properly\n",
          "31": "        some_task.cancel()\n",
          "32": "        with pytest.raises(asyncio.CancelledError):\n",
          "33": "            await some_task\n",
          "34": "\n",
          "35": "        assert q.empty()"
        },
        "content_change": {
          "31": "        some_task.cancel()\n",
          "32": "        with pytest.raises(asyncio.CancelledError):\n",
          "33": "            await some_task"
        }
      }
    ],
    "location_ground_exp": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "789": "test_put_cancelled_race"
        },
        "content_all": {
          "785": "        put_a = loop.create_task(queue.put(1))",
          "786": "        put_b = loop.create_task(queue.put(2))",
          "787": "        put_c = loop.create_task(queue.put(3))",
          "788": "        await asyncio.sleep(0.1)",
          "789": "        put_b.cancel()",
          "790": "        await put_b",
          "791": "        assert put_b.cancelled()",
          "792": "        assert not put_c.cancelled()"
        },
        "content_change": {
          "789": "        put_c.cancel()",
          "790": "        await put_c",
          "791": "        assert put_c.cancelled()"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_sync.py",
        "function": {
          "300": "TestLifoQueueJoin"
        },
        "content_all": {
          "297": "\n",
          "298": "",
          "299": "class TestLifoQueueJoin(TestQueueJoin):",
          "300": "    pass",
          "301": "\n",
          "302": ""
        },
        "content_change": {
          "299": "class TestLifoQueueJoin(_QueueJoinTestMixin, TestQueueJoin):"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_sync.py",
        "function": {
          "320": "TestPriorityQueueJoin"
        },
        "content_all": {
          "317": "\n",
          "318": "",
          "319": "class TestPriorityQueueJoin(TestQueueJoin):",
          "320": "    pass",
          "321": "\n",
          "322": ""
        },
        "content_change": {
          "319": "class TestPriorityQueueJoin(_QueueJoinTestMixin, TestQueueJoin):"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": {
          "620": "wait_for_empty_queue"
        },
        "content_all": {
          "615": "            # Wait for the task to complete",
          "616": "            await task",
          "617": "",
          "618": "        finally:",
          "619": "            # Ensure task is canceled",
          "620": "            task.cancel()",
          "621": "            with pytest.raises(asyncio.CancelledError):",
          "622": "                await task"
        },
        "content_change": {
          "618": "        finally:",
          "619": "            # Ensure task is canceled",
          "620": "            task.cancel()",
          "621": "            with pytest.raises(asyncio.CancelledError):",
          "622": "                await task"
        }
      }
    ],
    "location_ground_truth": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "420": "test_put_cancelled_race"
        },
        "content_all": {
          "429": "        await put_a\n",
          "430": "        assert not put_b.done()\n",
          "431": "\n",
          "432": "        put_b.cancel()\n",
          "433": "\n",
          "434": "        with pytest.raises(asyncio.CancelledError):\n",
          "435": "            await put_b\n",
          "436": "\n",
          "437": "        async def go():\n",
          "438": "            a = await q.get()\n"
        },
        "content_change": {
          "432": "        put_b.cancel()\n",
          "435": "            await put_b\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": null,
        "content_all": {
          "581": "    q_class = janus.Queue\n",
          "582": "\n",
          "583": "\n",
          "584": "class TestLifoQueueJoin(TestQueueJoin):\n",
          "585": "    q_class = janus.LifoQueue\n",
          "586": "\n",
          "587": "\n",
          "588": "class TestPriorityQueueJoin(TestQueueJoin):\n",
          "589": "    q_class = janus.PriorityQueue\n"
        },
        "content_change": {
          "584": "class TestLifoQueueJoin(TestQueueJoin):\n",
          "588": "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@@ -18,48 +18,62 @@\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        try:\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                await _q.close()\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                await _q.close()\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        finally:\n            await _q.close()\n",
    "patch_im": "\n--- a/janus/__init__.py\n+++ b/janus/__init__.py\n@@ -100,6 +100,16 @@ async def put(self, item: T) -> None:\n             return\n         self._queue.append(item)\n         self._putters_condition.notify()\n+        if self._cancel_put(task):\n+            return\n \n+    def _cancel_put(self, task):\n+        try:\n+            task.result()\n+        except asyncio.CancelledError:\n+            return True\n+        return False\n+\n __import__(\"pkg_resources\").declare_namespace(__name__)\n\n\n--- a/tests/test_async.py\n+++ b/tests/test_async.py\n@@ -434,7 +434,12 @@ class TestQueuePut:\n \n     @pytest.mark.asyncio\n     async def test_put_cancelled_race(self):\n+        import asyncio\n         loop = asyncio.get_running_loop()\n+        # Ensure the creation of the tasks within the 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@@ -451,6 +456,9 @@ class TestQueuePut:\n             assert a == \"a\"\n             b = await q.get()\n             assert b == \"b\"\n+            assert not put_c.done()  # Ensure put_c has not executed\n+            c = await q.get()\n+            assert c == \"X\"\n             assert put_b.done()\n \n         await go()\n",
    "patch_il": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -18,12 +18,17 @@\n         _q = janus.Queue()\n         q = _q.async_q\n         try:\n+            async def close_queue(queue):\n+                queue.close()\n+                await queue.wait_closed()\n+\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+                await close_queue(_q)\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+                await close_queue(_q)\n \n             await add_putter()\n \n             _q = janus.Queue()\n             q = _q.async_q\n             q.put_nowait(1)\n         finally:\n+            await close_queue(_q)\n",
    "patch_iml": "\n--- a/aio-libs_janus/janus/__init__.py\n+++ b/aio-libs_janus/janus/__init__.py\n@@ -324,8 +324,10 @@\n             self._putters.append((self._loop, item))\n             return\n         self._queue.append(item)\n         self._putters_condition.notify()\n\n+        # This ensures the cancelled tasks do not put items in the queue\n+        self._handle_cancelled_putters()\n+\n     def _handle_cancelled_putters(self):\n         with self._putters_condition:\n             new_putters = []\n             for waiter, item in self._putters:\n                 if not waiter.cancelled():\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -429,7 +429,13 @@\n         put_c = loop.create_task(q.put(\"X\"))\n \n         await put_a\n         assert not put_b.done()\n \n+        async def cancel_put_task(task):\n+            if not task.done():\n+                task.cancel()\n+                try:\n+                    await task\n+                except asyncio.CancelledError:\n+                    pass\n\n         put_b.cancel()\n \n         with pytest.raises(asyncio.CancelledError):\n             await put_b\n@@ -439,6 +443,7 @@\n             assert a == \"a\"\n             b = await q.get()\n             assert b == \"b\"\n             assert put_b.done()\n \n+        await cancel_put_task(put_b)\n         await go()\n",
    "patch_ground": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -250,16 +250,16 @@\n     await q.put(f\"item {item}\")\n \n @pytest.mark.asyncio\n async def test_put_cancelled_race():\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_c = loop.create_task(q.put(\"b\"))\n-    put_b = loop.create_task(q.put(\"X\"))\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_b.cancel()\n \n     with pytest.raises(asyncio.CancelledError):\n         await put_b\n\n--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -568,7 +568,7 @@\n class TestLifoQueueJoin(_QueueJoinTestMixin):\n-    pass  \n\n class TestPriorityQueueJoin(_QueueJoinTestMixin):\n-    pass\n\n @pytest.mark.asyncio\n async def wait_for_empty_queue(self):\n     q = janus.Queue(maxsize=1)\n     ...\n     some_task = asyncio.create_task(q.join())\n     await q.put(1)\n     ...\n     some_task.cancel()\n     with pytest.raises(asyncio.CancelledError):\n         await some_task\n\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -100,7 +100,9 @@\n     ...\n     # After the join operation, we should ensure the task is cancelled properly.\n@@ -227,11 +227,7 @@\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+    some_task = asyncio.create_task(q.join())\n     await q.put(1)\n     ...\n     some_task.cancel()\n     with pytest.raises(asyncio.CancelledError):\n         await some_task\n",
    "patch_ground_location": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -80,7 +80,7 @@\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_b.cancel()\n \n     with pytest.raises(asyncio.CancelledError):\n-        await put_c\n+        await put_b\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--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -329,7 +329,7 @@\n \n class TestLifoQueueJoin(TestQueueJoin):\n     pass\n \n class TestPriorityQueueJoin(TestQueueJoin):\n     pass\n+class TestLifoQueueJoin(_QueueJoinTestMixin):\n+    pass\n \n+class TestPriorityQueueJoin(_QueueJoinTestMixin):\n+    pass\n \n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -28,8 +28,9 @@\n         # After the join\n         await q.join()\n         # Ensure task is cancelled properly\n         some_task.cancel()\n         with pytest.raises(asyncio.CancelledError):\n             await some_task\n \n         assert q.empty()\n",
    "patch_ground_exp": "\n--- aio-libs_janus/tests/test_async.py\n+++ aio-libs_janus/tests/test_async.py\n@@ -786,9 +786,9 @@\n         put_a = loop.create_task(queue.put(1))\n         put_b = loop.create_task(queue.put(2))\n         put_c = loop.create_task(queue.put(3))\n         await asyncio.sleep(0.1)\n-        put_b.cancel()\n-        await put_b\n-        assert put_b.cancelled()\n-        assert not put_c.cancelled()\n+        put_c.cancel()\n+        await put_c\n+        assert put_c.cancelled()\n \n \n--- aio-libs_janus/tests/test_sync.py\n+++ aio-libs_janus/tests/test_sync.py\n@@ -299,7 +299,7 @@\n \n \n class TestLifoQueueJoin(TestQueueJoin):\n+class TestLifoQueueJoin(_QueueJoinTestMixin, TestQueueJoin):\n     pass\n \n \n@@ -319,7 +319,7 @@\n \n \n class TestPriorityQueueJoin(TestQueueJoin):\n+class TestPriorityQueueJoin(_QueueJoinTestMixin, TestQueueJoin):\n     pass\n \n \n--- aio-libs_janus/tests/test_mixed.py\n+++ aio-libs_janus/tests/test_mixed.py\n@@ -618,9 +618,9 @@\n         finally:\n             # Ensure task is canceled\n             task.cancel()\n             with pytest.raises(asyncio.CancelledError):\n                 await task\n",
    "patch_ground_all": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -429,9 +429,9 @@ async def test_put_cancelled_race():\n         await put_a\n         assert not put_b.done()\n \n-        put_b.cancel()\n+        put_c.cancel()\n \n         with pytest.raises(asyncio.CancelledError):\n-            await put_b\n+            await put_c\n \n         async def go():\n             a = await q.get()\n@@ -581,10 +581,10 @@ class TestQueueJoin:\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 \n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -263,7 +263,9 @@ async def test_wait_for_empty_queue():\n \n         await q.join()\n \n-        assert q.empty()\n+        # Cancel the pending task\n+        task.cancel()\n+        assert await q.empty()\n \n-        assert task.cancelled()\n+        assert task.cancelled()\n",
    "patch_ground_truth": "--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -429,10 +429,10 @@\n         await put_a\n         assert not put_b.done()\n \n-        put_b.cancel()\n+        put_c.cancel()\n \n         with pytest.raises(asyncio.CancelledError):\n-            await put_b\n+            await put_c\n \n         async def go():\n             a = await q.get()\n@@ -581,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_put_cancelled_race ___________________________________________________________________________\n\nself = <test_async.TestQueuePut object at 0x7acab0cbf580>\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_b.cancel()\n    \n        with pytest.raises(asyncio.CancelledError):\n            await put_b\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\ntests/test_async.py:444: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\n    async def go():\n        a = await q.get()\n        assert a == \"a\"\n        b = await q.get()\n>       assert b == \"b\"\nE       AssertionError: assert 'X' == 'b'\nE         \nE         - b\nE         + X\n\ntests/test_async.py:441: AssertionError\n================================================================================= short test summary info ==================================================================================\nFAILED tests/test_async.py::TestQueuePut::test_put_cancelled_race - AssertionError: assert 'X' == 'b'\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": 7,
      "Explanation": 7,
      "Overall": 6
    },
    "issue_message": {
      "Title": 6,
      "Description": 5,
      "Reproducibility": 5,
      "Relevance": 7,
      "Explanation": 6,
      "Overall": 6
    },
    "issue_ground": {
      "Title": 8,
      "Description": 8,
      "Reproducibility": 8,
      "Relevance": 7,
      "Explanation": 7,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Incorrect Test Case References and Class Inheritance in Async Testing",
      "description": "There are issues in the asynchronous tests which include incorrect task references and class inheritance problems. Specifically:\n\n1. In the test case where multiple put tasks are created (test_put_cancelled_race), the cancellation and assertion checks are being applied to `put_b` instead of `put_c`. Initially, `put_b` is created as a put task, but the subsequent cancellation and `await` operation incorrectly reference `put_c`, which causes failures and incorrect test behavior.\n\n2. The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes do not inherit from a mixin that provides essential queue testing functionality. Instead of inheriting from `TestQueueJoin`, these classes should inherit from `_QueueJoinTestMixin` to properly leverage the mixin's functionality and ensure comprehensive testing of LIFO and priority queue join operations.\n\n3. In the mixed mode tests, when waiting for an empty queue, there is an issue where the task is not cancelled after joining (wait_for_empty_queue). This could potentially result in the task hanging indefinitely, thereby affecting the reliability and robustness of the test suite.\n\nAddressing these issues will ensure that the tests accurately reflect the intended scenarios and provide reliable coverage for the queue functionalities.",
      "explanation": "### Summary of the Issue\n\nThe issue involves several problems with the asynchronous tests in a Python project, specifically in the `aio-libs_janus` repository. The key points are:\n\n1. **Incorrect task reference in tests:** In some test cases, cancellation and assertion checks are applied to the wrong tasks, leading to test failures and incorrect behaviors.\n2. **Incorrect class inheritance:** Some test classes (`TestLifoQueueJoin` and `TestPriorityQueueJoin`) do not inherit from the correct mixin class that provides essential queue functionalities.\n3. **Hanging tasks in the test suite:** There is an issue in mixed mode tests where a task might not be canceled properly after joining, potentially causing the test suite to hang indefinitely.\n\n### Commit Changes\n\nThe commit addresses these issues as follows:\n\n1. **Fixed Incorrect Task References:**\n   - The commit updates the asynchronous test case to ensure that cancellation and assertion checks are applied to the correct task (`put_c` instead of `put_b`). This corrects the logic and ensures that the assertions correctly reflect the intended behavior of the tasks, preventing false test failures.\n\n2. **Corrected Class Inheritance:**\n   - The commit changes the inheritance of `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes to inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This ensures that these test classes leverage the proper functionality provided by the mixin, leading to comprehensive and accurate testing of LIFO and priority queue join operations.\n\n3. **Resolved Hanging Tasks:**\n   - In the mixed mode tests, particularly when waiting for an empty queue, the commit introduces an additional step to cancel the task explicitly after joining. This prevents the task from hanging indefinitely, thereby improving the reliability and robustness of the test suite.\n\n### Cause and Solution Explained\n\n- **Cause of Incorrect Task References:**\n  The issue in the test was caused by referencing `put_b` for cancellation and assertion checks when it should have been `put_c`. This mismatch led to incorrect behaviors in the tests as they were not checking the proper tasks.\n\n  **Solution:**\n  The commit correctly identifies the task that should be referred to (`put_c`) and updates the test accordingly. This small but critical correction ensures that the intended task is canceled and awaited, leading to accurate test outcomes.\n\n- **Cause of Incorrect Class Inheritance:**\n  The test classes `TestLifoQueueJoin` and `TestPriorityQueueJoin` were inheriting from `TestQueueJoin` instead of `_QueueJoinTestMixin`. This incorrect inheritance caused the tests to miss essential functionalities provided by the mixin, consequently affecting the comprehensiveness and accuracy of queue join tests.\n\n  **Solution:**\n  The fix involves modifying the inheritance so that these classes inherit from `_QueueJoinTestMixin`. This ensures that they include the necessary queue operations and validations, leading to thorough testing.\n\n- **Cause of Hanging Tasks:**\n  In the mixed mode tests, there wasn't a proper mechanism to cancel tasks after the join operation, leading to scenarios where tasks could hang indefinitely. Such issues affect the robustness and reliability of the test suite.\n\n  **Solution:**\n  The commit adds a step to cancel the task explicitly after the join operation. This ensures that all tasks are properly concluded, preventing any potential indefinite hanging and enhancing the test suite's robustness.\n\n### Conclusion\n\nThe commit effectively addresses the issues by:\n1. Correcting task references in the test cases to ensure proper cancellation and assertions.\n2. Adjusting class inheritance to include the correct mixin, thereby enabling comprehensive queue join tests.\n3. Explicitly canceling tasks in mixed mode tests to prevent indefinite hanging.\n\nThese changes collectively ensure that the asynchronous tests behave as intended, reliable, and cover the necessary functionalities thoroughly. This improvement enhances the overall test accuracy and robustness, contributing to the project's reliability and maintainability."
    }
  }
}