Doctesting functions that receive and display user input - Python (tearing my hair out)
The simplest way to make this testable would be parameter injection:
def getFiveNums(input_func=input):
print("Howdy. Please enter five numbers, hit <enter> after each one")
for i in range(5):
newNum = input_func("Please type in a number:")
numbers.append(newNum)
print("Here are your numbers: ", numbers)
You can't realistically be expected to unit test input/output like that -- you cannot be concerned that the call to input
might somehow fail. Your best option is to pass in a stub method of some nature; something like
def fake_input(str):
print(str)
return 3
So that in your doctest, you actually test getFiveNums(fake_input)
.
Moreover, by breaking the direct dependency on input
now, if you were to port this code to something else later that didn't use a command line, you could just drop in the new code to retrieve input (whether that would be a dialog box in a GUI application, or a Javascript popup in a web-based application, etc.).
I know you are asking for a doctest answer but may I suggest that this type of function may not be a good candidate for doctest. I use doctests for documentation more than testing and the doctest for this wouldn't make good documentation IMHO.
A unitest approach may look like:
import unittest
# stores 5 user-entered numbers (strings, for now) in a list
def getFiveNums():
numbers = []
print "Howdy. Please enter five numbers, hit <enter> after each one"
for i in range(5):
newNum = input("Please type in a number:")
numbers.append(newNum)
return numbers
def mock_input(dummy_prompt):
return 1
class TestGetFiveNums(unittest.TestCase):
def setUp(self):
self.saved_input = __builtins__.input
__builtins__.input = mock_input
def tearDown(self):
__builtins__.input = self.saved_input
def testGetFiveNums(self):
printed_lines = getFiveNums()
self.assertEquals(printed_lines, [1, 1, 1, 1, 1])
if __name__ == "__main__":
unittest.main()
It's maybe not exactally testing the function you put forward but you get the idea.
I found a different way.
"""
>>> get_five_nums(testing=True)
Howdy. Please enter five numbers, hit <enter> after each one.
Please type in a number: 1
Please type in a number: 1
Please type in a number: 1
Please type in a number: 1
Please type in a number: 1
Here is a list of the numbers you entered: [1, 1, 1, 1, 1]
>>>
"""
import doctest
numbers = []
def get_five_nums(testing=False):
"""Stores 5 user-entered numbers (strings, for now) in a list."""
print("Howdy. Please enter five numbers, hit <enter> after each one.")
for i in range(5):
new_num = int(input("Please type in a number: "))
if testing:
print(new_num)
numbers.append(new_num)
print("Here is a list of the numbers you entered: ", numbers)
if __name__ == "__main__":
doctest.testmod(verbose=True)
Save the above code in a file called foo.py. Now make a file called input.txt.
All it needs in it is.
1
1
1
1
1
Five ones. One on each line.
To test you program do the following, at terminal or command prompt (I'm using a mac):
$ python foo.py < input.txt
This is easily changeable for any kind of user input on any program. With this you can now copy the output of terminal session and use it as your doctest.
NOTE: the function call in terminal would be get_five_nums(). In you doctest it needs to be get_five_nums(testing=True).
Even though doctest doesn't appear to be intended to be used in this way it is still a handy hack.