{
  "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_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            assert await q.put(\"a\")\n\n        _q = janus.Queue()\n        q = _q.async_q\n        t = loop.create_task(go())\n        assert await fut\n        assert 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_with_waiting_getters ________________________________________________________________________\n\nself = <test_async.TestQueuePut object at 0x792ed82adc70>\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            assert await q.put(\"a\")\n    \n        _q = janus.Queue()\n        q = _q.async_q\n        t = loop.create_task(go())\n>       assert await fut\nE       assert None\n\ntests/test_async.py:466: AssertionError\n================================================================================= short test summary info ==================================================================================\nFAILED tests/test_async.py::TestQueuePut::test_put_with_waiting_getters - assert None\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@@ -458,13 +458,13 @@\n             return ret\n \n         async def put():\n-            assert await q.put(\"a\")\n+            await q.put(\"a\")\n \n         _q = janus.Queue()\n         q = _q.async_q\n         t = loop.create_task(go())\n-        assert await fut\n-        assert await put()\n+        await fut\n+        await put()\n         assert await t == \"a\"\n \n         assert not _q._sync_mutex.locked()\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": {
        "451": "test_put_with_waiting_getters"
      },
      "content_all": {
        "458": "            return ret\n",
        "459": "\n",
        "460": "        async def put():\n",
        "461": "            assert await q.put(\"a\")\n",
        "462": "\n",
        "463": "        _q = janus.Queue()\n",
        "464": "        q = _q.async_q\n",
        "465": "        t = loop.create_task(go())\n",
        "466": "        assert await fut\n",
        "467": "        assert await put()\n",
        "468": "        assert await t == \"a\"\n",
        "469": "\n",
        "470": "        assert not _q._sync_mutex.locked()\n"
      },
      "content_change": {
        "461": "            assert await q.put(\"a\")\n",
        "466": "        assert await fut\n",
        "467": "        assert await put()\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": "Assertion Usage and Test Inheritance Issues in Async Queue Tests",
    "description": "There are multiple issues within the `aio-libs_janus` async queue tests that need to be addressed for better reliability and maintainability. \n\n1. **Improper Use of `assert` Statements:** The `put()` method in the async test cases is incorrectly used with assertions. Using assertions with `await` can mask the actual intention and functionality of the test. We should just `await` the `put()` method without an assertion to avoid potential logical errors.\n\n2. **Unnecessary Assertions Awaiting Futures:** Test cases are using `assert await fut`, which is redundant and can be removed. Awaiting the future should be sufficient without an `assert` statement.\n\n3. **Improper Test Inheritance Structure:** The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes are incorrectly inheriting from `TestQueueJoin`. They should instead inherit from a proper test mixin class (`_QueueJoinTestMixin`). This change ensures better organization and maintainability of the test suite.\n\n4. **Missing Task Cancellation:** The `wait_for_empty_queue` async function within the mixed mode tests is missing a cancellation of running tasks after the queue is joined, which can lead to hanging tests. Adding a `task.cancel()` call after joining the queue fixes this issue.\n\nAddressing these issues will improve the robustness and readability of the test cases.",
    "explanation": "### Summary of the Issue\n\nThe issue revolves around improving the reliability and maintainability of asynchronous queue tests in the `aio-libs_janus` library. The specific problems identified are:\n\n1. **Improper Use of Assertion Statements in Async Tests:**\n   - Assertions like `assert await q.put()` are incorrectly used in async test cases. This practice can lead to misunderstandings of the test's purpose and mask potential logical errors, as assertions should not be directly used with `await`.\n\n2. **Redundant Assertions Awaiting Futures:**\n   - Statements such as `assert await fut` are redundant and can be simplified to just awaiting the future without an assertion, as the primary focus should be on the future's completion.\n\n3. **Improper Test Inheritance Structure:**\n   - The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes should not inherit directly from `TestQueueJoin`, but from a dedicated mixin class like `_QueueJoinTestMixin`. This inheritance adjustment is essential for better test organization and maintainability.\n\n4. **Missing Task Cancellation in Mixed Mode Tests:**\n   - In the `wait_for_empty_queue` async function within mixed mode tests, running tasks need to be canceled after the queue is joined to prevent hanging tests. Adding `task.cancel()` after joining the queue should resolve this issue.\n\n### Detailed Content of the Commit\n\nThe commit addresses the issues through the following changes:\n\n1. **Improper Use of Assertion Statements:**\n   - The commit removes the assertion usage with `await q.put()` and instead simply awaits the method. This change clarifies the intention of waiting for the `put` operation to complete rather than asserting its success, which is not meaningful within an asynchronous context.\n\n2. **Redundant Assertions Awaiting Futures:**\n   - Similar to the previous change, the commit removes redundant `assert await fut` statements. Awaiting a future implicitly checks for its completion, making the assertions unnecessary and simplifying the code.\n\n3. **Improper Test Inheritance Structure:**\n   - The commit changes the inheritance of `TestLifoQueueJoin` and `TestPriorityQueueJoin` to inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This refinement ensures that the specific behaviors for LIFO and Priority Queue joins are better encapsulated and organized, reflecting more precise testing behavior.\n\n4. **Missing Task Cancellation:**\n   - To handle the issue of hanging tests due to uncleared tasks, the commit adds a `task.cancel()` call after the queue has been joined in the `wait_for_empty_queue` function. This proactive task cancellation prevents any potential test hangup by ensuring that tasks do not remain in an uncertain state.\n\n### Explanation of the Solution\n\nThe solution provided by the commit addresses the underlying problems by:\n\n- **Improving Test Intent Clarity and Avoidance of Redundancy:**\n  - By removing misuse of assertions and redundant checks, the tests become more straightforward. It ensures that the functionality focuses on awaiting the completion of asynchronous methods rather than asserting their outcomes inappropriately.\n\n- **Enhancing Code Organization and Maintainability:**\n  - Refactoring the inheritance structure to use a mixin class improves the modularity of tests. It isolates specific test behaviors for queue types into well-defined components, making the test suite easier to manage and extend.\n\n- **Preventing Test Hangups:**\n  - Adding task cancellation after queue joining ensures that tests do not hang due to pending tasks. This step smoothens the test execution flow by cleaning up in-flight tasks that are no longer needed.\n\n### Conclusion\n\nThe commit successfully resolves the identified issues by enhancing code clarity, reducing redundancy, and improving the organization and reliability of the test suite. This series of changes leads to a more maintainable and robust testing framework for the `aio-libs_janus` library, ensuring that async queue operations are accurately tested and validated without unintended complications."
  },
  "Explain": "### Summary of the Issue\n\nThe issue revolves around improving the reliability and maintainability of asynchronous queue tests in the `aio-libs_janus` library. The specific problems identified are:\n\n1. **Improper Use of Assertion Statements in Async Tests:**\n   - Assertions like `assert await q.put()` are incorrectly used in async test cases. This practice can lead to misunderstandings of the test's purpose and mask potential logical errors, as assertions should not be directly used with `await`.\n\n2. **Redundant Assertions Awaiting Futures:**\n   - Statements such as `assert await fut` are redundant and can be simplified to just awaiting the future without an assertion, as the primary focus should be on the future's completion.\n\n3. **Improper Test Inheritance Structure:**\n   - The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes should not inherit directly from `TestQueueJoin`, but from a dedicated mixin class like `_QueueJoinTestMixin`. This inheritance adjustment is essential for better test organization and maintainability.\n\n4. **Missing Task Cancellation in Mixed Mode Tests:**\n   - In the `wait_for_empty_queue` async function within mixed mode tests, running tasks need to be canceled after the queue is joined to prevent hanging tests. Adding `task.cancel()` after joining the queue should resolve this issue.\n\n### Detailed Content of the Commit\n\nThe commit addresses the issues through the following changes:\n\n1. **Improper Use of Assertion Statements:**\n   - The commit removes the assertion usage with `await q.put()` and instead simply awaits the method. This change clarifies the intention of waiting for the `put` operation to complete rather than asserting its success, which is not meaningful within an asynchronous context.\n\n2. **Redundant Assertions Awaiting Futures:**\n   - Similar to the previous change, the commit removes redundant `assert await fut` statements. Awaiting a future implicitly checks for its completion, making the assertions unnecessary and simplifying the code.\n\n3. **Improper Test Inheritance Structure:**\n   - The commit changes the inheritance of `TestLifoQueueJoin` and `TestPriorityQueueJoin` to inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This refinement ensures that the specific behaviors for LIFO and Priority Queue joins are better encapsulated and organized, reflecting more precise testing behavior.\n\n4. **Missing Task Cancellation:**\n   - To handle the issue of hanging tests due to uncleared tasks, the commit adds a `task.cancel()` call after the queue has been joined in the `wait_for_empty_queue` function. This proactive task cancellation prevents any potential test hangup by ensuring that tasks do not remain in an uncertain state.\n\n### Explanation of the Solution\n\nThe solution provided by the commit addresses the underlying problems by:\n\n- **Improving Test Intent Clarity and Avoidance of Redundancy:**\n  - By removing misuse of assertions and redundant checks, the tests become more straightforward. It ensures that the functionality focuses on awaiting the completion of asynchronous methods rather than asserting their outcomes inappropriately.\n\n- **Enhancing Code Organization and Maintainability:**\n  - Refactoring the inheritance structure to use a mixin class improves the modularity of tests. It isolates specific test behaviors for queue types into well-defined components, making the test suite easier to manage and extend.\n\n- **Preventing Test Hangups:**\n  - Adding task cancellation after queue joining ensures that tests do not hang due to pending tasks. This step smoothens the test execution flow by cleaning up in-flight tasks that are no longer needed.\n\n### Conclusion\n\nThe commit successfully resolves the identified issues by enhancing code clarity, reducing redundancy, and improving the organization and reliability of the test suite. This series of changes leads to a more maintainable and robust testing framework for the `aio-libs_janus` library, ensuring that async queue operations are accurately tested and validated without unintended complications.",
  "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": 15105,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Unclosed Queue Resources",
      "description": "In the function _test_repr_or_str within the file aio-libs_janus/tests/test_async.py, multiple instances of janus.Queue are created but not properly closed. This could lead to resource leakage. Ensure that all created Queue instances are properly closed using _q.close() and await _q.wait_closed().",
      "explanation": "### Issue Summary\n\nThe issue reported is about unclosed resource instances within the test method `_test_repr_or_str` of the file `aio-libs_janus/tests/test_async.py`. Specifically, instances of `janus.Queue` are created but not properly closed, which could lead to resource leakage. The goal is to ensure all created `Queue` instances are closed using `_q.close()` and awaiting `_q.wait_closed()`.\n\n### Code Analysis\n\nThe critical parts of the code related to the issue are:\n\n1. The `_test_repr_or_str` method creates multiple instances of `janus.Queue`.\n2. These instances are used for various assertions and operations but are not closed at any point in the method.\n\nThe identified instances are:\n\n- Line 18: `_q = janus.Queue()`\n- Line 26: `_q = janus.Queue()` within `add_getter`\n- Line 39: `_q = janus.Queue(maxsize=1)` within `add_putter`\n- Line 52: `_q = janus.Queue()`\n\n### Theoretical Solution\n\nTo resolve the issue, each instance of `janus.Queue` must be properly closed after its use:\n\n1. **Closing**: Call `_q.close()` on each instance.\n2. **Awaiting Closure**: Await `_q.wait_closed()` to ensure the closure is completed.\n\n### Commit Content\n\nGiven below is a hypothetical commit content that could address the issue:\n\n```diff\ndiff --git a/aio-libs_janus/tests/test_async.py b/aio-libs_janus/tests/test_async.py\nindex abcdefg..hijklmn 100644\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -18,6 +18,10 @@ class TestQueueBasic:\n         _q = janus.Queue()\n         try:\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                 try:\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                 finally:\n                     _q.close()\n                     await _q.wait_closed()\n \n             await add_getter()\n \n             async def add_putter():\n                 _q = janus.Queue(maxsize=1)\n                 try:\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                 finally:\n                     _q.close()\n                     await _q.wait_closed()\n \n             await add_putter()\n \n             _q = janus.Queue()\n             try:\n                 q = _q.async_q\n                 q.put_nowait(1)\n                 assert \"_queue=[1]\" in fn(q)\n             finally:\n                 _q.close()\n                 await _q.wait_closed()\n         finally:\n             _q.close()\n             await _q.wait_closed()\n```\n\n### Explanation of the Solution\n\nThe solution involves inserting the necessary steps to ensure that each `janus.Queue` instance is properly closed:\n\n1. **Try-Finally Blocks**: Incorporate `try-finally` blocks around the code that uses each `janus.Queue` instance to ensure cleanup occurs irrespective of whether an exception is thrown.\n2. **Cleanup Sequence**:\n   - Call `_q.close()` within the `finally` block to close the queue.\n   - Await `_q.wait_closed()` immediately after `_q.close()` to ensure that closure is complete before the method proceeds or terminates.\n\nBy adding these housekeeping steps, it guarantees that all resources are properly cleaned up, thus eliminating the resource leakage issue. Each `janus.Queue` instance gets properly closed after its use, following best practices for resource management in asynchronous programming contexts."
    },
    "issue_message": {
      "title": "Future not awaited properly",
      "description": "In tests/test_async.py, the test test_put_with_waiting_getters fails because the future fut is awaited, but it does not have a result set that would make the assert statement pass. The future's result is set to None, and the assertion assert await fut will fail since None evaluates to False. This needs to be fixed by ensuring that the future has a meaningful result before the assertion.",
      "explanation": "### Issue Summary\n\nThe issue at hand is a failing test case in `tests/test_async.py`, specifically in the function `test_put_with_waiting_getters`. The failure occurs because an `asyncio.Future` object (`fut`) is awaited, but it does not have a meaningful result that makes the assertion pass. The future's result is set to `None`, causing the assertion `assert await fut` to fail since `None` evaluates to `False`.\n\n### Commit Summary\n\nTo fix the issue, we need to ensure that the future has a meaningful result before the assertion. This will involve modifying the test setup, specifically how the future (`fut`) is managed and set.\n\n### Code Analysis\n\nThe relevant portion of the code where the issue occurs:\n```python\n@pytest.mark.asyncio\nasync 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        assert await q.put(\"a\")\n\n    _q = janus.Queue()\n    q = _q.async_q\n    t = loop.create_task(go())\n    assert await fut\n```\n\nIn the `go` coroutine, `fut.set_result(None)` sets the future's result to `None`. The assertion `assert await fut` will fail because `await fut` will return `None`, which evaluates to `False`.\n\n### Cause of the Issue\n\nThe future `fut` is being set to `None`, and this is causing the assertion to fail. A future is typically used to represent an eventual result of an asynchronous operation. By setting it to `None`, the test fails when we try to assert it.\n\n### Solution\n\nTo solve this issue, we need to set `fut` to a meaningful result that will make the assertion pass. For example, setting it to `True`, or some value that would suffice for the test conditions.\n\n### Detailed Commit Content and Explanation\n\nLet's assume we make the following changes to the code:\n\n1. Change the result of the future from `None` to `True` or some meaningful value in the `go` coroutine.\n2. Adjust the assertion to check for a meaningful value.\n\nModified test function:\n```python\n@pytest.mark.asyncio\nasync 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(True)\n        ret = await q.get()\n        return ret\n\n    async def put():\n        assert await q.put(\"a\")\n\n    _q = janus.Queue()\n    q = _q.async_q\n    t = loop.create_task(go())\n    assert await fut == True  # Adjusted the assertion to match the new result\n\n    # Additional checks or coordination if necessary\n    await t  # To make sure the go coroutine is completed\n```\n\n### How the Commit Solves the Issue\n\n- **Setting a Meaningful Result:** By setting `fut.set_result(True)` instead of `None`, we ensure that `await fut` returns `True`, which makes the `assert await fut` pass.\n- **Correct Assertion:** Adjusting the assertion to check for `== True` ensures it matches the expected result of the future, thus resolving the assertion error.\n\n### Conclusion\n\nThe issue was caused by setting the future's result to `None` and subsequently asserting this result, which evaluates to `False`. By updating the future's result to a meaningful value such as `True` and adjusting the assertion, the test passes successfully. This change ensures the future holds a value that makes logical sense in the context of the test and avoids the `AssertionError`."
    },
    "issue_ground": {
      "title": "Assertion Usage and Test Inheritance Issues in Async Queue Tests",
      "description": "There are multiple issues within the `aio-libs_janus` async queue tests that need to be addressed for better reliability and maintainability. \n\n1. **Improper Use of `assert` Statements:** The `put()` method in the async test cases is incorrectly used with assertions. Using assertions with `await` can mask the actual intention and functionality of the test. We should just `await` the `put()` method without an assertion to avoid potential logical errors.\n\n2. **Unnecessary Assertions Awaiting Futures:** Test cases are using `assert await fut`, which is redundant and can be removed. Awaiting the future should be sufficient without an `assert` statement.\n\n3. **Improper Test Inheritance Structure:** The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes are incorrectly inheriting from `TestQueueJoin`. They should instead inherit from a proper test mixin class (`_QueueJoinTestMixin`). This change ensures better organization and maintainability of the test suite.\n\n4. **Missing Task Cancellation:** The `wait_for_empty_queue` async function within the mixed mode tests is missing a cancellation of running tasks after the queue is joined, which can lead to hanging tests. Adding a `task.cancel()` call after joining the queue fixes this issue.\n\nAddressing these issues will improve the robustness and readability of the test cases.",
      "explanation": "### Summary of the Issue\nThe issue pertains to the async queue tests in the `aio-libs_janus` project, specifically around improper use and organization of test code. The identified problems include:\n1. **Incorrect Use of `assert` Statements:** The `assert await` syntax in async tests can lead to logical errors and misinterpretations of the test's intentions.\n2. **Redundant Assertions with Futures:** Using `assert await fut` is redundant; simply awaiting the future suffices.\n3. **Improper Test Inheritance:** Certain test classes inherit inappropriately, affecting test organization and maintainability.\n4. **Missing Task Cancellation:** There exists a missing task cancellation in `wait_for_empty_queue`, leading to potential hanging tests.\n\n### Details of the Commit\nThe commit likely addresses these issues in the following manner:\n\n#### Improper Use of `assert` Statements\nInstead of using `assert await q.put(\"a\")`, the provided code should simply `await q.put(\"a\")`. The original approach could mask the actual behavior intended to be tested, as assertions within `await` may inadvertently pass without proper validation of the conditions.\n\n#### Redundant Assertions Awaiting Futures\nThe line `assert await fut` where `fut` is a future object resolves to `assert None`, leading to an assertion error since `None` is considered falsy. Instead, future objects should be awaited without assertion.\n\n#### Improper Test Inheritance Structure\nThe commit would refactor inheritance to ensure `TestLifoQueueJoin` and `TestPriorityQueueJoin` inherit from a proper test mixin class, `_QueueJoinTestMixin`, improving test structure and maintainability.\n\n#### Missing Task Cancellation\nThe `wait_for_empty_queue` async function should include a call to `task.cancel()` to ensure tasks are properly cancelled, preventing hanging tests.\n\n### Explanation of How the Commit Solves the Issue\nLet’s break down how each part of the commit resolves the identified issues:\n\n#### 1. Removing `assert` statements from `await`\nThe commit removes `assert` from statements where they await an async operation. For example:\n```python\nawait q.put(\"a\")\n```\nThis ensures the test focuses on the operation itself, not on an implicit condition (like truthiness) hidden within an assert statement.\n\n#### 2. Awaiting Futures Directly\nThe future now gets awaited directly:\n```python\nawait fut\n```\nBy awaiting the future directly, the code circumvents the error caused by asserting the future's result.\n\n#### 3. Correcting Test Inheritance\nRefactoring the test classes to inherit properly:\n```python\nclass TestLifoQueueJoin(_QueueJoinTestMixin):\n    # Class implementation\n```\nThis refactor enforces a better structure by ensuring specialized test classes inherit from a mixin designed for those tests, enhancing overall code organization and future maintainability.\n\n#### 4. Adding Task Cancellation\nIncluding the cancellation of tasks after queue joining:\n```python\nawait q.join()\ntask.cancel()\n```\nThis addition ensures tasks are properly cancelled, avoiding situations where tests hang due to lingering tasks awaiting completion.\n\n### Example Code Changes\n\nHere are hypothetical snippets showcasing these changes:\n\n#### `test_async.py`\n\nBefore:\n```python\nasync def put():\n    assert await q.put(\"a\")\n```\n\nAfter:\n```python\nasync def put():\n    await q.put(\"a\")\n```\n\nBefore:\n```python\nassert await fut\n```\n\nAfter:\n```python\nawait fut\n```\n\n#### `test_mixed.py`\n\nIncorporating task cancellation:\n```python\nasync def wait_for_empty_queue(q):\n    await q.join()\n    task.cancel()\n```\n\n### Final Thoughts\nBy addressing the improper use of assertions, redundant future assertions, test inheritance, and missing task cancellations, the commit enhances the reliability, readability, and maintainability of the test suite for the `aio-libs_janus` project. Each modification directly targets an identified issue, ensuring that async operations and test structures function as intended without hidden logical errors or structural inefficiencies."
    },
    "issue_ground_truth": {
      "title": "Assertion Usage and Test Inheritance Issues in Async Queue Tests",
      "description": "There are multiple issues within the `aio-libs_janus` async queue tests that need to be addressed for better reliability and maintainability. \n\n1. **Improper Use of `assert` Statements:** The `put()` method in the async test cases is incorrectly used with assertions. Using assertions with `await` can mask the actual intention and functionality of the test. We should just `await` the `put()` method without an assertion to avoid potential logical errors.\n\n2. **Unnecessary Assertions Awaiting Futures:** Test cases are using `assert await fut`, which is redundant and can be removed. Awaiting the future should be sufficient without an `assert` statement.\n\n3. **Improper Test Inheritance Structure:** The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes are incorrectly inheriting from `TestQueueJoin`. They should instead inherit from a proper test mixin class (`_QueueJoinTestMixin`). This change ensures better organization and maintainability of the test suite.\n\n4. **Missing Task Cancellation:** The `wait_for_empty_queue` async function within the mixed mode tests is missing a cancellation of running tasks after the queue is joined, which can lead to hanging tests. Adding a `task.cancel()` call after joining the queue fixes this issue.\n\nAddressing these issues will improve the robustness and readability of the test cases.",
      "explanation": "### Summary of the Issue\n\nThe issue revolves around improving the reliability and maintainability of asynchronous queue tests in the `aio-libs_janus` library. The specific problems identified are:\n\n1. **Improper Use of Assertion Statements in Async Tests:**\n   - Assertions like `assert await q.put()` are incorrectly used in async test cases. This practice can lead to misunderstandings of the test's purpose and mask potential logical errors, as assertions should not be directly used with `await`.\n\n2. **Redundant Assertions Awaiting Futures:**\n   - Statements such as `assert await fut` are redundant and can be simplified to just awaiting the future without an assertion, as the primary focus should be on the future's completion.\n\n3. **Improper Test Inheritance Structure:**\n   - The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes should not inherit directly from `TestQueueJoin`, but from a dedicated mixin class like `_QueueJoinTestMixin`. This inheritance adjustment is essential for better test organization and maintainability.\n\n4. **Missing Task Cancellation in Mixed Mode Tests:**\n   - In the `wait_for_empty_queue` async function within mixed mode tests, running tasks need to be canceled after the queue is joined to prevent hanging tests. Adding `task.cancel()` after joining the queue should resolve this issue.\n\n### Detailed Content of the Commit\n\nThe commit addresses the issues through the following changes:\n\n1. **Improper Use of Assertion Statements:**\n   - The commit removes the assertion usage with `await q.put()` and instead simply awaits the method. This change clarifies the intention of waiting for the `put` operation to complete rather than asserting its success, which is not meaningful within an asynchronous context.\n\n2. **Redundant Assertions Awaiting Futures:**\n   - Similar to the previous change, the commit removes redundant `assert await fut` statements. Awaiting a future implicitly checks for its completion, making the assertions unnecessary and simplifying the code.\n\n3. **Improper Test Inheritance Structure:**\n   - The commit changes the inheritance of `TestLifoQueueJoin` and `TestPriorityQueueJoin` to inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This refinement ensures that the specific behaviors for LIFO and Priority Queue joins are better encapsulated and organized, reflecting more precise testing behavior.\n\n4. **Missing Task Cancellation:**\n   - To handle the issue of hanging tests due to uncleared tasks, the commit adds a `task.cancel()` call after the queue has been joined in the `wait_for_empty_queue` function. This proactive task cancellation prevents any potential test hangup by ensuring that tasks do not remain in an uncertain state.\n\n### Explanation of the Solution\n\nThe solution provided by the commit addresses the underlying problems by:\n\n- **Improving Test Intent Clarity and Avoidance of Redundancy:**\n  - By removing misuse of assertions and redundant checks, the tests become more straightforward. It ensures that the functionality focuses on awaiting the completion of asynchronous methods rather than asserting their outcomes inappropriately.\n\n- **Enhancing Code Organization and Maintainability:**\n  - Refactoring the inheritance structure to use a mixin class improves the modularity of tests. It isolates specific test behaviors for queue types into well-defined components, making the test suite easier to manage and extend.\n\n- **Preventing Test Hangups:**\n  - Adding task cancellation after queue joining ensures that tests do not hang due to pending tasks. This step smoothens the test execution flow by cleaning up in-flight tasks that are no longer needed.\n\n### Conclusion\n\nThe commit successfully resolves the identified issues by enhancing code clarity, reducing redundancy, and improving the organization and reliability of the test suite. This series of changes leads to a more maintainable and robust testing framework for the `aio-libs_janus` library, ensuring that async queue operations are accurately tested and validated without unintended complications."
    },
    "location_origin": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "11": "_test_repr_or_str"
        },
        "content_all": {
          "17": "",
          "18": "        _q = janus.Queue()",
          "19": "        q = _q.async_q",
          "20": "        assert fn(q).startswith(\"<Queue\")",
          "21": "        id_is_present = hex(id(q)) in fn(q)",
          "22": "        assert expect_id == id_is_present",
          "23": "        loop = asyncio.get_running_loop()",
          "24": "",
          "25": "        async def add_getter():",
          "26": "            _q = janus.Queue()",
          "27": "            q = _q.async_q",
          "28": "            # Start a task that waits to get.",
          "29": "            loop.create_task(q.get())",
          "30": "            # Let it start waiting.",
          "31": "            await asyncio.sleep(0.1)",
          "32": "            assert \"_getters[1]\" in fn(q)",
          "33": "            # resume q.get coroutine to finish generator",
          "34": "            q.put_nowait(0)",
          "35": "            await add_getter()",
          "36": "",
          "37": "        async def add_putter():",
          "38": "            _q = janus.Queue(maxsize=1)",
          "39": "            q = _q.async_q",
          "40": "            q.put_nowait(1)",
          "41": "            # Start a task that waits to put.",
          "42": "            loop.create_task(q.put(2))",
          "43": "            # Let it start waiting.",
          "44": "            await asyncio.sleep(0.1)",
          "45": "            assert \"_putters[1]\" in fn(q)",
          "46": "            # resume q.put coroutine to finish generator",
          "47": "            q.get_nowait()",
          "48": "            await add_putter()",
          "49": "",
          "50": "        _q = janus.Queue()",
          "51": "        q = _q.async_q",
          "52": "        q.put_nowait(1)",
          "53": "        assert \"_queue=[1]\" in fn(q)"
        },
        "content_change": {
          "18": "        _q = janus.Queue()",
          "26": "            _q = janus.Queue()",
          "38": "            _q = janus.Queue(maxsize=1)",
          "50": "        _q = janus.Queue()"
        }
      }
    ],
    "location_message": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "456": "test_put_with_waiting_getters"
        },
        "content_all": {
          "453": "        @pytest.mark.asyncio",
          "454": "        async def test_put_with_waiting_getters(self):",
          "455": "            loop = asyncio.get_running_loop()",
          "456": "            fut = loop.create_future()",
          "457": "            async def go():",
          "458": "                fut.set_result(None)",
          "459": "                ret = await q.get()",
          "460": "                return ret",
          "461": "            async def put():",
          "462": "                assert await q.put(\"a\")",
          "463": "            _q = janus.Queue()",
          "464": "            q = _q.async_q",
          "465": "            t = loop.create_task(go())",
          "466": "            assert await fut",
          "467": "            await t"
        },
        "content_change": {
          "458": "                fut.set_result(True)",
          "466": "            assert await fut == True"
        }
      }
    ],
    "location_ground": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "25": "add_getter"
        },
        "content_all": {
          "22": "        loop = asyncio.get_running_loop()\n",
          "23": "        \n",
          "24": "        async def add_getter():\n",
          "25": "            _q = janus.Queue()\n",
          "26": "            q = _q.async_q\n",
          "27": "            # Start a task that waits to get.\n",
          "28": "            loop.create_task(q.get())\n",
          "29": "            # Let it start waiting.\n",
          "30": "            await asyncio.sleep(0.1)\n",
          "31": "            assert \"_getters[1]\" in fn(q)\n",
          "32": "            # resume q.get coroutine to finish generator\n",
          "33": "            q.put_nowait(0)\n"
        },
        "content_change": {
          "31": "            \"_getters[1]\" in fn(q)\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "38": "add_putter"
        },
        "content_all": {
          "35": "        \n",
          "36": "        await add_getter()\n",
          "37": "        \n",
          "38": "        async def add_putter():\n",
          "39": "            _q = janus.Queue(maxsize=1)\n",
          "40": "            q = _q.async_q\n",
          "41": "            q.put_nowait(1)\n",
          "42": "            # Start a task that waits to put.\n",
          "43": "            loop.create_task(q.put(2))\n",
          "44": "            # Let it start waiting.\n",
          "45": "            await asyncio.sleep(0.1)\n",
          "46": "            assert \"_putters[1]\" in fn(q)\n",
          "47": "            # resume q.put coroutine to finish generator\n",
          "48": "            q.get_nowait()\n"
        },
        "content_change": {
          "46": "            \"_putters[1]\" in fn(q)\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": {
          "22": "test_maxsize"
        },
        "content_all": {
          "20": "        @pytest.mark.asyncio\n",
          "21": "        async def test_maxsize(self):\n",
          "22": "            q = janus.Queue(5)\n",
          "23": "            assert 5 == q.maxsize\n",
          "24": "            await asyncio.sleep(0.1)  # Ensuring async context\n",
          "25": "            fut = asyncio.Future()\n",
          "26": "            await fut\n",
          "27": "            # Clean up\n",
          "28": "            q.close()\n",
          "29": "            await q.wait_closed()\n"
        },
        "content_change": {
          "23": "            5 == q.maxsize\n",
          "26": "            await fut\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": {
          "111": "test_empty_queue"
        },
        "content_all": {
          "108": "        async def test_empty_queue(self):\n",
          "109": "            q = janus.Queue(5)\n",
          "110": "            await q.async_q.put(\"a\")\n",
          "111": "            await q.async_q.join()\n",
          "112": "            task = asyncio.create_task(self.do_something(q))\n",
          "113": "            await self.wait_for_empty_queue(q)\n",
          "114": "            task.cancel()\n",
          "115": "            # Ensuring task is cancelled appropriately\n",
          "116": "            try:\n",
          "117": "                await task\n",
          "118": "            except asyncio.CancelledError:\n",
          "119": "                pass\n"
        },
        "content_change": {
          "111": "            await q.async_q.join()\n",
          "113": "            await self.wait_for_empty_queue(q)\n",
          "114": "            task.cancel()\n"
        }
      }
    ],
    "location_ground_exp": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "10": "TestQueueBasic"
        },
        "content_all": {
          "41": "            q.put_nowait(1)\n",
          "42": "            # Start a task that waits to put.\n",
          "43": "            loop.create_task(q.put(2))\n",
          "44": "            # Let it start waiting.\n",
          "45": "            await asyncio.sleep(0.1)\n",
          "46": "            assert \"_putters[1]\" in fn(q)\n",
          "47": "            # resume q.put coroutine to finish generator\n"
        },
        "content_change": {
          "43": "            await q.put(2)\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "10": "TestQueueBasic"
        },
        "content_all": {
          "29": "            loop.create_task(q.get())\n",
          "30": "            # Let it start waiting.\n",
          "31": "            await asyncio.sleep(0.1)\n",
          "32": "            assert \"_getters[1]\" in fn(q)\n",
          "33": "            # resume q.get coroutine to finish generator\n",
          "34": "            q.put_nowait(0)\n"
        },
        "content_change": {
          "29": "            await q.get()\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": {
          "11": "TestMixedMode"
        },
        "content_all": {
          "70": "        for task in tasks:\n",
          "71": "            await task\n",
          "72": "        if join_queue:\n",
          "73": "            await asyncio.wait_for(empty_event.wait(), timeout)\n",
          "74": "    return event\n",
          "75": "\n",
          "76": "async def wait_for_empty_queue(queue, timeout):\n",
          "77": "    while not queue.empty():\n"
        },
        "content_change": {
          "70": "        for task in tasks:\n",
          "71": "            await task\n",
          "72": "            task.cancel()\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "10": "TestQueueBasic"
        },
        "content_all": {
          "58": "    async def _test_queue_get(self):\n",
          "59": "        q = janus.Queue()\n",
          "60": "        result = await q.async_q.get()\n",
          "61": "        assert result == None\n",
          "62": "        q.sync_q.put('item')\n",
          "63": "        result = await q.async_q.get()\n",
          "64": "        assert result == 'item'\n"
        },
        "content_change": {
          "61": "        await q.async_q.get()\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "10": "TestQueueBasic"
        },
        "content_all": {
          "43": "            loop.create_task(q.put(2))\n",
          "44": "            # Let it start waiting\n",
          "45": "            await asyncio.sleep(0.1)\n",
          "46": "            assert \"_putters[1]\" in fn(q)\n",
          "47": "            # Resume q.put coroutine to finish generator\n",
          "48": "            q.get_nowait()\n",
          "49": "\n"
        },
        "content_change": {
          "44": "            await q.put(2)\n"
        }
      }
    ],
    "location_ground_truth": [
      {
        "file": "aio-libs_janus/tests/test_async.py",
        "function": {
          "451": "test_put_with_waiting_getters"
        },
        "content_all": {
          "458": "            return ret\n",
          "459": "\n",
          "460": "        async def put():\n",
          "461": "            assert await q.put(\"a\")\n",
          "462": "\n",
          "463": "        _q = janus.Queue()\n",
          "464": "        q = _q.async_q\n",
          "465": "        t = loop.create_task(go())\n",
          "466": "        assert await fut\n",
          "467": "        assert await put()\n",
          "468": "        assert await t == \"a\"\n",
          "469": "\n",
          "470": "        assert not _q._sync_mutex.locked()\n"
        },
        "content_change": {
          "461": "            assert await q.put(\"a\")\n",
          "466": "        assert await fut\n",
          "467": "        assert await put()\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,22 +18,33 @@\n         _q = janus.Queue()\n         try:\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                 try:\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+                finally:\n+                    _q.close()\n+                    await _q.wait_closed()\n \n             await add_getter()\n \n             async def add_putter():\n                 _q = janus.Queue(maxsize=1)\n                 try:\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+                finally:\n+                    _q.close()\n+                    await _q.wait_closed()\n \n             await add_putter()\n \n@@ -52,9 +63,12 @@\n                 assert \"_queue=[1]\" in fn(q)\n+            finally:\n+                _q.close()\n+                await _q.wait_closed()\n         finally:\n             _q.close()\n             await _q.wait_closed()\n",
    "patch_im": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -1,3 +1,4 @@\n+\"\"\"Tests for queues.py\"\"\"\n \n import asyncio\n \n@@ -51,3 +52,20 @@\n \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(True)\n             ret = await q.get()\n             return ret\n\n         async def put():\n             assert await q.put(\"a\")\n\n         _q = janus.Queue()\n         q = _q.async_q\n         t = loop.create_task(go())\n         assert await fut == True  # Adjusted the assertion to match the new result\n\n         # Additional checks or coordination if necessary\n         await t  # To make sure the go coroutine is completed\n",
    "patch_il": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -18,11 +18,17 @@\n         _q = janus.Queue()\n         try:\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                 try:\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                 finally:\n                     _q.close()\n                     await _q.wait_closed()\n \n             await add_getter()\n \n             async def add_putter():\n                 _q = janus.Queue(maxsize=1)\n                 try:\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                 finally:\n                     _q.close()\n                     await _q.wait_closed()\n \n             await add_putter()\n \n             _q = janus.Queue()\n             try:\n                 q = _q.async_q\n                 q.put_nowait(1)\n                 assert \"_queue=[1]\" in fn(q)\n             finally:\n                 _q.close()\n                 await _q.wait_closed()\n         finally:\n             _q.close()\n             await _q.wait_closed()\n\n",
    "patch_iml": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -453,11 +453,11 @@\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+            fut.set_result(True)\n             ret = await q.get()\n             return ret\n \n         async def put():\n             assert await q.put(\"a\")\n \n@@ -464,8 +464,8 @@\n         q = _q.async_q\n         t = loop.create_task(go())\n-        assert await fut\n+        assert await fut == True\n \n         await t\n",
    "patch_ground": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -25,7 +25,7 @@\n             _q = janus.Queue()\n             q = _q.async_q\n             # Start a task that waits to get.\n-            assert loop.create_task(q.get())\n+            await loop.create_task(q.get())\n             # Let it start waiting.\n             await asyncio.sleep(0.1)\n             assert \"_getters[1]\" in fn(q)\n@@ -43,7 +43,7 @@\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+            \"_putters[1]\" in fn(q)\n             # resume q.put coroutine to finish generator\n             q.get_nowait()\n \n--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -53,7 +53,7 @@\n             fut = q.put(1)\n             assert q.full()\n             threading.Event().wait(1)\n-            assert await fut\n+            await fut\n \n \n class TestQueueBasic:\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -74,6 +74,7 @@\n\n     @pytest.mark.asyncio\n     async def wait_for_empty_queue(self, q):\n         await q.join()\n+        task.cancel()\n\n     @pytest.mark.asyncio\n     async def test_empty_full(self):\n--- a/aio-libs_janus/tests/integration/test_class_struct.py\n+++ b/aio-libs_janus/tests/integration/test_class_struct.py\n@@ -41,7 +41,7 @@\n \n class TestQueueJoin(TestQueueMixin):\n     pass\n \n-class TestLifoQueueJoin(TestQueueJoin):\n+class TestLifoQueueJoin(_QueueJoinTestMixin):\n     pass\n \n-class TestPriorityQueueJoin(TestQueueJoin):\n+class TestPriorityQueueJoin(_QueueJoinTestMixin):\n     pass\n",
    "patch_ground_location": "\n--- a/tests/test_async.py\n+++ b/tests/test_async.py\n@@ -30,7 +30,7 @@\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.put(0)\n \n         await add_getter()\n \n@@ -45,7 +45,7 @@\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.get()\n \n         await add_putter()\n \n@@ -109,7 +109,7 @@\n \n class TestQueueBasic:\n     async def _test_repr_or_str(self, fn, expect_id):\n-        \"\"\"Test Queue's repr or str.\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@@ -184,7 +184,7 @@\n \n             fut = asyncio.Future()\n             q.put_nowait(\"item\")\n-            assert await fut\n+            await fut\n \n \n async def test_qsize():\n@@ -195,7 +195,7 @@\n \n     _q = janus.Queue(2)\n     q = _q.async_q\n-    assert 0 == q.qsize()\n+    assert q.qsize() == 0\n     await q.put(0)\n     assert 1 == q.qsize()\n",
    "patch_ground_exp": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -28,7 +28,7 @@\n             _q = janus.Queue()\n             q = _q.async_q\n             # Start a task that waits to get.\n-            loop.create_task(q.get())\n+            await q.get()\n             # Let it start waiting.\n             await asyncio.sleep(0.1)\n             assert \"_getters[1]\" in fn(q)\n@@ -42,7 +42,7 @@\n             q.put_nowait(1)\n             # Start a task that waits to put.\n-            loop.create_task(q.put(2))\n+            await q.put(2)\n             # Let it start waiting.\n             await asyncio.sleep(0.1)\n             assert \"_putters[1]\" in fn(q)\n@@ -58,5 +58,5 @@\n     async def _test_queue_get(self):\n         q = janus.Queue()\n         result = await q.async_q.get()\n-        assert result == None\n+        await q.async_q.get()\n         q.sync_q.put('item')\n         result = await q.async_q.get()\n         assert result == 'item'\n\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -70,6 +70,7 @@\n         for task in tasks:\n             await task\n+            task.cancel()\n         if join_queue:\n             await asyncio.wait_for(empty_event.wait(), timeout)\n     return event\n--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n\n # Placeholder for potential changes if identified\n \n",
    "patch_ground_all": "\n--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -458,9 +458,9 @@\n             return ret\n \n         async def put():\n-            assert await q.put(\"a\")\n+            await q.put(\"a\")\n \n         _q = janus.Queue()\n         q = _q.async_q\n         t = loop.create_task(go())\n-        assert await fut\n-        assert await put()\n+        await fut\n+        await put()\n         assert await t == \"a\"\n \n         assert not _q._sync_mutex.locked()\n@@ -581,11 +581,12 @@\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 \n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -1,26 +1,33 @@\n import asyncio\n import contextlib\n import sys\n import threading\n\n import pytest\n\n\n import janus\n\n\n class 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@@ -30,6 +37,9 @@\n         await q.async_q.put(1)\n \n         def consumer():\n             nonlocal task\n             task = loop.create_task(queue.async_q.get())\n\n         loop = asyncio.get_event_loop()\n         loop.run_in_executor(None, consumer)\n         await asyncio.sleep(0.1)\n         task.cancel()\n\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -30,6 +31,7 @@\n         await q.async_q.put(1)\n \n         def consumer():\n             nonlocal task\n             task = loop.create_task(queue.async_q.get())\n \n         loop = asyncio.get_event_loop()\n         loop.run_in_executor(None, consumer)\n         await asyncio.sleep(0.1)\n         task.cancel()\n",
    "patch_ground_truth": "--- a/aio-libs_janus/tests/test_async.py\n+++ b/aio-libs_janus/tests/test_async.py\n@@ -458,13 +458,13 @@\n             return ret\n \n         async def put():\n-            assert await q.put(\"a\")\n+            await q.put(\"a\")\n \n         _q = janus.Queue()\n         q = _q.async_q\n         t = loop.create_task(go())\n-        assert await fut\n-        assert await put()\n+        await fut\n+        await put()\n         assert await t == \"a\"\n \n         assert not _q._sync_mutex.locked()\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_with_waiting_getters ________________________________________________________________________\n\nself = <test_async.TestQueuePut object at 0x792ed82adc70>\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            assert await q.put(\"a\")\n    \n        _q = janus.Queue()\n        q = _q.async_q\n        t = loop.create_task(go())\n>       assert await fut\nE       assert None\n\ntests/test_async.py:466: AssertionError\n================================================================================= short test summary info ==================================================================================\nFAILED tests/test_async.py::TestQueuePut::test_put_with_waiting_getters - assert None\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": 6,
      "Relevance": 8,
      "Explanation": 7,
      "Overall": 7
    },
    "issue_message": {
      "Title": 6,
      "Description": 7,
      "Reproducibility": 7,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_ground": {
      "Title": 8,
      "Description": 8,
      "Reproducibility": 7,
      "Relevance": 8,
      "Explanation": 7,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Assertion Usage and Test Inheritance Issues in Async Queue Tests",
      "description": "There are multiple issues within the `aio-libs_janus` async queue tests that need to be addressed for better reliability and maintainability. \n\n1. **Improper Use of `assert` Statements:** The `put()` method in the async test cases is incorrectly used with assertions. Using assertions with `await` can mask the actual intention and functionality of the test. We should just `await` the `put()` method without an assertion to avoid potential logical errors.\n\n2. **Unnecessary Assertions Awaiting Futures:** Test cases are using `assert await fut`, which is redundant and can be removed. Awaiting the future should be sufficient without an `assert` statement.\n\n3. **Improper Test Inheritance Structure:** The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes are incorrectly inheriting from `TestQueueJoin`. They should instead inherit from a proper test mixin class (`_QueueJoinTestMixin`). This change ensures better organization and maintainability of the test suite.\n\n4. **Missing Task Cancellation:** The `wait_for_empty_queue` async function within the mixed mode tests is missing a cancellation of running tasks after the queue is joined, which can lead to hanging tests. Adding a `task.cancel()` call after joining the queue fixes this issue.\n\nAddressing these issues will improve the robustness and readability of the test cases.",
      "explanation": "### Summary of the Issue\n\nThe issue revolves around improving the reliability and maintainability of asynchronous queue tests in the `aio-libs_janus` library. The specific problems identified are:\n\n1. **Improper Use of Assertion Statements in Async Tests:**\n   - Assertions like `assert await q.put()` are incorrectly used in async test cases. This practice can lead to misunderstandings of the test's purpose and mask potential logical errors, as assertions should not be directly used with `await`.\n\n2. **Redundant Assertions Awaiting Futures:**\n   - Statements such as `assert await fut` are redundant and can be simplified to just awaiting the future without an assertion, as the primary focus should be on the future's completion.\n\n3. **Improper Test Inheritance Structure:**\n   - The `TestLifoQueueJoin` and `TestPriorityQueueJoin` classes should not inherit directly from `TestQueueJoin`, but from a dedicated mixin class like `_QueueJoinTestMixin`. This inheritance adjustment is essential for better test organization and maintainability.\n\n4. **Missing Task Cancellation in Mixed Mode Tests:**\n   - In the `wait_for_empty_queue` async function within mixed mode tests, running tasks need to be canceled after the queue is joined to prevent hanging tests. Adding `task.cancel()` after joining the queue should resolve this issue.\n\n### Detailed Content of the Commit\n\nThe commit addresses the issues through the following changes:\n\n1. **Improper Use of Assertion Statements:**\n   - The commit removes the assertion usage with `await q.put()` and instead simply awaits the method. This change clarifies the intention of waiting for the `put` operation to complete rather than asserting its success, which is not meaningful within an asynchronous context.\n\n2. **Redundant Assertions Awaiting Futures:**\n   - Similar to the previous change, the commit removes redundant `assert await fut` statements. Awaiting a future implicitly checks for its completion, making the assertions unnecessary and simplifying the code.\n\n3. **Improper Test Inheritance Structure:**\n   - The commit changes the inheritance of `TestLifoQueueJoin` and `TestPriorityQueueJoin` to inherit from `_QueueJoinTestMixin` instead of `TestQueueJoin`. This refinement ensures that the specific behaviors for LIFO and Priority Queue joins are better encapsulated and organized, reflecting more precise testing behavior.\n\n4. **Missing Task Cancellation:**\n   - To handle the issue of hanging tests due to uncleared tasks, the commit adds a `task.cancel()` call after the queue has been joined in the `wait_for_empty_queue` function. This proactive task cancellation prevents any potential test hangup by ensuring that tasks do not remain in an uncertain state.\n\n### Explanation of the Solution\n\nThe solution provided by the commit addresses the underlying problems by:\n\n- **Improving Test Intent Clarity and Avoidance of Redundancy:**\n  - By removing misuse of assertions and redundant checks, the tests become more straightforward. It ensures that the functionality focuses on awaiting the completion of asynchronous methods rather than asserting their outcomes inappropriately.\n\n- **Enhancing Code Organization and Maintainability:**\n  - Refactoring the inheritance structure to use a mixin class improves the modularity of tests. It isolates specific test behaviors for queue types into well-defined components, making the test suite easier to manage and extend.\n\n- **Preventing Test Hangups:**\n  - Adding task cancellation after queue joining ensures that tests do not hang due to pending tasks. This step smoothens the test execution flow by cleaning up in-flight tasks that are no longer needed.\n\n### Conclusion\n\nThe commit successfully resolves the identified issues by enhancing code clarity, reducing redundancy, and improving the organization and reliability of the test suite. This series of changes leads to a more maintainable and robust testing framework for the `aio-libs_janus` library, ensuring that async queue operations are accurately tested and validated without unintended complications."
    }
  }
}