Contribution Guide#
Thank you for looking to contribute to MC/DC! We are really excited to see what you bring to this exciting open source project! Whether you are here to make a single PR and never return, or want to become a maintainer we are pumped to work with you. We have regular developers meetings for any and all who are interested to discuss contributions to this code base.
This describes the processes of contributing to MC/DC for both internal (CEMeNT) and external developers.
Please note our code of conduct which we take seriously
Code Styling#
Our code is auto-linted for the Black code style. Your contributions will not be merged unless you follow this code style. It’s pretty easy to do this locally, just run,
pip install black
black .
in the top level MC/DC directory and all necessary changes will be automatically made for you.
Debugging#
MCDC includes options to debug the Numba JIT code. It does this by toggling Numba options using the numba.config submodule. This will result in less performant code and longer compile times but will allow for better error messages from Numba and other packages. See Numba documentation of a list of all possible debug and compiler options. The most useful set of debug options for MC/DC can be enabled with
python input.py --mode=numba_debug
Which will toggle the following debug and compiler options in Numba:
DISABLE_JIT=False
turns on the jitterNUMBA_OPT=0
Forces the compilers to form un-optimized code (other options for this are1
,2
, and3
with3
being the most optimized). This option might need to be changed if errors only result from more optimization.DEBUG=False
turns on all debugging options. This is still disabled inmcdc numba_debug
as it will print ALOT of info on your terminal screenNUMBA_FULL_TRACEBACKS=1
allows errors from sub-packages to be printed (i.e. Numpy)NUMBA_BOUNDSCHECK=1
numba will check vectors for bounds errors. If this is disabled it bound errors will result in aseg_fault
. This in consort with the previous option allows for the exact location of a bound error to be printed from Numpy subroutinesNUMBA_DEBUG_NRT=1
enables the Numba run time (NRT) statistics counter This helps with debugging memory leaks.NUMBA_DEBUG_TYPEINFER= 1
print out debugging information about type inferences that numba might need to make if a function is ill-definedNUMBA_ENABLE_PROFILING=1
enables profiler useNUMBA_DUMP_CFG=1
prints out a control flow diagram
If extra debug options or alteration to these options are required they can be toggled and passed under the mode==numba_debug
option tree near the top of mcdc/main.py
.
Caching#
MC/DC is a just-in-time (JIT) compiled code. This is sometimes disadvantageous, especially for users who might run many versions of the same simulation with slightly different parameters. As the JIT compilation scheme will only compile functions that are actually used in a given simulation, it is not a grantee that any one function will be compiled.
Developers should be very cautious about using caching features. Numba has a few documented errors around caching. The most critical of which is that functions in other files that are called by cached functions will not force a recompile, even if there are changes in those sub-functions. In this case caching should be disabled.
In MC/DC the outer most loop functions (in mcdc/loop.py
) are called to be cached.
This is done with a option on the jit flag above the individual function declarations like
nb.njit(cache=True)
def loop_fixed_source(mcdc):
# Loop over
...
To disable caching toggle these jit flags from True
to False
.
Alteratively a developer could delete the __pycache__
directory or other cache directory which is system dependent (see more about clearing the numba cache)
At some point MC/DC will enable Numba’s Ahead of Time compilation abilities. But the core development team is holding off until scheduled upgrades to AOT functionality in Numba are implemented. However if absolutely required by users numba does allow for some cache sharing.
Testing#
MC/DC has a robust testing suite that your changes must be able to pass before a PR is accepted. Unit tests for functions that have them are ran in a pure python from. Mostly this is for ensuring input operability A regression test suite (including models with analytical and experimental solutions) is provided to ensure accuracy and precision of MC/DC.
Our test suite runs on every PR, and Push. Our github based CI runs for,
linux-64 (x86)
osx-64 (x86, intel based macs)
while we do not have continuos integration we have validated MC/DC on other systems.
To run the regression tests locally, navigate to \MCDC\tests\regression
and run,
python run.py <OPTION_FLAG(s)>
and all the tests will run. Various option OPTION_FLAG
are accepted to control the tests ran,
Run a specific test (with wildcard
*
support):--name=<test_name>
Run in Numba mode:
--mode=numba
Run in multiple MPI ranks (currently support
mpiexec
andsrun
):--mpiexec=<number of ranks>
Note that flags can be combined. To add a new test:
Create a folder. The name of the folder will be the test name.
Add the input file. Name it`input.py`.
Add the answer key file. Name it answer.h5.
Make sure that the number of particles run is large enough for a good test.
If the test runs longer than 5 seconds, consider decreasing the number of particles.
When adding a new hardware backend a new instantiation of the test suit should be made.
This is done with github actions.
See the (.github/workflows
) for examples.
If a new simulation type is added (e.g. quasi montecarlo w/ davidson’s method, residual monte carlo, intrusive uq) more regression tests should be added with your PR. If you are wondering accommodations.
Adding Documentation#
It’s not everything it needs to be but we are trying!
If your contribution changes the behavior of the input deck, instillation process, or testing infrastructure your contribution must include alteration to this documentaiton.
That can be done by editing the RST files in /MCDC/docs/source/<FILENAME>.rst
.
To add a new page to the documentation,
Add a new file for example
<FILE_NAME>.rst
Add the necessary file header (for example this file is:
.. _contributions:
)Add
<FILE_NAME>
(without file extension to the.. toctree::
section ofindex.rst
)Write your contributions using
.rst
format (see this cheat sheet)
To build changes you’ve made locally before committing,
Install dependencies (we recommend:
conda install sphinx
andpip install furo
). Note that these dependencies are not installed as a part of base MC/DCRun
make html
to compileThen launch
build/html/index.html
with your browser of choice
Pull Requests#
MC/DC works off of a fork workflow in which contributors fork our repo, make alterations, and submit a pull requests. You should only submit a pull request once your code passes all tests, is properly linted, you have edited documentation (if necessary), and added any new tests (if needed).
Within your pull request documentation please list:
Type of PR (e.g. enhancement, bugfix, etc);
Link to any theory to understand what you are doing;
Link to any open/closed issues if applicable;
New functionalities implemented
Depreciated functionalities
New dependencies needed (we don’t add these lightly)
Anything else we need to give you the thorough code review you deserve!
If these things aren’t listed we will ask for clarifying questions!