|
|
"""Decorators for labeling test objects.
|
|
|
|
|
|
Decorators that merely return a modified version of the original
|
|
|
function object are straightforward. Decorators that return a new
|
|
|
function object need to use
|
|
|
nose.tools.make_decorator(original_function)(decorator) in returning
|
|
|
the decorator, in order to preserve metadata such as function name,
|
|
|
setup and teardown functions and so on - see nose.tools for more
|
|
|
information.
|
|
|
|
|
|
This module provides a set of useful decorators meant to be ready to use in
|
|
|
your own tests. See the bottom of the file for the ready-made ones, and if you
|
|
|
find yourself writing a new one that may be of generic use, add it here.
|
|
|
|
|
|
NOTE: This file contains IPython-specific decorators and imports the
|
|
|
numpy.testing.decorators file, which we've copied verbatim. Any of our own
|
|
|
code will be added at the bottom if we end up extending this.
|
|
|
"""
|
|
|
|
|
|
# Stdlib imports
|
|
|
import inspect
|
|
|
import sys
|
|
|
|
|
|
# Third-party imports
|
|
|
|
|
|
# This is Michele Simionato's decorator module, also kept verbatim.
|
|
|
from decorator_msim import decorator, update_wrapper
|
|
|
|
|
|
# Grab the numpy-specific decorators which we keep in a file that we
|
|
|
# occasionally update from upstream: decorators_numpy.py is an IDENTICAL copy
|
|
|
# of numpy.testing.decorators.
|
|
|
from decorators_numpy import *
|
|
|
|
|
|
##############################################################################
|
|
|
# Local code begins
|
|
|
|
|
|
# Utility functions
|
|
|
|
|
|
def apply_wrapper(wrapper,func):
|
|
|
"""Apply a wrapper to a function for decoration.
|
|
|
|
|
|
This mixes Michele Simionato's decorator tool with nose's make_decorator,
|
|
|
to apply a wrapper in a decorator so that all nose attributes, as well as
|
|
|
function signature and other properties, survive the decoration cleanly.
|
|
|
This will ensure that wrapped functions can still be well introspected via
|
|
|
IPython, for example.
|
|
|
"""
|
|
|
import nose.tools
|
|
|
|
|
|
return decorator(wrapper,nose.tools.make_decorator(func)(wrapper))
|
|
|
|
|
|
|
|
|
def make_label_dec(label,ds=None):
|
|
|
"""Factory function to create a decorator that applies one or more labels.
|
|
|
|
|
|
:Parameters:
|
|
|
label : string or sequence
|
|
|
One or more labels that will be applied by the decorator to the functions
|
|
|
it decorates. Labels are attributes of the decorated function with their
|
|
|
value set to True.
|
|
|
|
|
|
:Keywords:
|
|
|
ds : string
|
|
|
An optional docstring for the resulting decorator. If not given, a
|
|
|
default docstring is auto-generated.
|
|
|
|
|
|
:Returns:
|
|
|
A decorator.
|
|
|
|
|
|
:Examples:
|
|
|
|
|
|
A simple labeling decorator:
|
|
|
>>> slow = make_label_dec('slow')
|
|
|
>>> print slow.__doc__
|
|
|
Labels a test as 'slow'.
|
|
|
|
|
|
And one that uses multiple labels and a custom docstring:
|
|
|
>>> rare = make_label_dec(['slow','hard'],
|
|
|
... "Mix labels 'slow' and 'hard' for rare tests.")
|
|
|
>>> print rare.__doc__
|
|
|
Mix labels 'slow' and 'hard' for rare tests.
|
|
|
|
|
|
Now, let's test using this one:
|
|
|
>>> @rare
|
|
|
... def f(): pass
|
|
|
...
|
|
|
>>>
|
|
|
>>> f.slow
|
|
|
True
|
|
|
>>> f.hard
|
|
|
True
|
|
|
"""
|
|
|
|
|
|
if isinstance(label,basestring):
|
|
|
labels = [label]
|
|
|
else:
|
|
|
labels = label
|
|
|
|
|
|
# Validate that the given label(s) are OK for use in setattr() by doing a
|
|
|
# dry run on a dummy function.
|
|
|
tmp = lambda : None
|
|
|
for label in labels:
|
|
|
setattr(tmp,label,True)
|
|
|
|
|
|
# This is the actual decorator we'll return
|
|
|
def decor(f):
|
|
|
for label in labels:
|
|
|
setattr(f,label,True)
|
|
|
return f
|
|
|
|
|
|
# Apply the user's docstring, or autogenerate a basic one
|
|
|
if ds is None:
|
|
|
ds = "Labels a test as %r." % label
|
|
|
decor.__doc__ = ds
|
|
|
|
|
|
return decor
|
|
|
|
|
|
#-----------------------------------------------------------------------------
|
|
|
# Decorators for public use
|
|
|
|
|
|
skip_doctest = make_label_dec('skip_doctest',
|
|
|
"""Decorator - mark a function or method for skipping its doctest.
|
|
|
|
|
|
This decorator allows you to mark a function whose docstring you wish to
|
|
|
omit from testing, while preserving the docstring for introspection, help,
|
|
|
etc.""")
|
|
|
|
|
|
def skip(msg=''):
|
|
|
"""Decorator - mark a test function for skipping from test suite.
|
|
|
|
|
|
This function *is* already a decorator, it is not a factory like
|
|
|
make_label_dec or some of those in decorators_numpy.
|
|
|
|
|
|
:Parameters:
|
|
|
|
|
|
func : function
|
|
|
Test function to be skipped
|
|
|
|
|
|
msg : string
|
|
|
Optional message to be added.
|
|
|
"""
|
|
|
|
|
|
import nose
|
|
|
|
|
|
def inner(func):
|
|
|
|
|
|
def wrapper(*a,**k):
|
|
|
if msg: out = '\n'+msg
|
|
|
else: out = ''
|
|
|
raise nose.SkipTest("Skipping test for function: %s%s" %
|
|
|
(func.__name__,out))
|
|
|
|
|
|
return apply_wrapper(wrapper,func)
|
|
|
|
|
|
return inner
|
|
|
|
|
|
# Decorators to skip certain tests on specific platforms.
|
|
|
skip_win32 = skipif(sys.platform=='win32',"This test does not run under Windows")
|
|
|
skip_linux = skipif(sys.platform=='linux2',"This test does not run under Linux")
|
|
|
skip_osx = skipif(sys.platform=='darwin',"This test does not run under OSX")
|
|
|
|