Test Driven Development in Python
Test Driven Development in Python
Kevin Dahlhausen
kevin.dahlhausen@keybank.com
My (pythonic) Background
learned of python in 96 < Vim Editor Fast-Light Toolkit python wrappers PyGallery one of the early online photo gallery generators wxGlade plugin despise HDD Hype-Driven Design
design methodology -> artifacts for testing short cycle repeated many times:
write a test watch it fail make it compile make it pass refactor the code refactor the test (and elaborate) rinse and repeat
objects: simpler, stand-alone, minimal dependencies tends to result in extensible architectures instant feedback safety net same feeling as w/scc
unittest
batteries included since 2.1 PyUnit before that python version of JUnit TestCase
assert_, equality, almost equality, raises exception assertEqual( a, b, msg) setUp / tearDown methods doc strings addTest, run auto-discovery tools vs. maintenance by hand of test-suites
TestSuite
unittest/TDD example
Game needs a Tank Pygame -> Sprite -> image, rect attr start with test:
testTankImage:
create a tank object and assert that tank has image and image is not None
import unittest from Tank import Tank class TestTank(unittest.TestCase): -> def testTankHasImage(self): tank = Tank() assert tank.image != None, "tank image is None" def testTankHasRectangle(self): tank = Tank() assert tank.rect != None, "tank rectangle is None" __name__=="__main__": unittest.main()
-> if
PyMock
most objects collaborate with others mock object is a test-oriented replacement (object) for a collaborator javas EasyMock website verifies method calls/parameters (vs. stub)
simulates objects (function calls, properties, functions, and generators) that interact with your object under test and verify the calls record playback verify (demo 2)
PyMock - sample
TestTankDraw.py: import unittest import pymock import pygame from Tank import Tank class TestTankDraw(unittest.TestCase): def testDrawTank(self): tankGroup = pygame.sprite.Group() tank = Tank() controller = pymock.Controller() 1 2 3 surface = controller.mock() surface.blit( tank.image, tank.rect) controller.replay() tankGroup.add(tank) tankGroup.draw(surface) 4 controller.verify()
if __name__=="__main__": unittest.main()
PyMock
import pymock import os class TestOverride(pymock.PyMockTestCase): def testOverride(self): 1 2 3 self.override(os, 'listdir') # object, method name self.expectAndReturn(os.listdir(), ['file-one', 'file-two']) self.replay() result = os.listdir() self.assertEqual( result, ['file-one', 'file-two'], "wrong files returned") self.verify()
Nose
plugins:
Nose
test function yields a function and parameters to apply to that function will automatically call the function for each set returned by the generator
Results:
c:/cygwin/bin/sh.exe -c "nosetests -v" TestNoseGenerator.test_tankXPositioning:(0,) ... ok TestNoseGenerator.test_tankXPositioning:(160,) ... ok TestNoseGenerator.test_tankXPositioning:(161,) ... ok -----------------------------------------------------Ran 3 tests in 0.219s
just download and place on path optionally run coverage for tests can check source under tree not imported by tests specify particular package
features:
--with-coverage
uses hotshot profiler very configurable requires some studying best practice:
save to file with profile-stats-file post process with python scripts for reporting
example:
nosetests --with-profile --profile-statsfile=stats.dat
26409 function calls (26158 primitive calls) in 1.997 CPU seconds Ordered by: internal time ncalls tottime percall cumtime percall filename:lineno(function) 1 0.422 0.422 0.651 0.651 c:\python24\lib\sitepackages\numeric\numeric.py:85(?) 1 0.238 0.238 0.896 0.896 c:\python24\lib\site-packages\pygame\__init__.py:25(?) 1 0.159 0.159 0.166 0.166 c:\python24\lib\pickle.py:25(?) 56 0.131 0.002 0.135 0.002 c:\kpd\proj\codemashpresentation\supertank2\src\tank.py:11(__init__) 1 0.095 0.095 1.051 1.051 c:\kpd\proj\codemashpresentation\supertank2\src\testborder.py:1(?)
on error
on failure
Nosy
similar to Cruise Control run in a window while you work save file -> view unit test output in nosy window mod in comments adds directory recursion
live demo
py.test
by Holger Krekel (and others) nose is very similar no api model -> test_ funcs, TestXXX class collects tests module, class, and member-function setup/teardown test generators test ordering test for deprecation warnings
history:
late 70s coin-op 1982 TI-99/4a ext. basic 1992 C w/Mode 13 VGA 2007 pygame!
tanks, shots, borders, key/joystick input, movement, collision detection, explosions 92 tests run in 0.6s 773 lines in 28 files
471 lines game code in 10 files (37%) 813 lines test code in 18 files (63%)
4.
background paint when sprite moves image conversion to current screen format tank movement delay / FPS timing
progress starts off slow then accelerates the resulting app is very extensible
working with a safety net is *really* nice unit tests (can) run fast
mock objects are almost indispensable when dealing with interacting libraries PyMock is amazing ratio of source to test code stayed fairly constant at about 1:2 ( 1:1.9 -> 1:1.7) nosetest / nosy combination is quite handy thinking about how to test for a behavior drives nicer implementations tests first no shortcuts
Resources
unittest module
Pymock:
Nose:
http://www.nedbatchelder.com/code/modules/coverage.html Nosy:
http://jeffwinkler.net/2006/04/27/keeping-your-nose-green/ http://codespeak.net/py/current/doc/test.html
py.test