模拟用户输入()

nbb*_*nbb 3 python testing input mocking flexmock

我正在尝试模拟我将要使用的 python 脚本的用户输入py.test。下面是一些代表我想要完成的基本代码:

def ask():
    while True:
        age = input("Enter your age: ")
        if int(age) < 13:
            print("You are too young")
        else:
            name = input("Enter your name: ")
            break
    print("Welcome!")
Run Code Online (Sandbox Code Playgroud)

我想模仿用户输入并读取输出。一个例子可能是这样的:

@mock.patch('builtins.input', side_effect=['11'])
def test_invalid_age():
    ask()
    assert stdout == "You are too young"
Run Code Online (Sandbox Code Playgroud)

我也听说这flexmock可能是内置模拟unittest系统的更好替代方案,但此时我会采取任何解决方案。

更新:

我玩了更多,并进行了一个测试:

@mock.patch('builtins.input', side_effect=['11'])
def test_bad_params(self, input):
    ask()
    output = sys.stdout.getline().strip()
    assert output == "You are too young"
Run Code Online (Sandbox Code Playgroud)

当我运行 py.test 我得到这个结果:

E StopIteration /usr/lib/python3.3/unittest/mock.py:904:
StopIteration
Run Code Online (Sandbox Code Playgroud)

它确实捕获了适当的标准输出调用“你太年轻了”。

che*_*ner 5

ask在第一个太年轻的年龄之后不再回来;它循环直到输入适当的年龄。正如所写,您需要提供它可能读取的所有字符串,然后在ask返回后执行所有断言。

@mock.patch('builtins.input', side_effect=['11', '13', 'Bob'])
def test_bad_params(self, input):
    ask()
    output = sys.stdout.getline().strip()
    assert output == "You are too young"
    # Check the output after "13" and "Bob" are entered as well!
    assert sys.stdout.getline().strip() == "Welcome!"
Run Code Online (Sandbox Code Playgroud)