Skip to content

Commit

Permalink
Add presets and adapt code to plot FLEXPART-ICON output. (#37)
Browse files Browse the repository at this point in the history
* Added ICON model names.

* Fix bug for FLEXPART-ICON output plotting, increase patch level

* Add modifications from black.

* Update README.md, fix bug that caused species attributes to always have the default values.

* Make README.md adhere to MD032.

* Use pinned environment for gihub actions.
  • Loading branch information
pirmink authored Feb 28, 2024
1 parent d27a626 commit 9128017
Show file tree
Hide file tree
Showing 30 changed files with 787 additions and 91 deletions.
8 changes: 4 additions & 4 deletions .github/workflows/pre-commit.yml
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
name: Run pre-commit in blueprint
name: Run pre-commit in pyflexplot

on:
push:
Expand All @@ -9,7 +9,7 @@ on:
- main

jobs:
blueprint-pre-commit:
pyflexplot-pre-commit:
runs-on: ubuntu-latest
defaults:
run:
Expand All @@ -26,9 +26,9 @@ jobs:
channels: conda-forge
channel-priority: flexible
show-channel-urls: true
- name: Create env from unpinned reqs
- name: Create env from pinned reqs
run: |
conda env create --name dev_env --file requirements/requirements.yml
conda env create --name dev_env --file requirements/environment.yml
- name: Install project into env
run: |
conda run --name dev_env pip install --no-deps .
Expand Down
126 changes: 84 additions & 42 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,9 @@
# PyFlexPlot

PyFlexPlot is a Python-based tool to visualize/post-process FLEXPART dispersion simulation results stored in NetCDF format.

## Table of Contents

- [Prerequisites and Cloning the Repository](#prerequisites-and-cloning-the-repository)
- [Getting Started](#getting-started)
- [Usage](#usage)
Expand All @@ -16,9 +19,13 @@ PyFlexPlot is a Python-based tool to visualize/post-process FLEXPART dispersion
- [Credits](#credits)
- [External Links](#external-links)
- [License](#license)

## Prerequisites and Cloning the Repository
Before you get started with this repository, ensure you have the following software/tools installed on your system: Git, Python and conda/mamba.
To get a local copy of this repository, run following commands and naviate into the repository:

Before you get started with this repository, ensure you have the following software tools installed on your system: Git, Python and Conda.

To get a local copy of this repository, run the following commands and naviate into the repository:

```bash
git clone https://github.com/MeteoSwiss-APN/pyflexplot <custom_directory_name>
cd <custom_directory_name>
Expand All @@ -28,26 +35,25 @@ cd <custom_directory_name>

Once you created or cloned this repository, make sure the installation is running properly. Install the package dependencies with the provided script `setup_env.sh`.
Check available options with

```bash
tools/setup_env.sh -h
```

We distinguish pinned installations based on exported (reproducible) environments and free installations where the installation
is based on top-level dependencies listed in `requirements/requirements.yml`. If you start developing, you might want to do an unpinned installation and export the environment:

```bash
tools/setup_env.sh -u -e -n <custom_environment_name>
```
*Hint*: If you are the package administrator, it is a good idea to understand what this script does, you can do everything manually with `conda` instructions.

*Hint*: Use the flag `-m` to speed up the installation using mamba. Of course you will have to install mamba first (we recommend to install mamba into your base
environment `conda install -c conda-forge mamba`. If you install mamba in another (maybe dedicated) environment, environments installed with mamba will be located
in `<miniconda_root_dir>/envs/mamba/envs`, which is not very practical.
*Hint*: If you are the package administrator, it is a good idea to understand what this script does, you can do everything manually with `conda` instructions.

The package itself is installed with `pip`. For development, install in editable mode:
The package itself is installed with `pip`. As all dependencies have already been installed by Conda, use the `--no-deps` option for pip. For development, install the package in "editable" mode:

```bash
conda activate <custom_environment_name>
pip install --editable .
pip install --no-deps --editable .
```

*Warning:* Make sure you use the right pip, i.e. the one from the installed conda environment (`which pip` should point to something like `path/to/miniconda/envs/<custom_environment_name>/bin/pip`).
Expand All @@ -64,63 +70,91 @@ If the tests pass, you are good to go. If not, contact the package administrator
every time you add new imports while developing. Check the next section to find some guidance on the development process if you are new to Python and/or SEN.

## Usage

To utilize pyflexplot, first ensure you are in the root directory of the project and have activated the necessary conda environment:

```bash
cd <custom_directory_name>
conda activate <custom_environment_name>
```

The primary command for pyflexplot follows this structure:

```bash
pyflexplot [OPTIONS] CONFIG_FILE_DIRECTORY
```

To see the available options, run:

```bash
pyflexplot -h
pyflexplot --help
```
To utilize all available CPUs for the command, use the option:

To utilize all available CPUs for the command, add the option:

```bash
--num-procs=$SLURM_CPUS_PER_TASK
--num-procs=$SLURM_CPUS_PER_TASK
```

### Usage Example

After you've set up pyflexplot ([Prerequisites and cloning the repository](#prerequisites-and-cloning-the-repository) and [Getting started](#getting-started)),
you'll need to specify a configuration file and an output directory.
Create an output directory using:
Define the variable `dest` to contain the name of the output directory, e.g.:

```bash
dest=test_output/
dest=plot-test/
```

Note: The directory will be automatically created if it doesn't already exist.

There are several default config files available under ```src/pyflexplot/data/presets/opr```.
Furthermore, there are already several default config files available in the directory ```src/pyflexplot/data/presets/opr```.
To run the program for all presets in the PDF graphics format with the default input data, use:
There are several default config files available under [`src/pyflexplot/data/presets/opr`](src/pyflexplot/data/presets/opr/).
To run the program for all presets that produce the graphics in PDF format, define the `preset` variable as:

```bash
preset='opr/*/all_pdf'
```
Alternatively, select a specific preset from the table below:
| Model | Type | Preset |
|------------------|----------------------|----------------------------------|
| FLEXPART-IFS | Global output: | preset=opr/ifs-hres/all_pdf |
| FLEXPART-IFS | Europe output: | preset=opr/ifs-hres-eu/all_pdf |
| FLEXPART-COSMO | deterministic output:| preset=opr/cosmo-1e-ctrl/all_pdf |
| FLEXPART-COSMO | deterministic output:| preset=opr/cosmo-2e-ctrl/all_pdf |
| FLEXPART-COSMO-1E| ensemble output: | preset=opr/cosmo-1e/all_pdf |
| FLEXPART-COSMO-2E| ensemble output: | preset=opr/cosmo-2e/all_pdf |

After selecting a preset, run pyflexplot interactively:
```bash

This preset however only works for the default (test) input files.

To produce graphics for a specific FLEXPART output, select the fitting specific preset from the table below and define the `preset` variable accordingly:

| Model | Type | Define Preset Variable |
|-----------------------|--------------|------------------------------------|
| FLEXPART-IFS | global | `preset=opr/ifs-hres/all_pdf` |
| FLEXPART-IFS | Europe | `preset=opr/ifs-hres-eu/all_pdf` |
| FLEXPART-COSMO-1E-CTRL| deterministic| `preset=opr/cosmo-1e-ctrl/all_pdf` |
| FLEXPART-COSMO-2E-CTRL| deterministic| `preset=opr/cosmo-2e-ctrl/all_pdf` |
| FLEXPART-COSMO-1E | ensemble | `preset=opr/cosmo-1e/all_pdf` |
| FLEXPART-COSMO-2E | ensemble | `preset=opr/cosmo-2e/all_pdf` |
| FLEXPART-ICON-CH1-CTRL| deterministic| `preset=opr/icon-ch1-ctrl/all_pdf` |
| FLEXPART-ICON-CH2-EPS | ensemble | `preset=opr/icon-ch2-eps/all_pdf` |

After selecting a preset, run pyflexplot interactively for the default test data:

```bash
pyflexplot --preset "$preset" --merge-pdfs --dest=$dest
```

Alternatively, specify the FLEXPART output file in NetCDF format as input data:

```bash
pyflexplot --preset "$preset" --merge-pdfs --dest=$dest --setup infile <netcdf-file>
```

For ensemble input, the placeholder `{ens_member:03}` may be used within the path of `<netcdf-file>`. Instead of `03` for `%03d`, other formats for the ensemble member field can be used.

## Developer Notes

As this package was created with the SEN Python blueprint, it comes with a stack of development tools, which are described in more detail on the [Website](https://meteoswiss-apn.github.io/mch-python-blueprint/). Here, we give a brief overview on what is implemented.

### Implemented Debugging Features

pyflexplot offers several debugging options to assist in troubleshooting and
refining your workflow (see ```pyflexplot -h```).
Here are some of the key debugging features:
```

```text
--pdb / --no-pdb Drop into debugger when an exception is raised. [default: no-pdb]
--preset-cat PATTERN Show the contents of preset setup files
--only N Only create the first N plots based on the given setup.
Expand All @@ -146,19 +180,19 @@ pytest

If you use the tools provided by the blueprint as is, pre-commit will not be triggered locally but only if you push to the main branch
(or push to a PR to the main branch). If you consider it useful, you can set up pre-commit to run locally before every commit by initializing it once. In the root directory of
your package, type:
your package, type `pre-commit install`. If you run `pre-commit` without having it installed before, it will fail, and the only way to recover it is to do a forced reinstallation (`conda install --force-reinstall pre-commit`).

Alternatively, you can instead run `pre-commit` selectively, without installing and whenever you want, by typing

```bash
pre-commit install
pre-commit run --all-files
```

If you run `pre-commit` without installing it before (line above), it will fail and the only way to recover it, is to do a forced reinstallation (`conda install --force-reinstall pre-commit`).
You can also just run pre-commit selectively, whenever you want by typing (`pre-commit run --all-files`). Note that mypy and pylint take a bit of time, so it is really
Note that mypy and pylint take a bit of time, so it is really
up to you, if you want to use pre-commit locally or not. In any case, after running pytest, you can commit and the linters will run at the latest on the GitHub actions server,
when you push your changes to the main branch. Note that pytest is currently not invoked by pre-commit, so it will not run automatically. Automated testing can be set up with
GitHub Actions or be implemented in a Jenkins pipeline (template for a plan available in `jenkins/`. See the next section for more details.


### Testing and Coding Standards

Testing your code and compliance with the most important Python standards is a requirement for Python software written in SEN. To make the life of package
Expand All @@ -179,35 +213,43 @@ automatically. If you need to run your tests on CSCS machines, contact DevOps to
and exclusively run your tests and checks on GitHub actions.

### Updating the Test References

Pyflexplot includes a set of functionality tests that compare generated output against predefined reference data.
These reference files, which are in the .py format, are derived from and stored alongside the original data in the tests/data directory.
To update these references, uncomment the lines of code in the test file you wish to update and run the test.

## Key Features

### PDF and PNG Files

Pyflexplot allows to visualize data on a map plot and save the output in either PDF or PNG format. To utilize this feature, simply adjust the outfile variable with the appropriate file extension.
![Example Image Output](img/integrated_concentration_site-Goesgen_species-1_domain-full_lang-de_ts-20200217T0900.png)

### Shape File Generation

Furthermore, Pyflexplot provides the functionality to export data into shape files (.shp) to utilize them in GIS programs such as QGIS 3. The output is a ZIP archive containing the essential components of a shapefile: .shp, .dbf, .shx, .prj, and .shp.xml.
Key aspects of this feature include:
* __Filtering Zero Values__: The tool initially removes zero values from fields (e.g., concentration) before processing.
* __Logarithmic Transformation__: Field values undergo a log_10 transformation to optimize the visualization of data ranges.
* __Precision Handling__: The transformed field values are recorded with 15 decimal places, accommodating the precision limitations of some GIS software.
* __Metadata Storage__: Information, such as details about released materials, are stored within a .shp.xml file as metadata.

- __Filtering Zero Values__: The tool initially removes zero values from fields (e.g., concentration) before processing.
- __Logarithmic Transformation__: Field values undergo a log_10 transformation to optimize the visualization of data ranges.
- __Precision Handling__: The transformed field values are recorded with 15 decimal places, accommodating the precision limitations of some GIS software.
- __Metadata Storage__: Information, such as details about released materials, are stored within a .shp.xml file as metadata.

### Scaling the field values

Another feature is to manipulate the field values by scaling with an arbitrary factor. This factor can be set in the preset with the variable `multiplier`.

## Credits

This package was created with [`copier`](https://github.com/copier-org/copier) and the [`MeteoSwiss-APN/mch-python-blueprint`](https://github.com/MeteoSwiss-APN/mch-python-blueprint) project template.

## External Links

* [copier](https://github.com/copier-org/copier) - Based library and CLI app for rendering project templates.
* [mch-python-blueprint](https://github.com/MeteoSwiss-APN/mch-python-blueprint) - Project template embedded in this project based on copier.
* [pyshp](https://github.com/GeospatialPython/pyshp) - Python module to generate Shapefiles

- [copier](https://github.com/copier-org/copier) - Based library and CLI app for rendering project templates.
- [mch-python-blueprint](https://github.com/MeteoSwiss-APN/mch-python-blueprint) - Project template embedded in this project based on copier.
- [pyshp](https://github.com/GeospatialPython/pyshp) - Python module to generate Shapefiles

## License

This project is licensed under the terms of the MeteoSwiss. The full license text can be found in the [LICENSE](LICENSE) file.
In essence, you are free to use, modify, and distribute the software, provided the associated copyright notice and disclaimers are included.
2 changes: 1 addition & 1 deletion VERSION
Original file line number Diff line number Diff line change
@@ -1 +1 @@
1.1.0
1.1.1
6 changes: 3 additions & 3 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,8 @@ build-backend = "setuptools.build_meta"

[project]
name = "pyflexplot"
version = "1.1.0"
description = "Stefan Ruedisuehli's PyFlexPlot"
version = "1.1.1"
description = "PyFlexPlot - Visualize and post-process FLEXPART dispersion simulation results stored in NetCDF format"
readme = "README.md"
keywords = [
"Dispersion",
Expand All @@ -22,7 +22,7 @@ classifiers = [
"Intended Audience :: Developers",
"Natural Language :: English",
"Programming Language :: Python :: 3",
"Programming Language :: Python :: 3.10",
"Programming Language :: Python :: 3.11",
]
requires-python = ">=3.10"
authors = [
Expand Down
2 changes: 1 addition & 1 deletion src/pyflexplot/data/presets/opr/cosmo-1e-ctrl/all_pdf.toml
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# PyFlexPlot setup file to create deterministic COSMO-1E-CTRL plots

[_base]
# Data source: tsa:/scratch/ruestefa/shared/test/pyflexplot/
# Test data source: /store/mch/msopr/pyflexplot_testdata/
infile = "data/cosmo-1e-ctrl/grid_conc_0910_20200216000000.nc"
model = "COSMO-1E"
lang = "de"
Expand Down
2 changes: 1 addition & 1 deletion src/pyflexplot/data/presets/opr/cosmo-1e-ctrl/all_png.toml
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# PyFlexPlot setup file to create deterministic COSMO-1E-CTRL plots

[_base]
# Data source: tsa:/scratch/ruestefa/shared/test/pyflexplot/
# Test data source: /store/mch/msopr/pyflexplot_testdata/
infile = "data/cosmo-1e-ctrl/grid_conc_0910_20200216000000.nc"
model = "COSMO-1E"
lang = "de"
Expand Down
49 changes: 49 additions & 0 deletions src/pyflexplot/data/presets/opr/cosmo-1e-ctrl/all_shp.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
# PyFlexPlot setup file to create deterministic COSMO-1E-CTRL shapefiles

[_base]
# Test data source: /store/mch/msopr/pyflexplot_testdata/
infile = "data/cosmo-1e-ctrl/grid_conc_0910_20200216000000.nc"
model = "COSMO-1E"
lang = "de"
outfile_time_format = "%Y%m%dT%H%M"
combine_species = false

[_base._concentration]
plot_variable = "concentration"
level = 0

[_base._concentration._integr]
integrate = true
time = -1

[_base._tot_deposition]
plot_variable = "tot_deposition"
integrate = true

[_base._affected_area]
plot_variable = "affected_area"
integrate = true
level = 0
time = -1
combine_species = true

[_base._concentration."_shp+"]
outfile = "concentration_site-{release_site}_species-{species_id}_domain-{domain}_lang-{lang}_ts-{time_step}.shp"
[_base._concentration._integr."_shp+"]
outfile = "integrated_concentration_site-{release_site}_species-{species_id}_domain-{domain}_lang-{lang}_ts-{time_step}.shp"
[_base._tot_deposition."_shp+"]
outfile = "deposition_tot_site-{release_site}_species-{species_id}_domain-{domain}_lang-{lang}_ts-{time_step}.shp"
[_base._affected_area."_shp+"]
outfile = "affected_area_site-{release_site}_species-{species_id}_domain-{domain}_lang-{lang}_ts-{time_step}.shp"

[_base._concentration."_multipanel_time+"]
outfile = "concentration_site-{release_site}_species-{species_id}_domain-{domain}_lang-{lang}_ts-{time_step}.shp"
plot_type = "multipanel"
multipanel_param = "time"
time = [2, 4, 8, -1]

["**".full]
domain = "full"

["**".zoom]
domain = "ch"
2 changes: 1 addition & 1 deletion src/pyflexplot/data/presets/opr/cosmo-1e/all_pdf.toml
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# PyFlexPlot setup file to create ensemble CCOSMO-1E plots

[_base]
# Data source: tsa:/scratch/ruestefa/shared/test/pyflexplot/
# Test data source: /store/mch/msopr/pyflexplot_testdata/
infile = "data/cosmo-1e/2021030503/{ens_member:03}/grid_conc_20210305030000.nc_GOE"
ens_member_id = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
model = "COSMO-1E"
Expand Down
2 changes: 1 addition & 1 deletion src/pyflexplot/data/presets/opr/cosmo-1e/all_png.toml
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# PyFlexPlot setup file to create ensemble CCOSMO-1E plots

[_base]
# Data source: tsa:/scratch/ruestefa/shared/test/pyflexplot/
# Test data source: /store/mch/msopr/pyflexplot_testdata/
infile = "data/cosmo-1e/2021030503/{ens_member:03}/grid_conc_20210305030000.nc_GOE"
ens_member_id = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
model = "COSMO-1E"
Expand Down
4 changes: 2 additions & 2 deletions src/pyflexplot/data/presets/opr/cosmo-1e/all_shp.toml
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# PyFlexPlot setup file to create ensemble CCOSMO-1E plots
# PyFlexPlot setup file to create ensemble COSMO-1E shapefiles

[_base]
# Data source: tsa:/scratch/ruestefa/shared/test/pyflexplot/
# Test data source: /store/mch/msopr/pyflexplot_testdata/
infile = "data/cosmo-1e/2021030503/{ens_member:03}/grid_conc_20210305030000.nc_GOE"
ens_member_id = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
model = "COSMO-1E"
Expand Down
Loading

0 comments on commit 9128017

Please sign in to comment.