Python SQLAlchemy 模拟

nej*_*obi 8 python unit-testing sqlalchemy python-unittest python-unittest.mock

这是我进行链式数据库查询调用的一种方法。

import math
def get_all_keys():
    db_session = DBSession()
    keys = db_session.query(SomeClass).all()
Run Code Online (Sandbox Code Playgroud)

我应该嘲笑DBSession().query(SomeClass).all()。我尝试了一些,但没有任何效果。这是我的试验。

@patch('app.modules.somemodule.DBSession')
def test_asd(self, DBSession):
    DBSession.execute.query.execute.all.return_value = [1, 2, 3]
    self.assertListEqual(DBSession.query('qwe').all(), [1, 2, 3])
Run Code Online (Sandbox Code Playgroud)

在这次尝试中,如果我尝试打印,DBSession()我会得到一个MagicMock看起来不错的对象。但是当我运行时pytest出现以下错误。

    self = <test_some_module.SomeModuleTests testMethod=test_asd>, DBSession = <MagicMock name='DBSession' id='140028663111976'>

    @patch('app.modules.somemodule.DBSession')
    def test_asd(self, DBSession):
        DBSession.execute.query.execute.all.return_value = [1, 2, 3]
        print(DBSession().query('qwe').all())
>       self.assertListEqual(DBSession.query('qwe').all(), [1, 2, 3])
E       AssertionError: First sequence is not a list: <MagicMock name='DBSession.query().all()' id='140028662649184'>

tests/test_some_module.py:21: AssertionError
Run Code Online (Sandbox Code Playgroud)

好像我在嘲笑时做错了什么。我的错误是什么?我检查了文档。

小智 7

您可以使用 alchemy 模拟来模拟 SQLAlchemy 的会话和数据

from alchemy_mock.mocking import UnifiedAlchemyMagicMock
db_session = UnifiedAlchemyMagicMock()
db_session.query.return_value.all.return_value = [1,2,3]

mock_res = test_asd(db_session)
self.assertEqual(mock_res, [1,2,3])
Run Code Online (Sandbox Code Playgroud)

您也可以选择模拟数据

db_session.add(SomeClass(column1=1, column2=2, column3=3))
Run Code Online (Sandbox Code Playgroud)

  • 现在你可以使用[mock-alchemy](https://pypi.org/project/mock-alchemy/) (5认同)

Ilj*_*ilä 6

您只是在嘲笑错误的事情,并断言调用链应该返回具有给定项目的实际列表,而不是魔术模拟对象。代替execute, 只是模拟对同名属性的访问,您应该使用return_value,正如您在最后一步中所做的那样。鉴于链

DBSession().query(...).all()
Run Code Online (Sandbox Code Playgroud)

你应该嘲笑

DBSession.return_value.query.return_value.all.return_value = [1, 2, 3]
Run Code Online (Sandbox Code Playgroud)

另一方面,test_asd()在你正在做的断言中

DBSession.query(...).all()
Run Code Online (Sandbox Code Playgroud)

所以

DBSession.query.return_value.all.return_value = [1, 2, 3]
Run Code Online (Sandbox Code Playgroud)