为了账号安全,请及时绑定邮箱和手机立即绑定

pytest - 修补类不起作用,而是调用类

pytest - 修补类不起作用,而是调用类

绝地无双 2022-01-05 10:57:04
不知道为什么,但这是我的代码片段:stats_collector.pyclass StatsCollector( object ) :def __init__( self, user_id, app_id ) :    logging.info( "Stats: APP_ID = {0}, ID = {1}".format( app_id, user_id ) )mydb.pyfrom namespacename.mylibname.stats_collector import StatsCollectorclass Db( object ) :    # constructor/destructor    def __init__( self, dbname ) :        ....    def beginTransaction( self, user_id, app_id ) :        logging.info( "Begin" )        self._stats = StatsCollector( user_id, app_id ) test_mylibname_mydbfrom namespacename.mylibname.mydb import Dbfrom namespacename.mylibname.stats_collector import StatsCollector@pytest.mark.mydb_temp@mock.patch( 'namespacename.mylibname.stats_collector.StatsCollector')def test_db_beginTransaction( mock_stats_collector ) :    db = Db( TEST_DB_NAME )    mock_stats_collector.return_value = mock.Mock()    db.beginTransaction( TEST_ID, TEST_APP_ID )    ......    ......我可以在我的登录中看到我的stats_collector.__init__- 为什么我要输入它?不应该在我的内部beginTransaction调用StatsCollector返回值是 MockObject 并且我不应该看到任何日志吗?结构如下:tests/├── mylibname│   ├── test_mylibname_mydb.pynamespacename/mylibname├── stats_collector│   ├── mylibname_stats_collector.py│   ├── __init__.py├── mydb│   ├── mylibname_mydb.py│   ├── __init__.py** 编辑 **遵循有关评论的建议 -@mock.patch( 'namespacename.mylibname.mydb.StatsCollector')def test_db_beginTransaction( mock_stats_init ) :    db = Db( TEST_DB_NAME )    db.beginTransaction( TEST_UUID, TEST_APP_ID )    print db._transaction    assert db._transaction is mock_stats_init让我:E       AssertionError: assert <namespacename.mylibname.stats_collector.mylibname_stats_collector.StatsCollector object at 0x7f42d837b110> is <MagicMock name='StatsCollector' id='139925072008976'>E        +  where <namespacename.mylibname.stats_collector.mylibname_stats_collector.StatsCollector object at 0x7f42d837b110> = <namespacename.mylibname.mydb.mylibname_mydb.Db object at 0x7f42d8365850>._transaction
查看完整描述

3 回答

?
MYYA

TA贡献1868条经验 获得超4个赞

您需要修补正在测试的模块中的符号“A”,即“B”。

当你这样做时@mock.patch("full.path.A"),它应该是:

@mock.patch("full.path.to.B.A")

现在A模块中的符号B用你的模拟打了补丁。


查看完整回答
反对 回复 2022-01-05
?
米琪卡哇伊

TA贡献1998条经验 获得超6个赞

你不是错过了文件本身的名称吗?

@mock.patch( 'namespacename.mylibname.stats_collector.mylibname_stats_collector.StatsCollector')



查看完整回答
反对 回复 2022-01-05
?
小怪兽爱吃肉

TA贡献1852条经验 获得超1个赞

当我写这篇文章时,你可能已经想通了。


经验法则:不要在定义类或函数的地方打补丁,而是在使用它们的地方打补丁。


a.py


class A:

    def exponent(self, a, b)

        return a ** b

    


b.py    



from a import A

class B:

   def add_constat(a, b, c)

      return A().exponent(a, b) + c

      

  

  

  

为了测试 add_constant 方法,您可能会想像这样修补 A


TestB:


@patch('a.A.exponent', mock_a)

    test_add_constant(self, mock_a) 

这是错误的,因为您正在修补给出定义的文件中的类。


A 在类 B 的文件 b 中使用。因此,您应该修补该类。


TestB:


    @patch('b.A')

    test_add_constant(self, mock_a):

    # mock_a is fake class of A, the return_value of mock_a gives us an instance (object) of the class(A)

    instance_a = mock_a.return_value # 

   

    # we now have instance of the class i.e A, hence it is possible to call the methods of class A

   

    instance_a.exponent.return_value = 16

   

    assert 26 = B().add_constant(2,4,10)

我对您的代码进行了一些修改,以便它可以在我的 Python 环境中使用。


stats_collector.py


class StatsCollector:

    def __init__(self, user_id, app_id):

        self.user_id = user_id

        self.app_id = app_id


    def stat(self):

        return self.user_id + ':' + self.app_id



mydb.py


from stats_collector import StatsCollector

import logging


class Db:

    # constructor

    def __init__(self, db_name):

        self.db_name = db_name


    def begin_transaction(self, user_id, app_id):

        logging.info("Begin")

        stat = StatsCollector(user_id, app_id).stat()


        if stat:

            return user_id + ':' + app_id

        return "wrong User"

使用类似的比喻:为了测试文件 mydb.py 的 Db 类中的“begin_transaction”,您需要修补 mydb.py 文件中使用的 StatsCollector 类


test_mydb.py


from unittest.mock import patch

from unittest import TestCase



class TestDb(TestCase):


    @patch('mydb.StatsCollector')

    def test_begin_transaction(self, db_class_mock):


        # db_class_mock is the mock of the class, it is not an instance of the DB class.

        # to create an instance of the class, you need to call return_value

       

        db_instance = db_class_mock.return_value

        db_instance.stat.return_value = 1

        # i prefere to do the above two lines in just one line as


        #db_class_mock.return_value.stat.return_value = 1

        


        db = Db('stat')

        expected = db.begin_transaction('Addis', 'Abeba')


        assert expected == 'Addis' + ':' + 'Abeba'


        # set the return value of stat method

        db_class_mock.return_value.stat.return_value = 0

        expected = db.begin_transaction('Addis', 'Abeba')


        assert expected == "wrong User"

我希望这可以帮助网络中的某人


查看完整回答
反对 回复 2022-01-05
  • 3 回答
  • 0 关注
  • 186 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
微信客服

购课补贴
联系客服咨询优惠详情

帮助反馈 APP下载

慕课网APP
您的移动学习伙伴

公众号

扫描二维码
关注慕课网微信公众号