jwst
is an open source python package, and the source code is available in the
JWST Github repository. New contributions
and contributors are very welcome! Do not hesitate to reach out to the package
maintainers if you are new to open-source development
or if you have any questions/concerns. We just ask that all contributors adhere
to the Space Telescope Code of Conduct.
If you would like to report a bug or request a new feature, this can be done by opening a new issue.
If you would like to contribute code, this is done by submitting a pull request
to the "main" branch of spacetelescope/jwst
. To do this, we recommend the
following workflow (which assumes you already have a Github account / command line tools).
If you are also new to git, please refer to the git reference manual
for an overview of git basics.
First, to clarify some terms that will be commonly used here:
upstream
refers to the mainjwst
repository. this is where code from all contributors is ultimately merged into and where releases of the package will be made from.origin
refers to the online fork you made of the upstreamjwst
repository.local
refers to the clone you made of the origin on your computer.- The term
remote
in this context can refer to origin, or upstream. in general, it means anything hosted online on Github.
The first step is to create your own 'remote' (online) and 'local' (on your machine)
clones of the central spacetelescope/jwst
repository. You will make code changes
on your machine to your 'local' clone, push these to 'origin' (your online fork),
and finally, open a pull request to the ''main'' branch of spacetelescope/jwst
.
-
On the 'spacetelescope/jwst' Github repository page, 'fork' the JWST repository to your own account space by clicking the appropriate button on the upper right-hand side. This will create an online clone of the main JWST repository but instead of being under the 'spacetelescope' organization, it will be under your personal account space. It is necessary to fork the repository so that many contributors aren't making branches directly on 'spacetelescope/jwst'.
-
Now that you have remotely forked
jwst
, it needs to be downloaded to your machine. To create this 'local' clone, choose an area on your file system and use thegit clone
command to dowload your remote fork on to your machine.>> cd directory >> git clone git@github.com:<your_username>/jwst.git
-
Make sure that your references to 'origin' and 'upstream' are set correctly - you will need this to keep everything in sync and push your changes online. While your inital local clone will be an exact copy of your remote, which is an exact copy of the 'upstream'
spacetelescope/jwst
, these all must be kept in sync manually (via git fetch/pull/push).To check the current status of these references:
>> git remote -v
After your inital clone, you will likely be missing the reference to 'upstream'
(which is just the most commonly used name in git to refer to the main project repository - you
can call this whatever you want but the origin/upstream conventions are most commonly used) - to
set this, use the add
git command:
If you are using an SSH key to authenticate.
>> git remote add upstream git@github.com:spacetelescope/jwst.git
Otherwise, you can simply set it to the repository URL but you will have to enter your password every time you fetch/push
>> git remote add upstream https://github.com/spacetelescope/jwst
If you ever want to reset these URLs, add references to other remote forks of
jwst
for collaboration, or change from URL to SSH, you can use the related
git remote set-url
command.
It is a standard practice in git to create a new 'branch' (off upstream/main
)
for each new feature or bug fix. You can call this branch whatever you like - in
this example, we'll call it 'my_feature'. First, make sure you
have all recent changes to upstream by 'fetching' them:
>> git fetch upstream
The following will create a new branch off local/main called 'my_feature', and automatically switch you over to your new branch.
>> git checkout -b my_feature upstream/main
Next, you will create a new Python environment where you will install your new
branch of jwst
. We will show here how to use conda to manage environments, but
there are other options.
- Create a conda environment.
It is good practice to maintain different environments for different versions
of JWST and its dependencies. You will likely want to maintain one, for example,
for the latest released version of JWST (i.e. what you get by doing pip install jwst
),
as well as one for development. Assuming the user has conda installed,
here we will create a new conda environment called 'jwst_dev' where we can install the new branch of our cloned repository.
>> conda create -n jwst_dev python
Doing this will create a new environment with just some basic packages (i.e setuptools, pip) installed.
- Installing
jwst
in this environment
Make sure you are in your new environment:
>> conda activate jwst_dev
And now in the top level of your local jwst
repository, ensuring you're
on the 'my_feature' branch:
>> pip install -e .
This will install jwst
from this cloned source code in 'editable' mode,
meaning that you can import the code from this directory when within a Python
session. This makes it easier for development because you can have the code
you're editing somewhere convenient in your file system vs. with other packages
in 'site-packages'. If you cloned the repository on your Desktop, for example,
you can modify it there and Python will know that is where the source code is
when you're importing it within a Python session.
Note : If you use it, make sure to install iPython in your new environment as well. Otherwise, it will pick up packages from the base environment instead.
Now that you've forked, cloned, made a new branch for your feature, and installed
it in a new environment for development of jwst
, you are ready to make changes
to the code. As you make changes, make sure to git commit -m <"some message">
frequently
(in case you need to undo something by reverting back to a previous commit - you
cant do this if you commit everything at once!). After you've made your desired
changes, and committed these changes, you will need to push them online to your
'remote' fork of jwst
:
>> git push origin my_feature
If the changes are significant, please make an entry in CHANGES.rst
in the top
level jwst
directory with a short description of the changes you've made and, once
you open a pull request, add the corresponding PR number.
Now, you can open a pull request on the main branch of the upstream jwst
repository.
-
On the
spacetelescope/jwst
web page, after you push your changes you should see a large green banner appear at the top prompting you to open a pull request with your recently pushed changes. You can also open a pull request from the pull request tab on that page. Select your fork and your 'my_feature' branch, and open a pull request against the 'main' branch. -
There is now a checklist of items that need to be done before your PR can be merged.
- The continuous integration (CI) tests must complete and pass. The CI runs several different checks including running the unit tests, ensuring the documentation builds, checking for code style issues (see the PEP8 style guide), and ensuring any changes are covered by unit tests. The CI runs upon opening a PR, and will re-run any time you push commits to that branch.
- You will need to add a change log entry in CHANGES.rst if your contribution is a new feature or bug fix. An entry is not required for small fixes like typos.
- Your PR will need to be reviewed and approved by at least two maintainers. They may require changes from you before your code can be merged, in which case you will need to go back and make these changes and push them (they will automatically appear in the PR when they're pushed to origin/my_feature).
As jwst
is constantly evolving, you will often encounter the situation where you've
made changes to your branch off 'main', but in the time its taken you to make those
changes, 'upstream/main' has evolved with new commits from other developers. In this
situation, you will want to make sure you incorporate these changes into your branch.
Rebasing allows you to do two things - 1. apply others changes on top of yours, and 2.
squash your commits, even if there aren't new changes to apply.
Periodically, while writing code, to keep your branch up to date you will want to do an interactive rebase against upstream/main to apply any new changes on top of yours:
>> git rebase -i upstream/main
This will then prompt you to select commits and commit messages - if you select
just the top commit, this will 'squash' the others and combine them into one. You
will be prompted to resolve any conflicts if you've made modifications to the same
file someone else has. Once you've completed your rebase, you must force push
your
branch to origin/my_feature to make your local and remote match.
>> git push -f origin/my_feature
Before merging a PR, we typically would like you to rebase and squash all of your
commits into a single one, which is also done with git rebase
jwst
uses sphinx to generate documentation, which is then hosted online on readthedocs.
You can access two versions of the documentation on the JWST readthedocs website
- the 'latest' version is whatever is currently on the main branch, and the 'stable' version is the last released version. If you successfully merge a PR with documentation changes, they will only appear on 'latest' until the next JWST release.
All of the documentation resides in the jwst/docs
subdirectories (mainly within
directories in jwst/docs/jwst
, organized by module). The documentation is written
in .rst
(reStructured text) files - if you wish to make changes or add to the documentation,
do so in these files in the appropriate location. reStructured text is the markup
language used by sphinx, for information on the syntax refer to the sphinx documentation.
While writing documentation, you will want to make sure the documentation builds
successfully (i.e, produces a working .html file). This happens on the CI when you
open a pull request, but it is a good idea to build the documentation yourself locally
as well. Before you do this, you will need to make sure you have the correct dependencies
installed in your current environment. All of these optional dependencies are specified
in pyproject.toml
and include things like the correct version of sphinx, as well as the
necessary sphinx themes that the project uses. These do not install automatically
when you install jwst
unless directly specified. To do this, while in the top level
directory of jwst
on your my_feature branch:
>> pip install -e ".[docs]"
Now, with the correct documentation dependencies installed, you can attempt to build
the documentation locally. To do this, enter into the jwst/docs
subdirectory and do:
>> make html
If the documentation successfully builds, the output HTML files will be output in
the _build/html
subdirectory. You can open the main index.html
file in your browser
and explore the full documentation pages just like the readthedocs site. If there were
any errors or warnings, a traceback will be printed on your terminal. Small errors, like
a typo in a link to another section, can cause frustrating errors so it is good practice
to build the docs frequently when editing them.
Unit tests are located in each module in jwst
in a tests
subdirectory (for example,
jwst/jwst/ramp_fitting/tests
). These tests run the code on simplified datasets to make
sure there are no breaking changes introduced. Most lines of jwst
should be covered
by a unit test, so when adding code you will often need to write a new test or add
to an existing test to ensure adequate coverage.
Take a look around at the existing tests for a template - a majority of these tests
use a @pytest.fixture to set up a common dataset (usually a very simple datamodel
with some observational parameters and simplified data/dq arrays) for tests in that
file, and the test functions themselves set up a scenario to run a pipeline/step
under certain conditions, and culminate in a set of assertions that need to pass
(or fail if the test is marked as xfail
).
The CI will run the unit tests on your branch when you open a pull request. They
will re-run every time you push commits to this remote branch as well. Unit tests
must pass on any changes you make, and if you're introducing new code that isn't
covered by an existing test, you will need to write one. The codecoverage
check
that runs on the CI when you open a pull request will tell you if you've introduced
any new lines that aren't covered by a test.
You can also run unit tests locally, and you should do this periodically to test your code. To do this, you will need to install the optional dependencies needed for running tests.
>> pip install -e ".[test]"
This will install the optional 'test' dependencies specified in pyproject.toml
that
don't install by default. The package pytest
is one of these and is what's used
to run the tests. pytest
searches through all the directories in your repository
(underneath the directory from which it was invoked command line) and looks for any
directories called 'test' or .py files with the word 'test' in the name. Functions
in these files will be executed.
To run all of the jwst
unit tests, while in the jwst/jwst
level directory, simply
run the command:
>> pytest
If you want to run all the tests for a single module, for example ramp_fitting
,
you can run this from either the 'jwst/jwst/ramp_fitting' OR the 'jwst/jwst/ramp_fitting/tests' directory/.
To run all tests within a single test file (for example, all tests in jwst/jwst/jump/test_detect_jumps.py
).
>> pytest test_detect_jumps.py
pytest
can be configured with many different flags - see the pytest documentation
to see all of these. Here we will summarize a few of the most useful options.
If you are writing and debugging a test, you may find it helpful to have print statements printed to the terminal while running tests (which doesn't happen by default). To do this,
>> pytest -s
If you want to only run a specific test function within a file (or only tests with names that contain a certain string):
>> pytest -k testname.
This will search all files under the directory you're in for files or functions with
the string 'test' in them, and within those files run only the test functions that
contain the string testname
.
Within the test files themselves, decorators can be used to control the behavior of the test. Some of the more useful decorators include: 1. @pytest.mark.parametrize can be used to run a single test on multiples sets of input parameters 2. @pytest.skip can be used to skip tests altogether, or under specific conditions (for example, only when being run by the CI) 3. @pytest.fixture to declare a fixture 4. @pytest.mark.xfail will make a test pass only if it fails.
If you encounter the scenario where you wish to simultaneously make changes in
jwst
and also in one of its dependencies like stcal
or stpipe
, we recommend
that you create a new environment with development versions of both of these packages.
To do this, you can follow the same workflow outlined in the 'Contributing code' section
of this guide. To summarize, you will want to create a new Python environment
(called, for example, jwst_stcal_dev), fork and clone a local copy of jwst
if you
haven't already and install this (doing pip install -e .
in the top level directory
of jwst
), and then fork and clone stcal
and install it in this environment in
the same way. To double check that you have the correct dev versions of these packages
in your environment, you can check their versions by doing:
>> conda list
When opening up two dependent pull requests in jwst
and one of its dependency packages,
unit tests will not pass on the CI because the pyproject.toml
file in jwst
points to the
last released version of stcal
, and stcal points to the last version of jwst
, so the
issue becomes circular. What you will need to do is modify the pyproject.toml
files in both
packages to point to the other to demonstrate that CI tests pass (and make a comment
noting this in your PR), and then change it back before the PR is merge so that changes
to pyproject.toml
are not merged into main. In your jwst
branch, to point to your
branch in the dependent package (in this example stcal
), change the required stcal
version in pyproject.toml
to:
>> stcal @ git+https://github.com/<your_username>/stcal.git@<your_branch>
And similarly, in stcal
, change the required jwst
version to:
>> jwst @ git+https://github.com/<your_username>/jwst.git@<your_branch>
Let the CI run and ensure it passes, comment this in your PR and make sure the reviewers confirm, and then change the versions back before your PR is merged (which will again cause the CI to fail, but that’s OK).