# *StormReactor*: Python package for modeling any pollutant generation or treatment method in EPA SWMM
[![Code style: black](https://img.shields.io/badge/code%20style-black-000000.svg)](https://github.com/python/black)
[![Python package](https://github.com/kLabUM/StormReactor/actions/workflows/python-package.yml/badge.svg)](https://github.com/kLabUM/StormReactor/actions/workflows/python-package.yml)
## Overview
*StormReactor* was developed to expand the ability to model stormwater quality and water quality based real-time control in EPA's Stormwater Management Model (SWMM). It is a natural extension of *Open-Storm*'s(http://open-storm.org/) mission to open up and ease access into the technical world of smart stormwater systems. *StormReactor* enables users to model any stormwater pollutant treatment or generation method in any node or link in a stormwater network. A user can implement any SWMM treatment function defined in the *SWMM Reference Manual Volume III: Water Quality*, select from a few additional methods we built, or create their own.
Note: In order to use *StormReactor*'s ability to change the pollutant concentration in a node, you MUST have treatment enabled in the SWMM input file. This means if you open your input file in a text editor, it should have a treatment listed for the node you want to access. When *StormReactor* runs, it will replace the treatment in the input file with the method you want. See snippet from an example input file below.
```example1.inp
STORAGE]
;;Name Elev. MaxDepth InitDepth Shape Curve Name/Params N/A Fevap Psi Ksat IMD
;;-------------- -------- ---------- ----------- ---------- ---------------------------- -------- -------- -------- --------
Tank 10 5 0 TABULAR Tank_Curve 0 0
[POLLUTANTS]
;;Name Units Crain Cgw Crdii Kdecay SnowOnly Co-Pollutant Co-Frac Cdwf Cinit
;;-------------- ------ ---------- ---------- ---------- ---------- ---------- ---------------- ---------- ---------- ----------
P1 MG/L 0.0 0.0 0 0.0 NO * 0.0 0.0 0
[TREATMENT]
;;Node Pollutant Function
;;-------------- ---------------- ----------
Tank P1 R = 0.5
```
## Installation
**Requirements**
- python 3.6+
- numpy
- pyswmm 1.2.0+
- scipy
**PyPI**
*StormReactor* is available through PyPI at https://pypi.python.org/pypi/StormReactor/ or you can install it directly in your terminal using the command below. Please raise an issue on the repository or reach out if you run into any issues installing or using the package.
```bash
$ pip install StormReactor
```
## How to Use *StormReactor*
### Example 1
Here is a simple example on how to use *StormReactor* for modeling a variety of water quality methods (e.g., gravity settling, event mean concentration) for a pollutant (e.g., TSS) in several stormwater assets (e.g., basin, channel). This example covers all existing pollutant treatment and generation methods in *StormReactor* except a completely stirred tank reactor (CSTR). Please see the next example for modeling a CSTR.
```python
# import packages
import StormReactor
from pyswmm import Simulation
# build water quality configuration dictionary
config = {'basin': { 'type': 'node', 'pollutant': 'P1', 'method': 'GravitySettling', 'parameters': {'k': 0.0005, 'C_s': 21.0}},\
'channel': { 'type': 'link', 'pollutant': 'P1', 'method': 'EventMeanConc', 'parameters': {'C': 10.0}}}
# initialize water quality
with Simulation('example1.inp') as sim:
WQ = waterQuality(sim, config)
for step in sim:
# update each time step
WQ.updateWQState()
```
### Example 2
Here is a simple example for modeling a CSTR for a pollutant (e.g., nitrate) in several stormwater assets (e.g., basin, wetland). Note you must call `updateWQState_CSTR(index)` instead of `updateWQState()` because the CSTR code requires the additonal input of `index`. This is the only difference for modeling a CSTR.
```python
# import packages
import StormReactor
from pyswmm import Simulation
# build water quality configuration dictionary
config = {'basin': { 'type': 'node', 'pollutant': 'P1', 'method': 'CSTR', 'parameters': {'k': -0.0005, 'n': 1.0, 'Co': 10.0}},\
'wetland': { 'type': 'node', 'pollutant': 'P1', 'method': 'CSTR', 'parameters': {'k': -0.000089, 'n': 3.0, 'Co': 10.0}}}
# initialize water quality
with Simulation('example2.inp') as sim:
WQ = waterQuality(sim, config)
for step in sim:
# update each time step
WQ.updateWQState_CSTR(index)
```
## Water Quality Methods
- `EventMeanConc`:
Event Mean Concentration -- treatment results in a constant concentration.
Treatment method parameters required:
`C` = constant treatment concentration for each pollutant (SI/US: mg/L)
- `ConstantRemoval`
Constant Removal -- treatment results in a constant percent removal.
Treatment method parameters required:
`R` = pollutant removal fraction (unitless)
- `CoRemoval`:
Co-Removal Treatment -- removal of some pollutant is proportional to the removal of some other pollutant
Treatment method parameters required:
`R1` = pollutant removal fraction (unitless)
`R2` = pollutant removal fraction for other pollutant (unitless)
- `ConcDependRemoval`:
Concentration-Dependent Removal -- when higher pollutant removal efficiencies occur with higher influent concentrations
Treatment method parameters required:
`R_l` = lower removal rate (unitless)
`BC` = boundary concentration that determines removal rate (SI/US: mg/L)
`R_u` = upper removal rate (unitless)
- `NthOrderReaction`:
Nth Order Reaction Kinetics -- when treatemtn of pollutant X exhibits n-th order reaction kinetics where the instantaneous reaction rate is kC^n
Treatment method parameters required:
`k` = reaction rate constant (SI: m/hr, US: ft/hr)
`n` = reaction order (first order, second order, etc.) (unitless)
- `kCModel`:
K-C Star Model -- the first-order model with background concentration made popular by Kadlec and Knight (1996) for long-term treatment performance of wetlands
Treatment method parameters required:
`k` = reaction rate constant (SI: m/hr, US: ft/hr)
`C_s` = constant residual concentration that always remains (SI/US: mg/L)
- `GravitySettling`:
Gravity Settling -- during a quiescent period of time within a storage volume, a fraction of suspended particles will settle out
Treatment method parameters required:
`k` = reaction rate constant (SI: m/hr, US: ft/hr)
`C_s` = constant residual concentration that always remains (SI/US: mg/L)
StormReactor also includes a few additional water quality methods, like a CSTR and a bioretention cell phosphorus model. Users can also create their own water quality methods. Please see the [StormReactor repository](https://github.com/kLabUM/StormReactor) above for more details.
## Creating Your Own Water Quality Method
To create a new water quality method, follow the steps below:
1. Fork the repository to your own personal repository.
2. Add the name of your new method to the water quality methods definition in `waterQuality()` within waterQuality.py
```python
# Water quality methods
self.method = {
"EventMeanConc": self._EventMeanConc,
"ConstantRemoval": self._ConstantRemoval,
"CoRemoval": self._CoRemoval,
"ConcDependRemoval": self._ConcDependRemoval,
"NthOrderReaction": self._NthOrderReaction,
"kCModel": self._kCModel,
"GravitySettling": self._GravitySettling,
"CSTR": self._CSTRSolver,
"Phosphorus": self._Phosphorus,
"NewMethod": self._NewMethod
}
```
3. Add the definition of your new water quality method to the end of `waterQuality()` within waterQuality.py. Be sure to include all the necessary method inputs: `self, ID, pollutant_ID, dictionary, flag`. You can use any of the PySWMM/SWMMM getters to get the necessary water quantity and quality values for your model. Also be sure to set `parameters = dictionary` so that you can access your inputs in your dictionary. Once your model code is added, don't forget to set the new node and link concentrations in SWMM using the appropriate setters.
```python
def _NewMethod(self, ID, pollutant_ID, dictionary, flag):
"""
Add method description and required parameters.
"""
# Set parameters = dictionary so you can access your dictionary parameters.
parameters = dictionary
"""
CODE BLOCK
New method code to calculate new pollutant concentration, here referred to as Cnew.
Set the concentration in SWMM using the appropriate setters using the flag feature.
"""
if self.flag == 0:
self.sim._model.setNodePollut(ID, pollutant_ID, Cnew)
else:
self.sim._model.setLinkPollut(ID, pollutant_ID, Cnew)
```
4. Now run your new model! Modify code as needed.
## Bugs
Our issue tracker is at https://github.com/kLabUM/StormReactor/issues. Please report any bugs that you find. Or even better, fork the repository on GitHub and create a pull request. All changes are welcome, big or small, and we will help you make the pull request if you are new to git (just ask on the issue).
## Contributions
If you want to contribute your water quality methods to *StormReactor*, please follow these steps:
1. Raise an issue on the issue tracker at https://github.com/kLabUM/StormReactor/issues to describe the new method you are proposing to add.
2. Follow the steps above in "Creating Your Own Water Quality Method" to build your new method.
3. Create tests to confirm your new method works. Please follow the format for node and link tests as shown at https://github.com/kLabUM/StormReactor/tree/master/tests.
4. Submit a pull request at https://github.com/kLabUM/StormReactor/pulls to merge your edits with the existing *StormReactor* code base.
Note: There might be comments, suggestions, and questions. We're all working together to produce cohesive and high-quality software.
## Citing StormReactor
If you use our package, please cite our software as follows:
Mason, Brooke, Mullapudi, Abhiram, & Kerkez, Branko. (2021). StormReactor (v1.1). Zenodo. https://doi.org/10.5281/zenodo.7110344
Raw data
{
"_id": null,
"home_page": "https://github.com/kLabUM/StormReactor",
"name": "StormReactor",
"maintainer": "",
"docs_url": null,
"requires_python": ">=3.6",
"maintainer_email": "",
"keywords": "swmm pyswmm pollutants modeling water-quality",
"author": "Brooke Mason, Abhiram Mullapudi",
"author_email": "bemason@umich.edu, abhiramm@umich.edu",
"download_url": "https://files.pythonhosted.org/packages/26/f3/45f10017ea0b22dd4b3815d4b369a5c11771b688b93a0ecb531c7bc60bdd/StormReactor-1.3.0.tar.gz",
"platform": null,
"description": "# *StormReactor*: Python package for modeling any pollutant generation or treatment method in EPA SWMM\n[![Code style: black](https://img.shields.io/badge/code%20style-black-000000.svg)](https://github.com/python/black)\n[![Python package](https://github.com/kLabUM/StormReactor/actions/workflows/python-package.yml/badge.svg)](https://github.com/kLabUM/StormReactor/actions/workflows/python-package.yml)\n## Overview \n\n*StormReactor* was developed to expand the ability to model stormwater quality and water quality based real-time control in EPA's Stormwater Management Model (SWMM). It is a natural extension of *Open-Storm*'s(http://open-storm.org/) mission to open up and ease access into the technical world of smart stormwater systems. *StormReactor* enables users to model any stormwater pollutant treatment or generation method in any node or link in a stormwater network. A user can implement any SWMM treatment function defined in the *SWMM Reference Manual Volume III: Water Quality*, select from a few additional methods we built, or create their own. \n\nNote: In order to use *StormReactor*'s ability to change the pollutant concentration in a node, you MUST have treatment enabled in the SWMM input file. This means if you open your input file in a text editor, it should have a treatment listed for the node you want to access. When *StormReactor* runs, it will replace the treatment in the input file with the method you want. See snippet from an example input file below.\n\n```example1.inp\nSTORAGE]\n;;Name Elev. MaxDepth InitDepth Shape Curve Name/Params N/A Fevap Psi Ksat IMD \n;;-------------- -------- ---------- ----------- ---------- ---------------------------- -------- -------- -------- --------\nTank 10 5 0 TABULAR Tank_Curve 0 0 \n\n[POLLUTANTS]\n;;Name Units Crain Cgw Crdii Kdecay SnowOnly Co-Pollutant Co-Frac Cdwf Cinit \n;;-------------- ------ ---------- ---------- ---------- ---------- ---------- ---------------- ---------- ---------- ----------\nP1 MG/L 0.0 0.0 0 0.0 NO * 0.0 0.0 0 \n[TREATMENT]\n;;Node Pollutant Function \n;;-------------- ---------------- ----------\nTank P1 R = 0.5\n```\n\n\n## Installation \n\n**Requirements**\n- python 3.6+\n- numpy\n- pyswmm 1.2.0+\n- scipy\n\n**PyPI**\n\n*StormReactor* is available through PyPI at https://pypi.python.org/pypi/StormReactor/ or you can install it directly in your terminal using the command below. Please raise an issue on the repository or reach out if you run into any issues installing or using the package. \n\n```bash \n$ pip install StormReactor\n```\n\n\n## How to Use *StormReactor*\n\n### Example 1\n\nHere is a simple example on how to use *StormReactor* for modeling a variety of water quality methods (e.g., gravity settling, event mean concentration) for a pollutant (e.g., TSS) in several stormwater assets (e.g., basin, channel). This example covers all existing pollutant treatment and generation methods in *StormReactor* except a completely stirred tank reactor (CSTR). Please see the next example for modeling a CSTR.\n\n```python \n# import packages\nimport StormReactor\nfrom pyswmm import Simulation\n\n# build water quality configuration dictionary\nconfig = {'basin': { 'type': 'node', 'pollutant': 'P1', 'method': 'GravitySettling', 'parameters': {'k': 0.0005, 'C_s': 21.0}},\\\n\t\t\t'channel': { 'type': 'link', 'pollutant': 'P1', 'method': 'EventMeanConc', 'parameters': {'C': 10.0}}}\n\n\n# initialize water quality\nwith Simulation('example1.inp') as sim:\n\tWQ = waterQuality(sim, config)\n\n\tfor step in sim:\n\t\t# update each time step\n\t\tWQ.updateWQState()\n```\n\n### Example 2\n\nHere is a simple example for modeling a CSTR for a pollutant (e.g., nitrate) in several stormwater assets (e.g., basin, wetland). Note you must call `updateWQState_CSTR(index)` instead of `updateWQState()` because the CSTR code requires the additonal input of `index`. This is the only difference for modeling a CSTR.\n\n```python \n# import packages\nimport StormReactor\nfrom pyswmm import Simulation\n\n# build water quality configuration dictionary\nconfig = {'basin': { 'type': 'node', 'pollutant': 'P1', 'method': 'CSTR', 'parameters': {'k': -0.0005, 'n': 1.0, 'Co': 10.0}},\\\n\t\t\t'wetland': { 'type': 'node', 'pollutant': 'P1', 'method': 'CSTR', 'parameters': {'k': -0.000089, 'n': 3.0, 'Co': 10.0}}}\n\n\n# initialize water quality\nwith Simulation('example2.inp') as sim:\n\tWQ = waterQuality(sim, config)\n\n\tfor step in sim:\n\t\t# update each time step\n\t\tWQ.updateWQState_CSTR(index)\n\n```\n## Water Quality Methods\n\n- `EventMeanConc`: \n Event Mean Concentration -- treatment results in a constant concentration.\n Treatment method parameters required:\n `C` = constant treatment concentration for each pollutant (SI/US: mg/L)\n\n- `ConstantRemoval`\n Constant Removal -- treatment results in a constant percent removal.\n Treatment method parameters required:\n `R` = pollutant removal fraction (unitless)\n\n- `CoRemoval`:\n Co-Removal Treatment -- removal of some pollutant is proportional to the removal of some other pollutant\n Treatment method parameters required:\n `R1` = pollutant removal fraction (unitless) \n `R2` = pollutant removal fraction for other pollutant (unitless)\n\n- `ConcDependRemoval`:\n Concentration-Dependent Removal -- when higher pollutant removal efficiencies occur with higher influent concentrations\n Treatment method parameters required:\n `R_l` = lower removal rate (unitless)\n `BC` = boundary concentration that determines removal rate (SI/US: mg/L)\n `R_u` = upper removal rate (unitless)\n\n- `NthOrderReaction`:\n Nth Order Reaction Kinetics -- when treatemtn of pollutant X exhibits n-th order reaction kinetics where the instantaneous reaction rate is kC^n\n Treatment method parameters required:\n `k` = reaction rate constant (SI: m/hr, US: ft/hr)\n `n` = reaction order (first order, second order, etc.) (unitless)\n\n- `kCModel`:\n K-C Star Model -- the first-order model with background concentration made popular by Kadlec and Knight (1996) for long-term treatment performance of wetlands\n Treatment method parameters required:\n `k` = reaction rate constant (SI: m/hr, US: ft/hr)\n `C_s` = constant residual concentration that always remains (SI/US: mg/L)\n\n- `GravitySettling`:\n Gravity Settling -- during a quiescent period of time within a storage volume, a fraction of suspended particles will settle out\n Treatment method parameters required:\n `k` = reaction rate constant (SI: m/hr, US: ft/hr)\n `C_s` = constant residual concentration that always remains (SI/US: mg/L)\n\nStormReactor also includes a few additional water quality methods, like a CSTR and a bioretention cell phosphorus model. Users can also create their own water quality methods. Please see the [StormReactor repository](https://github.com/kLabUM/StormReactor) above for more details.\n\n\n\n## Creating Your Own Water Quality Method\n\nTo create a new water quality method, follow the steps below:\n1. Fork the repository to your own personal repository.\n2. Add the name of your new method to the water quality methods definition in `waterQuality()` within waterQuality.py\n```python \n# Water quality methods\nself.method = {\n \"EventMeanConc\": self._EventMeanConc,\n \"ConstantRemoval\": self._ConstantRemoval,\n \"CoRemoval\": self._CoRemoval,\n \"ConcDependRemoval\": self._ConcDependRemoval,\n \"NthOrderReaction\": self._NthOrderReaction,\n \"kCModel\": self._kCModel,\n \"GravitySettling\": self._GravitySettling,\n \"CSTR\": self._CSTRSolver,\n \"Phosphorus\": self._Phosphorus,\n \"NewMethod\": self._NewMethod\n }\n```\n3. Add the definition of your new water quality method to the end of `waterQuality()` within waterQuality.py. Be sure to include all the necessary method inputs: `self, ID, pollutant_ID, dictionary, flag`. You can use any of the PySWMM/SWMMM getters to get the necessary water quantity and quality values for your model. Also be sure to set `parameters = dictionary` so that you can access your inputs in your dictionary. Once your model code is added, don't forget to set the new node and link concentrations in SWMM using the appropriate setters.\n```python \ndef _NewMethod(self, ID, pollutant_ID, dictionary, flag):\n\t\"\"\"\n\tAdd method description and required parameters.\n\t\"\"\"\n\t# Set parameters = dictionary so you can access your dictionary parameters.\n\tparameters = dictionary\n\n\t\"\"\"\n\tCODE BLOCK\n\tNew method code to calculate new pollutant concentration, here referred to as Cnew.\n\tSet the concentration in SWMM using the appropriate setters using the flag feature.\n\t\"\"\"\n\tif self.flag == 0:\n\t\tself.sim._model.setNodePollut(ID, pollutant_ID, Cnew)\n\telse:\n\t\tself.sim._model.setLinkPollut(ID, pollutant_ID, Cnew)\n\t\n```\n4. Now run your new model! Modify code as needed.\n\n\n## Bugs\n\nOur issue tracker is at https://github.com/kLabUM/StormReactor/issues. Please report any bugs that you find. Or even better, fork the repository on GitHub and create a pull request. All changes are welcome, big or small, and we will help you make the pull request if you are new to git (just ask on the issue).\n\n\n## Contributions\n\nIf you want to contribute your water quality methods to *StormReactor*, please follow these steps:\n1. Raise an issue on the issue tracker at https://github.com/kLabUM/StormReactor/issues to describe the new method you are proposing to add. \n2. Follow the steps above in \"Creating Your Own Water Quality Method\" to build your new method. \n3. Create tests to confirm your new method works. Please follow the format for node and link tests as shown at https://github.com/kLabUM/StormReactor/tree/master/tests. \n4. Submit a pull request at https://github.com/kLabUM/StormReactor/pulls to merge your edits with the existing *StormReactor* code base.\nNote: There might be comments, suggestions, and questions. We're all working together to produce cohesive and high-quality software.\n\n## Citing StormReactor\n\nIf you use our package, please cite our software as follows:\nMason, Brooke, Mullapudi, Abhiram, & Kerkez, Branko. (2021). StormReactor (v1.1). Zenodo. https://doi.org/10.5281/zenodo.7110344\n",
"bugtrack_url": null,
"license": "",
"summary": "StormReactor: Python package for modelling any pollutant generation or treatment method in SWMM",
"version": "1.3.0",
"split_keywords": [
"swmm",
"pyswmm",
"pollutants",
"modeling",
"water-quality"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "08db7453e8d89f2f44a7ff582ca67927928075753be206a78150d72dec26f504",
"md5": "784be2f5a1a3411a786cc8869a53ccdb",
"sha256": "fd61bf25aadd220cbab9c8f850cd2963d42bcfaa0d6b41e9c97c6609c5816dbb"
},
"downloads": -1,
"filename": "StormReactor-1.3.0-py3-none-any.whl",
"has_sig": false,
"md5_digest": "784be2f5a1a3411a786cc8869a53ccdb",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": ">=3.6",
"size": 12578,
"upload_time": "2023-01-03T13:14:13",
"upload_time_iso_8601": "2023-01-03T13:14:13.597302Z",
"url": "https://files.pythonhosted.org/packages/08/db/7453e8d89f2f44a7ff582ca67927928075753be206a78150d72dec26f504/StormReactor-1.3.0-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "26f345f10017ea0b22dd4b3815d4b369a5c11771b688b93a0ecb531c7bc60bdd",
"md5": "706e14e42e23a9a5405027dce98b8a2c",
"sha256": "61337171f105bdf2c756368650745c1ffab22dabf8a8929ace875f3951c77945"
},
"downloads": -1,
"filename": "StormReactor-1.3.0.tar.gz",
"has_sig": false,
"md5_digest": "706e14e42e23a9a5405027dce98b8a2c",
"packagetype": "sdist",
"python_version": "source",
"requires_python": ">=3.6",
"size": 14648,
"upload_time": "2023-01-03T13:14:15",
"upload_time_iso_8601": "2023-01-03T13:14:15.003404Z",
"url": "https://files.pythonhosted.org/packages/26/f3/45f10017ea0b22dd4b3815d4b369a5c11771b688b93a0ecb531c7bc60bdd/StormReactor-1.3.0.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2023-01-03 13:14:15",
"github": true,
"gitlab": false,
"bitbucket": false,
"github_user": "kLabUM",
"github_project": "StormReactor",
"travis_ci": false,
"coveralls": false,
"github_actions": true,
"requirements": [
{
"name": "numpy",
"specs": [
[
"==",
"1.22.0"
]
]
},
{
"name": "pyswmm",
"specs": [
[
"==",
"1.2.0"
]
]
},
{
"name": "scipy",
"specs": [
[
"==",
"1.7.1"
]
]
},
{
"name": "pytest",
"specs": [
[
"==",
"7.1.2"
]
]
},
{
"name": "scikit-learn",
"specs": [
[
"==",
"1.1.2"
]
]
}
],
"lcname": "stormreactor"
}