{
  "RepoName": "https://github.com/aio-libs/janus.git",
  "CommitSHA": "e46a1dfa166ebc2484716ee36cb99763f6247d52",
  "Time": "",
  "Difficulty": "Medium",
  "Type": "logic error",
  "BuggyCode": [
    {
      "path": "aio-libs_janus/setup.py",
      "content": "from setuptools import setup\n\nsetup()\n"
    },
    {
      "path": "aio-libs_janus/tests/test_sync.py",
      "content": "# Some simple queue module tests, plus some failure conditions\n# to ensure the Queue locks remain stable.\nimport asyncio\nimport queue\nimport threading\nimport time\nfrom unittest.mock import patch\n\nimport pytest\n\nimport janus\n\nQUEUE_SIZE = 5\n\n\ndef qfull(q):\n    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n\n\n# A thread to run a function that unclogs a blocked Queue.\nclass _TriggerThread(threading.Thread):\n    def __init__(self, fn, args):\n        self.fn = fn\n        self.args = args\n        self.startedEvent = threading.Event()\n        threading.Thread.__init__(self)\n\n    def run(self):\n        # The sleep isn't necessary, but is intended to give the blocking\n        # function in the main thread a chance at actually blocking before\n        # we unclog it.  But if the sleep is longer than the timeout-based\n        # tests wait in their blocking functions, those tests will fail.\n        # So we give them much longer timeout values compared to the\n        # sleep here (I aimed at 10 seconds for blocking functions --\n        # they should never actually wait that long - they should make\n        # progress as soon as we call self.fn()).\n        time.sleep(0.1)\n        self.startedEvent.set()\n        self.fn(*self.args)\n\n\n# Execute a function that blocks, and in a separate thread, a function that\n# triggers the release.  Returns the result of the blocking function.  Caution:\n# block_func must guarantee to block until trigger_func is called, and\n# trigger_func must guarantee to change queue state so that block_func can make\n# enough progress to return.  In particular, a block_func that just raises an\n# exception regardless of whether trigger_func is called will lead to\n# timing-dependent sporadic failures, and one of those went rarely seen but\n# undiagnosed for years.  Now block_func must be unexceptional.  If block_func\n# is supposed to raise an exception, call do_exceptional_blocking_test()\n# instead.\n\n\nclass BlockingTestMixin:\n    def do_blocking_test(self, block_func, block_args, trigger_func, trigger_args):\n        self.t = _TriggerThread(trigger_func, trigger_args)\n        self.t.start()\n        self.result = block_func(*block_args)\n        # If block_func returned before our thread made the call, we failed!\n        if not self.t.startedEvent.is_set():\n            pytest.fail(\"blocking function '%r' appeared not to block\" % block_func)\n        self.t.join(10)  # make sure the thread terminates\n        if self.t.is_alive():\n            pytest.fail(\"trigger function '%r' appeared to not return\" % trigger_func)\n        return self.result\n\n    # Call this instead if block_func is supposed to raise an exception.\n    def do_exceptional_blocking_test(\n        self,\n        block_func,\n        block_args,\n        trigger_func,\n        trigger_args,\n        expected_exception_class,\n    ):\n        self.t = _TriggerThread(trigger_func, trigger_args)\n        self.t.start()\n        try:\n            try:\n                block_func(*block_args)\n            except expected_exception_class:\n                raise\n            else:\n                pytest.fail(\"expected exception of kind %r\" % expected_exception_class)\n        finally:\n            self.t.join(10)  # make sure the thread terminates\n            if self.t.is_alive():\n                pytest.fail(\n                    \"trigger function '%r' appeared to not return\" % trigger_func\n                )\n            if not self.t.startedEvent.is_set():\n                pytest.fail(\"trigger thread ended but event never set\")\n\n\nclass BaseQueueTestMixin(BlockingTestMixin):\n    cum = 0\n    cumlock = threading.Lock()\n\n    def simple_queue_test(self, _q):\n        q = _q.sync_q\n        if q.qsize():\n            raise RuntimeError(\"Call this function with an empty queue\")\n        assert q.empty()\n        assert not q.full()\n        # I guess we better check things actually queue correctly a little :)\n        q.put(111)\n        q.put(333)\n        q.put(222)\n        target_order = dict(\n            Queue=[111, 333, 222],\n            LifoQueue=[222, 333, 111],\n            PriorityQueue=[111, 222, 333],\n        )\n        actual_order = [q.get(), q.get(), q.get()]\n        assert actual_order == target_order[_q.__class__.__name__]\n        for i in range(QUEUE_SIZE - 1):\n            q.put(i)\n            assert q.qsize()\n        assert not qfull(q)\n        last = 2 * QUEUE_SIZE\n        full = 3 * 2 * QUEUE_SIZE\n        q.put(last)\n        assert qfull(q)\n        assert not q.empty()\n        assert q.full()\n        try:\n            q.put(full, block=0)\n            pytest.fail(\"Didn't appear to block with a full queue\")\n        except queue.Full:\n            pass\n        try:\n            q.put(full, timeout=0.01)\n            pytest.fail(\"Didn't appear to time-out with a full queue\")\n        except queue.Full:\n            pass\n        # Test a blocking put\n        self.do_blocking_test(q.put, (full,), q.get, ())\n        self.do_blocking_test(q.put, (full, True, 10), q.get, ())\n        # Empty it\n        for i in range(QUEUE_SIZE):\n            q.get()\n        assert not q.qsize()\n        try:\n            q.get(block=0)\n            pytest.fail(\"Didn't appear to block with an empty queue\")\n        except queue.Empty:\n            pass\n        try:\n            q.get(timeout=0.01)\n            pytest.fail(\"Didn't appear to time-out with an empty queue\")\n        except queue.Empty:\n            pass\n        # Test a blocking get\n        self.do_blocking_test(q.get, (), q.put, (\"empty\",))\n        self.do_blocking_test(q.get, (True, 10), q.put, (\"empty\",))\n\n    def worker(self, q):\n        try:\n            while True:\n                x = q.get()\n                if x < 0:\n                    q.task_done()\n                    return\n                with self.cumlock:\n                    self.cum += x\n                q.task_done()\n        except Exception as ex:\n            from traceback import print_exc\n\n            print_exc(ex)\n\n    def queue_join_test(self, q):\n        self.cum = 0\n        for i in (0, 1):\n            threading.Thread(target=self.worker, args=(q,)).start()\n        for i in range(100):\n            q.put(i)\n        q.join()\n        assert self.cum == sum(range(100))\n        for i in (0, 1):\n            q.put(-1)  # instruct the threads to close\n        q.join()  # verify that you can join twice\n\n    @pytest.mark.asyncio\n    async def test_queue_task_done(self):\n        # Test to make sure a queue task completed successfully.\n        _q = self.type2test()\n        q = _q.sync_q\n        try:\n            q.task_done()\n        except ValueError:\n            pass\n        else:\n            pytest.fail(\"Did not detect task count going negative\")\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_queue_join(self):\n        # Test that a queue join()s successfully, and before anything else\n        # (done twice for insurance).\n        _q = self.type2test()\n        q = _q.sync_q\n        self.queue_join_test(q)\n        self.queue_join_test(q)\n        try:\n            q.task_done()\n        except ValueError:\n            pass\n        else:\n            pytest.fail(\"Did not detect task count going negative\")\n        finally:\n            _q.close()\n            await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_simple_queue(self):\n        # Do it a couple of times on the same queue.\n        # Done twice to make sure works with same instance reused.\n        _q = self.type2test(QUEUE_SIZE)\n        self.simple_queue_test(_q)\n        self.simple_queue_test(_q)\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_negative_timeout_raises_exception(self):\n        _q = self.type2test(QUEUE_SIZE)\n        q = _q.sync_q\n        with pytest.raises(ValueError):\n            q.put(1, timeout=-1)\n        with pytest.raises(ValueError):\n            q.get(1, timeout=-1)\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nowait(self):\n        _q = self.type2test(QUEUE_SIZE)\n        q = _q.sync_q\n        for i in range(QUEUE_SIZE):\n            q.put_nowait(1)\n        with pytest.raises(queue.Full):\n            q.put_nowait(1)\n\n        for i in range(QUEUE_SIZE):\n            q.get_nowait()\n        with pytest.raises(queue.Empty):\n            q.get_nowait()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_shrinking_queue(self):\n        # issue 10110\n        _q = self.type2test(3)\n        q = _q.sync_q\n        q.put(1)\n        q.put(2)\n        q.put(3)\n        with pytest.raises(queue.Full):\n            q.put_nowait(4)\n        assert q.qsize() == 3\n        q._maxsize = 2  # shrink the queue\n        with pytest.raises(queue.Full):\n            q.put_nowait(4)\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_maxsize(self):\n        # Test to make sure a queue task completed successfully.\n        _q = self.type2test(5)\n        q = _q.sync_q\n        assert q.maxsize == 5\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestQueue(BaseQueueTestMixin):\n    type2test = janus.Queue\n\n\nclass TestLifoQueue(BaseQueueTestMixin):\n    type2test = janus.LifoQueue\n\n\nclass TestPriorityQueue(BaseQueueTestMixin):\n    type2test = janus.PriorityQueue\n\n\n# A Queue subclass that can provoke failure at a moment's notice :)\nclass FailingQueueException(Exception):\n    pass\n\n\nclass FailingQueue(janus.Queue):\n    def __init__(self, *args, **kwargs):\n        self.fail_next_put = False\n        self.fail_next_get = False\n        super().__init__(*args, **kwargs)\n\n    def _put(self, item):\n        if self.fail_next_put:\n            self.fail_next_put = False\n            raise FailingQueueException(\"You Lose\")\n        return super()._put(item)\n\n    def _get(self):\n        if self.fail_next_get:\n            self.fail_next_get = False\n            raise FailingQueueException(\"You Lose\")\n        return super()._get()\n\n\nclass TestFailingQueue(BlockingTestMixin):\n    def failing_queue_test(self, _q):\n        q = _q.sync_q\n        if q.qsize():\n            raise RuntimeError(\"Call this function with an empty queue\")\n        for i in range(QUEUE_SIZE - 1):\n            q.put(i)\n        # Test a failing non-blocking put.\n        _q.fail_next_put = True\n        try:\n            q.put(\"oops\", block=0)\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        _q.fail_next_put = True\n        try:\n            q.put(\"oops\", timeout=0.1)\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        q.put(\"last\")\n        assert qfull(q)\n        # Test a failing blocking put\n        _q.fail_next_put = True\n        try:\n            self.do_blocking_test(q.put, (\"full\",), q.get, ())\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        # Check the Queue isn't damaged.\n        # put failed, but get succeeded - re-add\n        q.put(\"last\")\n        # Test a failing timeout put\n        _q.fail_next_put = True\n        try:\n            self.do_exceptional_blocking_test(\n                q.put, (\"full\", True, 10), q.get, (), FailingQueueException\n            )\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        # Check the Queue isn't damaged.\n        # put failed, but get succeeded - re-add\n        q.put(\"last\")\n        assert qfull(q)\n        q.get()\n        assert not qfull(q)\n        q.put(\"last\")\n        assert qfull(q)\n        # Test a blocking put\n        self.do_blocking_test(q.put, (\"full\",), q.get, ())\n        # Empty it\n        for i in range(QUEUE_SIZE):\n            q.get()\n        assert not q.qsize()\n        q.put(\"first\")\n        _q.fail_next_get = True\n        try:\n            q.get()\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        assert q.qsize()\n        _q.fail_next_get = True\n        try:\n            q.get(timeout=0.1)\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        assert q.qsize()\n        q.get()\n        assert not q.qsize()\n        _q.fail_next_get = True\n        try:\n            self.do_exceptional_blocking_test(\n                q.get, (), q.put, (\"empty\",), FailingQueueException\n            )\n            pytest.fail(\"The queue didn't fail when it should have\")\n        except FailingQueueException:\n            pass\n        # put succeeded, but get failed.\n        assert q.qsize()\n        q.get()\n        assert not q.qsize()\n\n    @pytest.mark.asyncio\n    async def test_failing_queue(self):\n        # Test to make sure a queue is functioning correctly.\n        # Done twice to the same instance.\n        q = FailingQueue(QUEUE_SIZE)\n        self.failing_queue_test(q)\n        self.failing_queue_test(q)\n        q.close()\n        await q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_closed_loop_non_failing(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(QUEUE_SIZE)\n        q = _q.sync_q\n        # we are pacthing loop to follow setUp/tearDown agreement\n        with patch.object(loop, \"call_soon_threadsafe\") as func:\n            func.side_effect = RuntimeError()\n            q.put_nowait(1)\n            assert func.call_count == 1\n        _q.close()\n        await _q.wait_closed()\n"
    },
    {
      "path": "aio-libs_janus/tests/test_async.py",
      "content": "\"\"\"Tests for queues.py\"\"\"\n\nimport asyncio\n\nimport pytest\n\nimport janus\n\n\nclass TestQueueBasic:\n    async def _test_repr_or_str(self, fn, expect_id):\n        \"\"\"Test Queue's repr or str.\n\n        fn is repr or str. expect_id is True if we expect the Queue's id to\n        appear in fn(Queue()).\n        \"\"\"\n\n        _q = janus.Queue()\n        q = _q.async_q\n        assert fn(q).startswith(\"<Queue\")\n        id_is_present = hex(id(q)) in fn(q)\n        assert expect_id == id_is_present\n        loop = asyncio.get_running_loop()\n\n        async def add_getter():\n            _q = janus.Queue()\n            q = _q.async_q\n            # Start a task that waits to get.\n            loop.create_task(q.get())\n            # Let it start waiting.\n            await asyncio.sleep(0.1)\n            assert \"_getters[1]\" in fn(q)\n            # resume q.get coroutine to finish generator\n            q.put_nowait(0)\n\n        await add_getter()\n\n        async def add_putter():\n            _q = janus.Queue(maxsize=1)\n            q = _q.async_q\n            q.put_nowait(1)\n            # Start a task that waits to put.\n            loop.create_task(q.put(2))\n            # Let it start waiting.\n            await asyncio.sleep(0.1)\n            assert \"_putters[1]\" in fn(q)\n            # resume q.put coroutine to finish generator\n            q.get_nowait()\n\n        await add_putter()\n\n        _q = janus.Queue()\n        q = _q.async_q\n        q.put_nowait(1)\n        assert \"_queue=[1]\" in fn(q)\n\n    # def test_repr(self):\n    #     self._test_repr_or_str(repr, True)\n\n    # def test_str(self):\n    #     self._test_repr_or_str(str, False)\n\n    @pytest.mark.asyncio\n    async def test_empty(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        assert q.empty()\n        q.put_nowait(1)\n        assert not q.empty()\n        assert 1 == q.get_nowait()\n        assert q.empty()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_full(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        assert not q.full()\n\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n        q.put_nowait(1)\n        assert q.full()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_order(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        for i in [1, 3, 2]:\n            q.put_nowait(i)\n\n        items = [q.get_nowait() for _ in range(3)]\n        assert [1, 3, 2] == items\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_maxsize(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=2)\n        q = _q.async_q\n        assert 2 == q.maxsize\n        have_been_put = []\n\n        fut = loop.create_future()\n\n        async def putter():\n            for i in range(3):\n                await q.put(i)\n                have_been_put.append(i)\n                if i == q.maxsize - 1:\n                    fut.set_result(None)\n            return True\n\n        async def test():\n            t = loop.create_task(putter())\n            await fut\n\n            # The putter is blocked after putting two items.\n            assert [0, 1] == have_been_put\n            assert 0 == q.get_nowait()\n\n            # Let the putter resume and put last item.\n            await t\n            assert [0, 1, 2] == have_been_put\n            assert 1 == q.get_nowait()\n            assert 2 == q.get_nowait()\n\n        await test()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestQueueGetTests:\n    @pytest.mark.asyncio\n    async def test_blocking_get(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        q.put_nowait(1)\n\n        res = await q.get()\n        assert 1 == res\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_get_with_putters(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(1)\n        q = _q.async_q\n        q.put_nowait(1)\n\n        fut = loop.create_future()\n\n        async def put():\n            t = asyncio.ensure_future(q.put(2))\n            await asyncio.sleep(0.01)\n            fut.set_result(None)\n            return t\n\n        t = await put()\n\n        res = await q.get()\n        assert 1 == res\n\n        await t\n        assert 1 == q.qsize()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_blocking_get_wait(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue()\n        q = _q.async_q\n        started = asyncio.Event()\n        finished = False\n\n        async def queue_get():\n            nonlocal finished\n            started.set()\n            res = await q.get()\n            finished = True\n            return res\n\n        async def queue_put():\n            loop.call_later(0.01, q.put_nowait, 1)\n            queue_get_task = loop.create_task(queue_get())\n            await started.wait()\n            assert not finished\n            res = await queue_get_task\n            assert finished\n            return res\n\n        res = await queue_put()\n        assert 1 == res\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nonblocking_get(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        q.put_nowait(1)\n        assert 1 == q.get_nowait()\n\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nonblocking_get_exception(self):\n        _q = janus.Queue()\n        pytest.raises(asyncio.QueueEmpty, _q.async_q.get_nowait)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_get_cancelled(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue()\n        q = _q.async_q\n\n        async def queue_get():\n            return await asyncio.wait_for(q.get(), 0.051)\n\n        async def test():\n            get_task = loop.create_task(queue_get())\n            await asyncio.sleep(0.01)  # let the task start\n            q.put_nowait(1)\n            return await get_task\n\n        assert 1 == await test()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_get_cancelled_race(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue()\n        q = _q.async_q\n\n        f1 = loop.create_future()\n\n        async def g1():\n            f1.set_result(None)\n            await q.get()\n\n        t1 = loop.create_task(g1())\n        t2 = loop.create_task(q.get())\n\n        await f1\n        await asyncio.sleep(0.01)\n        t1.cancel()\n\n        with pytest.raises(asyncio.CancelledError):\n            await t1\n        assert t1.done()\n        q.put_nowait(\"a\")\n\n        await t2\n        assert t2.result() == \"a\"\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_get_with_waiting_putters(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n\n        loop.create_task(q.put(\"a\"))\n        loop.create_task(q.put(\"b\"))\n\n        await asyncio.sleep(0.01)\n\n        assert await q.get() == \"a\"\n        assert await q.get() == \"b\"\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n\nclass TestQueuePut:\n    @pytest.mark.asyncio\n    async def test_blocking_put(self):\n        _q = janus.Queue()\n        q = _q.async_q\n\n        # No maxsize, won't block.\n        await q.put(1)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_blocking_put_wait(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n        started = asyncio.Event()\n        finished = False\n\n        async def queue_put():\n            nonlocal finished\n            started.set()\n            await q.put(1)\n            await q.put(2)\n            finished = True\n\n        async def queue_get():\n            loop.call_later(0.01, q.get_nowait)\n            queue_put_task = loop.create_task(queue_put())\n            await started.wait()\n            assert not finished\n            await queue_put_task\n            assert finished\n\n        await queue_get()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nonblocking_put(self):\n        _q = janus.Queue()\n        q = _q.async_q\n        q.put_nowait(1)\n        assert 1 == q.get_nowait()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_nonblocking_put_exception(self):\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n        q.put_nowait(1)\n        pytest.raises(asyncio.QueueFull, q.put_nowait, 2)\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_float_maxsize(self):\n        _q = janus.Queue(maxsize=1.3)\n        q = _q.async_q\n        q.put_nowait(1)\n        q.put_nowait(2)\n        assert q.full()\n        pytest.raises(asyncio.QueueFull, q.put_nowait, 3)\n\n        _q.close()\n        await _q.wait_closed()\n\n        _q = janus.Queue(maxsize=1.3)\n        q = _q.async_q\n\n        async def queue_put():\n            await q.put(1)\n            await q.put(2)\n            assert q.full()\n\n        await queue_put()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_put_cancelled(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue()\n        q = _q.async_q\n\n        async def queue_put():\n            await q.put(1)\n            return True\n\n        async def test():\n            return await q.get()\n\n        t = loop.create_task(queue_put())\n        assert 1 == await test()\n        assert t.done()\n        assert t.result()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_put_cancelled_race(self):\n        loop = asyncio.get_running_loop()\n        _q = janus.Queue(maxsize=1)\n        q = _q.async_q\n\n        put_a = loop.create_task(q.put(\"a\"))\n        put_b = loop.create_task(q.put(\"b\"))\n        put_c = loop.create_task(q.put(\"X\"))\n\n        await put_a\n        assert not put_b.done()\n\n        put_c.cancel()\n\n        with pytest.raises(asyncio.CancelledError):\n            await put_c\n\n        async def go():\n            a = await q.get()\n            assert a == \"a\"\n            b = await q.get()\n            assert b == \"b\"\n            assert put_b.done()\n\n        await go()\n\n        assert not _q._sync_mutex.locked()\n        _q.close()\n        await _q.wait_closed()\n\n    @pytest.mark.asyncio\n    async def test_put_with_waiting_getters(self):\n        loop = asyncio.get_running_loop()\n        fut = loop.create_future()\n\n        async def go():\n            fut.set_result(None)\n            ret = await q.get()\n            return ret\n\n        async def put():\n            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\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________________________________________________________________________ TestMixedMode.test_close_after_async_join _________________________________________________________________________\n\nself = <test_mixed.TestMixedMode object at 0x7ceecb0e3f70>\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)\nE           Failed: DID NOT RAISE <class 'RuntimeError'>\n\ntests/test_mixed.py:255: Failed\n================================================================================= short test summary info ==================================================================================\nFAILED tests/test_mixed.py::TestMixedMode::test_close_after_async_join - Failed: DID NOT RAISE <class 'RuntimeError'>\n=============================================================================== 1 failed, 71 passed in 4.80s ===============================================================================",
  "Patch": "--- 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@@ -245,6 +246,7 @@\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",
  "BuggyCodeLocation": [
    {
      "file": "aio-libs_janus/tests/test_mixed.py",
      "function": null,
      "content_all": {},
      "content_change": {}
    }
  ],
  "Issue": {
    "title": "Tasks not getting canceled correctly and incorrect queue state handling",
    "description": "There are two prominent issues that need attention:\n\n1. **Async Task Not Getting Canceled Properly**:\n   In some scenarios, tasks placed in the queue are not getting canceled correctly. When using `q.async_q.join()`, the tasks remain in a blocking state if certain conditions are not met, leading to potential deadlock situations or unresponsive behavior. Proper cancellation of these tasks is crucial to ensure the queue can be managed effectively without leaving orphaned tasks running indefinitely.\n\n2. **Improper Handling of Queue State in Test Scenarios**:\n   The tests for queue-related functionalities sometimes do not reflect the actual state changes, especially when operations like `put` are used on synchronous queues within asynchronous tests. For instance, using `q.sync_q.put(1)` before `asyncio.ensure_future(q.async_q.join())` helps simulate a more realistic blocking condition, helping to ensure that the queue’s empty or full state is managed accurately across async and sync contexts.\n\nBoth issues can lead to inconsistent behavior in production scenarios where managing the queue's state accurately is critical for operations.",
    "explanation": "### Summary of the Issue\n\nThe issue comprises two distinct but related problems in an asynchronous and synchronous mixed queueing system:\n\n1. **Async Task Not Getting Canceled Properly**: Certain asynchronous tasks placed in the queue are not being canceled correctly. This can cause tasks to remain in a blocking state, potentially leading to deadlocks or unresponsive behavior. Proper task cancellation is essential to maintain queue stability and prevent orphaned tasks from running indefinitely.\n\n2. **Improper Handling of Queue State in Test Scenarios**: Some test cases for the queue functionalities do not accurately reflect the real state changes, especially when switching between synchronous and asynchronous operations. Specifically, there are problems with simulating realistic blocking conditions which are necessary to test the queue’s behavior correctly under different states (empty or full).\n\n### Analysis of the Commit\n\nThe commits made to address the issues involve changes to the test files, mainly `test_mixed.py`. Let's break down these changes and understand how they address the identified problems.\n\n1. **Cancellation of Async Tasks**:\n   - **Issue**: Tasks that were awaiting completion via `q.async_q.join()` were not being canceled correctly, leaving them in a blocking state.\n   - **Solution**: The commit adds a `task.cancel()` call after the `await q.async_q.join()` statement in the `wait_for_empty_queue` function. This ensures that the async task is appropriately canceled after the queue work is completed.\n\n2. **Handling of Queue State in Test Scenarios**:\n   - **Issue**: In some tests, operations like `put` were not being correctly positioned within the asynchronous tests, preventing accurate simulation of queue states.\n   - **Solution**: The commit ensures that `put` operations are performed on the synchronous queue before starting the async join operation. For instance, adding `q.sync_q.put(1)` ensures that there is an entry in the queue, which helps in creating a more realistic blocking condition when `asyncio.ensure_future(q.async_q.join())` is called.\n\n### Explanation of the Solution\n\n1. **Task Cancellation**:\n   - **Problem**: In a scenario where async tasks join the queue, the absence of explicit cancellation could result in tasks being left unattended in a waiting state. This can lead to deadlocks or unresponsive applications because the tasks are indefinitely suspending the execution flow.\n   - **Commit**: By explicitly calling `task.cancel()`, the developer has ensured that any remaining active tasks are forcefully canceled after their intended operations are completed. This prevents any potential blocking caused by unfinished tasks.\n\n2. **Queue State Simulation in Tests**:\n   - **Problem**: Testing the state changes in mixed synchronous and asynchronous environments can be tricky. If the synchronous operations (such as `put`) are not executed at the right moment, it can lead to inaccurate test scenarios that do not reflect real-world usage, where queues might have items and consumers waiting on both sync and async sides.\n   - **Commit**: Adjusting the sequence by adding `q.sync_q.put(1)` before asynchronous joining operation (`asyncio.ensure_future(q.async_q.join())`) ensures that tests are more reflective of actual queue states. This helps in simulating more accurate blocking conditions, which is crucial for robust and reliable testing of queue behavior under different states and usage patterns.\n\n### Conclusion\n\nThe primary issue was twofold: ensuring proper cancellation of async tasks and accurate handling of queue states in test scenarios. The commit effectively resolved these issues by explicitly canceling async tasks to prevent deadlocks and adjusting the order of operations in tests to better simulate real-world conditions. This ensures that the queue system operates reliably both in production and under test conditions, maintaining its integrity and responsiveness without unintended blocking or state inaccuracies."
  },
  "Explain": "### Summary of the Issue\n\nThe issue comprises two distinct but related problems in an asynchronous and synchronous mixed queueing system:\n\n1. **Async Task Not Getting Canceled Properly**: Certain asynchronous tasks placed in the queue are not being canceled correctly. This can cause tasks to remain in a blocking state, potentially leading to deadlocks or unresponsive behavior. Proper task cancellation is essential to maintain queue stability and prevent orphaned tasks from running indefinitely.\n\n2. **Improper Handling of Queue State in Test Scenarios**: Some test cases for the queue functionalities do not accurately reflect the real state changes, especially when switching between synchronous and asynchronous operations. Specifically, there are problems with simulating realistic blocking conditions which are necessary to test the queue’s behavior correctly under different states (empty or full).\n\n### Analysis of the Commit\n\nThe commits made to address the issues involve changes to the test files, mainly `test_mixed.py`. Let's break down these changes and understand how they address the identified problems.\n\n1. **Cancellation of Async Tasks**:\n   - **Issue**: Tasks that were awaiting completion via `q.async_q.join()` were not being canceled correctly, leaving them in a blocking state.\n   - **Solution**: The commit adds a `task.cancel()` call after the `await q.async_q.join()` statement in the `wait_for_empty_queue` function. This ensures that the async task is appropriately canceled after the queue work is completed.\n\n2. **Handling of Queue State in Test Scenarios**:\n   - **Issue**: In some tests, operations like `put` were not being correctly positioned within the asynchronous tests, preventing accurate simulation of queue states.\n   - **Solution**: The commit ensures that `put` operations are performed on the synchronous queue before starting the async join operation. For instance, adding `q.sync_q.put(1)` ensures that there is an entry in the queue, which helps in creating a more realistic blocking condition when `asyncio.ensure_future(q.async_q.join())` is called.\n\n### Explanation of the Solution\n\n1. **Task Cancellation**:\n   - **Problem**: In a scenario where async tasks join the queue, the absence of explicit cancellation could result in tasks being left unattended in a waiting state. This can lead to deadlocks or unresponsive applications because the tasks are indefinitely suspending the execution flow.\n   - **Commit**: By explicitly calling `task.cancel()`, the developer has ensured that any remaining active tasks are forcefully canceled after their intended operations are completed. This prevents any potential blocking caused by unfinished tasks.\n\n2. **Queue State Simulation in Tests**:\n   - **Problem**: Testing the state changes in mixed synchronous and asynchronous environments can be tricky. If the synchronous operations (such as `put`) are not executed at the right moment, it can lead to inaccurate test scenarios that do not reflect real-world usage, where queues might have items and consumers waiting on both sync and async sides.\n   - **Commit**: Adjusting the sequence by adding `q.sync_q.put(1)` before asynchronous joining operation (`asyncio.ensure_future(q.async_q.join())`) ensures that tests are more reflective of actual queue states. This helps in simulating more accurate blocking conditions, which is crucial for robust and reliable testing of queue behavior under different states and usage patterns.\n\n### Conclusion\n\nThe primary issue was twofold: ensuring proper cancellation of async tasks and accurate handling of queue states in test scenarios. The commit effectively resolved these issues by explicitly canceling async tasks to prevent deadlocks and adjusting the order of operations in tests to better simulate real-world conditions. This ensures that the queue system operates reliably both in production and under test conditions, maintaining its integrity and responsiveness without unintended blocking or state inaccuracies.",
  "Source": "Human",
  "Token": 1150,
  "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_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.asyncio\n27     async def test_maxsize_named_param(self):\n28         q = janus.Queue(maxsize=7)\n29         assert 7 == q.maxsize\n30 \n31     @pytest.mark.asyncio\n32     async def test_maxsize_default(self):\n33         q = janus.Queue()\n34         assert 0 == q.maxsize\n35 \n36     @pytest.mark.asyncio\n37     async def test_unfinished(self):\n38         q = janus.Queue()\n39         assert q.sync_q.unfinished_tasks == 0\n40         assert q.async_q.unfinished_tasks == 0\n41         q.sync_q.put(1)\n42         assert q.sync_q.unfinished_tasks == 1\n43         assert q.async_q.unfinished_tasks == 1\n44         q.sync_q.get()\n45         assert q.(...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_do(...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.ge(...truncated)"
    }
  ],
  "TokenAll": 15097,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Potential use of private attributes in qfull function",
      "description": "The qfull function defined at line 16 in aio-libs_janus/tests/test_sync.py utilizes private attributes _parent and _maxsize of the queue object, which can lead to issues if the internal implementation of the queue changes. Consider using public methods or properties to achieve the same functionality.",
      "explanation": "### Issue Summary\nThe issue in question concerns a function `qfull` defined in the `aio-libs_janus/tests/test_sync.py` file. This function uses the private attributes `_parent` and `_maxsize` of a queue object. The use of private attributes poses a potential risk because any changes to the internal structure of the queue class could render this function inoperable. It is recommended to use public methods or properties to ensure greater stability and maintainability of the code.\n\n### Commit Content\nTo resolve this issue, the developer should replace the usage of private attributes with public properties or methods. Here's an example of what the commit might look like:\n\n```python\n16 def qfull(q):\n17     return q.maxsize > 0 and q.qsize() == q.maxsize\n```\n\nThis commit replaces the usage of `q._parent._maxsize` with `q.maxsize`, assuming that `maxsize` is a public property of the queue that returns the maximum allowed size.\n\n### Detailed Explanation\n1. **Understanding the Problem:**\n   - Private attributes (those that start with an underscore) are intended for internal use within a class and its subclasses.\n   - Using private attributes outside of the class they are defined in is generally a bad practice because these attributes are subject to change without notice. This could break the code that depends on them.\n   - In the `qfull` function defined at line 16 in `aio-libs_janus/tests/test_sync.py`, private attributes `_parent` and `_maxsize` are being accessed:\n     ```python\n     def qfull(q):\n         return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n     ```\n\n2. **The Proposed Commit:**\n   The commit changes the function to use a public property, `maxsize`, that should provide the necessary information.\n   ```python\n   def qfull(q):\n       return q.maxsize > 0 and q.qsize() == q.maxsize\n   ```\n\n3. **How the Commit Solves the Issue:**\n   - **Replacement of Private Attributes:**\n     - Instead of accessing the private `q._parent._maxsize`, the function now uses `q.maxsize`.\n     - This property `maxsize` is expected to provide the maximum size of the queue, serving the same purpose as the previous private attribute `_maxsize`.\n\n   - **Ensuring Stability and Maintainability:**\n     - By employing public properties, the function is more resilient to changes in the internal structure of the queue class.\n     - If the internal implementation changes but the public interface remains the same, the `qfull` function will continue to operate without modification.\n\n4. **Code Safety and Future-Proofing:**\n   - **Public API Usage:**\n     - Public attributes and methods are defined with the intent of being accessed and manipulated from outside the class. They are part of the class’s public interface and are more reliable for use in other parts of the code.\n     - This change adheres to the principle of encapsulation in object-oriented programming, which helps in maintaining the integrity of data and easing the maintainability of the code.\n\n   - **Mitigation of Risks:**\n     - Reducing dependency on internal attributes lowers the risk of future changes breaking the existing code, ensuring that any update to the internal implementation won't necessitate changes to the dependent code as long as the public API remains consistent.\n\nBy implementing this change, the `qfull` function becomes more stable and maintainable, reducing the likelihood of future breakages due to internal changes in the queue's implementation. This solution effectively resolves the issue while adhering to best practices in software development."
    },
    "issue_message": {
      "title": "Failed to Raise RuntimeError in test_close_after_async_join",
      "description": "In the test function `test_close_after_async_join` within `tests/test_mixed.py`, the code expects a `RuntimeError` to be raised when awaiting the result of `asyncio.wait_for`. However, the error is not being raised as expected, indicating a potential issue in the test logic or the behavior of the `janus.Queue` implementation.",
      "explanation": "## Summary of the Issue\n\nIn the provided test function `test_close_after_async_join` within the `tests/test_mixed.py`, the expectation is to receive a `RuntimeError` when awaiting the result of `asyncio.wait_for`. The specific steps taken in the test involve creating a `janus.Queue`, starting a task with `async_q.join()`, and then closing the queue. When awaiting the result of this join operation with a specified timeout using `asyncio.wait_for`, the test expects a `RuntimeError` to be raised. However, this error does not occur, indicating a potential problem either in the test logic or with the `janus.Queue` implementation.\n\n## Analysis of the Commit\n\n1. **Detailed Breakdown of the Commit:**\n\n   The fix ensures that the logic properly raises a `RuntimeError` when attempting to await the `asyncio.wait_for` on a join operation after the queue is closed.\n\n   ```python\n   @pytest.mark.asyncio\n   async def test_close_after_async_join(self):\n       q = janus.Queue()\n       # Ensuring that there is some task to join on.\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\n2. **Changes Made:**\n\n   - A line is added to put an item into the `sync_q` of the `janus.Queue`: `q.sync_q.put(1)`.\n\n## Explanation of How the Commit Solves the Issue\n\n1. **Understanding the Queue Behavior:**\n\n   - `janus.Queue` is a synchronized queue that bridges between async and sync contexts.\n   - The `join()` method on an async queue will block until all items in the queue have been processed.\n\n2. **Original Issue Root Cause:**\n\n   - In the original test function, there was no item put into the queue before calling `async_q.join()`. This means the join operation might complete immediately because there are no tasks to wait for.\n   - When the queue is closed without any pending tasks, awaiting `asyncio.wait_for` doesn’t face the scenario that should trigger a `RuntimeError`.\n\n3. **Commit Solution:**\n\n   - By adding `q.sync_q.put(1)`, an item is now placed into the synchronous part of the queue which then needs to be processed, causing the `join()` method to genuinely block.\n   - When `q.close()` is subsequently called, the queue is closed while there is still an unfinished task, making the state of the queue invalid for awaiting further `join` operations.\n   - This invalid state correctly triggers the `RuntimeError` that the test is trying to validate.\n\n## Conclusion\n\nThe commit adds an item to the queue before initiating the join operation. This ensures that the `join` method has an actual task to wait for, leading to the expected block. When the queue is then closed, this sets up the correct conditions for the `RuntimeError` to be raised, thereby aligning the test's expectation with the behavior being tested. This precisely addresses the issue observed where the error was not previously raised due to the absence of any pending tasks in the queue."
    },
    "issue_ground": {
      "title": "Tasks not getting canceled correctly and incorrect queue state handling",
      "description": "There are two prominent issues that need attention:\n\n1. **Async Task Not Getting Canceled Properly**:\n   In some scenarios, tasks placed in the queue are not getting canceled correctly. When using `q.async_q.join()`, the tasks remain in a blocking state if certain conditions are not met, leading to potential deadlock situations or unresponsive behavior. Proper cancellation of these tasks is crucial to ensure the queue can be managed effectively without leaving orphaned tasks running indefinitely.\n\n2. **Improper Handling of Queue State in Test Scenarios**:\n   The tests for queue-related functionalities sometimes do not reflect the actual state changes, especially when operations like `put` are used on synchronous queues within asynchronous tests. For instance, using `q.sync_q.put(1)` before `asyncio.ensure_future(q.async_q.join())` helps simulate a more realistic blocking condition, helping to ensure that the queue’s empty or full state is managed accurately across async and sync contexts.\n\nBoth issues can lead to inconsistent behavior in production scenarios where managing the queue's state accurately is critical for operations.",
      "explanation": "### Issue Summary\n\nThe issue reports two main problems in the `aio-libs/janus` library:\n\n1. **Async Task Not Getting Canceled Properly:**\n   - The tasks queued are not being properly canceled when required. Specifically, when using `q.async_q.join()`, some tasks remain in a blocking state, leading to potential deadlock situations.\n\n2. **Improper Handling of Queue State in Test Scenarios:**\n   - Test cases do not accurately reflect the state changes in the queue when it transitions between empty or full states, especially when mixing synchronous (`put`) and asynchronous (`join`) operations.\n\n### Analysis of Commit\n\nHere is a hypothetical commit that addresses these issues:\n\n```diff\ncommit abc123def\nAuthor: Developer <developer@example.com>\nDate:   Tue, 24 Oct 2023 17:58:00 +0000\n\n    Fix task cancellation and improve queue state handling in testing scenarios\n    \n    - Properly cancel async tasks in the queue and ensure they do not remain in a blocking state.\n    - Improve test cases to simulate realistic queue conditions by adding `sync_q.put` operations before `async_q.join`.\n\ndiff --git a/aio-libs_janus/janus/__init__.py b/aio-libs_janus/janus/__init__.py\nindex e234567..d345678 100644\n--- a/aio-libs_janus/janus/__init__.py\n+++ b/aio-libs_janus/janus/__init__.py\n@@ -100,6 +100,10 @@ class Queue(BaseQueue[T]):\n         self._unfinished_tasks = 0\n+    def close(self):\n+        with self._mutex:\n+            self._closed = True\n+            self._not_empty.notify_all()\n         .\n\n@@ -120,6 +124,10 @@ class Queue(BaseQueue[T]):\n         return self._not_full.wait(timeout)\n\n+    def cancel_join(self):\n+        with self._mutex:\n+            self._not_empty.notify_all()\n+\ndiff --git a/aio-libs_janus/tests/test_mixed.py b/aio-libs_janus/tests/test_mixed.py\nindex efg9999..hij0000 100644\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -254,8 +254,7 @@ class TestMixedMode:\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+        q.sync_q.put(1)\n         \n         task = asyncio.ensure_future(q.async_q.join())\n         await asyncio.sleep(0.1)\n\n         q.close()\n+        q.cancel_join()\n         with pytest.raises(RuntimeError):\n             with contextlib.suppress(asyncio.TimeoutError):\n                 await asyncio.wait_for(task, timeout=0.1)\n```\n\n### Explanation\n\n#### Cause of the Issue\n\n1. **Task Cancellation:**\n   - The `async_q.join()` is supposed to block until all tasks are processed. However, without proper task cancellation handling, tasks might remain in a blocking state. This results in potential deadlocks or unresponsive behaviors.\n\n2. **Queue State in Test Scenarios:**\n   - The tests did not properly simulate real-life conditions. Particularly, the absence of blocking conditions (`put` operations on synchronous queues) before calling `async_q.join` can result in inaccurate state changes, making the tests less reliable.\n\n#### Content of the Commit\n\n1. **Proper Task Cancellation:**\n   - Introduced `cancel_join` method in the `Queue` class. This method uses a notify-all mechanism to unblock any remaining tasks waiting in the queue by calling `_not_empty.notify_all()` within a locked context. Also, ensured that on queue closing (`close` method), the `cancel_join` would be invoked to notify all waiting tasks.\n\n2. **Improved Test Cases:**\n   - Uncommented and used `q.sync_q.put(1)` in `test_close_after_async_join` to simulate a more realistic blocking scenario where tasks are actually pending. This provides a better validation context for the `async_q.join` behavior under normal operation conditions.\n\n#### How the Commit Solves the Issue\n\n- **Task Cancellation:**\n  - By adding the `cancel_join` method and calling it within the `close` method, the commit ensures that all tasks waiting in `async_q.join()` are properly unblocked when a queue closure event occurs. This prevents tasks from lingering in a blocked state and avoids potential deadlocks.\n\n- **Test Adjustments:**\n  - The test `test_close_after_async_join` now simulates a realistic scenario where the synchronous queue (`sync_q`) has a task put in it before triggering an asynchronous join (`async_q.join`). This more accurate simulation ensures that the queue's state transition during mixed synchronous and asynchronous operations is reliable and accurately tested.\n\nIn summary, the commit makes"
    },
    "issue_ground_truth": {
      "title": "Tasks not getting canceled correctly and incorrect queue state handling",
      "description": "There are two prominent issues that need attention:\n\n1. **Async Task Not Getting Canceled Properly**:\n   In some scenarios, tasks placed in the queue are not getting canceled correctly. When using `q.async_q.join()`, the tasks remain in a blocking state if certain conditions are not met, leading to potential deadlock situations or unresponsive behavior. Proper cancellation of these tasks is crucial to ensure the queue can be managed effectively without leaving orphaned tasks running indefinitely.\n\n2. **Improper Handling of Queue State in Test Scenarios**:\n   The tests for queue-related functionalities sometimes do not reflect the actual state changes, especially when operations like `put` are used on synchronous queues within asynchronous tests. For instance, using `q.sync_q.put(1)` before `asyncio.ensure_future(q.async_q.join())` helps simulate a more realistic blocking condition, helping to ensure that the queue’s empty or full state is managed accurately across async and sync contexts.\n\nBoth issues can lead to inconsistent behavior in production scenarios where managing the queue's state accurately is critical for operations.",
      "explanation": "### Summary of the Issue\n\nThe issue comprises two distinct but related problems in an asynchronous and synchronous mixed queueing system:\n\n1. **Async Task Not Getting Canceled Properly**: Certain asynchronous tasks placed in the queue are not being canceled correctly. This can cause tasks to remain in a blocking state, potentially leading to deadlocks or unresponsive behavior. Proper task cancellation is essential to maintain queue stability and prevent orphaned tasks from running indefinitely.\n\n2. **Improper Handling of Queue State in Test Scenarios**: Some test cases for the queue functionalities do not accurately reflect the real state changes, especially when switching between synchronous and asynchronous operations. Specifically, there are problems with simulating realistic blocking conditions which are necessary to test the queue’s behavior correctly under different states (empty or full).\n\n### Analysis of the Commit\n\nThe commits made to address the issues involve changes to the test files, mainly `test_mixed.py`. Let's break down these changes and understand how they address the identified problems.\n\n1. **Cancellation of Async Tasks**:\n   - **Issue**: Tasks that were awaiting completion via `q.async_q.join()` were not being canceled correctly, leaving them in a blocking state.\n   - **Solution**: The commit adds a `task.cancel()` call after the `await q.async_q.join()` statement in the `wait_for_empty_queue` function. This ensures that the async task is appropriately canceled after the queue work is completed.\n\n2. **Handling of Queue State in Test Scenarios**:\n   - **Issue**: In some tests, operations like `put` were not being correctly positioned within the asynchronous tests, preventing accurate simulation of queue states.\n   - **Solution**: The commit ensures that `put` operations are performed on the synchronous queue before starting the async join operation. For instance, adding `q.sync_q.put(1)` ensures that there is an entry in the queue, which helps in creating a more realistic blocking condition when `asyncio.ensure_future(q.async_q.join())` is called.\n\n### Explanation of the Solution\n\n1. **Task Cancellation**:\n   - **Problem**: In a scenario where async tasks join the queue, the absence of explicit cancellation could result in tasks being left unattended in a waiting state. This can lead to deadlocks or unresponsive applications because the tasks are indefinitely suspending the execution flow.\n   - **Commit**: By explicitly calling `task.cancel()`, the developer has ensured that any remaining active tasks are forcefully canceled after their intended operations are completed. This prevents any potential blocking caused by unfinished tasks.\n\n2. **Queue State Simulation in Tests**:\n   - **Problem**: Testing the state changes in mixed synchronous and asynchronous environments can be tricky. If the synchronous operations (such as `put`) are not executed at the right moment, it can lead to inaccurate test scenarios that do not reflect real-world usage, where queues might have items and consumers waiting on both sync and async sides.\n   - **Commit**: Adjusting the sequence by adding `q.sync_q.put(1)` before asynchronous joining operation (`asyncio.ensure_future(q.async_q.join())`) ensures that tests are more reflective of actual queue states. This helps in simulating more accurate blocking conditions, which is crucial for robust and reliable testing of queue behavior under different states and usage patterns.\n\n### Conclusion\n\nThe primary issue was twofold: ensuring proper cancellation of async tasks and accurate handling of queue states in test scenarios. The commit effectively resolved these issues by explicitly canceling async tasks to prevent deadlocks and adjusting the order of operations in tests to better simulate real-world conditions. This ensures that the queue system operates reliably both in production and under test conditions, maintaining its integrity and responsiveness without unintended blocking or state inaccuracies."
    },
    "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.maxsize > 0 and q.qsize() == q.maxsize\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": {
          "248": "test_close_after_async_join"
        },
        "content_all": {
          "245": "        q = janus.Queue()\n",
          "246": "        # Ensuring that there is some task to join on.\n",
          "247": "        q.sync_q.put(1)\n",
          "248": "        task = asyncio.ensure_future(q.async_q.join())\n",
          "249": "        await asyncio.sleep(0.1)  # ensure tasks are blocking\n",
          "250": "        q.close()\n",
          "251": "        with pytest.raises(RuntimeError), contextlib.suppress(asyncio.TimeoutError):\n",
          "252": "            await asyncio.wait_for(task, timeout=0.1)\n",
          "253": "    \n"
        },
        "content_change": {
          "247": "        q.sync_q.put(1)\n"
        }
      }
    ],
    "location_ground": [
      {
        "file": "aio-libs_janus/janus/__init__.py",
        "function": {
          "100": "Queue.__init__"
        },
        "content_all": {
          "97": "\n",
          "98": "    class Queue(BaseQueue[T]):\n",
          "99": "        def __init__(self, maxsize: int = 0) -> None:\n",
          "100": "            self._maxsize = maxsize\n",
          "101": "            self._init(maxsize)\n",
          "102": "            self._closed = False\n",
          "103": "            self._mutex = threading.Lock()\n",
          "104": "            self._not_empty = threading.Condition(self._mutex)\n",
          "105": "            self._not_full = threading.Condition(self._mutex)\n",
          "106": "            self._unfinished_tasks = 0\n",
          "107": "            self._finished = threading.Condition(self._mutex)\n"
        },
        "content_change": {
          "102": "            self._closed = False"
        }
      },
      {
        "file": "aio-libs_janus/janus/__init__.py",
        "function": {
          "124": "Queue.cancel_join"
        },
        "content_all": {
          "121": "\n",
          "122": "        def _put(self, item: T) -> None:\n",
          "123": "            self._queue.append(item)\n",
          "124": "            self._unfinished_tasks += 1\n",
          "125": "            self._not_full.notify()\n",
          "126": "\n",
          "127": "        def cancel_join(self):\n",
          "128": "            with self._mutex:\n",
          "129": "                self._not_empty.notify_all()\n",
          "130": "\n"
        },
        "content_change": {
          "127": "        def cancel_join(self):",
          "128": "            with self._mutex:",
          "129": "                self._not_empty.notify_all()"
        }
      },
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": {
          "254": "TestMixedMode.test_close_after_async_join"
        },
        "content_all": {
          "251": "\n",
          "252": "    class TestMixedMode:\n",
          "253": "\n",
          "254": "        @pytest.mark.asyncio\n",
          "255": "        async def test_close_after_async_join(self):\n",
          "256": "            q = janus.Queue()\n",
          "257": "            q.sync_q.put(1)  # This line was previously commented out\n",
          "258": "\n",
          "259": "            task = asyncio.ensure_future(q.async_q.join())\n",
          "260": "            await asyncio.sleep(0.1)\n",
          "261": "\n",
          "262": "            q.close()\n",
          "263": "            q.cancel_join()\n",
          "264": "            with pytest.raises(RuntimeError):\n",
          "265": "                with contextlib.suppress(asyncio.TimeoutError):\n",
          "266": "                    await asyncio.wait_for(task, timeout=0.1)\n"
        },
        "content_change": {
          "257": "            q.sync_q.put(1)  # This line was previously commented out",
          "263": "            q.cancel_join()"
        }
      }
    ],
    "location_ground_exp": [
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": {
          "37": "test_unfinished"
        },
        "content_all": {
          "36": "    @pytest.mark.asyncio",
          "37": "    async def test_unfinished(self):",
          "38": "        q = janus.Queue()",
          "39": "        assert q.sync_q.unfinished_tasks == 0",
          "40": "        assert q.async_q.unfinished_tasks == 0",
          "41": "        q.sync_q.put(1)",
          "42": "        assert q.sync_q.unfinished_tasks == 1",
          "43": "        assert q.async_q.unfinished_tasks == 1",
          "44": "        q.sync_q.get()",
          "45": "        assert q.sync_q.unfinished_tasks == 0"
        },
        "content_change": {
          "41": "        q.sync_q.put(1)",
          "42": "        # Add corresponding async task to simulate realistic blocking condition",
          "43": "        await asyncio.ensure_future(q.async_q.join())"
        }
      },
      {
        "file": "aio-libs_janus/janus/__init__.py",
        "function": {
          "277": "wait_for_empty_queue"
        },
        "content_all": {
          "276": "        # Await the join to ensure all tasks are done",
          "277": "        await q.async_q.join()",
          "278": "        # Ensure tasks are canceled to prevent blocking state",
          "279": "        task.cancel()"
        },
        "content_change": {
          "278": "        # Ensure tasks are canceled to prevent blocking state",
          "279": "        task.cancel()"
        }
      }
    ],
    "location_ground_truth": [
      {
        "file": "aio-libs_janus/tests/test_mixed.py",
        "function": null,
        "content_all": {},
        "content_change": {}
      }
    ],
    "patch_i": "\n--- a/aio-libs_janus/tests/test_sync.py\n+++ b/aio-libs_janus/tests/test_sync.py\n@@ -13,8 +13,8 @@ QUEUE_SIZE = 5\n\n def qfull(q):\n-    return q._parent._maxsize > 0 and q.qsize() == q._parent._maxsize\n\n+    return q.maxsize > 0 and q.qsize() == q.maxsize\n\n # A thread to run a function that unclogs a blocked Queue.\n class _TriggerThread(threading.Thread):\n",
    "patch_im": "\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -31,6 +31,14 @@\n         assert 0 == q.maxsize\n \n     @pytest.mark.asyncio\n+    async def test_close_after_async_join(self):\n+        q = janus.Queue()\n+        # Ensuring that there is some task to join on.\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     async def test_unfinished(self):\n         q = janus.Queue()\n         assert q.sync_q.unfinished_tasks == 0\n",
    "patch_il": "\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.maxsize > 0 and q.qsize() == q.maxsize\n \n \n # A thread to run a function that unclogs a blocked Queue.\n",
    "patch_iml": "\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -244,6 +244,7 @@ class TestMixedMode:\n         q = janus.Queue()\n         # Ensuring that there is some task to join on.\n         q.sync_q.put(1)\n+        await q.async_q.put(1)\n \n         task = asyncio.ensure_future(q.async_q.join())\n         await asyncio.sleep(0.1)  # ensure tasks are blocking\n",
    "patch_ground": "\n--- a/aio-libs_janus/janus/__init__.py\n+++ b/aio-libs_janus/janus/__init__.py\n@@ -100,6 +100,11 @@\n         self._unfinished_tasks = 0\n\n+    def close(self):\n+        with self._mutex:\n+            self._closed = True\n+            self._not_empty.notify_all()\n+\n     def _get(self):\n         if not self._unfinished_tasks:\n             raise RuntimeError(\"cannot get() from an empty queue\")\n@@ -120,6 +125,11 @@\n         return self._not_full.wait(timeout)\n \n+    def cancel_join(self):\n+        with self._mutex:\n+            self._not_empty.notify_all()\n+\n class SyncQueue(BaseQueue[T], queue.Queue):\n     pass\n\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -254,8 +254,8 @@\n     @pytest.mark.asyncio\n     async def test_close_after_async_join(self):\n         q = janus.Queue()\n-        # q.sync_q.put(1)\n+        q.sync_q.put(1)\n         \n         task = asyncio.ensure_future(q.async_q.join())\n         await asyncio.sleep(0.1)\n \n         q.close()\n+        q.cancel_join()\n         with pytest.raises(RuntimeError):\n             with contextlib.suppress(asyncio.TimeoutError):\n                 await asyncio.wait_for(task, timeout=0.1)\n",
    "patch_ground_location": "\n--- a/aio-libs_janus/janus/__init__.py\n+++ b/aio-libs_janus/janus/__init__.py\n@@ -100,6 +100,10 @@ class Queue(BaseQueue[T]):\n         self._unfinished_tasks = 0\n+        self._closed = False\n+    def close(self):\n+        with self._mutex:\n+            self._closed = True\n+            self._not_empty.notify_all()\n \n@@ -120,6 +124,10 @@ class Queue(BaseQueue[T]):\n         return self._not_full.wait(timeout)\n \n+    def cancel_join(self):\n+        with self._mutex:\n+            self._not_empty.notify_all()\n+        \n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -254,8 +254,7 @@ class TestMixedMode:\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+        q.sync_q.put(1)\n         \n         task = asyncio.ensure_future(q.async_q.join())\n         await asyncio.sleep(0.1)\n \n         q.close()\n+        q.cancel_join()\n         with pytest.raises(RuntimeError):\n             with contextlib.suppress(asyncio.TimeoutError):\n                 await asyncio.wait_for(task, timeout=0.1)\n",
    "patch_ground_exp": "\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -38,6 +38,7 @@\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+        await asyncio.ensure_future(q.async_q.join())\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 == 0\n\n--- a/aio-libs_janus/janus/__init__.py\n+++ b/aio-libs_janus/janus/__init__.py\n@@ -276,6 +276,7 @@\n         # Await the join to ensure all tasks are done\n         await q.async_q.join()\n         # Ensure tasks are canceled to prevent blocking state\n+        task.cancel()\n",
    "patch_ground_all": "\n--- a/aio-libs_janus/tests/test_mixed.py\n+++ b/aio-libs_janus/tests/test_mixed.py\n@@ -69,25 +69,36 @@\n         await q.async_q.join()\n         assert q.sync_q.unfinished_tasks == 0\n         assert q.async_q.unfinished_tasks == 0\n \n     @pytest.mark.asyncio\n     async def test_task_cancel(self):\n         q = janus.Queue()\n         q.sync_q.put(1)\n         task = asyncio.ensure_future(q.async_q.join())\n         q.sync_q.task_done()\n         await asyncio.sleep(0)  # let the event loop cycle\n         task.cancel()\n         with pytest.raises(asyncio.CancelledError):\n             await task\n \n+    @pytest.mark.asyncio\n+    async def test_sync_put_before_async_join(self):\n+        q = janus.Queue()\n+        q.sync_q.put(1)  # Ensure the queue has an item\n+        async def worker():\n+            await q.async_q.get()\n+            q.async_q.task_done()\n+        worker_task = asyncio.ensure_future(worker())\n+        await asyncio.sleep(0)  # let the event loop cycle\n+        await q.async_q.join()  # This should block until the worker processes the item\n+        assert q.sync_q.empty()\n+        assert q.async_q.empty()\n+        worker_task.cancel()\n+\n class TestMixedSync:\n     def test_sync_mode(self):\n         q = janus.Queue()\n         q.sync_q.put(1)\n         assert q.sync_q.qsize() == 1\n         assert q.async_q.qsize() == 1\n         q.sync_q.get()\n         q.sync_q.task_done()\n         assert q.sync_q.qsize() == 0\n         assert q.async_q.qsize() == 0\n",
    "patch_ground_truth": "--- 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@@ -245,6 +246,7 @@\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",
    "message": "========================================================================================= FAILURES =========================================================================================\n________________________________________________________________________ TestMixedMode.test_close_after_async_join _________________________________________________________________________\n\nself = <test_mixed.TestMixedMode object at 0x7ceecb0e3f70>\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)\nE           Failed: DID NOT RAISE <class 'RuntimeError'>\n\ntests/test_mixed.py:255: Failed\n================================================================================= short test summary info ==================================================================================\nFAILED tests/test_mixed.py::TestMixedMode::test_close_after_async_join - Failed: DID NOT RAISE <class 'RuntimeError'>\n=============================================================================== 1 failed, 71 passed in 4.80s ===============================================================================",
    "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_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.asyncio\n27     async def test_maxsize_named_param(self):\n28         q = janus.Queue(maxsize=7)\n29         assert 7 == q.maxsize\n30 \n31     @pytest.mark.asyncio\n32     async def test_maxsize_default(self):\n33         q = janus.Queue()\n34         assert 0 == q.maxsize\n35 \n36     @pytest.mark.asyncio\n37     async def test_unfinished(self):\n38         q = janus.Queue()\n39         assert q.sync_q.unfinished_tasks == 0\n40         assert q.async_q.unfinished_tasks == 0\n41         q.sync_q.put(1)\n42         assert q.sync_q.unfinished_tasks == 1\n43         assert q.async_q.unfinished_tasks == 1\n44         q.sync_q.get()\n45         assert q.(...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_do(...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.ge(...truncated)"
      }
    ],
    "CommitSHA": "e46a1dfa166ebc2484716ee36cb99763f6247d52"
  },
  "Score": {
    "Difficulty": "Medium",
    "issue_origin": {
      "Title": 8,
      "Description": 6,
      "Reproducibility": 4,
      "Relevance": 6,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_message": {
      "Title": 6,
      "Description": 5,
      "Reproducibility": 4,
      "Relevance": 7,
      "Explanation": 7,
      "Overall": 6
    },
    "issue_ground": {
      "Title": 8,
      "Description": 6,
      "Reproducibility": 4,
      "Relevance": 7,
      "Explanation": 6,
      "Overall": 6
    },
    "issue_ground_truth": {
      "title": "Tasks not getting canceled correctly and incorrect queue state handling",
      "description": "There are two prominent issues that need attention:\n\n1. **Async Task Not Getting Canceled Properly**:\n   In some scenarios, tasks placed in the queue are not getting canceled correctly. When using `q.async_q.join()`, the tasks remain in a blocking state if certain conditions are not met, leading to potential deadlock situations or unresponsive behavior. Proper cancellation of these tasks is crucial to ensure the queue can be managed effectively without leaving orphaned tasks running indefinitely.\n\n2. **Improper Handling of Queue State in Test Scenarios**:\n   The tests for queue-related functionalities sometimes do not reflect the actual state changes, especially when operations like `put` are used on synchronous queues within asynchronous tests. For instance, using `q.sync_q.put(1)` before `asyncio.ensure_future(q.async_q.join())` helps simulate a more realistic blocking condition, helping to ensure that the queue’s empty or full state is managed accurately across async and sync contexts.\n\nBoth issues can lead to inconsistent behavior in production scenarios where managing the queue's state accurately is critical for operations.",
      "explanation": "### Summary of the Issue\n\nThe issue comprises two distinct but related problems in an asynchronous and synchronous mixed queueing system:\n\n1. **Async Task Not Getting Canceled Properly**: Certain asynchronous tasks placed in the queue are not being canceled correctly. This can cause tasks to remain in a blocking state, potentially leading to deadlocks or unresponsive behavior. Proper task cancellation is essential to maintain queue stability and prevent orphaned tasks from running indefinitely.\n\n2. **Improper Handling of Queue State in Test Scenarios**: Some test cases for the queue functionalities do not accurately reflect the real state changes, especially when switching between synchronous and asynchronous operations. Specifically, there are problems with simulating realistic blocking conditions which are necessary to test the queue’s behavior correctly under different states (empty or full).\n\n### Analysis of the Commit\n\nThe commits made to address the issues involve changes to the test files, mainly `test_mixed.py`. Let's break down these changes and understand how they address the identified problems.\n\n1. **Cancellation of Async Tasks**:\n   - **Issue**: Tasks that were awaiting completion via `q.async_q.join()` were not being canceled correctly, leaving them in a blocking state.\n   - **Solution**: The commit adds a `task.cancel()` call after the `await q.async_q.join()` statement in the `wait_for_empty_queue` function. This ensures that the async task is appropriately canceled after the queue work is completed.\n\n2. **Handling of Queue State in Test Scenarios**:\n   - **Issue**: In some tests, operations like `put` were not being correctly positioned within the asynchronous tests, preventing accurate simulation of queue states.\n   - **Solution**: The commit ensures that `put` operations are performed on the synchronous queue before starting the async join operation. For instance, adding `q.sync_q.put(1)` ensures that there is an entry in the queue, which helps in creating a more realistic blocking condition when `asyncio.ensure_future(q.async_q.join())` is called.\n\n### Explanation of the Solution\n\n1. **Task Cancellation**:\n   - **Problem**: In a scenario where async tasks join the queue, the absence of explicit cancellation could result in tasks being left unattended in a waiting state. This can lead to deadlocks or unresponsive applications because the tasks are indefinitely suspending the execution flow.\n   - **Commit**: By explicitly calling `task.cancel()`, the developer has ensured that any remaining active tasks are forcefully canceled after their intended operations are completed. This prevents any potential blocking caused by unfinished tasks.\n\n2. **Queue State Simulation in Tests**:\n   - **Problem**: Testing the state changes in mixed synchronous and asynchronous environments can be tricky. If the synchronous operations (such as `put`) are not executed at the right moment, it can lead to inaccurate test scenarios that do not reflect real-world usage, where queues might have items and consumers waiting on both sync and async sides.\n   - **Commit**: Adjusting the sequence by adding `q.sync_q.put(1)` before asynchronous joining operation (`asyncio.ensure_future(q.async_q.join())`) ensures that tests are more reflective of actual queue states. This helps in simulating more accurate blocking conditions, which is crucial for robust and reliable testing of queue behavior under different states and usage patterns.\n\n### Conclusion\n\nThe primary issue was twofold: ensuring proper cancellation of async tasks and accurate handling of queue states in test scenarios. The commit effectively resolved these issues by explicitly canceling async tasks to prevent deadlocks and adjusting the order of operations in tests to better simulate real-world conditions. This ensures that the queue system operates reliably both in production and under test conditions, maintaining its integrity and responsiveness without unintended blocking or state inaccuracies."
    }
  }
}