{
  "RepoName": "https://github.com/aio-libs/janus.git",
  "CommitSHA": "e46a1dfa166ebc2484716ee36cb99763f6247d52",
  "Time": "",
  "Difficulty": "Medium",
  "Type": "indexing 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__]\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\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_______________________________________________________________________________ TestQueue.test_simple_queue ________________________________________________________________________________\n\nself = <test_sync.TestQueue object at 0x7ba449e981f0>\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\ntests/test_sync.py:221: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\nself = <test_sync.TestQueue object at 0x7ba449e981f0>, _q = <janus.Queue object at 0x7ba449e52a60>\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__]\nE       KeyError: <class 'janus.Queue'>\n\ntests/test_sync.py:115: KeyError\n______________________________________________________________________________ TestLifoQueue.test_queue_join _______________________________________________________________________________\n\ncls = <class '_pytest.runner.CallInfo'>, func = <function call_and_report.<locals>.<lambda> at 0x7ba44856db80>, when = 'call'\nreraise = (<class '_pytest.outcomes.Exit'>, <class 'KeyboardInterrupt'>)\n\n    @classmethod\n    def from_call(\n        cls,\n        func: Callable[[], TResult],\n        when: Literal[\"collect\", \"setup\", \"call\", \"teardown\"],\n        reraise: Optional[\n            Union[Type[BaseException], Tuple[Type[BaseException], ...]]\n        ] = None,\n    ) -> \"CallInfo[TResult]\":\n        \"\"\"Call func, wrapping the result in a CallInfo.\n    \n        :param func:\n            The function to call. Called without arguments.\n        :param when:\n            The phase in which the function is called.\n        :param reraise:\n            Exception or exceptions that shall propagate if raised by the\n            function, instead of being wrapped in the CallInfo.\n        \"\"\"\n        excinfo = None\n        start = timing.time()\n        precise_start = timing.perf_counter()\n        try:\n>           result: Optional[TResult] = func()\n\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/_pytest/runner.py:341: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/_pytest/runner.py:241: in <lambda>\n    lambda: runtest_hook(item=item, **kwds), when=when, reraise=reraise\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/pluggy/_hooks.py:513: in __call__\n    return self._hookexec(self.name, self._hookimpls.copy(), kwargs, firstresult)\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/pluggy/_manager.py:120: in _hookexec\n    return self._inner_hookexec(hook_name, methods, kwargs, firstresult)\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/_pytest/threadexception.py:87: in pytest_runtest_call\n    yield from thread_exception_runtest_hook()\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/_pytest/threadexception.py:63: in thread_exception_runtest_hook\n    yield\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/_pytest/unraisableexception.py:90: in pytest_runtest_call\n    yield from unraisable_exception_runtest_hook()\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\n    def unraisable_exception_runtest_hook() -> Generator[None, None, None]:\n        with catch_unraisable_exception() as cm:\n            try:\n                yield\n            finally:\n                if cm.unraisable:\n                    if cm.unraisable.err_msg is not None:\n                        err_msg = cm.unraisable.err_msg\n                    else:\n                        err_msg = \"Exception ignored in\"\n                    msg = f\"{err_msg}: {cm.unraisable.object!r}\\n\\n\"\n                    msg += \"\".join(\n                        traceback.format_exception(\n                            cm.unraisable.exc_type,\n                            cm.unraisable.exc_value,\n                            cm.unraisable.exc_traceback,\n                        )\n                    )\n>                   warnings.warn(pytest.PytestUnraisableExceptionWarning(msg))\nE                   pytest.PytestUnraisableExceptionWarning: Exception ignored in: <coroutine object Queue._notify_async_not_full.<locals>.f at 0x7ba449ed3ec0>\nE                   \nE                   Traceback (most recent call last):\nE                     File \"/home/user/anaconda3/envs/py39/lib/python3.9/warnings.py\", line 506, in _warn_unawaited_coroutine\nE                       warn(msg, category=RuntimeWarning, stacklevel=2, source=coro)\nE                   RuntimeWarning: coroutine 'Queue._notify_async_not_full.<locals>.f' was never awaited\n\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/_pytest/unraisableexception.py:80: PytestUnraisableExceptionWarning\n------------------------------------------------------------------------------------ Captured log call -------------------------------------------------------------------------------------\nERROR    asyncio:base_events.py:1753 Task was destroyed but it is pending!\ntask: <Task pending name='Task-897' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nERROR    asyncio:base_events.py:1753 Task was destroyed but it is pending!\ntask: <Task pending name='Task-898' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nERROR    asyncio:base_events.py:1753 Task was destroyed but it is pending!\ntask: <Task pending name='Task-899' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nERROR    asyncio:base_events.py:1753 Task was destroyed but it is pending!\ntask: <Task pending name='Task-900' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nERROR    asyncio:base_events.py:1753 Task was destroyed but it is pending!\ntask: <Task pending name='Task-901' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nERROR    asyncio:base_events.py:1753 Task was destroyed but it is pending!\ntask: <Task pending name='Task-902' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\n_____________________________________________________________________________ TestLifoQueue.test_simple_queue ______________________________________________________________________________\n\nself = <test_sync.TestLifoQueue object at 0x7ba449e76f70>\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\ntests/test_sync.py:221: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\nself = <test_sync.TestLifoQueue object at 0x7ba449e76f70>, _q = <janus.LifoQueue object at 0x7ba448548e80>\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__]\nE       KeyError: <class 'janus.LifoQueue'>\n\ntests/test_sync.py:115: KeyError\n___________________________________________________________________________ TestPriorityQueue.test_simple_queue ____________________________________________________________________________\n\nself = <test_sync.TestPriorityQueue object at 0x7ba449e987f0>\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\ntests/test_sync.py:221: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\nself = <test_sync.TestPriorityQueue object at 0x7ba449e987f0>, _q = <janus.PriorityQueue object at 0x7ba449e20220>\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__]\nE       KeyError: <class 'janus.PriorityQueue'>\n\ntests/test_sync.py:115: KeyError\n================================================================================= short test summary info ==================================================================================\nFAILED tests/test_sync.py::TestQueue::test_simple_queue - KeyError: <class 'janus.Queue'>\nFAILED tests/test_sync.py::TestLifoQueue::test_queue_join - pytest.PytestUnraisableExceptionWarning: Exception ignored in: <coroutine object Queue._notify_async_not_full.<locals>.f at 0x7ba449ed3ec0>\nFAILED tests/test_sync.py::TestLifoQueue::test_simple_queue - KeyError: <class 'janus.LifoQueue'>\nFAILED tests/test_sync.py::TestPriorityQueue::test_simple_queue - KeyError: <class 'janus.PriorityQueue'>\n=============================================================================== 4 failed, 68 passed in 2.27s ===============================================================================\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1331' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1332' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1333' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1334' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1335' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1336' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nsys:1: RuntimeWarning: coroutine 'Queue._notify_async_not_empty.<locals>.f' was never awaited\nRuntimeWarning: Enable tracemalloc to get the object allocation traceback\nsys:1: RuntimeWarning: coroutine 'Queue._notify_async_not_full.<locals>.f' was never awaited\nRuntimeWarning: Enable tracemalloc to get the object allocation traceback\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1765' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1766' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1767' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1768' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1769' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1770' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>",
  "Patch": "--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -112,7 +112,7 @@\n             PriorityQueue=[111, 222, 333],\n         )\n         actual_order = [q.get(), q.get(), q.get()]\n-        assert actual_order == target_order[_q.__class__]\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--- 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_mixed.py",
      "function": null,
      "content_all": {},
      "content_change": {}
    },
    {
      "file": "aio-libs_janus/tests/test_sync.py",
      "function": {
        "99": "simple_queue_test"
      },
      "content_all": {
        "112": "            PriorityQueue=[111, 222, 333],\n",
        "113": "        )\n",
        "114": "        actual_order = [q.get(), q.get(), q.get()]\n",
        "115": "        assert actual_order == target_order[_q.__class__]\n",
        "116": "        for i in range(QUEUE_SIZE - 1):\n",
        "117": "            q.put(i)\n",
        "118": "            assert q.qsize()\n"
      },
      "content_change": {
        "115": "        assert actual_order == target_order[_q.__class__]\n"
      }
    },
    {
      "file": "aio-libs_janus/tests/test_sync.py",
      "function": null,
      "content_all": {},
      "content_change": {}
    }
  ],
  "Issue": {
    "title": "Inconsistent Type Handling and Unresponsive Task Cancellation in Queue Tests",
    "description": "The following issues have been identified in the `janus` testing suite:\n\n1. **Inconsistent Type Handling in Simple Queue Tests**:\n    - In `test_sync.py`, when verifying the order of items in different queue types, the current implementation uses `_q.__class__`, comparing it against `target_order`. This can lead to issues if the class representation changes or differs subtly. It should use the class name instead for a more consistent comparison.\n    \n2. **Async Task Cancellation Not Properly Integrated**:\n    - In `test_mixed.py`, during the `wait_for_empty_queue()` async function, when it finishes joining on the async queue, there might be lingering tasks that were supposed to be cancelled but weren't. This can lead to unresponsive or hanging tests as the expected cancellation does not occur.\n    \nThe above issues affect the maintainability and reliability of the testing suite. Ensuring consistent and type-safe comparisons will prevent subtle bugs from creeping in, while proper cancellation of tasks ensures that tests do not hang indefinitely, improving the reliability of the CI pipeline.",
    "explanation": "### Summary of the Issue\nThe issue in the `janus` testing suite pertains to two primary problems:\n1. **Inconsistent Type Handling in Simple Queue Tests**: In `test_sync.py`, comparisons are done using the class representation (`_q.__class__`), which can lead to inconsistencies if the class representation changes subtly. The correct approach should be to use the class name for comparison.\n2. **Async Task Cancellation Not Properly Integrated**: In `test_mixed.py`, during the `wait_for_empty_queue()` function, there's a possibility of tasks lingering after their supposed cancellation. This can lead to unresponsive or hanging tests because the expected task cancellations do not occur as they should.\n\n### Explanation of the Commit\nThe commit made changes in two files: `test_sync.py` and `test_mixed.py`.\n\n#### Changes in `test_sync.py`\nThe commit modified the comparisons in the simple queue tests from using the class representation to using the class name:\n- Before the change, the comparison used `_q.__class__` to compare against `target_order`.\n- After the change, it uses `_q.__class__.__name__`, which is more stable as it compares the actual class name as a string.\n\n*Explanation*: By using the class name instead of the class representation, we avoid potential issues caused by subtle differences or changes in how the class might be represented internally. This ensures more consistent and predictable behavior in the tests.\n\n#### Changes in `test_mixed.py`\nThe commit added a task cancellation step in the `wait_for_empty_queue()` function:\n- A `task.cancel()` call was added to ensure that all tasks are properly cancelled when the async queue has finished joining.\n\n*Explanation*: Adding the `task.cancel()` call ensures that any lingering tasks are properly cancelled. This prevents the tests from hanging or becoming unresponsive, leading to more reliable test execution and improved maintainability.\n\n### Detailed Explanation of the Issues and Solutions\n#### Inconsistent Type Handling in Simple Queue Tests\n**Issue**: \n- The test was using `_q.__class__` to verify the order of items in different queue types against a `target_order`. This is problematic as class representations can change and might not always be consistent, leading to potential mismatches and false test failures.\n\n**Solution**:\n- The commit corrected this by using `_q.__class__.__name__`, which directly compares the class names as strings. This is a more consistent and reliable way to verify the type, as class names as strings are less likely to have subtle changes compared to class representations.\n\n**Benefit**: \n- This change improves the robustness of the tests, reducing the likelihood of encountering errors due to class representation issues and ensuring that the comparisons are done in a type-safe manner.\n\n#### Async Task Cancellation Not Properly Integrated\n**Issue**:\n- During the execution of `wait_for_empty_queue()` in `test_mixed.py`, some tasks that are supposed to be cancelled might not be, potentially leading to unresponsive or hanging tests. This happens because there was no explicit cancellation call for tasks that might remain after the queue operation is completed.\n\n**Solution**:\n- The commit introduced a `task.cancel()` call to ensure that any tasks that linger after the async queue operations are explicitly cancelled.\n\n**Benefit**: \n- Adding this explicit cancellation helps in cleaning up tasks properly, preventing them from hanging or causing the test suite to be unresponsive. This improves test reliability and the stability of the CI pipeline.\n\n### Conclusion\n- The inconsistencies in type handling were resolved by ensuring the tests compared class names as strings, providing more stable and predictable comparisons.\n- The task cancellation logic was introduced to cleanly cancel any remaining tasks, preventing potential hangs and improving test execution reliability.\n\nThese changes together enhance the maintainability and reliability of the `janus` testing suite, ensuring more accurate and consistent test outcomes while preventing potential issues related to unresponsive tasks."
  },
  "Explain": "### Summary of the Issue\nThe issue in the `janus` testing suite pertains to two primary problems:\n1. **Inconsistent Type Handling in Simple Queue Tests**: In `test_sync.py`, comparisons are done using the class representation (`_q.__class__`), which can lead to inconsistencies if the class representation changes subtly. The correct approach should be to use the class name for comparison.\n2. **Async Task Cancellation Not Properly Integrated**: In `test_mixed.py`, during the `wait_for_empty_queue()` function, there's a possibility of tasks lingering after their supposed cancellation. This can lead to unresponsive or hanging tests because the expected task cancellations do not occur as they should.\n\n### Explanation of the Commit\nThe commit made changes in two files: `test_sync.py` and `test_mixed.py`.\n\n#### Changes in `test_sync.py`\nThe commit modified the comparisons in the simple queue tests from using the class representation to using the class name:\n- Before the change, the comparison used `_q.__class__` to compare against `target_order`.\n- After the change, it uses `_q.__class__.__name__`, which is more stable as it compares the actual class name as a string.\n\n*Explanation*: By using the class name instead of the class representation, we avoid potential issues caused by subtle differences or changes in how the class might be represented internally. This ensures more consistent and predictable behavior in the tests.\n\n#### Changes in `test_mixed.py`\nThe commit added a task cancellation step in the `wait_for_empty_queue()` function:\n- A `task.cancel()` call was added to ensure that all tasks are properly cancelled when the async queue has finished joining.\n\n*Explanation*: Adding the `task.cancel()` call ensures that any lingering tasks are properly cancelled. This prevents the tests from hanging or becoming unresponsive, leading to more reliable test execution and improved maintainability.\n\n### Detailed Explanation of the Issues and Solutions\n#### Inconsistent Type Handling in Simple Queue Tests\n**Issue**: \n- The test was using `_q.__class__` to verify the order of items in different queue types against a `target_order`. This is problematic as class representations can change and might not always be consistent, leading to potential mismatches and false test failures.\n\n**Solution**:\n- The commit corrected this by using `_q.__class__.__name__`, which directly compares the class names as strings. This is a more consistent and reliable way to verify the type, as class names as strings are less likely to have subtle changes compared to class representations.\n\n**Benefit**: \n- This change improves the robustness of the tests, reducing the likelihood of encountering errors due to class representation issues and ensuring that the comparisons are done in a type-safe manner.\n\n#### Async Task Cancellation Not Properly Integrated\n**Issue**:\n- During the execution of `wait_for_empty_queue()` in `test_mixed.py`, some tasks that are supposed to be cancelled might not be, potentially leading to unresponsive or hanging tests. This happens because there was no explicit cancellation call for tasks that might remain after the queue operation is completed.\n\n**Solution**:\n- The commit introduced a `task.cancel()` call to ensure that any tasks that linger after the async queue operations are explicitly cancelled.\n\n**Benefit**: \n- Adding this explicit cancellation helps in cleaning up tasks properly, preventing them from hanging or causing the test suite to be unresponsive. This improves test reliability and the stability of the CI pipeline.\n\n### Conclusion\n- The inconsistencies in type handling were resolved by ensuring the tests compared class names as strings, providing more stable and predictable comparisons.\n- The task cancellation logic was introduced to cleanly cancel any remaining tasks, preventing potential hangs and improving test execution reliability.\n\nThese changes together enhance the maintainability and reliability of the `janus` testing suite, ensuring more accurate and consistent test outcomes while preventing potential issues related to unresponsive tasks.",
  "Source": "Human",
  "Token": 1142,
  "Command": [
    "pytest -v tests"
  ],
  "FilteredCode": [
    {
      "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 blocking functions --\n35         # they should never actually wait that long - they should make\n36         # progress as soon as we call self.fn()).\n37         time.sleep(0.1)\n38         self.startedEvent.set()\n39         self.fn(*self.args)\n40 \n41 \n42 # Execute a function that blocks, and in a separate thread, a function that\n43 # triggers the release.  Returns the result of the blocking function.  Caution:\n44 # block_func must guarantee to block until trigger_func is called, a(...truncated)"
    },
    {
      "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 wait(...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_done(se(...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.mark.as(...truncated)"
    }
  ],
  "TokenAll": 15104,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Use of potentially unsafe private attribute in qfull function",
      "description": "The function qfull uses the private attribute q._parent._maxsize on line 17. Accessing private attributes is generally discouraged as they may change without warning in future versions, thus potentially breaking the code.",
      "explanation": "### Issue Summary\n\nThe issue reported involves the use of a potentially unsafe private attribute in the `qfull` function. Specifically, the function accesses the private attribute `q._parent._maxsize` on line 17. Accessing private attributes (those prefixed with an underscore) can cause issues because these attributes are considered implementation details and can change without warning in future versions, potentially breaking the code.\n\n### Content of the Commit\n\nThe commit likely aims to address this issue by eliminating the need to access the private attribute directly. Instead, it will probably introduce a safe, public method or property to retrieve the necessary information, ensuring future compatibility and adhering to good programming practices.\n\nThe exact commit change is not provided, but based on the issue and the existing code, we can infer the following changes would be made:\n\n### Analyzing the Provided Code\n\nLet's analyze the provided code in more detail, focusing on the piece where the issue resides.\n\n#### Original Code\n\n**File: aio-libs_janus/tests/test_sync.py**\n```python\n16 def qfull(q):\n17     return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n```\n\nHere, `qfull` checks if the queue `q` is full by comparing its size (`q.qsize()`) with `q._parent._maxsize`. The use of `q._parent` is a way to access the parent queue and its private attribute `_maxsize`.\n\n#### Issues Identified:\n1. Accessing `q._parent._maxsize` directly is discouraged as `_maxsize` is a private attribute.\n2. This approach could break if the internal structure of the queue changes in future updates.\n3. It violates encapsulation principles.\n\n#### Proposed Solution\n\nTo fix this issue, we should modify `qfull` to use a public interface for accessing the queue's `maxsize`. This is typically done by providing a public property or method in the queue class that returns `_maxsize`.\n\n### Expected Changes in the Commit\n\nLet's outline the changes that would be expected in the commit:\n\n#### Step 1: Adding a Public Method or Property\n\nThe public interface should be added to the queue class to expose the `_maxsize` safely. This would typically be added in the `janus` queue implementation.\n\n**Modified File: aio-libs_janus/janus/__init__.py**\nAssuming `janus.Queue` class is where `_maxsize` is defined:\n```python\nclass Queue:\n    def __init__(self, maxsize=0):\n        self._maxsize = maxsize\n        # other initializations\n\n    @property\n    def maxsize(self):\n        return self._maxsize\n```\n\n#### Step 2: Adjusting the `qfull` Function\n\nNow, the `qfull` function should be updated to use this new property.\n\n**Modified File: aio-libs_janus/tests/test_sync.py**\n```python\n16 def qfull(q):\n17     return q._parent.maxsize > 0 and q.qsize() == q._parent.maxsize\n```\n\n### Explanation of the Solution\n\n1. **Encapsulation**: By introducing a public property `maxsize`, we ensure that the internal representation of the queue size is encapsulated and can be changed without breaking external code. \n\n2. **Safety**: This approach avoids direct access to `_parent._maxsize`, making the code more robust against future changes in the `janus.Queue`.\n\n3. **Maintainability**: Code that adheres to encapsulation and uses public interfaces is easier to maintain and understand.\n\n### Summary\n\n- **Issue**: The `qfull` function was using the private attribute `q._parent._maxsize`, which is unsafe and against good practice.\n- **Solution**: Introduced a public property `maxsize` in the queue class to access `_maxsize` safely.\n- **Commit Analysis**: The changes encapsulated the `_maxsize` inside a public property, modified the `qfull` function to use this property, thus solving the issue by adhering to principles of encapsulation and future-proofing the code against changes in `janus.Queue` internal implementation."
    },
    "issue_message": {
      "title": "KeyError in test_simple_queue",
      "description": "The function `simple_queue_test` in `tests/test_sync.py` raises a KeyError because the class used for `_q` (janus.Queue, janus.LifoQueue, janus.PriorityQueue) is not handled in the `target_order` dictionary. Ensure that the key `_q.__class__` exists in `target_order` or handle the exception appropriately.",
      "explanation": "### Issue Summary\n\nThe issue is related to a `KeyError` raised in the `simple_queue_test` function within `tests/test_sync.py`. The root cause of the error is that the class instance used for `_q` (`janus.Queue`, `janus.LifoQueue`, `janus.PriorityQueue`) is not included as a key in the `target_order` dictionary. This leads to an unhandled exception when the class is looked up in the dictionary.\n\n### Analysis\n\n1. **Error Details:**\n   - The test case `test_simple_queue` is calling the `simple_queue_test` method.\n   - Within `simple_queue_test`, a key lookup is performed in the `target_order` dictionary using `_q.__class__` as the key.\n   - The keys in the `target_order` dictionary are expected to match classes such as `janus.Queue`, `janus.LifoQueue`, and `janus.PriorityQueue`.\n   - The lookup fails because these class instances are not correctly represented in the dictionary, causing a `KeyError`.\n\n2. **Relevant Code (Highlighted Section):**\n   ```python\n   def simple_queue_test(self, _q):\n       ...\n       target_order = dict(\n           Queue=[111, 333, 222],\n           LifoQueue=[222, 333, 111],\n           PriorityQueue=[111, 222, 333],\n       )\n       ...\n       assert actual_order == target_order[_q.__class__]\n   ```\n\n### Content of Commit Fixing the Issue\n\n**Commit Message:**\n```plaintext\nHandle KeyError by including class types in target_order dictionary\n```\n\n**Code Changes:**\nThe `target_order` dictionary is updated to use class types as keys, ensuring the lookup will succeed.\n\n```python\ndef simple_queue_test(self, _q):\n    ...\n    target_order = dict(\n        janus.Queue=[111, 333, 222],\n        janus.LifoQueue=[222, 333, 111],\n        janus.PriorityQueue=[111, 222, 333],\n    )\n    ...\n    assert actual_order == target_order[_q.__class__]\n```\n\n### Explanation of the Fix\n\n1. **Summary of Fix:**\n   - The fix updates the `target_order` dictionary to use the actual class types (`janus.Queue`, `janus.LifoQueue`, `janus.PriorityQueue`) as keys. This change aligns the dictionary keys with the types of the `_q` instances being used in the tests.\n\n2. **Details and Reasoning:**\n   - The original dictionary used string representations of class names (`'Queue'`, `'LifoQueue'`, `'PriorityQueue'`). These strings do not match the actual classes returned by `_q.__class__`, leading to a `KeyError`.\n   - By switching the dictionary keys to the actual class types (`janus.Queue`, etc.), the code ensures that the lookup `_q.__class__` will always find the correct entry in the dictionary.\n   \n3. **How the Fix Solves the Issue:**\n   - The updated `target_order` dictionary now correctly maps each specific queue class to its expected order of elements. For instance, when `_q` is an instance of `janus.LifoQueue`, `_q.__class__` translates to `janus.LifoQueue`, which is now a valid key in the `target_order` dictionary.\n   - This prevents the `KeyError` from occurring and ensures that the tests correctly validate the order of elements dequeued from each queue type.\n\n### Example Explanation for Developers\n\nImagine having a lookup table in your code where you want to associate specific behaviors with certain types of objects. If your lookup keys are not correctly aligned with the objects you are testing, your program will fail to find the necessary values, causing errors. This fix ensures that the object types (like `janus.Queue` or `janus.LifoQueue`) directly match the keys in your lookup table, preventing such errors.\n\nThis change ensures the following:\n- Correct association between queue types and their expected behaviors.\n- Prevention of `KeyError` exceptions by ensuring that every possible class type used in the test has a corresponding key in the `target_order` dictionary.\n\nBy making this adjustment, the tests can now accurately and reliably verify that different queue implementations (`Queue`, `LifoQueue`, `PriorityQueue`) behave as expected without encountering unexpected errors."
    },
    "issue_ground": {
      "title": "Inconsistent Type Handling and Unresponsive Task Cancellation in Queue Tests",
      "description": "The following issues have been identified in the `janus` testing suite:\n\n1. **Inconsistent Type Handling in Simple Queue Tests**:\n    - In `test_sync.py`, when verifying the order of items in different queue types, the current implementation uses `_q.__class__`, comparing it against `target_order`. This can lead to issues if the class representation changes or differs subtly. It should use the class name instead for a more consistent comparison.\n    \n2. **Async Task Cancellation Not Properly Integrated**:\n    - In `test_mixed.py`, during the `wait_for_empty_queue()` async function, when it finishes joining on the async queue, there might be lingering tasks that were supposed to be cancelled but weren't. This can lead to unresponsive or hanging tests as the expected cancellation does not occur.\n    \nThe above issues affect the maintainability and reliability of the testing suite. Ensuring consistent and type-safe comparisons will prevent subtle bugs from creeping in, while proper cancellation of tasks ensures that tests do not hang indefinitely, improving the reliability of the CI pipeline.",
      "explanation": "### Summary of the Issue\n\nThe issues in the `janus` testing suite revolve around two primary areas:\n\n1. **Inconsistent Type Handling in Simple Queue Tests**:\n   - **File: `test_sync.py`**\n   - In the `simple_queue_test` function, type checking is done using `_q.__class__`, which causes failures when the representation of the class differs. \n   \n2. **Async Task Cancellation Not Properly Integrated**:\n   - **File: `test_mixed.py`**\n   - In the asynchronous function `wait_for_empty_queue()`, there can be lingering tasks which were intended to be canceled but weren't. This results in unresponsive or hanging tests, producing runtime warnings and causing the CI pipeline to be unreliable.\n\n### Commit Summary\n\nThe commit simply addresses these problems by:\n\n1. **Changing the Type Comparison Method**:\n   - Modifying the `simple_queue_test` in `test_sync.py` to use class names (strings) instead of class objects for comparison.\n   \n2. **Ensuring Proper Task Cancellation**:\n   - In `test_mixed.py`, enhancing the `wait_for_empty_queue()` to ensure all pending tasks that are supposed to be canceled are indeed canceled.\n\n### Detailed Explanation of Commit Changes\n\n1. **Type Comparison Change:**\n   \n   ```python\n   # Before Change\n   target_order = dict(\n       Queue=[111, 333, 222],\n       LifoQueue=[222, 333, 111],\n       PriorityQueue=[111, 222, 333],\n   )\n   ...\n   assert actual_order == target_order[_q.__class__]\n   ```\n\n   **After Change:**\n   ```python\n   target_order = {\n       'Queue': [111, 333, 222],\n       'LifoQueue': [222, 333, 111],\n       'PriorityQueue': [111, 222, 333],\n   }\n   ...\n   assert actual_order == target_order[type(_q).__name__]\n   ```\n\n   - **Cause of Issue:**\n     Using `_q.__class__` in the dictionary access can lead to `KeyError` if the class representation changes due to any reason, such as subclassing or changes in the name mangling.\n   - **Solution:**\n     Switching to using class names (`type(_q).__name__`) ensures a consistent string-based comparison which avoids the pitfalls of class object mismatch and improves robustness.\n   - **How it Solves the Issue:**\n     This change prevents `KeyError` by ensuring the dictionary always matches on straightforward class names like `'Queue'`, `'LifoQueue'`, and `'PriorityQueue'` instead of potentially complex class object representations.\n\n2. **Ensuring Proper Async Task Cancellation:**\n\n   In the `wait_for_empty_queue()` and related functions in `test_mixed.py`, the update is:\n\n   ```python\n   # Assume an example function wait_for_empty_queue definition\n   async def wait_for_empty_queue(queue):\n       # Before fix\n       await queue.join()\n       # Possibly pending tasks not being cancelled here leading to corruption\n       \n       # After fix: Ensure to cancel lingering tasks\n       ...\n       tasks = [t for t in asyncio.all_tasks() if t is not asyncio.current_task()]\n       for task in tasks:\n           task.cancel()\n       await asyncio.gather(*tasks, return_exceptions=True)\n   ```\n\n   - **Cause of Issue:**\n     After awaiting `queue.join()`, there could still be pending tasks that were supposed to get canceled but didn’t, leading to unawaited coroutine warnings.\n   - **Solution:**\n     The updated code ensures that after the `queue.join()` call, all remaining tasks are retrieved and explicitly canceled, and then gathered to handle the cancellation properly.\n   - **How it Solves the Issue:**\n     By explicitly canceling and awaiting on all tasks, the function ensures no pending tasks remain unnoticed. This prevents runtime warnings about unawaited coroutines and reduces the risk of tests hanging, thereby improving the reliability of the tests.\n\n### Conclusion\n\nThe commit resolves two key issues related to type handling in synchronous queue tests and proper task cancellation in asynchronous queue tests:\n\n1. **Type Handling Issue:**\n   - Switching from class object-based comparison to class name-based comparison in `test_sync.py`.\n\n2. **Task Cancellation Issue:**\n   - Ensuring all pending tasks are canceled correctly in `test_mixed.py`.\n\nThese changes improve test robustness, reduce `KeyError`s, prevent hanging tests, and ensure a more reliable CI pipeline."
    },
    "issue_ground_truth": {
      "title": "Inconsistent Type Handling and Unresponsive Task Cancellation in Queue Tests",
      "description": "The following issues have been identified in the `janus` testing suite:\n\n1. **Inconsistent Type Handling in Simple Queue Tests**:\n    - In `test_sync.py`, when verifying the order of items in different queue types, the current implementation uses `_q.__class__`, comparing it against `target_order`. This can lead to issues if the class representation changes or differs subtly. It should use the class name instead for a more consistent comparison.\n    \n2. **Async Task Cancellation Not Properly Integrated**:\n    - In `test_mixed.py`, during the `wait_for_empty_queue()` async function, when it finishes joining on the async queue, there might be lingering tasks that were supposed to be cancelled but weren't. This can lead to unresponsive or hanging tests as the expected cancellation does not occur.\n    \nThe above issues affect the maintainability and reliability of the testing suite. Ensuring consistent and type-safe comparisons will prevent subtle bugs from creeping in, while proper cancellation of tasks ensures that tests do not hang indefinitely, improving the reliability of the CI pipeline.",
      "explanation": "### Summary of the Issue\nThe issue in the `janus` testing suite pertains to two primary problems:\n1. **Inconsistent Type Handling in Simple Queue Tests**: In `test_sync.py`, comparisons are done using the class representation (`_q.__class__`), which can lead to inconsistencies if the class representation changes subtly. The correct approach should be to use the class name for comparison.\n2. **Async Task Cancellation Not Properly Integrated**: In `test_mixed.py`, during the `wait_for_empty_queue()` function, there's a possibility of tasks lingering after their supposed cancellation. This can lead to unresponsive or hanging tests because the expected task cancellations do not occur as they should.\n\n### Explanation of the Commit\nThe commit made changes in two files: `test_sync.py` and `test_mixed.py`.\n\n#### Changes in `test_sync.py`\nThe commit modified the comparisons in the simple queue tests from using the class representation to using the class name:\n- Before the change, the comparison used `_q.__class__` to compare against `target_order`.\n- After the change, it uses `_q.__class__.__name__`, which is more stable as it compares the actual class name as a string.\n\n*Explanation*: By using the class name instead of the class representation, we avoid potential issues caused by subtle differences or changes in how the class might be represented internally. This ensures more consistent and predictable behavior in the tests.\n\n#### Changes in `test_mixed.py`\nThe commit added a task cancellation step in the `wait_for_empty_queue()` function:\n- A `task.cancel()` call was added to ensure that all tasks are properly cancelled when the async queue has finished joining.\n\n*Explanation*: Adding the `task.cancel()` call ensures that any lingering tasks are properly cancelled. This prevents the tests from hanging or becoming unresponsive, leading to more reliable test execution and improved maintainability.\n\n### Detailed Explanation of the Issues and Solutions\n#### Inconsistent Type Handling in Simple Queue Tests\n**Issue**: \n- The test was using `_q.__class__` to verify the order of items in different queue types against a `target_order`. This is problematic as class representations can change and might not always be consistent, leading to potential mismatches and false test failures.\n\n**Solution**:\n- The commit corrected this by using `_q.__class__.__name__`, which directly compares the class names as strings. This is a more consistent and reliable way to verify the type, as class names as strings are less likely to have subtle changes compared to class representations.\n\n**Benefit**: \n- This change improves the robustness of the tests, reducing the likelihood of encountering errors due to class representation issues and ensuring that the comparisons are done in a type-safe manner.\n\n#### Async Task Cancellation Not Properly Integrated\n**Issue**:\n- During the execution of `wait_for_empty_queue()` in `test_mixed.py`, some tasks that are supposed to be cancelled might not be, potentially leading to unresponsive or hanging tests. This happens because there was no explicit cancellation call for tasks that might remain after the queue operation is completed.\n\n**Solution**:\n- The commit introduced a `task.cancel()` call to ensure that any tasks that linger after the async queue operations are explicitly cancelled.\n\n**Benefit**: \n- Adding this explicit cancellation helps in cleaning up tasks properly, preventing them from hanging or causing the test suite to be unresponsive. This improves test reliability and the stability of the CI pipeline.\n\n### Conclusion\n- The inconsistencies in type handling were resolved by ensuring the tests compared class names as strings, providing more stable and predictable comparisons.\n- The task cancellation logic was introduced to cleanly cancel any remaining tasks, preventing potential hangs and improving test execution reliability.\n\nThese changes together enhance the maintainability and reliability of the `janus` testing suite, ensuring more accurate and consistent test outcomes while preventing potential issues related to unresponsive tasks."
    },
    "location_origin": [
      {
        "file": "aio-libs_janus/tests/test_sync.py",
        "function": {
          "16": "qfull"
        },
        "content_all": {
          "13": "QUEUE_SIZE = 5\n",
          "14": "\n",
          "15": "\n",
          "16": "def qfull(q):\n",
          "17": "    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n",
          "18": "\n",
          "19": "\n",
          "20": "# A thread to run a function that unclogs a blocked Queue.\n"
        },
        "content_change": {
          "17": "    return q._parent.maxsize > 0 and q.qsize() == q._parent.maxsize\n"
        }
      },
      {
        "file": "aio-libs_janus/janus/__init__.py",
        "function": {
          "29": "BaseQueue"
        },
        "content_all": {
          "23": "\n",
          "24": "\n",
          "25": "T = TypeVar(\"T\")\n",
          "26": "OptFloat = Optional[float]\n",
          "27": "\n",
          "28": "\n",
          "29": "class BaseQueue(Protocol[T]):\n",
          "30": "    @property\n",
          "31": "    def maxsize(self) -> int:\n",
          "32": "        ...\n",
          "33": "\n",
          "34": "    @property\n",
          "35": "    def closed(self) -> bool:\n"
        },
        "content_change": {
          "31": "    def maxsize(self) -> int:\n",
          "32": "        ...\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "aio-libs_janus/tests/test_sync.py",
        "function": {
          "107": "simple_queue_test"
        },
        "content_all": {
          "106": "\n",
          "107": "    def simple_queue_test(self, _q):\n",
          "108": "        q = _q.sync_q\n",
          "109": "        if q.qsize():\n",
          "110": "            raise RuntimeError(\"Call this function with an empty queue\")\n",
          "111": "        assert q.empty()\n",
          "112": "        assert not q.full()\n",
          "113": "        # I guess we better check things actually queue correctly a little :)\n",
          "114": "        q.put(111)\n",
          "115": "        q.put(333)\n",
          "116": "        q.put(222)\n",
          "117": "        target_order = dict(\n",
          "118": "            Queue=[111, 333, 222],\n",
          "119": "            LifoQueue=[222, 333, 111],\n",
          "120": "            PriorityQueue=[111, 222, 333],\n",
          "121": "        )\n",
          "122": "        actual_order = [q.get(), q.get(), q.get()]\n",
          "123": "        assert actual_order == target_order[_q.__class__]\n",
          "124": "\n"
        },
        "content_change": {
          "123": "        assert actual_order == target_order[_q.__class__]\n"
        }
      }
    ],
    "location_ground": [
      {
        "file": "aio-libs_janus/tests/test_sync.py",
        "function": {
          "16": "simple_queue_test"
        },
        "content_all": {
          "10": "\n",
          "11": "import janus\n",
          "12": "\n",
          "13": "QUEUE_SIZE = 5\n",
          "14": "\n",
          "15": "\n",
          "16": "def qfull(q):\n",
          "17": "    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n",
          "18": "\n",
          "19": "\n",
          "20": "# A thread to run a function that unclogs a blocked Queue.\n"
        },
        "content_change": {
          "16": "    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_sync.py",
        "function": {
          "42": "simple_queue_test"
        },
        "content_all": {
          "40": "\n",
          "41": "\n",
          "42": "# Execute a function that blocks, and in a separate thread, a function that\n",
          "43": "# triggers the release.  Returns the result of the blocking function.  Caution:\n",
          "44": "# block_func must guarantee to block until trigger_func is called, and trigger_func\n",
          "45": "# must release block_func within a week :-O  Returns the result of calling\n",
          "46": "# block_func, or raises an exception if trigger_func raises one.\n",
          "47": "def _do_blocking_test(block_func, trigger_func, trigger_args=[]):\n"
        },
        "content_change": {
          "42": "        assert actual_order == target_order[type(_q).__name__]\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": {
          "22": "wait_for_empty_queue"
        },
        "content_all": {
          "20": "\n",
          "21": "async def wait_for_empty_queue(queue):\n",
          "22": "    await queue.join()\n",
          "23": "    tasks = [t for t in asyncio.all_tasks() if t is not asyncio.current_task()]\n",
          "24": "    for task in tasks:\n",
          "25": "        task.cancel()\n",
          "26": "    await asyncio.gather(*tasks, return_exceptions=True)\n"
        },
        "content_change": {
          "22": "    await queue.join()\n",
          "23": "    tasks = [t for t in asyncio.all_tasks() if t is not asyncio.current_task()]\n",
          "24": "    for task in tasks:\n",
          "25": "        task.cancel()\n",
          "26": "    await asyncio.gather(*tasks, return_exceptions=True)\n"
        }
      }
    ],
    "location_ground_exp": [
      {
        "file": "aio-libs_janus/tests/test_sync.py",
        "function": {
          "16": "qfull"
        },
        "content_all": {
          "13": "QUEUE_SIZE = 5\n",
          "14": "\n",
          "15": "\n",
          "16": "def qfull(q):\n",
          "17": "    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n",
          "18": "\n",
          "19": "\n",
          "20": "# A thread to run a function that unclogs a blocked Queue.\n"
        },
        "content_change": {
          "17": "    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": {
          "22": "test_maxsize"
        },
        "content_all": {
          "19": "            janus.Queue()\n",
          "20": "\n",
          "21": "    @pytest.mark.asyncio\n",
          "22": "    async def test_maxsize(self):\n",
          "23": "        q = janus.Queue(5)\n",
          "24": "        assert 5 == q.maxsize\n",
          "25": "\n",
          "26": "    @pytest.mark.as\n"
        },
        "content_change": {
          "23": "        q = janus.Queue(5)\n",
          "24": "        assert 5 == q.maxsize\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_sync.py",
        "function": {
          "16": "Modified_SimpleQueue_Tests_Class_Name_Comparison"
        },
        "content_all": {
          "13": "QUEUE_SIZE = 5\n",
          "14": "\n",
          "15": "\n",
          "16": "def qfull(q):\n",
          "17": "    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n",
          "18": "\n",
          "19": "\n",
          "20": "# A thread to run a function that unclogs a blocked Queue.\n"
        },
        "content_change": {
          "17": "    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n"
        }
      }
    ],
    "location_ground_truth": [
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": null,
        "content_all": {},
        "content_change": {}
      },
      {
        "file": "aio-libs_janus/tests/test_sync.py",
        "function": {
          "99": "simple_queue_test"
        },
        "content_all": {
          "112": "            PriorityQueue=[111, 222, 333],\n",
          "113": "        )\n",
          "114": "        actual_order = [q.get(), q.get(), q.get()]\n",
          "115": "        assert actual_order == target_order[_q.__class__]\n",
          "116": "        for i in range(QUEUE_SIZE - 1):\n",
          "117": "            q.put(i)\n",
          "118": "            assert q.qsize()\n"
        },
        "content_change": {
          "115": "        assert actual_order == target_order[_q.__class__]\n"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_sync.py",
        "function": null,
        "content_all": {},
        "content_change": {}
      }
    ],
    "patch_i": "\n--- a/aio-libs_janus/janus/__init__.py\n+++ b/aio-libs_janus/janus/__init__.py\n@@ -29,6 +29,10 @@\n \n class Queue:\n     def __init__(self, maxsize=0):\n         self._maxsize = maxsize\n         # other initializations\n\n+    @property\n+    def maxsize(self):\n+        return self._maxsize\n+\n--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -16,7 +16,7 @@\n \n def qfull(q):\n-    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\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.\n",
    "patch_im": "\n--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -1,44 +1,52 @@\n # Some simple queue module tests, plus some failure conditions\n # to ensure the Queue locks remain stable.\n import asyncio\n import queue\n import threading\n import time\n from unittest.mock import patch\n \n import pytest\n \n import janus\n \n QUEUE_SIZE = 5\n \n \n def 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.\n class _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@@ -60,13 +68,13 @@ class TestSimpleQueue:\n         loop.create_task(test(_q))\n         loop.run_until_complete(_q._parent.join())\n \n-    def simple_queue_test(self, _q):\n+    def simple_queue_test(self, _q): \n         produced = []\n         consumed = []\n         def produce():\n             produced.extend(range(11, 12))  # to drive the timeout condition\n         janus._trigger_thread(produce)\n-        target_order = dict(\n-            Queue=[111, 333, 222],\n-            LifoQueue=[222, 333, 111],\n-            PriorityQueue=[111, 222, 333],\n+        target_order = {\n+            janus.Queue: [111, 333, 222],\n+            janus.LifoQueue: [222, 333, 111],\n+            janus.PriorityQueue: [111, 222, 333],\n+        }\n         actual_order = []\n         while not qfull(_q):\n             _q.put_nowait(111)\n",
    "patch_il": "\n--- a/aio-libs_janus/janus/__init__.py\n+++ b/aio-libs_janus/janus/__init__.py\n@@ -43,6 +43,10 @@\n \n     def __init__(self, maxsize=0):\n         self._maxsize = maxsize\n+        # other initializations\n+\n+    @property\n+    def maxsize(self):\n+        return self._maxsize\n \n # (rest of the code remains unchanged)\n \n--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -16,7 +16,7 @@\n def qfull(q):\n-    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n+    return q._parent.maxsize > 0 and q.qsize() == q._parent.maxsize\n \n \n # (rest of the code remains unchanged)\n",
    "patch_iml": "\n--- a/tests/test_sync.py\n+++ b/tests/test_sync.py\n@@ -116,8 +116,8 @@\n         q.put(222)\n         target_order = dict(\n             janus.Queue: [111, 333, 222],\n-            janus.LifoQueue: [222, 333, 111],\n-            janus.PriorityQueue: [111, 222, 333],\n+            janus.LifoQueue: [222, 333, 111],\n+            janus.PriorityQueue: [111, 222, 333],\n         )\n         actual_order = [q.get(), q.get(), q.get()]\n         assert actual_order == target_order[_q.__class__]\n",
    "patch_ground": "\n--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -1,11 +1,11 @@\n # Some simple queue module tests, plus some failure conditions\n # to ensure the Queue locks remain stable.\n import asyncio\n import queue\n import threading\n import time\n from unittest.mock import patch\n \n import pytest\n@@ -42,28 +42,28 @@ class _TriggerThread(threading.Thread):\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 wake it.\n #\n # Used by these helpers because it's so easy to misuse _TriggerThread directly.\n def blocking_test(block_func, args, trigger_func, trigger_args):\n     t = _TriggerThread(trigger_func, trigger_args)\n     t.start()\n     t.startedEvent.wait()\n     result = block_func(*args)\n     t.join()\n     return result\n \n \n-@pytest.mark.parametrize(\"queue_type, target_order\", [\n-    (queue.Queue, [111, 333, 222]),\n-    (queue.LifoQueue, [222, 333, 111]),\n-    (queue.PriorityQueue, [111, 222, 333]),\n+@pytest.mark.parametrize(\"queue_type, target_order\", [\n+    ('Queue', [111, 333, 222]),\n+    ('LifoQueue', [222, 333, 111]),\n+    ('PriorityQueue', [111, 222, 333]),\n ])\n def test_simple_queue(queue_type, target_order):\n     q = queue_type()\n     for item in target_order:\n         q.put(item)\n     actual_order = [q.get() for _ in range(3)]\n-    assert actual_order == target_order[q.__class__]\n+    assert actual_order == target_order[type(q).__name__]\n \n \n @pytest.mark.parametrize(\"queue_type\", [\n     queue.Queue,\n     janus.SyncQueue,\n ])\n def test_bad_max_size(queue_type):\n     # Can't reconfigure maxsize to an explicit zero, with a queue that\n     # already has room.\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -1,10 +1,20 @@\n import asyncio\n import contextlib\n import sys\n import threading\n \n import pytest\n \n import janus\n \n \n class TestMixedMode:\n     @pytest.mark.skipif(\n         sys.version_info < (3, 7),\n@@ -134,11 +134,19 @@ class TestMixedMode:\n     )\n     async def test_background_thread_async_wait(self, caplog):\n         loop = asyncio.get_running_loop()\n \n         q = janus.Queue()\n         thread = threading.Thread(\n             target=lambda: q.async_q.put(\"X\", timeout=2), daemon=True\n         )\n         thread.start()\n \n+        async def wait_for_empty_queue(queue):\n+            await queue.join()\n+            tasks = [t for t in asyncio.all_tasks() if t is not asyncio.current_task()]\n+            for task in tasks:\n+                task.cancel()\n+            await asyncio.gather(*tasks, return_exceptions=True)\n+\n+        await wait_for_empty_queue(q.async_q)\n+\n         await asyncio.sleep(0.1)\n",
    "patch_ground_location": "\n--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -1,7 +1,7 @@\n # Some simple queue module tests, plus some failure conditions\n # to ensure the Queue locks remain stable.\n import asyncio\n import queue\n import threading\n import time\n from unittest.mock import patch\n@@ -42,7 +42,7 @@ def _do_blocking_test(block_func, trigger_func, trigger_args=[]):\n \n # The test body that verifies that insertions in a queue variant class\n # end up in the expected order.\n def simple_queue_test(sync_q_factory, expected_order):\n     _q = sync_q_factory()\n     for i in [111, 222, 333]:\n         _q.put(i)\n     actual_order = []\n     while not qfull(_q):\n         actual_order.append(_q.get())\n-    assert actual_order == expected_order[_q.__class__]\n+    assert actual_order == expected_order[type(_q).__name__]\n \n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -20,7 +20,7 @@ class TestMixedMode:\n \n @pytest.mark.asyncio\n async def wait_for_empty_queue(queue):\n     await queue.join()\n+    tasks = [t for t in asyncio.all_tasks() if t is not asyncio.current_task()]\n     for task in tasks:\n         task.cancel()\n     await asyncio.gather(*tasks, return_exceptions=True)\n",
    "patch_ground_exp": "\n--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -1,7 +1,9 @@\n # Some simple queue module tests, plus some failure conditions\n # to ensure the Queue locks remain stable.\n import asyncio\n import queue\n+import janus\n+import threading\n import time\n from unittest.mock import patch\n \n@@ -70,7 +72,7 @@ def test_simple_queue():\n \n     q.put(2)\n     q.put(3)\n-    assert q._q.__class__ == queue.Queue\n+    assert q._q.__class__.__name__ == \"Queue\"\n     assert list(q._q.queue) == target_order\n \n \n@@ -94,7 +96,7 @@ def test_full_simple_queue():\n     q.put(2)\n     q.put(3)\n     with pytest.raises(janus.QueueFull):\n-        q.put(4, timeout=0.1)\n+        q.put(4, timeout=0.1)\n     assert q._q.__class__.__name__ == \"Queue\"\n     assert list(q._q.queue) == target_order\n \n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -1,7 +1,8 @@\n import asyncio\n import contextlib\n import sys\n import threading\n+import janus\n import pytest\n \n \n@@ -41,6 +42,7 @@ async def wait_for_empty_queue(q):\n         if q.empty():\n             break\n         await asyncio.sleep(0.01)\n+        task.cancel()\n \n async def test_task_cancellation():\n     q = janus.Queue()\n@@ -49,6 +51,7 @@ async def test_task_cancellation():\n \n         await q.async_q.join()\n \n         for task in tasks:\n+            task.cancel()\n             with contextlib.suppress(asyncio.CancelledError):\n                 await task\n",
    "patch_ground_all": "\n--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -112,7 +112,7 @@ \n             PriorityQueue=[111, 222, 333],\n         )\n         actual_order = [q.get(), q.get(), q.get()]\n-        assert actual_order == target_order[_q.__class__]\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--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -26,6 +26,14 @@ \n         assert 5 == q.maxsize\n \n     @pytest.mark.asyncio\n+    async def wait_for_empty_queue(self, q):\n+        while not q.empty():\n+            await asyncio.sleep(0.1)\n+        for task in asyncio.all_tasks():\n+            if task is not asyncio.current_task():\n+                task.cancel()\n+\n+    @pytest.mark.asyncio\n     async def test_async_task_cancellation(self):\n         q = janus.Queue()\n         await q.async_q.put(1)\n",
    "patch_ground_truth": "--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -112,7 +112,7 @@\n             PriorityQueue=[111, 222, 333],\n         )\n         actual_order = [q.get(), q.get(), q.get()]\n-        assert actual_order == target_order[_q.__class__]\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--- 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_______________________________________________________________________________ TestQueue.test_simple_queue ________________________________________________________________________________\n\nself = <test_sync.TestQueue object at 0x7ba449e981f0>\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\ntests/test_sync.py:221: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\nself = <test_sync.TestQueue object at 0x7ba449e981f0>, _q = <janus.Queue object at 0x7ba449e52a60>\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__]\nE       KeyError: <class 'janus.Queue'>\n\ntests/test_sync.py:115: KeyError\n______________________________________________________________________________ TestLifoQueue.test_queue_join _______________________________________________________________________________\n\ncls = <class '_pytest.runner.CallInfo'>, func = <function call_and_report.<locals>.<lambda> at 0x7ba44856db80>, when = 'call'\nreraise = (<class '_pytest.outcomes.Exit'>, <class 'KeyboardInterrupt'>)\n\n    @classmethod\n    def from_call(\n        cls,\n        func: Callable[[], TResult],\n        when: Literal[\"collect\", \"setup\", \"call\", \"teardown\"],\n        reraise: Optional[\n            Union[Type[BaseException], Tuple[Type[BaseException], ...]]\n        ] = None,\n    ) -> \"CallInfo[TResult]\":\n        \"\"\"Call func, wrapping the result in a CallInfo.\n    \n        :param func:\n            The function to call. Called without arguments.\n        :param when:\n            The phase in which the function is called.\n        :param reraise:\n            Exception or exceptions that shall propagate if raised by the\n            function, instead of being wrapped in the CallInfo.\n        \"\"\"\n        excinfo = None\n        start = timing.time()\n        precise_start = timing.perf_counter()\n        try:\n>           result: Optional[TResult] = func()\n\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/_pytest/runner.py:341: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/_pytest/runner.py:241: in <lambda>\n    lambda: runtest_hook(item=item, **kwds), when=when, reraise=reraise\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/pluggy/_hooks.py:513: in __call__\n    return self._hookexec(self.name, self._hookimpls.copy(), kwargs, firstresult)\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/pluggy/_manager.py:120: in _hookexec\n    return self._inner_hookexec(hook_name, methods, kwargs, firstresult)\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/_pytest/threadexception.py:87: in pytest_runtest_call\n    yield from thread_exception_runtest_hook()\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/_pytest/threadexception.py:63: in thread_exception_runtest_hook\n    yield\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/_pytest/unraisableexception.py:90: in pytest_runtest_call\n    yield from unraisable_exception_runtest_hook()\n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\n    def unraisable_exception_runtest_hook() -> Generator[None, None, None]:\n        with catch_unraisable_exception() as cm:\n            try:\n                yield\n            finally:\n                if cm.unraisable:\n                    if cm.unraisable.err_msg is not None:\n                        err_msg = cm.unraisable.err_msg\n                    else:\n                        err_msg = \"Exception ignored in\"\n                    msg = f\"{err_msg}: {cm.unraisable.object!r}\\n\\n\"\n                    msg += \"\".join(\n                        traceback.format_exception(\n                            cm.unraisable.exc_type,\n                            cm.unraisable.exc_value,\n                            cm.unraisable.exc_traceback,\n                        )\n                    )\n>                   warnings.warn(pytest.PytestUnraisableExceptionWarning(msg))\nE                   pytest.PytestUnraisableExceptionWarning: Exception ignored in: <coroutine object Queue._notify_async_not_full.<locals>.f at 0x7ba449ed3ec0>\nE                   \nE                   Traceback (most recent call last):\nE                     File \"/home/user/anaconda3/envs/py39/lib/python3.9/warnings.py\", line 506, in _warn_unawaited_coroutine\nE                       warn(msg, category=RuntimeWarning, stacklevel=2, source=coro)\nE                   RuntimeWarning: coroutine 'Queue._notify_async_not_full.<locals>.f' was never awaited\n\n../../../../anaconda3/envs/py39/lib/python3.9/site-packages/_pytest/unraisableexception.py:80: PytestUnraisableExceptionWarning\n------------------------------------------------------------------------------------ Captured log call -------------------------------------------------------------------------------------\nERROR    asyncio:base_events.py:1753 Task was destroyed but it is pending!\ntask: <Task pending name='Task-897' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nERROR    asyncio:base_events.py:1753 Task was destroyed but it is pending!\ntask: <Task pending name='Task-898' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nERROR    asyncio:base_events.py:1753 Task was destroyed but it is pending!\ntask: <Task pending name='Task-899' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nERROR    asyncio:base_events.py:1753 Task was destroyed but it is pending!\ntask: <Task pending name='Task-900' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nERROR    asyncio:base_events.py:1753 Task was destroyed but it is pending!\ntask: <Task pending name='Task-901' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nERROR    asyncio:base_events.py:1753 Task was destroyed but it is pending!\ntask: <Task pending name='Task-902' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\n_____________________________________________________________________________ TestLifoQueue.test_simple_queue ______________________________________________________________________________\n\nself = <test_sync.TestLifoQueue object at 0x7ba449e76f70>\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\ntests/test_sync.py:221: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\nself = <test_sync.TestLifoQueue object at 0x7ba449e76f70>, _q = <janus.LifoQueue object at 0x7ba448548e80>\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__]\nE       KeyError: <class 'janus.LifoQueue'>\n\ntests/test_sync.py:115: KeyError\n___________________________________________________________________________ TestPriorityQueue.test_simple_queue ____________________________________________________________________________\n\nself = <test_sync.TestPriorityQueue object at 0x7ba449e987f0>\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\ntests/test_sync.py:221: \n_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ \n\nself = <test_sync.TestPriorityQueue object at 0x7ba449e987f0>, _q = <janus.PriorityQueue object at 0x7ba449e20220>\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__]\nE       KeyError: <class 'janus.PriorityQueue'>\n\ntests/test_sync.py:115: KeyError\n================================================================================= short test summary info ==================================================================================\nFAILED tests/test_sync.py::TestQueue::test_simple_queue - KeyError: <class 'janus.Queue'>\nFAILED tests/test_sync.py::TestLifoQueue::test_queue_join - pytest.PytestUnraisableExceptionWarning: Exception ignored in: <coroutine object Queue._notify_async_not_full.<locals>.f at 0x7ba449ed3ec0>\nFAILED tests/test_sync.py::TestLifoQueue::test_simple_queue - KeyError: <class 'janus.LifoQueue'>\nFAILED tests/test_sync.py::TestPriorityQueue::test_simple_queue - KeyError: <class 'janus.PriorityQueue'>\n=============================================================================== 4 failed, 68 passed in 2.27s ===============================================================================\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1331' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1332' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1333' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1334' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1335' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1336' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nsys:1: RuntimeWarning: coroutine 'Queue._notify_async_not_empty.<locals>.f' was never awaited\nRuntimeWarning: Enable tracemalloc to get the object allocation traceback\nsys:1: RuntimeWarning: coroutine 'Queue._notify_async_not_full.<locals>.f' was never awaited\nRuntimeWarning: Enable tracemalloc to get the object allocation traceback\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1765' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1766' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1767' coro=<Queue._notify_async_not_empty.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:209> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1768' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1769' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>\nTask was destroyed but it is pending!\ntask: <Task pending name='Task-1770' coro=<Queue._notify_async_not_full.<locals>.f() running at /home/user/Documents/repoben/buggycode/aio-libs_janus/janus/__init__.py:224> cb=[set.discard()]>",
    "CodeBase": [
      {
        "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 blocking functions --\n35         # they should never actually wait that long - they should make\n36         # progress as soon as we call self.fn()).\n37         time.sleep(0.1)\n38         self.startedEvent.set()\n39         self.fn(*self.args)\n40 \n41 \n42 # Execute a function that blocks, and in a separate thread, a function that\n43 # triggers the release.  Returns the result of the blocking function.  Caution:\n44 # block_func must guarantee to block until trigger_func is called, a(...truncated)"
      },
      {
        "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 wait(...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_done(se(...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.mark.as(...truncated)"
      }
    ],
    "CommitSHA": "e46a1dfa166ebc2484716ee36cb99763f6247d52"
  },
  "Score": {
    "Difficulty": "Medium",
    "issue_origin": {
      "Title": 7,
      "Description": 6,
      "Reproducibility": 6,
      "Relevance": 7,
      "Explanation": 6,
      "Overall": 6
    },
    "issue_message": {
      "Title": 8,
      "Description": 6,
      "Reproducibility": 5,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_ground": {
      "Title": 9,
      "Description": 8,
      "Reproducibility": 8,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Inconsistent Type Handling and Unresponsive Task Cancellation in Queue Tests",
      "description": "The following issues have been identified in the `janus` testing suite:\n\n1. **Inconsistent Type Handling in Simple Queue Tests**:\n    - In `test_sync.py`, when verifying the order of items in different queue types, the current implementation uses `_q.__class__`, comparing it against `target_order`. This can lead to issues if the class representation changes or differs subtly. It should use the class name instead for a more consistent comparison.\n    \n2. **Async Task Cancellation Not Properly Integrated**:\n    - In `test_mixed.py`, during the `wait_for_empty_queue()` async function, when it finishes joining on the async queue, there might be lingering tasks that were supposed to be cancelled but weren't. This can lead to unresponsive or hanging tests as the expected cancellation does not occur.\n    \nThe above issues affect the maintainability and reliability of the testing suite. Ensuring consistent and type-safe comparisons will prevent subtle bugs from creeping in, while proper cancellation of tasks ensures that tests do not hang indefinitely, improving the reliability of the CI pipeline.",
      "explanation": "### Summary of the Issue\nThe issue in the `janus` testing suite pertains to two primary problems:\n1. **Inconsistent Type Handling in Simple Queue Tests**: In `test_sync.py`, comparisons are done using the class representation (`_q.__class__`), which can lead to inconsistencies if the class representation changes subtly. The correct approach should be to use the class name for comparison.\n2. **Async Task Cancellation Not Properly Integrated**: In `test_mixed.py`, during the `wait_for_empty_queue()` function, there's a possibility of tasks lingering after their supposed cancellation. This can lead to unresponsive or hanging tests because the expected task cancellations do not occur as they should.\n\n### Explanation of the Commit\nThe commit made changes in two files: `test_sync.py` and `test_mixed.py`.\n\n#### Changes in `test_sync.py`\nThe commit modified the comparisons in the simple queue tests from using the class representation to using the class name:\n- Before the change, the comparison used `_q.__class__` to compare against `target_order`.\n- After the change, it uses `_q.__class__.__name__`, which is more stable as it compares the actual class name as a string.\n\n*Explanation*: By using the class name instead of the class representation, we avoid potential issues caused by subtle differences or changes in how the class might be represented internally. This ensures more consistent and predictable behavior in the tests.\n\n#### Changes in `test_mixed.py`\nThe commit added a task cancellation step in the `wait_for_empty_queue()` function:\n- A `task.cancel()` call was added to ensure that all tasks are properly cancelled when the async queue has finished joining.\n\n*Explanation*: Adding the `task.cancel()` call ensures that any lingering tasks are properly cancelled. This prevents the tests from hanging or becoming unresponsive, leading to more reliable test execution and improved maintainability.\n\n### Detailed Explanation of the Issues and Solutions\n#### Inconsistent Type Handling in Simple Queue Tests\n**Issue**: \n- The test was using `_q.__class__` to verify the order of items in different queue types against a `target_order`. This is problematic as class representations can change and might not always be consistent, leading to potential mismatches and false test failures.\n\n**Solution**:\n- The commit corrected this by using `_q.__class__.__name__`, which directly compares the class names as strings. This is a more consistent and reliable way to verify the type, as class names as strings are less likely to have subtle changes compared to class representations.\n\n**Benefit**: \n- This change improves the robustness of the tests, reducing the likelihood of encountering errors due to class representation issues and ensuring that the comparisons are done in a type-safe manner.\n\n#### Async Task Cancellation Not Properly Integrated\n**Issue**:\n- During the execution of `wait_for_empty_queue()` in `test_mixed.py`, some tasks that are supposed to be cancelled might not be, potentially leading to unresponsive or hanging tests. This happens because there was no explicit cancellation call for tasks that might remain after the queue operation is completed.\n\n**Solution**:\n- The commit introduced a `task.cancel()` call to ensure that any tasks that linger after the async queue operations are explicitly cancelled.\n\n**Benefit**: \n- Adding this explicit cancellation helps in cleaning up tasks properly, preventing them from hanging or causing the test suite to be unresponsive. This improves test reliability and the stability of the CI pipeline.\n\n### Conclusion\n- The inconsistencies in type handling were resolved by ensuring the tests compared class names as strings, providing more stable and predictable comparisons.\n- The task cancellation logic was introduced to cleanly cancel any remaining tasks, preventing potential hangs and improving test execution reliability.\n\nThese changes together enhance the maintainability and reliability of the `janus` testing suite, ensuring more accurate and consistent test outcomes while preventing potential issues related to unresponsive tasks."
    }
  }
}