Friday, January 7, 2011

Running doctests in a django project.

Running doctests from a specific file (or files in a directory) could be painful in the django environment (specially with older versions of django).
Just uploaded a script as gist on github, which can help. This script lets you run all doctests from a given file or files in a directory. One would run this from the base directory of your site (i.e., the same directory with manage.py in it).
Note that
manage.py test xxx 
will let you run tests for your django application but will search for docstrings in
app/tests.py and models.py 
(at least of pre django 1.0). If you have a large project its unlikely that you would have such monolithic files.

#! /usr/bin/env python -Wall
"""
Run doctests from a given set of modules in django environment.
@author: Asad
@license: BSD
"""
import os, sys, glob
from sys import stderr
os.environ['DJANGO_SETTINGS_MODULE'] = 'settings'
from django.test.utils import setup_test_environment
import doctest
if len(sys.argv) < 2:
n = sys.argv[0]
print >> stderr, "Usage: %s path_to_test_file(s) [verbose 0 or 1]" % n
print >> stderr, "Example: %s appName/lib/a.py 1" % n
print >> stderr, "Example: %s appName/lib/ <== will search directory" % n
print >> stderr, "Example: %s appName/lib <== will search directory" % n
sys.exit(1)
path = sys.argv[1]
verbose = False
if len(sys.argv) == 3 and int(sys.argv[2]) == 1:
verbose = True
ls = []
if os.path.splitext(path)[1] == '.py' and os.path.isfile(path):
ls = [path]
elif os.path.isdir(path):
path = os.path.join(path, '*.py')
ls = glob.glob(path)
else:
print >> stderr, "Don't know how to deal with", path
if not os.path.exists(path):
print >> stderr, "Path does not exist"
sys.exit(1)
ls = [fn for fn in ls if not os.path.basename(fn).startswith('__')]
print "Finding doctests in", ls
results = []
for fn in ls:
setup_test_environment()
f = os.path.splitext(fn)[0].replace(os.sep, '.')
m = __import__(f, {}, {}, ['*'])
result = doctest.testmod(m, verbose=verbose)
if verbose: print >> stderr, f, result
results.append((f, result))
totalTests = 0
totalFailures = 0
for (f, result) in results:
totalFailures += result.failed
totalTests += result.attempted
if totalFailures > 0:
print >> stderr, "Summary:", \
totalFailures, "failures out of", \
totalTests, "attempts"
sys.exit(1)
else:
print "All", totalTests, "tests passed"
view raw runDocTests.py hosted with ❤ by GitHub

No comments:

Post a Comment