##// END OF EJS Templates
Unify command-line usage information in one place....
Unify command-line usage information in one place. Fixes: https://bugs.launchpad.net/ipython/+bug/505047

File last commit:

r2417:fc43d199
r2427:d81b8d54
Show More
iptest.py
393 lines | 14.3 KiB | text/x-python | PythonLexer
Fernando Perez
Add module I forgot
r1574 # -*- coding: utf-8 -*-
"""IPython Test Suite Runner.
Fernando Perez
Cleanup testing machinery.
r1851
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972 This module provides a main entry point to a user script to test IPython
itself from the command line. There are two ways of running this script:
1. With the syntax `iptest all`. This runs our entire test suite by
calling this script (with different arguments) or trial recursively. This
causes modules and package to be tested in different processes, using nose
or trial where appropriate.
2. With the regular nose syntax, like `iptest -vvs IPython`. In this form
the script simply calls nose, but with special command line flags and
plugins loaded.
For now, this script requires that both nose and twisted are installed. This
will change in the future.
Fernando Perez
Add module I forgot
r1574 """
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 from __future__ import absolute_import
Fernando Perez
Cleanup testing machinery.
r1851 #-----------------------------------------------------------------------------
# Module imports
#-----------------------------------------------------------------------------
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972 import os
import os.path as path
Fernando Perez
Improve test suite robustness by cleaning up stale processes when possible.
r2399 import signal
Fernando Perez
Add module I forgot
r1574 import sys
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972 import subprocess
Fernando Perez
Make iptest more reliable under Win32....
r2111 import tempfile
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972 import time
Fernando Perez
Add module I forgot
r1574 import warnings
import nose.plugins.builtin
Fernando Perez
Cleanup testing machinery.
r1851 from nose.core import TestProgram
Fernando Perez
Add module I forgot
r1574
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 from IPython.utils import genutils
from IPython.utils.platutils import find_cmd, FindCmdError
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 from . import globalipapp
from .plugin.ipdoctest import IPythonDoctest
Fernando Perez
Add module I forgot
r1574
Brian Granger
Making the doctest exclude paths os independent.
r1979 pjoin = path.join
Fernando Perez
Cleanup testing machinery.
r1851 #-----------------------------------------------------------------------------
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 # Warnings control
#-----------------------------------------------------------------------------
# Twisted generates annoying warnings with Python 2.6, as will do other code
# that imports 'sets' as of today
warnings.filterwarnings('ignore', 'the sets module is deprecated',
DeprecationWarning )
#-----------------------------------------------------------------------------
Administrator
Fixing bugs with the testing system.
r1981 # Logic for skipping doctests
Fernando Perez
Cleanup testing machinery.
r1851 #-----------------------------------------------------------------------------
Administrator
Fixing bugs with the testing system.
r1981 def test_for(mod):
"""Test to see if mod is importable."""
try:
__import__(mod)
except ImportError:
return False
else:
return True
have_curses = test_for('_curses')
have_wx = test_for('wx')
Fernando Perez
Small fixes for wx-dependent tests and include clearcmd....
r2091 have_wx_aui = test_for('wx.aui')
Administrator
Fixing bugs with the testing system.
r1981 have_zi = test_for('zope.interface')
have_twisted = test_for('twisted')
have_foolscap = test_for('foolscap')
have_objc = test_for('objc')
have_pexpect = test_for('pexpect')
Brian Granger
Updated iptest to skip inputhook*.py files for doctesting.
r2083 have_gtk = test_for('gtk')
have_gobject = test_for('gobject')
Administrator
Fixing bugs with the testing system.
r1981
Administrator
Fixing doctest EXCLUDES in iptest on win32....
r1980
Brian Granger
Refactored iptest.py to work with new package org....
r2079 def make_exclude():
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 # For the IPythonDoctest plugin, we need to exclude certain patterns that
# cause testing problems. We should strive to minimize the number of
# skipped modules, since this means untested code. As the testing
# machinery solidifies, this list should eventually become empty.
Fernando Perez
Fix config part of the test suite.
r2417 # These modules and packages will NOT get scanned by nose at all for tests
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions = [pjoin('IPython', 'external'),
Fernando Perez
Fix config part of the test suite.
r2417 pjoin('IPython', 'frontend', 'process', 'winprocess.py'),
pjoin('IPython_doctest_plugin'),
pjoin('IPython', 'quarantine'),
pjoin('IPython', 'deathrow'),
pjoin('IPython', 'testing', 'attic'),
pjoin('IPython', 'testing', 'tools'),
pjoin('IPython', 'testing', 'mkdoctests'),
pjoin('IPython', 'lib', 'inputhook'),
# Config files aren't really importable stand-alone
pjoin('IPython', 'config', 'default'),
pjoin('IPython', 'config', 'profile'),
]
Brian Granger
Refactored iptest.py to work with new package org....
r2079
if not have_wx:
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'gui'))
exclusions.append(pjoin('IPython', 'frontend', 'wx'))
exclusions.append(pjoin('IPython', 'lib', 'inputhookwx'))
Brian Granger
Updated iptest to skip inputhook*.py files for doctesting.
r2083
if not have_gtk or not have_gobject:
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'lib', 'inputhookgtk'))
Brian Granger
Refactored iptest.py to work with new package org....
r2079
Brian Granger
Merging -r 1177 from lp:ipython with fixes and resolutions....
r2124 if not have_wx_aui:
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'gui', 'wx', 'wxIPython'))
Brian Granger
Merging -r 1177 from lp:ipython with fixes and resolutions....
r2124
Brian Granger
Refactored iptest.py to work with new package org....
r2079 if not have_objc:
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'frontend', 'cocoa'))
Brian Granger
Refactored iptest.py to work with new package org....
r2079
if not sys.platform == 'win32':
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'utils', 'platutils_win32'))
Brian Granger
Refactored iptest.py to work with new package org....
r2079
# These have to be skipped on win32 because the use echo, rm, cd, etc.
# See ticket https://bugs.launchpad.net/bugs/366982
if sys.platform == 'win32':
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'testing', 'plugin', 'test_exampleip'))
exclusions.append(pjoin('IPython', 'testing', 'plugin', 'dtexample'))
Brian Granger
Refactored iptest.py to work with new package org....
r2079
if not os.name == 'posix':
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'utils', 'platutils_posix'))
Brian Granger
Refactored iptest.py to work with new package org....
r2079
if not have_pexpect:
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'scripts', 'irunner'))
Brian Granger
Refactored iptest.py to work with new package org....
r2079
Brian Granger
Merging -r 1196 from lp:ipython....
r2150 # This is scary. We still have things in frontend and testing that
# are being tested by nose that use twisted. We need to rethink
# how we are isolating dependencies in testing.
if not (have_twisted and have_zi and have_foolscap):
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'frontend', 'asyncfrontendbase'))
exclusions.append(pjoin('IPython', 'frontend', 'prefilterfrontend'))
exclusions.append(pjoin('IPython', 'frontend', 'frontendbase'))
exclusions.append(pjoin('IPython', 'frontend', 'linefrontendbase'))
exclusions.append(pjoin('IPython', 'frontend', 'tests',
Brian Granger
Merging -r 1196 from lp:ipython....
r2150 'test_linefrontend'))
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'frontend', 'tests',
Brian Granger
Merging -r 1196 from lp:ipython....
r2150 'test_frontendbase'))
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'frontend', 'tests',
Brian Granger
Merging -r 1196 from lp:ipython....
r2150 'test_prefilterfrontend'))
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'frontend', 'tests',
Brian Granger
Merging -r 1196 from lp:ipython....
r2150 'test_asyncfrontendbase')),
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions.append(pjoin('IPython', 'testing', 'parametric'))
exclusions.append(pjoin('IPython', 'testing', 'util'))
Brian Granger
Merging -r 1196 from lp:ipython....
r2150
Brian Granger
Refactored iptest.py to work with new package org....
r2079 # This is needed for the reg-exp to match on win32 in the ipdoctest plugin.
if sys.platform == 'win32':
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 exclusions = [s.replace('\\','\\\\') for s in exclusions]
Brian Granger
Refactored iptest.py to work with new package org....
r2079
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 return exclusions
Administrator
Fixing doctest EXCLUDES in iptest on win32....
r1980
Fernando Perez
Cleanup testing machinery.
r1851 #-----------------------------------------------------------------------------
# Functions and classes
#-----------------------------------------------------------------------------
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 class IPTester(object):
"""Call that calls iptest or trial in a subprocess.
"""
Fernando Perez
Improve test suite robustness by cleaning up stale processes when possible.
r2399 #: string, name of test runner that will be called
runner = None
#: list, parameters for test runner
params = None
#: list, arguments of system call to be made to call test runner
call_args = None
#: list, process ids of subprocesses we start (for cleanup)
pids = None
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 def __init__(self,runner='iptest',params=None):
Fernando Perez
Improve test suite robustness by cleaning up stale processes when possible.
r2399 """Create new test runner."""
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 if runner == 'iptest':
# Find our own 'iptest' script OS-level entry point
try:
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 iptest_path = os.path.abspath(find_cmd('iptest'))
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 except FindCmdError:
# Script not installed (may be the case for testing situations
# that are running from a source tree only), pull from internal
# path:
iptest_path = pjoin(genutils.get_ipython_package_dir(),
'scripts','iptest')
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 self.runner = ['python', iptest_path, '-v']
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 else:
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 self.runner = ['python', os.path.abspath(find_cmd('trial'))]
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 if params is None:
params = []
if isinstance(params,str):
params = [params]
self.params = params
# Assemble call
self.call_args = self.runner+self.params
Fernando Perez
Improve test suite robustness by cleaning up stale processes when possible.
r2399 # Store pids of anything we start to clean up on deletion, if possible
# (on posix only, since win32 has no os.kill)
self.pids = []
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 if sys.platform == 'win32':
def _run_cmd(self):
# On Windows, use os.system instead of subprocess.call, because I
# was having problems with subprocess and I just don't know enough
# about win32 to debug this reliably. Os.system may be the 'old
# fashioned' way to do it, but it works just fine. If someone
# later can clean this up that's fine, as long as the tests run
# reliably in win32.
return os.system(' '.join(self.call_args))
else:
def _run_cmd(self):
Fernando Perez
Improve test suite robustness by cleaning up stale processes when possible.
r2399 subp = subprocess.Popen(self.call_args)
self.pids.append(subp.pid)
# If this fails, the pid will be left in self.pids and cleaned up
# later, but if the wait call succeeds, then we can clear the
# stored pid.
retcode = subp.wait()
self.pids.pop()
return retcode
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398
def run(self):
"""Run the stored commands"""
try:
return self._run_cmd()
except:
import traceback
traceback.print_exc()
return 1 # signal failure
Fernando Perez
Improve test suite robustness by cleaning up stale processes when possible.
r2399 def __del__(self):
"""Cleanup on exit by killing any leftover processes."""
if not hasattr(os, 'kill'):
return
for pid in self.pids:
try:
print 'Cleaning stale PID:', pid
os.kill(pid, signal.SIGKILL)
except OSError:
# This is just a best effort, if we fail or the process was
# really gone, ignore it.
Fernando Perez
A few small fixes so ipythonx works, and PEP-8 cleanups I found along the way.
r2400 pass
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398
def make_runners():
"""Define the top-level packages that need to be tested.
"""
nose_packages = ['config', 'core', 'extensions', 'frontend', 'lib',
'scripts', 'testing', 'utils']
trial_packages = ['kernel']
if have_wx:
nose_packages.append('gui')
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 #nose_packages = ['core'] # dbg
#trial_packages = [] # dbg
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 nose_packages = ['IPython.%s' % m for m in nose_packages ]
trial_packages = ['IPython.%s' % m for m in trial_packages ]
# Make runners, most with nose
nose_testers = [IPTester(params=v) for v in nose_packages]
runners = dict(zip(nose_packages, nose_testers))
# And add twisted ones if conditions are met
if have_zi and have_twisted and have_foolscap:
trial_testers = [IPTester('trial',params=v) for v in trial_packages]
runners.update(dict(zip(trial_packages,trial_testers)))
return runners
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972 def run_iptest():
"""Run the IPython test suite using nose.
This function is called when this script is **not** called with the form
`iptest all`. It simply calls nose with appropriate command line flags
and accepts all of the standard nose arguments.
Fernando Perez
Add module I forgot
r1574 """
warnings.filterwarnings('ignore',
'This will be removed soon. Use IPython.testing.util instead')
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 argv = sys.argv + [ '--detailed-errors',
# Loading ipdoctest causes problems with Twisted, but
# our test suite runner now separates things and runs
# all Twisted tests with trial.
'--with-ipdoctest',
'--ipdoctest-tests','--ipdoctest-extension=txt',
#'-x','-s', # dbg
Fernando Perez
Fixes to testing system: ipdocetst plugin wasn't being properly loaded.
r1761 # We add --exe because of setuptools' imbecility (it
# blindly does chmod +x on ALL files). Nose does the
# right thing and it tries to avoid executables,
# setuptools unfortunately forces our hand here. This
# has been discussed on the distutils list and the
# setuptools devs refuse to fix this problem!
'--exe',
]
Fernando Perez
Add module I forgot
r1574
Fernando Perez
Update decorators and test scripts.
r1848 # Detect if any tests were required by explicitly calling an IPython
# submodule or giving a specific path
has_tests = False
Fernando Perez
Add module I forgot
r1574 for arg in sys.argv:
Fernando Perez
Update decorators and test scripts.
r1848 if 'IPython' in arg or arg.endswith('.py') or \
(':' in arg and '.py' in arg):
has_tests = True
Fernando Perez
Add module I forgot
r1574 break
Fernando Perez
- Make ipdoctest a little cleaner by giving it separate option names....
r1910
Fernando Perez
Update decorators and test scripts.
r1848 # If nothing was specifically requested, test full IPython
if not has_tests:
Fernando Perez
Add module I forgot
r1574 argv.append('IPython')
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 ## # Construct list of plugins, omitting the existing doctest plugin, which
## # ours replaces (and extends).
plugins = [IPythonDoctest(make_exclude())]
Fernando Perez
Fixes to testing system: ipdocetst plugin wasn't being properly loaded.
r1761 for p in nose.plugins.builtin.plugins:
plug = p()
if plug.name == 'doctest':
continue
plugins.append(plug)
Fernando Perez
Massive amount of work to improve the test suite, restores doctests....
r2414 # We need a global ipython running in this process
globalipapp.start_ipython()
# Now nose can run
Fernando Perez
Add module I forgot
r1574 TestProgram(argv=argv,plugins=plugins)
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972
def run_iptestall():
"""Run the entire IPython test suite by calling nose and trial.
This function constructs :class:`IPTester` instances for all IPython
modules and package and then runs each of them. This causes the modules
and packages of IPython to be tested each in their own subprocess using
nose or twisted.trial appropriately.
"""
Brian Granger
Refactored iptest.py to work with new package org....
r2079
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972 runners = make_runners()
Brian Granger
Refactored iptest.py to work with new package org....
r2079
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 # Run the test runners in a temporary dir so we can nuke it when finished
# to clean up any junk files left over by accident. This also makes it
# robust against being run in non-writeable directories by mistake, as the
# temp dir will always be user-writeable.
curdir = os.getcwd()
testdir = tempfile.gettempdir()
os.chdir(testdir)
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972 # Run all test runners, tracking execution time
failed = {}
t_start = time.time()
Fernando Perez
Work in multiple places to improve state of the test suite....
r2398 try:
for name,runner in runners.iteritems():
print '*'*77
print 'IPython test group:',name
res = runner.run()
if res:
failed[name] = res
finally:
os.chdir(curdir)
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972 t_end = time.time()
t_tests = t_end - t_start
nrunners = len(runners)
nfail = len(failed)
# summarize results
print
print '*'*77
Fernando Perez
Small fixes for wx-dependent tests and include clearcmd....
r2091 print 'Ran %s test groups in %.3fs' % (nrunners, t_tests)
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972 print
if not failed:
print 'OK'
else:
# If anything went wrong, point out what command to rerun manually to
# see the actual errors and individual summary
Fernando Perez
Small fixes for wx-dependent tests and include clearcmd....
r2091 print 'ERROR - %s out of %s test groups failed.' % (nfail, nrunners)
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972 for name in failed:
failed_runner = runners[name]
print '-'*40
print 'Runner failed:',name
print 'You may wish to rerun this one individually, with:'
print ' '.join(failed_runner.call_args)
print
def main():
Brian Granger
Fixing small bug in iptest. Can now be run as "iptest".
r1990 if len(sys.argv) == 1:
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972 run_iptestall()
else:
Brian Granger
Fixing small bug in iptest. Can now be run as "iptest".
r1990 if sys.argv[1] == 'all':
run_iptestall()
else:
run_iptest()
Brian Granger
Refactored iptest to include the iptestall capabilities....
r1972
if __name__ == '__main__':
Fernando Perez
Small fixes for wx-dependent tests and include clearcmd....
r2091 main()