Integration testing is an essential part of software development. It is performed when integrating code changes into the main development branch to verify that the code works as expected. The following sections describe how to run and add integration tests used to the svMultiPhysics program.
Running a test case requires
- Build svMultiPhysics
- Install Git LFS used to download test data
- Build svZeroDSolver (only required from certain tests)
svMultiPhysics can be built following these instructions.
To automatically run test cases using pytest
you must build svMultiPhysics in a folder named build
located at the svMultiPhysics repository root directory (i.e. the svMultiPhysics directory created when doing a git clone of the svMultiPhysics repository).
The svMultiPhysics tests require finite element mesh data stored in VTK-format VTP and VTU files. These large files are managed using the Git Large File Storage (LFS) extension. Git LFS stores files as text pointers inside git until the file contents are explicitly pulled from a remote server.
The file extensions currently tracked with Git LFS are stored in this file.
Git LFS is install following this guide.
To set up Git LFS for the svMultiPhysics repository run the following commands to activate Git lfs
git lfs install
and download file data
git lfs pull
These steps need to be performed only once. All large files are handled automatically during all Git operations, like push
, pull
, or commit
.
You can run an individual test by navigating to the ./tests/cases/<physics>/<test>
folder you want to run and execute svMultiPhysics
with the svFSI.xml
input file as an argument. A more elegant way, e.g., to run a whole group of tests, is using pytest
. By default, it will run all tests defined in the test_*.py
files in the ./tests folder. Tests and input files in ./tests/cases are grouped by physics type, e.g., struct, fluid, or fsi (using the naming convention from EquationType
). Here are a couple of useful Pytest
commands:
- Run only tests matching a pattern (can be physics or test case name):
pytest -k ustruct pytest -k block_compression
- List individual test cases that were run:
pytest -v
- Print the simulation output of all tests:
pytest -rP
For more options, simply call pytest -h
.
We expect that new code is fully covered with at least one integration test. We also strive to increase our coverage of existing code. You can have a look at our current code coverage with Codecov. It analyzes every pull request and checks the change of coverage (ideally increasing) and if any non-covered lines have been modified. We avoid modifying untested lines of codeas there is no guarantee that the code will still do the same thing as before.
Here are some steps you can follow to create a new test for the code you implemented. This will satisfy the coverage requirement (see above) and help other people who want to run your code. A test case is a great way to show what your code can do! Ideally, you do this early in your development. Then you can keep running your test case as you are refactoring and optimizing your code.
- Create an example (mesh and input file) that showcases what your code can do. If it doesn't cover everything, consider adding other tests.
- Verify the results: Since an analytical solution would be ideal but is rarely available, find other ways to ensure that your code is doing the right thing (reference solutions from other codes, manufactured solutions, convergence analysis, ...).
- Crank up the tolerances: Be as strict in your linear and nonlinear solver as you can be with the test still converging. This will avoid problems when running the test on different machines or with multiple processors.
- Reduce the computational effort: Try to make the test run in a few seconds by coarsening spatial and temporal discretization (but keeping the core function of the test).
- Set the maximum number of nonlinear iterations to the one where it currently achieves the set tolerance. This way, the test will fail if the linearization gets broken in the future (but the test still slowly converges to the correct solution).
- Test the test: Does it fail if you change parts of your input file that the test should be sensitive to (e.g., material parameters if you implemented a new solid material)?
- Put your files in the appropriate folder under
./tests/cases
and append the test to the Pythontest_*.py
file. If you created a new physics type, create new ones for both. - Check that the test is executed correctly in
GitHub Actions
when opening your pull request. You should see in your coverage report that your new code is covered.
If you want to parameterize values in your test case, you can use @pytest
fixtures. We currently use them to automatically loop different numbers of processors, meshes, or input files.