我如何模拟Python方法OptionParser.error(),它执行sys.exit()?

Dar*_*zer 14 python unit-testing mocking optparse

我正在尝试单元测试一些看起来像这样的代码:

def main():
    parser = optparse.OptionParser(description='This tool is cool', prog='cool-tool')
    parser.add_option('--foo', action='store', help='The foo option is self-explanatory')
    options, arguments = parser.parse_args()
    if not options.foo:
        parser.error('--foo option is required')
    print "Your foo is %s." % options.foo
    return 0

if __name__ == '__main__':
   sys.exit(main())
Run Code Online (Sandbox Code Playgroud)

使用如下代码:

@patch('optparse.OptionParser')
def test_main_with_missing_p4clientsdir_option(self, mock_optionparser):
    #
    # setup
    #
    optionparser_mock = Mock()
    mock_optionparser.return_value = optionparser_mock
    options_stub = Mock()
    options_stub.foo = None
    optionparser_mock.parse_args.return_value = (options_stub, sentinel.arguments)
    def parser_error_mock(message):
        self.assertEquals(message, '--foo option is required')
        sys.exit(2)
    optionparser_mock.error = parser_error_mock

    #
    # exercise & verify
    #
    self.assertEquals(sut.main(), 2)
Run Code Online (Sandbox Code Playgroud)

我正在使用迈克尔福德的模拟,并试图进行测试.

当我运行测试时,我得到:

  File "/Users/dspitzer/Programming/Python/test-optparse-error/tests/sut_tests.py", line 27, in parser_error_mock
    sys.exit(2)
SystemExit: 2

----------------------------------------------------------------------
Ran 1 test in 0.012s

FAILED (errors=1)
Run Code Online (Sandbox Code Playgroud)

问题是OptionParser.error执行sys.exit(2),因此main()自然依赖于它.但是nose或unittest检测到(预期的)sys.exit(2)并且未通过测试.

我可以通过在main()中的parser.error()调用下添加"return 2"并从parser_error_mock()中删除sys.exit()调试来进行测试,但是我觉得修改测试中的代码是令人讨厌的.允许测试通过.有更好的解决方案吗?

更新:df的答案有效,尽管正确的调用是"self.assertRaises(SystemExit,sut.main)".

这意味着测试会传递parser_error_mock()中sys.exit()中的数字.有没有办法测试退出代码?

顺便说一下,如果我补充一下,测试会更加强大:

self.assertEquals(optionparser_mock.method_calls, [('add_option', ('--foo',), {'action': 'store', 'help': 'The foo option is self-explanatory'}), ('parse_args', (), {})])
Run Code Online (Sandbox Code Playgroud)

在末尾.

更新2:我可以通过将"self.assertRaises(SystemExit,sut.main)"替换为以下内容来测试退出代码:

try:
    sut.main()
except SystemExit, e:
    self.assertEquals(type(e), type(SystemExit()))
    self.assertEquals(e.code, 2)
except Exception, e:
    self.fail('unexpected exception: %s' % e)
else:
    self.fail('SystemExit exception expected')
Run Code Online (Sandbox Code Playgroud)

dF.*_*dF. 12

这会有效assertEquals吗?

self.assertRaises(SystemExit, sut.main, 2)
Run Code Online (Sandbox Code Playgroud)

这应该捕获SystemExit异常并阻止脚本终止.


Dar*_*zer 1

正如我对问题的更新中所述,我必须将dF的答案修改为:

self.assertRaises(SystemExit, sut.main)
Run Code Online (Sandbox Code Playgroud)

...我想出了一些更长的代码片段来测试退出代码。

[注意:我接受了我自己的答案,但如果他更新了我的答案,我将删除此答案并接受dF的答案。]