Skip to content

roman-corgi/corgidrp

Repository files navigation

corgidrp: CORonaGraph Instrument Data Reduction Pipeline

This is the data reduction pipeline for the Nancy Grace Roman Space Telescope Coronagraph Instrument

Testing Badge

Install

As the code is very much still in development, clone this repository, enter the top-level folder, and run the following command:

pip install -e .

Then you can import corgidrp like any other python package!

The installation will create a configuration folder in your home directory called .corgidrp. That configuration directory will be used to locate things on your computer such as the location of the calibration database and the pipeline configuration file. The configuration files stores setting such as whether to track each individual error term added to the noise.

For Developers

Large binary files (used in tests) are stored in Git LFS. Install Git LFS if it isn't already installed. You may need to run git lfs pull after checking out the repository to download the latest large binary files, or the unit tests may fail.

To run the existing end-to-end tests, you also need the II&T code, which is used directly for comparing results. This also requires Git LFS to be installed first. Then install the II&T code by doing the following while in the top-level folder:

pip install -r requirements_e2etests.txt corgidrp

This will install the II&T repositories cal and proc_cgi_frame.

Troubleshooting

If you run into any issues with things in the .corgidrp directory not being found properly when you run the pipeline, such as a DetectorParams file, caldb, or configuration settings, your corgidrp is configured into a weird state. Report the bug to our Github issue tracker that includes both the error message, and the state of your .corgidrp folder. If you don't want to wait for us to troubleshoot the bug and deploy a fix, you can probably resolve the issue by completely deleting your .corgidrp folder and rerunning the code (the code will automatically remake it). This however means you will lose any changes you've made to your settings as well as your calibration database.

How to Contribute

We encourage you to chat with Jason, Max, and Marie (e.g., on Slack) to discuss what to do before you get started. Brainstorming about how to implement something is a very good use of time and makes sure you aren't going down the wrong path. Contact Jason is you have any questions on how to get started on programming details (e.g., git).

Below is a quick tutorial that outlines the general contribution process.

The basics of getting setup

Find a task to work on

Check out the Github issues page for tasks that need attention. Alternatively, contact Jason (@semaphoreP). Make sure to tag yourself on the issue and mention in the comments if you start working on it.

Clone the git repository and install

See install instructions above. Contact Jason (@semaphoreP) if you need write access to push changes as you make edits. If you do not have write access to the repository, you can still contribute by creating a fork of the repository under your own GitHub user. See here for details: https://docs.github.com/en/get-started/quickstart/fork-a-repo. You can then use the same commands given below, but just replace roman-corgi with your own GitHub username. If you fork the repository, you will need to make sure that your fork is up to date with the main repository (https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/working-with-forks/syncing-a-fork).

To quickly get up an running with the repository, execute the following commands in a terminal (or command prompt - you will need to have the git executable installed on your system):

> git clone https://github.com/roman-corgi/corgidrp.git
> cd corgidrp
> pip install -e .
Make a new git branch to work on

You will create a "feature branch" so you can develop your feature without impacting other people's code. Let's say I'm working on dark subtraction. I could create a feature branch and switch to it like this

> git branch dark-sub
> git checkout dark-sub

Write your pipeline step

In corgidrp, each pipeline step is a function, that is contained in one of the lX_to_lY.py files (where X and Y are various data levels). Think about how your feature can be implemented as a function that takes in some data and outputs processed data. Please see below for some corgidrp design principles.

All functions should follow this example:

def example_step(dataset, calib_data, tuneable_arg=1, another_arg="test"):
    """
    Function docstrings are required and should follow Google style docstrings. 
    We will not demonstrate it here for brevity.
    """
    # unless you don't alter the input dataset at all, plan to make a copy of the data
    # this is to ensure functions are reproducible
    processed_dataset = input_dataset.copy()

    ### Your code here that does the real work
    # here is a convience field to grab all the data in a dataset
    all_data = processed_dataset.all_data
    ### End of your code that does the real work

    # update the header of the new dataset with your processing step
    history_msg = "I did an example step"
    # update the output dataset with the new data and update the history
    processed_dataset.update_after_processing_step(history_msg, new_all_data=all_data)

    # return the processed data
    return processed_dataset

Inside the function can be nearly anything you want, but the function signature and start/end of the function should follow a few rules.

  • Each function should include a docstring that descibes what the function is doing, what the inputs (including units if appropriate) are and what the outputs (also with units). The dosctrings should be goggle style docstrings.
  • The input dataset should always be the first input
  • Additional arguments and keywords exist only if you need them--many relevant parameters might already by in Dataset headers. A pipeline step can only have a single argument (the input dataset) if needed.
  • All additional function arguments/keywords should only consist of the following types: int, float, str, or a class defined in corgidrp.Data.
    • (Long explaination for the curious: The reason for this is that pipeline steps can be written out as text files. Int/float/str are easily represented succinctly by textfiles. All classes in corgidrp.Data can be created simply by passing in a filepath. Therefore, all pipeline steps have easily recordable arguments for easy reproducibility.)
  • The first line of the function generally should be creating a copy of the dataset (which will be the output dataset). This way, the output dataset is not the same instance as the input dataset. This will make it easier to ensure reproducibility.
  • The function should always end with updating the header and (typically) the data of the output dataset. The history of running this pipeline step should be recorded in the header.

You can check out corgidrp.l2a_to_l2b.dark_subtraction function as an example of a basic pipeline step.

Write a unit test to debug your pipeline step

We are required to write tests to verify the functionality of the code. Instead of seeing this as an extra chore, I encourage you to write unit tests to be your debug script to get your code working (this is called "test-driven development").

All tests are stored in the tests folder and each test is a function that starts with test_. See tests/test_dark_sub.py as an example. Within each test, you will likely need to simulate some mock data, run it through your function you wrote, and verify it ran correctly using assert statements. Your tests should cover the primary use cases of your code, and check that the function outputs what you expect. You do not need high fidelity data for your test: focus on making sure the data is in the correct format as real data, and less on making sure the data values are simulated to high fidelity (see the examples in the mocks.py module).

Importantly, these tests will allow code reviewers to test and understand your code. We will also run these tests in an automated test suite (continuous integration) with the pipeline to verify the functions continue to work (e.g., as dependencies change).

How to run your tests locally

You can either run tests individually yourself (to debug individual tests) or run the entire test suite to make sure you didn't break anything.

To run an individual test, call the test function you want to test at the bottom of its test_*.py script. Then, you just need to run the test_*.py script. See tests/test_dark_sub.py for an example.

To run all the tests in the test suite, go to the base corgidrp folder in a terminal and run the pytest command.

End-to-End Testing

End-to-end testing refers to processing data as one would when we get the real data (i.e., starting from L1 data). If applicable, write an end-to-end test following the l1_to_l2a_e2e.py and l1_to_l2b_e2e.py examples in the tests/e2e_tests folder. For example, if you wrote a step that generates a calibration function, write an end-to-end test that produces the calibration file from L1 data. The steps are as follows:

  1. Write a recipe that produces the desired processed data product starting from L1 data. You will need to determine the series of step functions that need to be run, and what kind of arguments should be modified (e.g., whether prescan columns pixels should be cropped). Refer to the existing recipes in corgidrp/recipe_templates as examples and double check all the necessary steps in the FDD.
  2. Obtain TVAC L1 data from our Box folder (ask Alex Greenbaum or Jason if you don't have access). For some situations (e.g., boresight), there may not be appropriate TVAC data. In those cases, write a piece of code that uses the images from TVAC to provide realistic noise and add it to mock data (i.e., the ones generated for the unit testing) to create mock L1 data.
  3. Write an end-to-end test that processes the L1 data through the new recipe you created using the corgidrp.walker framework
    • You will probably need to modify the corgidrp.walker.guess_template() function to add logic for determining when to use your recipe based on header keywords (e.g., VISTYPE). Ask Jason, who developed this framework, if it is not clear what should be done.
    • Your recipe may require other calibratio files. For now, create them as part of the setup process during the script (see tests/e2e_tests/l1_to_l2b_e2e.py for examples of how to do this for each type of calibration)
    • if you need to create mock L1 data, please do it in the script as well.
    • See the existing tests in tests/e2e_tests/ for how to structure this script. You should only need to write a single script.
  4. Test that the script runs successfully on your local machine and produces the expected output. Debug as necessary. When appropriate, test your results against those obtained from the II&T/TVAC pipeline using the same input data.
  5. Determine how resource intensive your recipe is. There are many ways to do this, but Linux users can run /usr/bin/time -v python your_e2e_test.py and Mac userse can run /usr/bin/time -l -h -p python <your_e2e_test.py>. Record elapsed (wall clock) time, the percent of CPU this job got (only if parallelization was used), and total memory used (labelled "Maximum resident set size").
  6. Document your recipe on the "Corgi-DRP Implementation Document" on Confluence (see the big table in Section 2.0). You should fill out an entire row with your recipe. Under addition notes, note if your recipe took significant run time (> 1 minute) and significant memory (> 1 GB).
  7. PR!

To run the existing end-to-end tests, you need to have downloaded all the TVAC data to your computer. In a terminal, go to the base directory of the corgidrp repo and run the following command, substituting paths for paths on your computer as desired:

pytest --which e2e --tvacdata_path /path/to/CGI_TVAC_Data --e2eoutput_path tests/e2e_tests/ tests/e2e_tests/

Linting

In addition to unit tests, your code will need to pass a static analysis before being merged. corgidrp currently runs a subset of flake8 tests, which you can replicate on your local system by running:

flake8 . --count --select=E9,F63,F7,F82,DCO020,DCO021,DCO022,DCO023,DCO024,DCO030,DCO031,DCO032,DCO060,DCO061,DCO062,DCO063,DCO064,DCO065 --show-source --statistics

from the top-level directory of the repository. In order to run these tests you will need to have flake8 and flake8-docstrings-complete installed (both are pip-installable). Note that the test subset may be updated in the future. To see the current set of tests being applied, look in the continuous integration GitHub action, located in the repository in file .github/workflows/python-app.yml.

Create a pull request to merge your changes

Before creating a pull request, review the design Principles below. Use the Github pull request feature to request that your changes get merged into the main branch. Assign Jason/Max to be your reviewers. Your changes will be reviewed, and possibly some edits will be requested. You can simply make additional pushes to your branch to update the pull request with those changes (you don't need to delete the PR and make a new one). When the branch is satisfactory, we will pull your changes in. When preparing your pull request, you may find it helpful to follow this checklist:

  • If working with a fork, sync your fork to the upstream repository
  • Ensure that your pull can be merged automatically by merging main onto the branch you wish to pull from
  • Ensure that all of your new additions have properly formatted docstrings (this will happen automatically if you run the flake8 command given above
  • Ensure that all of the commits going in to your pull have informative messages
  • Ensure that all unit tests pass locally, and that you have provided new unit tests for all new functionality in your pull
  • Create a new pull request, fully describing all changes/additions

Overarching Design Principles

  • Minimize the use of external packages, unless it saves us a lot of time. If you need to use something external, default to well-established and maintained packages, such as numpy, scipy or Astropy. If you think you need to use something else, please check with Jason and Max.
  • Minimize the use of new classes, with the exception of new classes that extend the existing data framework.
  • The python module files (i.e. the *.py files) should typically hold on the order of 5-10 different functions. You can create new ones if need be, but the new files should be general enough in topic to encapsulate other future functions as well.
  • All the image data in the Dataset and Image objects should be saved as standard numpy arrays. Other types of arrays (such as masked arrays) can be used as intermediate products within a function.
  • Keep things simple
  • Use descriptive variable names always.
  • Comments should be used to describe a section of the code where it's not immediately obvious what the code is doing. Using descriptive variable names will minimize the amount of comments required.
  • Make pull requests as small as possible. It makes code easier to review, when only a small fraction of the code is being modified. You can do multiple PRs on the same task (e.g., a first very simplistic implementation, followed by a separate PR adding some new features and options.)

FAQ

  • Does my pipeline function need to save files?

    • Files will be saved by a higher level pipeline code. As long as you output an object that's an instance of a corgidrp.Data class, it will have a save() function that will be used.
  • Can I create new data classes?

    • Yes, you can feel free to make new data classes. Generally, they should be a subclass of the Image class, and you can look at the Dark class as an example. Each calibration type should have its own Image subclass defined. Talk with Jason and Max to discuss how your class should be implemented!
    • You do not necessarily need to write a copy function for subclasses of the Image class. If you need to copy calibration objects at all you can import and apply the copy module of python, see example:
      import copy
      flatfield = data.Flatfield('flatfield.fits')
      #reference copy
      flatfield_copy = copy.copy(flatfield)
      #deep data copy
      flatfield_copy = copy.deepcopy(flatfield)
      
  • What python version should I develop in?

    • Python 3.12
  • How should I treat different kinds of function parameters:

    • Constants - things that are highly unlikely to change - can be included in modules, like detector.py
    • Properties of the system - things we need to measure that might change in time - would go into a calibration file like DetectorParams
    • Function parameters - choices that we might make and want to have more flexibility that apply to a specific function (e.g. exactly what area of a detector we might use to calculate something) - would be keyword arguments to that function.
    • If it is a setting about pipeline behavior that may change, it should be implemented in the config file (examples are location of the calibration database and whether to save individual error terms in the output FITS files).
  • Where should I store computed variables so they can be referenced later in the pipeline?

    • If possible, in the header of the dataset being processed or in a hew HDU extension
    • If not possible, then let's chat!
  • Where do I save FITS files or other data files I need to use for my tests?

    • Auxiliary data to run tests should be stored in the tests/test_data folder
    • If they are larger than 1 MB, they should be stored using git lfs. Ask Jason about setting up git lfs (as of writing, we have not set up git lfs yet).

Change Log

v1.1.2

  • Flat field correction marks pixels divided by 0 as bad

v1.1.1

  • Fix unit test that wasn't cleaning up environment properly

v1.1

  • Bug fix so that corgidrp classes can be pickled
  • New corgidrp.ops interface
  • Improved agreement with II&T pipeline in updated e2e tests
  • Ability to embed just the illuminated part of the detector back into a full engineering frame 
  • Updated DRP throughout to handle recently updated data header specification

v1.0

  • First official pipeline release!
  • Step functions to produce the necessary calibration files for analog L1 to L2b step functions implemented and tested
  • Step function to produce boresight calibration implemented and tested
  • Automated data processing handling for analog L1/L2 calibration files and for boresight calibration
  • End-to-end testing demonstrating analog L1/L2 calibration files and boresight calibration file can be produced from realistic/real L1 data files

v0.2.1

  • Update end-to-end tests to handle updated test data filenames

v0.2

  • All L1 to L2b step functions implemented and tested
  • Automated data porcessing for analog L1 to L2b
  • End-to-end testing for analog L1 to L2b processing
  • Minor bug fixes throughout

v0.1.2

  • Added ability to change paths for analog L1 to L2a end-to-end test from command line
  • v0.1.1 was a partial release of this release, and should not be used.

v0.1

  • First preliminary release of pipeline including step functions (see next bullet), calibration database, walker (pipeline automation framework)
  • All L1 to L2a step funtions implemented and tested
  • Automated data processing for analog L1 to L2a
  • End-to-end test demonstrating analog L1 to L2a processing

About

Data Reduction Pipeline for the Roman Coronagraph Instrument

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages