1

在 Python 中,如何使用 mox 单元测试库模拟在 with 语句中创建的对象

代码

class MyCode:
    def generate_gzip_file(self):
        with gzip.GzipFile('file_name.txt.gz','wb') as f:
             f.write('data')

单元测试

class MyCodeTest(unittest.TestCase):
    def test_generate_gzip_file(self):
        mox = mox.Mox()
        mock_gzip_file = self.mox.CreateMock(gzip.GzipFile)
        mox.StubOutWithMock(gzip, 'GzipFile')
        gzip.GzipFile('file_name.txt.gz','wb').AndReturn(mock_file)
        mock_gzip_file.write('data')
        mox.ReplayAll()
        MyCode().generate_gzip_file()
        mox.VerifyAll()

AttributeError: __exit__在线收到错误

with gzip.GzipFile('file_name.txt.gz','wb') as f:
4

1 回答 1

3

DSM 是正确的,因为某种原因,模拟的实例gzip.GzipFile没有以方法结束。如果您忘记在与语句一起使用的类上__exit__进行定义,您将得到完全相同的错误。例如:__exit__with

>>> class C(object):
...   def __enter__(self):
...     return self
... 
>>> with C() as c:
...   pass
... 
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
AttributeError: __exit__

CreateMockAnything()幸运的是,您可以通过使用 Mox 的方法创建一个mock_gzip_file不强制执行特定接口的对象来解决此问题。您需要小心确保mock_gzip_file正确设置对象的期望(即,您设置对何时以及如何调用__enter__()and方法的期望)。__exit__(...)这是一个对我有用的例子:

import gzip
import mox
import unittest

class MyCode:
    def generate_gzip_file(self):
        with gzip.GzipFile('file_name.txt.gz', 'wb') as f:
             f.write('data')

class MyCodeTest(unittest.TestCase):
    def test_generate_gzip_file(self):
        mymox = mox.Mox()
        mock_gzip_file = mymox.CreateMockAnything()
        mymox.StubOutWithMock(gzip, 'GzipFile')
        gzip.GzipFile('file_name.txt.gz', 'wb').AndReturn(mock_gzip_file)
        mock_gzip_file.__enter__().AndReturn(mock_gzip_file)
        mock_gzip_file.write('data')
        mock_gzip_file.__exit__(None, None, None).AndReturn(None)
        mymox.ReplayAll()

        MyCode().generate_gzip_file()
        mymox.VerifyAll()

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

当我运行它时,我得到:

.
----------------------------------------------------------------------
Ran 1 test in 0.001s

OK
于 2012-01-19T01:23:38.960 回答