-
-
Notifications
You must be signed in to change notification settings - Fork 373
Testing: Unit Tests
The Stan repository contains functional and unit tests under the directory src/test
. There is a makefile for Gnu make that runs these targets, makefile
, and a helper Python script, runTests.py
, which calls this makefile to build and run the targets.
- Prerequisite tools and settings.
- How to run tests via script
runTests.py
. - How to run a single model integration test.
The following programs and tools are required:
- a C++ compiler, either g++ or clang++
- Gnu make, version 3.8 ([http://www.gnu.org/software/make/])
- Python: versions 2.7 and up
- Unix utilities
find
andgrep
To customize Stan's build options, variables can be set in a file called make/local
. This is the preferred way of customizing the build. You can also set it more permanently by setting variables in ~/.config/stan/make.local
.
Note: variables in make/local
override variables in ~/.config/stan/make/local
.
Nota bene: these build instructions are not in the released version yet. This is for the develop
branch (post v2.18.0. Prior to this, the build instructions are similar, but not identical.
There are a lot of make variables that can be set. In general, CXXFLAGS_*
is for C++ compiler flags, CPPFLAGS_*
is for C preprocessor flags, LDFLAGS_*
is for linker flags, and LDLBIS_*
is for libraries that need to be linked in.
These are the more common make flags that could be set:
-
CXX
: C++ compiler -
MATH
: location of the Stan Math library. -
CXXFLAGS_OS
: compiler flags specific to the operating system. -
CPPFLAGS_OS
: C preprocessor flags specific to the operating system. -
O
: optimization level. Defaults to3
. -
O_STANC
: optimization level for building the Stan compiler. Defaults to0
. -
INC_FIRST
: this is a C++ compiler option. If you need to include any headers before Math's headers, this is where to specify it. Default is empty. -
LDFLAGS_OS
: linker flags for the operating system -
LDLIBS_OS
: link libraries for the operating system
For additional make options, see Building and Running Tests in the Math wiki.
The make/local
file in the Stan repository is empty. To add CXX=clang++
and O=0
to make/local
type:
> echo "CXX=clang++" >> make/local
> echo "O=0" >> make/local
To run a single header test, add "-test" to the end of the file name. Example:
> make src/stan/math/constants.hpp-test
Test all source headers to ensure they are compilable and include enough header files:
> make test-headers
Run cpplint.py on source files (unfortunately requires python 2.7)
> make cpplint
The runTests.py
Python script is responsible for:
- building the test executables using Gnu make
- running the test executables
Test executables will be rebuilt if any changes are made to dependencies of the executable.
Running the script without any arguments brings up some help. This is the current help message:
> ./runTests.py
usage: ./runTests.py <path/test/dir(/files)>
or
./runTests.py -j<#cores> <path/test/dir(/files)>
Arguments:
- the first optional argument is
-j<#cores>
for building the test in parallel. If this argument is present, it will be passed along to themake
command and will override system defaults andMAKEFLAGS
set inmake/local
. - the rest of the arguments should either specify a single test file (
*_test.cpp
) or a test directory.
To run a single unit test, specify the path of the test as an argument. Example:
> ./runTests.py src/test/unit/math/prim/scal/fun/abs_test.cpp
To run multiple unit tests at once, specify the paths of the tests separated by a space. Example:
> ./runTests.py src/test/unit/math/prim/scal/fun/abs_test.cpp src/test/unit/math/prim/scal/fun/Phi_test.cpp
These can also take the -j
flag for parallel builds. For example:
> ./runTests.py -j4 src/test/unit/math/prim/scal/fun/abs_test.cpp
Note: On some systems it may be necessary to change the access permissions to make runTests.py
executable:
> chmod +x runTests.Py
This only needs to be done once.
To run a directory of tests (recursive), specify the directory as an argument. Note: there is no trailing slash (/
). Example:
> ./runTests.py src/test/unit/math/prim
To run multiple directories, separate the directories by a space. Example:
> ./runTests.py src/test/unit/math/prim src/test/unit/math/rev
To build in parallel, provide the -j
flag with the number of cores. Example:
> ./runTests.py -j4 src/test/unit/math
We have test Stan programs located in the src/test/test-models/good
folder. For example, src/test/test-models/good/funs1.stan
. We test that each of these can be generated into valid C++ header files that can be compiled.
The src/test/integration/compile_models_test.cpp
will test all Stan programs in src/test/test-models/good
are compilable.
We can also test that a single Stan program is valid. Let's say you wanted to test this Stan file: src/test/test-models/good/map_rect.stan
If you type:
> make test/test-models/good/map_rect.hpp-test
then it'll do a number of things (in this order):
-
build
test/test-models/stanc
--- it's a stripped down version of the compiler -
generate the C++: it goes from
src/test/test-models/good/map_rect.stan
totest/test-models/good/map_rect.hpp
-
the
*-test
target attempts to compile it by including it into a file with amain()
. That file with themain()
looks like this (it instantiates the log prob with the types we care about):
// test/test-model-main.cpp
int main() {
stan::io::var_context *data = NULL;
stan_model model(*data);
std::vector<stan::math::var> params;
std::vector<double> x_r;
std::vector<int> x_i;
model.log_prob<false, false>(params, x_i);
model.log_prob<false, true>(params, x_i);
model.log_prob<true, false>(params, x_i);
model.log_prob<true, true>(params, x_i);
model.log_prob<false, false>(x_r, x_i);
model.log_prob<false, true>(x_r, x_i);
model.log_prob<true, false>(x_r, x_i);
model.log_prob<true, true>(x_r, x_i);
return 0;
}
Stan Math is transitioning towards letting function working with Eigen types accept and return general expressions (not just plain matrices). Before we can return expressions from any function exposed to Stan language, all such functions need to be able to accept expressions. This is tested using expression testing framework. It tests that:
- functions can be compiled with expressions arguments
- results when given expressions are same as when given plain matrices (including derivatives)
- functions evaluate expressions at most once
Expression testing framework gets a list of all supported function signatures from stanc3 compiler. All functions on this list need to be tested. This way any newly added function is tested without any developer effort and without opportunity to forget on testing.
During the transitioning period we also have a list of exceptions. Those are functions that have not been modified to accept expressions yet. These are not tested. With time this list will grow shorter and will eventually be removed. Adding new fucntions to this list is not allowed.
For functions, not in the exception list, expression testing framework generates C++ tests in ./test/expressions/tests*_test.cpp
, compiles and runs them.
All expression tests can be run by:
./runTests.py ./test/expressions
Testing just one or a few functions is also supported. Listing some functions this way ignores the exception list, so a function one is working on can be tested regardless of whether it is supposed to work or not. The command is (This will not work if you put --only-functions
before ./test/expressions
as flag --only-functions
accepts multiple arguments and it will eat up ./test/expressions
too, unless there is some other flag, such as -j 1
inbetween.):
./runTests.py ./test/expressions --only-functions sum sin normal_id_glm_lpdf
To verify a function is being tested on can check it is not in the exceptions list in ./lib/stan_math/test/expressions/stan_math_sigs_exceptions.expected
. One can also check that test files generated by running the expression tests (./test/expressions/tests*_test.cpp
) contain tests for all supported signatures.
These guidelines were developed for the math library, but apply to all the other C++ repos, too.
Everything in Stan math is included in a single translation unit. Some of the external libraries require libraries to be linked in. These are:
- Intel Threading Building Blocks (also known as Intel oneTBB)
- SUNDIALS for ordinary differential equations solving
- Boost MPI
- OpenCL
Within the Math library, the only build targets are the tests. The makefile can be used to build other C++ executables, with all the correct compiler flags for Math. You can also use the standalone
make feature of the Math library. See here.
To customize how the tests are built in C++, variables can be set in a file called make/local
. This is the preferred way of customizing the build. You can also set it more permanently by setting variables in ~/.config/stan/make.local
.
Note: variables in make/local
override variables in ~/.config/stan/make/local
.
There are a lot of make variables that can be set. In general, CXXFLAGS_*
is for C++ compiler flags, CPPFLAGS_*
is for C preprocessor flags, LDFLAGS_*
is for linker flags, and LDLBIS_*
is for libraries that need to be linked in.
These are the more common make flags that could be set:
-
CXX
: C++ compiler -
CXXFLAGS_OS
: compiler flags specific to the operating system. -
CPPFLAGS_OS
: C preprocessor flags specific to the operating system. -
O
: optimization level. Defaults to3
. -
INC_FIRST
: this is a C++ compiler option. If you need to include any headers before Math's headers, this is where to specify it. Default is empty. -
LDFLAGS_OS
: linker flags for the operating system -
LDLIBS_OS
: link libraries for the operating system
These are the rest of the variables that can be set:
- C++ compiler flags
-
CXXFLAGS_LANG
: sets the language. Currently defaults to-std=c++1y
-
CXXFLAGS_WARNINGS
: compiler options to squash compiler warnings -
CXXFLAGS_BOOST
: Boost-specific compiler flags -
CXXFLAGS_EIGEN
: Eigen-specific compiler flags -
CXXFLAGS_OPENCL
: OpenCL-specific compiler flags -
CXXFLAGS_MPI
: MPI-specific compiler flags
-
- C preprocessor flags
-
CPPFLAGS_LANG
: CPPFLAGS_WARNINGS
CPPFLAGS_BOOST
CPPFLAGS_EIGEN
CPPFLAGS_OPENCL
CPPFLAGS_MPI
-
- Linker flags
LDFLAGS_LANG
LDFLAGS_WARNINGS
LDFLAGS_BOOST
LDFLAGS_EIGEN
LDFLAGS_OPENCL
LDFLAGS_MPI
- Link libraries
LDLIBS_LANG
LDLIBS_WARNINGS
LDLIBS_BOOST
LDLIBS_EIGEN
LDLIBS_OPENCL
LDLIBS_MPI
- Misc
-
OS
: operating system. Defaults toDarwin
,Linux
, orWindowsNT
. -
MATH
: the location of the math library. Defaults to empty variable (used for Stan and CmdStan). -
EIGEN
: location of the Eigen headers -
BOOST
: location of the Boost headers -
SUNDIALS
: location of the Sundials headers -
INC
: the includes for the C++ compiler -
INC_SUNDIALS
: the Sundials includes -
INC_GTEST
: the Google test includes -
EXE
: the executable file extension. On Windows, defaults to.EXE
. On other operating system, defaults to an empty variable.
-
For debugging, there's a useful target for printing Stan variables. It's print-*
where *
is the variable. For example, on a Mac:
> make print-CXX
CXX = clang++
We use the GoogleTest (also sometimes named GTest) framework for writing tests and GnuMake and Python for running them. We mostly rely on the EXPECT_*
macros for non-fatal assertion. These macros are used throughout the tests to test for expected behaviour. See link for an overiew of the macros provided by the GTEST library.
We extended the GTest macros with the following macros for simplified testing of expected values in Eigen matrices, vector and row vectors and also std::vector:
For matrices of doubles:
-
EXPECT_MATRIX_EQ(A,B)
which tests for elementwise equality of A and B usingEXPECT_EQ
-
EXPECT_MATRIX_FLOAT_EQ(A,B)
which tests for elementwise equality of A and B usingEXPECT_FLOAT_EQ
-
EXPECT_MATRIX_NEAR(A, B, DELTA)
which tests if the elementwise difference of A and B is larger than DELTA usingEXPECT_NEAR
For std::vectors:
-
EXPECT_STD_VECTOR_FLOAT_EQ(A, B)
which tests for elementwise equality of A and B usingEXPECT_FLOAT_EQ
Exceptions:
-
EXPECT_THROW_MSG(expr, T_e, msg)
which tests if the expression throws the expected exception with the expected message. -
EXPECT_THROW_MSG_WITH_COUNT(expr, T_e, msg, count)
which tests if the expression throws the expected exception with a specific number of occurrences of the expected message in the throw message.
Their implementation can be found here.
The main function of the makefiles is to build test executables. The typical way to build test executables is to use the runTests.py
python script, but we can use make to directly generate a test executable.
The name of the test executable is the name of the test with the file extension removed for Linux and Mac or replacing the file extension with .exe
for Windows. For example, to run the test test/unit/math_include_test.cpp
, we build on Linux and Mac with:
> make test/unit/math_include_test
or on Windows with:
> make test/unit/math_include_test.exe
We can then run the executable from the command line.
Along with the test executable, the makefiles generate dependency files. The dependency files are generated by using the C++ compiler to list the headers that the test depends on. The dependency file is automatically generated by make and included by make so it knows what header files the test depends on. If none of those files changed, then the executable doesn't need to be rebuilt.
We can build the dependency file directly. For the test/unit/math_include_test.cpp
example, we can build the dependency file using make:
> make test/unit/math_include_test.d
If you look at the first couple of lines, you'll see:
test/unit/math_include_test.o test/unit/math_include_test.d: \
test/unit/math_include_test.cpp stan/math.hpp stan/math/rev/mat.hpp \
Make uses the timestamp of the files to determine whether it needs to be rebuilt. If any of the files listed after the targets (after the :
) are updated, the executable will be rebuilt. If you've built all the unit tests and only change a single header file, building the unit tests again will selectively rebuild the tests that depends on that header.
The easiest way to build and run tests is to use the runTests.py
python script. To run unit tests:
> ./runTests.py test/unit
The Stan testing process depends on the makefiles in Math.
-
We're trying to make sure all the boundary conditions get called
- zero-size inputs for containers
- 0, NaN, inf, and -inf inputs for floating point
-
We want to make sure all tests get tickled and the appropriate exception types are thrown and documented.
-
We want to test that autodiff works
- reverse-mode
- forward mode: double (1st order), var (2nd order), fvar (3rd order)
There should be a finite difference testing framework soon. An alternative is to define a second, simpler templated versio of a function and test against that.
The place to start for doc for GoogleTest (other than looking at our existing examples) are:
-
Primer: an overview
-
Advance Guide: good description of available tests and best API guide I know
The documentation (above) is excellent so check there for details and the complete list of tests. The following is a small sample of commonly used test types in Stan and links to the relevant sections of the GoogleTest documentation.
- EXPECT/ASSERT EXPECT records the result but allows further tests in the same function, ASSERT exits the current function without cleanup. Stan unit tests use both forms.
- EXPECT_TRUE EXPECT_TRUE/FALSE test that the condition is TRUE/FALSE.
- EXPECT_EQ Tests that a calculated value equals (==) an expected result exactly, not appropriate for floating point comparison.
- EXPECT_FLOAT_EQ Tests that a floating point calculation results to known output values.
- EXPECT_NEAR Like EXPECT_FLOAT_EQ but a third argument sets the threshold used.
- ASSERT_THROW Tests that a specific exception type is thrown.
- ASSERT_NO_THROW Tests that the statement does not throw. Some parts of the code are not supposed to throw... not sure which.
Running tests in the Stan Math Library require these libraries, which are all included in the repository:
- Boost
- Eigen
- Google Test
- CppLint (optional)
No additional configuration is necessary to start running with the default libraries.
If you want to use custom locations for the library locations, set these makefile variables:
EIGEN
BOOST
GTEST
-
CPPLINT
(optional)
Example ~/.config/stan/make.local
file:
BOOST = ~/boost
To run tests, you will need a copy of the Math library, a C++ compiler, make, and python 2.x (for the test script).
To run the unit tests, type:
> ./runTests.py test/unit
To run the auto-generated distribution tests, type:
> ./runTests.py test/prob
To run the multiple translation unit tests, type:
> ./runTests.py test/unit/multiple_translation_units_test.cpp
If you see this message:
------------------------------------------------------------
make generate-tests -s
test/prob/generate_tests.cpp:9:10: fatal error: 'boost/algorithm/string.hpp' file not found
#include <boost/algorithm/string.hpp>
^
1 error generated.
make: *** [test/prob/generate_tests] Error 1
make generate-tests -s failed
the library paths have not been configured correctly.
To test headers,
> make test-headers
In order to cut down on the inconsistencies in stan/math non-probability unit tests and the menial labor that goes into writing the unit tests, we need to design a unified testing framework.
Right now we leave unit test writing to the developer. The developer must manually instantiate all combinations of admissible types for each function written. Next, it is up to the developer to write input tests, Nan tests, valid values, etc.
While we do have a framework for probability tests, the current code isn't testing second and third derivatives as was intended.
Generalize the current probability function testing framework so it can be used for math unit tests. The framework will implement the following tests:
- nan tests
- invalid inputs
- gradient tests
- hessian tests
- gradient of hessian tests
- values
Bob's first cut at a recursive template metaprogramming approach to automatically expanding the type combinations is pasted below in sections. The idea is that we'll move to a more general version of Nomad's functor-based testing suite. Instead of requiring explicitly written functors for each combination of types and test, we'll require a more generally written functor like so:
struct multiply_f {
template <typename T1, typename T2>
typename promote_args<T1,T2>::type
operator()(const cons<T1, cons<T2,nil> >& x) const {
return multiply_base(x.h_, x.t_.h_);
}
};
The supporting code is below, along with the TEST construction.
Supporting cons struct required for implementation:
#include <gtest/gtest.h>
#include <stan/agrad/rev.hpp>
#include <boost/math/tools/promotion.hpp>
#include <iostream>
#include <vector>
namespace stan {
namespace test {
struct nil {
nil() { }
};
template <typename H, typename T>
struct cons {
const H& h_;
const T& t_;
cons(const H& h, const T& t)
: h_(h),
t_(t)
{ }
};
// cons factory
const nil& cf() {
static const nil n;
return n;
}
template <typename H>
cons<H,nil>
cf(const H& h) {
return cons<H,nil>(h,cf());
}
template <typename H, typename T>
cons<H,T>
cf(const H& h, const T& t) {
return cons<H,T>(h,t);
}
template <typename F, typename T>
struct bind_head_var {
const F& f_;
const size_t n_;
bind_head_var(const F& f, size_t n)
: f_(f),
n_(n)
{ }
// T2 can't just be T because remaining types change
template <typename T2>
stan::agrad::var
operator()(const T2& t,
std::vector<stan::agrad::var>& xs) const {
using stan::agrad::var;
cons<var,T2> c(xs[n_],t);
return f_(c,xs);
}
};
template <typename F, typename T>
struct bind_head_dbl {
const F& f_;
const double x_;
bind_head_dbl(const F& f, double x)
: f_(f),
x_(x)
{ }
// T2 can't just be T because remaining types change
template <typename T2>
stan::agrad::var
operator()(const T2& t,
std::vector<stan::agrad::var>& xs) const {
cons<double,T2> c(x_,t);
return f_(c,xs);
}
};
template <typename T>
struct promote_cons {
};
template <>
struct promote_cons<nil> {
typedef double return_t;
};
template <typename H, typename T>
struct promote_cons<cons<H,T> > {
typedef
typename boost::math::tools::promote_args<H,
typename promote_cons<T>::return_t>::type
return_t;
};
template <typename F>
struct ignore_last_arg {
const F& f_;
ignore_last_arg(const F& f) : f_(f) { }
template <typename C>
typename promote_cons<C>::return_t
operator()(const C& x) const {
return f_(x);
}
template <typename C, typename T>
typename promote_cons<C>::return_t
operator()(const C& x, T& v) const {
return f_(x);
}
};
}
}
Examples of a value test:
using std::vector;
using boost::math::tools::promote_args;
using stan::agrad::var;
using stan::test::bind_head_dbl;
using stan::test::bind_head_var;
using stan::test::cf;
using stan::test::cons;
using stan::test::ignore_last_arg;
using stan::test::nil;
using stan::test::promote_cons;
template <typename V, typename F, typename T>
void test_funct(const V& fx,
const F& f,
const T& x);
template <typename V, typename F>
void test_eq_rev(const V& fx,
const F& f,
const nil& x,
vector<double>& xs) {
vector<var> xs_v(xs.size());
for (size_t i = 0; i < xs.size(); ++i)
xs_v[i] = xs[i];
nil n;
var fx_v = f(n,xs_v);
EXPECT_FLOAT_EQ(fx, fx_v.val());
}
template <typename V, typename F, typename T>
void test_eq_rev(const V& fx,
const F& f,
const cons<double,T>& x,
vector<double>& xs) {
bind_head_dbl<F,T> bh_rev(f,x.h_);
test_eq_rev(fx, bh_rev, x.t_, xs);
bind_head_var<F,T> bh_rev_var(f,xs.size());
vector<double> xs_copy(xs);
xs_copy.push_back(x.h_);
test_eq_rev(fx, bh_rev_var, x.t_, xs_copy);
}
// test reverse-mode in all args
template <typename V, typename F, typename C>
void test_eq_rev(const V& fx,
const F& f,
const C& x) {
vector<double> xs;
test_eq_rev(fx,ignore_last_arg<F>(f),x,xs);
}
template <typename V, typename F, typename T>
void test_funct(const V& fx,
const F& f,
const T& x) {
test_eq_rev(fx, f, x);
}
Example of how a developer would implement tests using the code above.
// ACTUAL TESTS START HERE
#include <typeinfo>
template <typename T1, typename T2>
typename boost::math::tools::promote_args<T1,T2>::type
multiply_base(const T1& x1, const T2& x2) {
std::cout << "T1=" << typeid(T1).name()
<< "; T2=" << typeid(T2).name() << std::endl;
return x1 * x2;
}
struct multiply_f {
template <typename T1, typename T2>
typename promote_args<T1,T2>::type
operator()(const cons<T1, cons<T2,nil> >& x) const {
return multiply_base(x.h_, x.t_.h_);
}
};
TEST(Foo, Bar) {
test_funct(6.0, multiply_f(), cf(3.0,cf(2.0)));
}
We have a few options for the interaction with the test framework. One option is to have the developer explicitly call each test, like so:
TEST(foo, bar1) {
test_funct(6.0, multiply_f(), (3.0, 2.0));
test_funct(-6.0, multiply_f(), (3.0, -2.0));
test_funct(6.0, multiply_f(), (3.0, -2.0));
test_funct(6.0, multiply_f(), (3.0, 2.0));
test_funct(6.0, multiply_f(), (3.0, 2.0));
}
TEST(foo, invalid) {
test_funct_invalid();
}
The other is to have a single call to the test framework but require subclass definitions, as in src/test/unit/prtest_fixture_distr.hpp.
In many tests, we compare values computed by two different mechanisms, without any assurance any one of them is "correct". For this case there is the stan::test::expect_near_rel
function which handles NA and Inf values and uses the stan::test::relative_tolerance
class to express tolerances.
The main idea about tolerances is that we care mainly about relative tolerance, but this fails around zero, so we use absolute tolerance around zero. I.e. the actual relative tolerance grows as we approach zero.
More details and reasoning behind this can be found in the code relative_tolerance
, on Dicourse and in PR #1657.
Once MPI is enabled, the runTests.py
script in the cmdstan/stan/lib/stan_math
directory will run all tests in an environment which resembles a MPI run. There are two types of tests:
-
conventional tests: This includes all unit tests which do not use any MPI parallelism. In order to run these tests in a MPI like way we compile these with the
mpicxx
command and execute them with thempirun
run command. However, we explicitly disable for these serial tests the use of multiple processes. That is, therunTests.py
script executes the tests withmpirun -np 1 test/unit/.../.../some_test
. Starting up multiple threads for serial only tests would lead to race conditions since these codes are not prepared for parallelism. -
dedicated MPI tests: All tests matching the regular expression
*mpi_*test.cpp
will be executed byrunTests.py
withmpirun -np #CPU test/unit/.../.../some_mpi_test.cpp
and #CPU will be set to the same argument as given to the-j
option ofrunTests.py
, but it will use at least 2 processes. This is to ensure that the MPI tests are actually run with multiple processes in parallel to emulate behavior under MPI execution. Note thatmpirun
is usually configured to disallow #CPU to exceed the number of physical CPUs found on the machine.
To illustrate what is happening let's consider two examples (assuming MPI is enabled as described above):
- conventional test:
./runTests.py test/unit/math/prim/mat/functor/map_rect_test.cpp
# => compilation with mpicxx
# => execution with mpirun using a single process
# mpirun -np 1 test/unit/math/prim/mat/functor/map_rect_test
- dedicated MPI test:
./runTests.py test/unit/math/prim/arr/functor/mpi_cluster_test.cpp
# => compilation with mpicxx
# => execution with mpirun using at least two processes
# mpirun -np 2 test/unit/math/prim/arr/functor/mpi_cluster_test
./runTests.py -j8 test/unit/math/prim/arr/functor/mpi_cluster_test.cpp
# => compilation with mpicxx
# => execution with mpirun using 8 processes
# mpirun -np 8 test/unit/math/prim/arr/functor/mpi_cluster_test