gpt4 book ai didi

python - 使用 aiohttp 和 asyncio 时编写单元测试

转载 作者:太空狗 更新时间:2023-10-29 21:53:59 25 4
gpt4 key购买 nike

我正在更新我的一个 Python 包,因此它是异步的(使用 aiohttp 而不是 requests)。我也在更新我的单元测试,以便它们可以使用新的异步版本,但我在这方面遇到了一些麻烦。

这是我的包裹中的一个片段:

async def fetch(session, url):
while True:
try:
async with session.get(url) as response:
assert response.status == 200
return await response.json()
except Exception as error:
pass


class FPL():
def __init__(self, session):
self.session = session

async def get_user(self, user_id, return_json=False):
url = API_URLS["user"].format(user_id)
user = await fetch(self.session, url)

if return_json:
return user
return User(user, session=self.session)

这样使用时似乎一切正常:

async def main():
async with aiohttp.ClientSession() as session:
fpl = FPL(session)
user = await fpl.get_user(3808385)
print(user)

loop = asynio.get_event_loop()
loop.run_until_complete(main())

>>> User 3808385

不幸的是,我的单元测试遇到了一些问题。我以为我可以简单地做一些事情

def _run(coroutine):
return asyncio.get_event_loop().run_until_complete(coroutine)


class FPLTest(unittest.TestCase):
def setUp(self):
session = aiohttp.ClientSession()
self.fpl = FPL(session)

def test_user(self):
user = _run(self.fpl.get_user("3523615"))
self.assertIsInstance(user, User)

user = _run(self.fpl.get_user("3523615", True))
self.assertIsInstance(user, dict)

if __name__ == '__main__':
unittest.main()

它会给出如下错误

DeprecationWarning: The object should be created from async function loop=loop)

ResourceWarning: Unclosed client session <aiohttp.client.ClientSession object at 0x7fbe647fd208>

我已经尝试将 _close() 函数添加到关闭 session 的 FPL 类,然后从测试中调用它,但这也没有工作,但仍然说有一个未关闭的客户端 session 。

是否可以这样做,我只是做错了什么,还是我最好使用 asynctestpytest-aiohttp 之类的东西来代替?

编辑:我还检查了aiohttp 的文档并找到了example。展示如何使用标准库的单元测试来测试应用程序。不幸的是,我无法让它工作,因为 AioHTTPTestCase 中提供的 loop 自 3.5 以来已被弃用并引发错误:

class FPLTest(AioHTTPTestCase):
def setUp(self):
session = aiohttp.ClientSession()
self.fpl = FPL(session)

@unittest_run_loop
async def test_user(self):
user = await self.fpl.get_user("3523615")
self.assertIsInstance(user, User)

user = await self.fpl.get_user("3523615", True)
self.assertIsInstance(user, dict)

给予

tests/test_fpl.py:20: DeprecationWarning: The object should be created from async function
session = aiohttp.ClientSession()
...
======================================================================
ERROR: test_user (__main__.FPLTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "/home/amos/Documents/fpl/venv/lib/python3.7/site-packages/aiohttp/test_utils.py", line 477, in new_func
return self.loop.run_until_complete(
AttributeError: 'FPLTest' object has no attribute 'loop'

======================================================================
ERROR: test_user (__main__.FPLTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "/home/amos/Documents/fpl/venv/lib/python3.7/site-packages/aiohttp/test_utils.py", line 451, in tearDown
self.loop.run_until_complete(self.tearDownAsync())
AttributeError: 'FPLTest' object has no attribute 'loop'

最佳答案

将 pytest 与 aiohttp-pytest 结合使用:

async def test_test_user(loop):
async with aiohttp.ClientSession() as session:
fpl = FPL(session)
user = await fpl.get_user(3808385)
assert isinstance(user, User)

现代 python 开发者的谚语:生命太短暂,不能不使用 pytest。

您可能还想设置一个模拟服务器以在测试期间接收您的 http 请求,我没有一个简单的示例,但可以看到一个完整的工作示例 here .

关于python - 使用 aiohttp 和 asyncio 时编写单元测试,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54278925/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com