Reshapr Package Development
Continuous Integration |
|
Documentation |
|
Package |
|
Meta |
The Reshapr package (Reshapr) is Command-line tool based on Xarray and Dask
for extraction of model variable time series from model products like
SalishSeaCast, HRDPS & CANESM2/CGCM4.
Python Versions
The Reshapr package is developed and tested using Python 3.14.
The package uses some Python language features that are not available in various earlier versions,
in particular:
formatted string literals (aka f-strings) with = specifiers
The minimum supported Python version is 3.12. The Continuous Integration workflow on GitHub ensures that the package is tested for all versions of Python>=3.12.
Getting the Code
Clone the code and documentation repository from GitHub with:
$ git clone git@github.com:UBC-MOAD/Reshapr.git
or copy the URI (the stuff after git clone above) from the Code button on the repository page.
Note
The git clone command above assumes that your are connecting to GitHub using SSH. If it fails, please follow the instructions in our Secure Remote Access docs to set up your SSH keys and Copy Your Public ssh Key to GitHub.
Development Environment
Setting up an isolated development environment using Conda is recommended. Assuming that you have Miniconda3 installed, you can create and activate an environment called reshapr-dev that will have all of the Python packages necessary for development, testing, and building the documentation with the commands below.
$ cd Reshapr
$ conda env create -f envs/environment-dev.yaml
$ conda activate reshapr-dev
Reshapr is installed in editable install mode as part of the conda environment
creation process.
That means that the package is installed from the cloned repo via symlinks so that
it will be automatically updated as the repo evolves.
To deactivate the environment use:
(reshapr-dev)$ conda deactivate
Coding Style
The Reshapr package uses Git pre-commit hooks managed by pre-commit
to maintain consistent code style and and other aspects of code,
docs,
and repo QA.
To install the pre-commit hooks in a newly cloned repo, activate the conda development environment, and run pre-commit install:
$ cd Reshapr
$ conda activate reshapr-dev
(reshapr-dev)$ pre-commit install
Note
You only need to install the hooks once immediately after you make a new clone of the Reshapr repository and build your Development Environment.
Building the Documentation
The documentation for the Reshapr package is written in reStructuredText
and converted to HTML using Sphinx.
Creating a Development Environment as described above includes the installation of Sphinx.
Building the documentation is driven by the docs/Makefile.
With your reshapr-dev development environment activated,
use:
(reshapr-dev)$ (cd docs && make clean html)
to do a clean build of the documentation. The output looks something like:
Removing everything under '_build'...
Running Sphinx v8.1.3
loading translations [en]... done
making output directory... done
loading intersphinx inventory 'arrow' from https://arrow.readthedocs.io/en/latest/objects.inv ...
loading intersphinx inventory 'dask' from https://docs.dask.org/en/stable/objects.inv ...
loading intersphinx inventory 'moaddocs' from https://ubc-moad-docs.readthedocs.io/en/latest/objects.inv ...
loading intersphinx inventory 'python' from https://docs.python.org/3/objects.inv ...
loading intersphinx inventory 'salishseanowcast' from https://salishsea-nowcast.readthedocs.io/en/latest/objects.inv ...
loading intersphinx inventory 'xarray' from https://docs.xarray.dev/en/stable/objects.inv ...
building [mo]: targets for 0 po files that are out of date
writing output...
building [html]: targets for 20 source files that are out of date
updating environment: [new config] 20 added, 0 changed, 0 removed
reading sources... [100%] subcommands/info
looking for now-outdated files... none found
pickling environment... done
checking consistency... done
preparing documents... done
copying assets...
copying static files...
Writing evaluated template result to /media/doug/warehouse/MOAD/Reshapr/docs/_build/html/_static/language_data.js
Writing evaluated template result to /media/doug/warehouse/MOAD/Reshapr/docs/_build/html/_static/documentation_options.js
Writing evaluated template result to /media/doug/warehouse/MOAD/Reshapr/docs/_build/html/_static/basic.css
Writing evaluated template result to /media/doug/warehouse/MOAD/Reshapr/docs/_build/html/_static/js/versions.js
copying static files: done
copying extra files...
copying extra files: done
copying assets: done
writing output... [100%] subcommands/info
generating indices... genindex py-modindex done
highlighting module code... [100%] reshapr.utils.date_formatters
writing additional pages... search done
dumping search index in English (code: en)... done
dumping object inventory... done
build succeeded.
The HTML pages are in _build/html.
The HTML rendering of the docs ends up in docs/_build/html/.
You can open the index.html file in that directory tree in your browser
to preview the results of the build.
If you use Firefox,
you can probably accomplish that with:
(reshapr-dev)$ firefox docs/_build/html/index.html
If you have write access to the repository on GitHub, whenever you push changes to GitHub the documentation is automatically re-built and rendered at https://reshapr.readthedocs.io/en/latest/.
Link Checking the Documentation
Sphinx also provides a link checker utility which can be run to find broken or redirected links in the docs. With your reshapr-dev) environment activated, use:
(reshapr-dev))$ cd Reshapr/docs/
(reshapr-dev)) docs$ make clean linkcheck
The output looks something like:
Removing everything under '_build'...
Running Sphinx v8.1.3
loading translations [en]... done
making output directory... done
loading intersphinx inventory 'arrow' from https://arrow.readthedocs.io/en/latest/objects.inv ...
loading intersphinx inventory 'dask' from https://docs.dask.org/en/stable/objects.inv ...
loading intersphinx inventory 'moaddocs' from https://ubc-moad-docs.readthedocs.io/en/latest/objects.inv ...
loading intersphinx inventory 'python' from https://docs.python.org/3/objects.inv ...
loading intersphinx inventory 'salishseanowcast' from https://salishsea-nowcast.readthedocs.io/en/latest/objects.inv ...
loading intersphinx inventory 'xarray' from https://docs.xarray.dev/en/stable/objects.inv ...
building [mo]: targets for 0 po files that are out of date
writing output...
building [linkcheck]: targets for 20 source files that are out of date
updating environment: [new config] 20 added, 0 changed, 0 removed
reading sources... [100%] subcommands/info
looking for now-outdated files... none found
pickling environment... done
checking consistency... done
preparing documents... done
copying assets...
copying assets: done
writing output... [100%] subcommands/info
( api: line 31) ok https://arrow.readthedocs.io/en/latest/api-guide.html#arrow.arrow.Arrow
( pkg_development: line 22) ok https://black.readthedocs.io/en/stable/
(design_notes/pkg_structure: line 68) ok https://click.palletsprojects.com/en/stable/
(design_notes/pkg_structure: line 57) ok https://click.palletsprojects.com/en/stable/quickstart/#nesting-commands
( pkg_development: line 452) ok https://coverage.readthedocs.io/en/latest/
( pkg_development: line 28) ok https://codecov.io/gh/UBC-MOAD/Reshapr/branch/main/graph/badge.svg
( pkg_development: line 22) ok https://app.codecov.io/gh/UBC-MOAD/Reshapr
( pkg_development: line 35) ok https://app.readthedocs.org/projects/reshapr/badge/?version=latest
(design_notes/motivation: line 53) ok https://docs.dask.org/en/latest/
( model_profiles: line 221) ok https://docs.dask.org/en/latest/array-chunks.html
( pkg_development: line 505) ok https://docs.github.com/en/actions
( installation: line 39) ok https://docs.github.com/en/authentication/connecting-to-github-with-ssh
( pkg_development: line 22) ok https://docs.python.org/3/
( installation: line 50) ok https://docs.conda.io/en/latest/miniconda.html
( installation: line 50) ok https://docs.conda.io/en/latest/
( api: line 3) ok https://docs.python.org/3/library/constants.html#None
( api: line 22) ok https://docs.python.org/3/library/exceptions.html#ValueError
( api: line 40) ok https://docs.python.org/3/library/pathlib.html#pathlib.Path
( pkg_development: line 412) ok https://docs.pytest.org/en/latest/
( api: line 31) ok https://docs.python.org/3/library/stdtypes.html#str
(design_notes/motivation: line 53) ok https://docs.xarray.dev/en/stable/
( pkg_development: line 87) ok https://docs.python.org/3/reference/lexical_analysis.html#f-strings
( api: line 40) ok https://docs.python.org/3/library/stdtypes.html#dict
( pkg_development: line 520) ok https://git-scm.com/
(design_notes/motivation: line 65) ok https://docs.xarray.dev/en/stable/generated/xarray.open_mfdataset.html#xarray.open_mfdataset
-rate limited- https://github.com/SalishSeaCast/analysis-doug/blob/main/notebooks/wastewater/extract_biology.yaml | sleeping...
(design_notes/history: line 25) ok https://github.com/SalishSeaCast/analysis-doug/blob/main/notebooks/dask-expts/dask_expts.ipynb
(examples/2xrez_physics_ONC_SCVIP: line 44) ok https://github.com/SalishSeaCast/analysis-doug/blob/main/notebooks/2xrez-2017/DeepWaterRenewal.ipynb
(design_notes/history: line 52) ok https://github.com/SalishSeaCast/analysis-doug/blob/main/notebooks/dask-expts/atlantis_nudge_diatoms.py
(design_notes/history: line 46) ok https://github.com/SalishSeaCast/analysis-doug/blob/main/notebooks/dask-expts/atlantis_nudge_diatoms.ipynb
( pkg_development: line 31) ok https://github.com/UBC-MOAD/Reshapr/actions/workflows/codeql-analysis.yaml/badge.svg
( pkg_development: line 25) ok https://github.com/UBC-MOAD/Reshapr/actions/workflows/pytest-with-coverage.yaml/badge.svg
(design_notes/motivation: line 129) ok https://github.com/UBC-MOAD/Reshapr
( pkg_development: line 38) ok https://github.com/UBC-MOAD/Reshapr/actions/workflows/sphinx-linkcheck.yaml/badge.svg
( pkg_development: line 492) ok https://github.com/UBC-MOAD/Reshapr/actions
( pkg_development: line 492) ok https://github.com/UBC-MOAD/Reshapr/commits/main
( pkg_development: line 22) ok https://github.com/UBC-MOAD/Reshapr/actions?query=workflow:CodeQL
( pkg_development: line 22) ok https://github.com/UBC-MOAD/Reshapr/issues
( pkg_development: line 64) ok https://img.shields.io/badge/%F0%9F%A5%9A-Hatch-4051b5.svg
( pkg_development: line 61) ok https://img.shields.io/badge/code%20style-black-000000.svg
( index: line 48) ok https://img.shields.io/badge/license-Apache%202-cb2533.svg
( pkg_development: line 58) ok https://img.shields.io/badge/pre--commit-enabled-brightgreen?logo=pre-commit&logoColor=white
( pkg_development: line 55) ok https://img.shields.io/badge/version%20control-git-blue.svg?logo=github
( pkg_development: line 48) ok https://img.shields.io/github/issues/UBC-MOAD/Reshapr?logo=github
( pkg_development: line 22) ok https://github.com/UBC-MOAD/Reshapr/actions?query=workflow:sphinx-linkcheck
( pkg_development: line 42) ok https://img.shields.io/github/v/release/UBC-MOAD/Reshapr?logo=github
( pkg_development: line 22) ok https://github.com/pypa/hatch
( pkg_development: line 45) ok https://img.shields.io/python/required-version-toml?tomlFilePath=https://raw.githubusercontent.com/UBC-MOAD/Reshapr/main/pyproject.toml&logo=Python&logoColor=gold&label=Python
( pkg_development: line 22) ok https://github.com/UBC-MOAD/Reshapr/releases
( pkg_development: line 22) ok https://github.com/UBC-MOAD/Reshapr/actions?query=workflow:pytest-with-coverage
(design_notes/motivation: line 53) ok https://pangeo.io/
(design_notes/motivation: line 53) ok https://pangeo.io/#ecosystem
( installation: line 65) ok https://pip.pypa.io/en/stable/topics/local-project-installs/#editable-installs
( pkg_development: line 91) ok https://peps.python.org/pep-0636/
( pkg_development: line 179) ok https://pre-commit.com/
( pkg_development: line 22) ok https://pre-commit.com
( pkg_development: line 452) ok https://pytest-cov.readthedocs.io/en/latest/
(examples/iona_wastewater_discharge_analysis: line 47) ok https://salishsea-nowcast.readthedocs.io/en/latest/workers.html#module-nowcast.workers.split_results
( installation: line 39) ok https://ubc-moad-docs.readthedocs.io/en/latest/ssh_access.html#copyyourpublicsshkeytogithub
( installation: line 39) ok https://ubc-moad-docs.readthedocs.io/en/latest/ssh_access.html#secureremoteaccess
( index: line 46) ok https://www.apache.org/licenses/LICENSE-2.0
( pkg_development: line 22) ok https://reshapr.readthedocs.io/en/latest/
( pkg_development: line 83) ok https://www.python.org/
( pkg_development: line 212) ok https://www.sphinx-doc.org/en/master/
( pkg_development: line 212) ok https://www.sphinx-doc.org/en/master/usage/restructuredtext/basics.html
(examples/prodigy_model_obs_assignment: line 40) ok https://www.frontiersin.org/journals/marine-science/articles/10.3389/fmars.2018.00536/full
(design_notes/pkg_structure: line 63) ok https://www.structlog.org/en/stable/index.html
(design_notes/history: line 46) ok https://nbviewer.org/github/SalishSeaCast/analysis-doug/blob/main/notebooks/dask-expts/atlantis_nudge_diatoms.ipynb
(examples/2xrez_physics_ONC_SCVIP: line 43) ok https://nbviewer.org/github/SalishSeaCast/analysis-doug/blob/main/notebooks/2xrez-2017/DeepWaterRenewal.ipynb
(design_notes/history: line 25) ok https://nbviewer.org/github/SalishSeaCast/analysis-doug/blob/main/notebooks/dask-expts/dask_expts.ipynb
-rate limited- https://github.com/SalishSeaCast/analysis-doug/blob/main/notebooks/wastewater/extract_biology.yaml | sleeping...
(examples/iona_wastewater_discharge_analysis: line 99) ok https://github.com/SalishSeaCast/analysis-doug/blob/main/notebooks/wastewater/extract_biology.yaml
(examples/iona_wastewater_discharge_analysis: line 95) ok https://github.com/SalishSeaCast/analysis-doug/blob/main/notebooks/wastewater/model_profiles/SalishSeaCast-202111-wastewater-salish.yaml
build succeeded.
Look for any errors in the above output or in _build/linkcheck/output.txt
Running the Unit Tests
The test suite for the Reshapr package is in Reshapr/tests/.
The pytest tool is used for test parametrization and as the test runner for the suite.
With your reshapr-dev development environment activated, use:
(reshapr-dev)$ cd Reshapr/
(reshapr-dev)$ pytest
to run the test suite. The output looks something like:
=============================== test session starts ================================
platform linux -- Python 3.14.0, pytest-9.0.1, pluggy-1.6.0
Using --randomly-seed=614873902
rootdir: /media/doug/warehouse/MOAD/Reshapr
configfile: pyproject.toml
plugins: cov-7.0.0, anyio-4.11.0, randomly-3.15.0
collected 303 items
tests/api/v1/test_extract_api_v1.py ......... [ 2%]
tests/core/test_info.py ............................... [ 13%]
tests/core/test_dask_cluster.py ......... [ 16%]
tests/utils/test_date_formatters.py ........... [ 19%]
tests/test_cluster_configs.py ..... [ 21%]
tests/test_model_profiles.py .........................................................
............................... [ 50%]
tests/cli/test_cli.py .. [ 51%]
tests/core/test_extract.py ...........................................................
......................................................................................
... [100%]
=============================== 303 passed in 8.42s ================================
You can monitor what lines of code the test suite exercises using the coverage.py and pytest-cov tools with the command:
(reshapr-dev)$ cd Reshapr/
(reshapr-dev)$ pytest --cov=./
and generate a test coverage report with:
(reshapr-dev)$ coverage report
to produce a plain text report, or
(reshapr-dev)$ coverage html
to produce an HTML report that you can view in your browser by opening
Reshapr/htmlcov/index.html.
Continuous Integration
The Reshapr package unit test suite is run and a coverage report is generated
whenever changes are pushed to GitHub.
The results are visible on the repo actions page,
from the green checkmarks beside commits on the repo commits page,
or from the green checkmark to the left of the “Latest commit” message on the
repo code overview page .
The testing coverage report is uploaded to codecov.io
The GitHub Actions workflow configuration that defines the continuous integration tasks
is in the .github/workflows/pytest-coverage.yaml file.
Version Control Repository
The Reshapr package code and documentation source files are available
as a Git repository at https://github.com/UBC-MOAD/Reshapr.
Issue Tracker
Development tasks, bug reports, and enhancement ideas are recorded and managed in the issue tracker at https://github.com/UBC-MOAD/Reshapr/issues.
License
The code and documentation of the Reshapr project are copyright 2022 – present by the UBC EOAS MOAD Group and The University of British Columbia.
They are licensed under the Apache License, Version 2.0. https://www.apache.org/licenses/LICENSE-2.0 Please see the LICENSE file for details of the license.
Release Process
Releases are done at Doug’s discretion when significant pieces of development work have been completed.
The release process steps are:
Use hatch version release to bump the version from
.devnto the next release version identifier; e.g.23.1.dev0to23.1Commit the version bump
Create an annotated tag for the release with Git -> New Tag… in PyCharm or git tag -e -a vyy.n; git tag -e -a v23.1
Push the version bump commit and tag to GitHub
Use the GitHub web interface to create a release, editing the auto-generated release notes into sections:
Features
Bug Fixes
Documentation
Maintenance
Dependency Updates
Use the GitHub Issues -> Milestones web interface to edit the release milestone:
Change the Due date to the release date
Delete the “when it’s ready” comment in the Description
Use the GitHub Issues -> Milestones web interface to create a milestone for the next release:
Set the Title to the next release version, prepended with a
v; e.g.v23.2Set the Due date to the end of the year of the next release
Set the Description to something like
v23.2 release - when it's ready :-)Create the next release milestone
Review the open issues, especially any that are associated with the milestone for the just released version, and update their milestone.
Close the milestone for the just released version.
Use hatch version minor,dev to bump the version for the next development cycle, or use hatch version major,minor,dev for a year rollover version bump
Commit the version bump
Push the version bump commit to GitHub