A set of Python scripts for delineating watersheds or drainage basins using data from MERIT-Hydro and MERIT-Basins. The script outputs geospatial data for subbasins and river reaches, and creates a river network graph representation, which can be useful for watershed modeling or machine learning applications.
This script also lets you subdivide the watershed at specific locations, such as gages, if you provide additional points that fall inside the main watershed, as shown in red in the image below.
These scripts are a heavily modified fork of delineator.py.
Geodata in a variety of formats -- shapefile, geopackage, GeoJSON, etc., for:
- sub-basin polygons
- sub-basin outlet points
- river reaches
and, optionally:
- A network graph representation of the river network.
The network graph can be saved in a variety of formats -- Python NetworkX Graph object (in a pickle file), JSON, GML, XML, etc.
You can also customize the size of the subbasins to make them larger; see Outputing larger subbasins below.
This repository includes sample data covering Iceland. To delineate watersheds in other parts of the world, you will need to download datasets from MERIT-Hydro and MERIT-Basins. Instructions on how to get the data and run the script are provided below.
To get started, download the latest release from this GitHub repository (or fork the repository).
These scripts were developed and tested with Python version 3.11 and 3.12.
I recommend creating a Python virtual environment in which to run the script. Here is a good [introduction to virtual environments] (https://python.land/virtual-environments/virtualenv) -- why they are useful, and how to use them. You can create and activate the virtual environment, then install all the required packages with the following commands.
Open the Terminal (Linux and Mac) or Command Prompt (Windows), cd
to the directory containing delineator.py
and related files, then enter these commands.
To create the virtual environment:
python -m venv venv
To activate the virtual environment (on Windows you may need to allow scripts using -- search for Set-ExecutionPolicy
).
# Windows Command Prompt or PowerShell
venv\Scripts\activate.bat
# Windows PowerShell
venv\Scripts\Activate.ps1
#Linux and MacOS venv activation
$ source venv/bin/activate
Next, install required packages:
$ pip install -r requirements.txt
This script uses the latest versions of Python packages circa April 2024.
Make sure that you have the correct version of numpy. The pysheds v0.4
library
is not compatible with numpy v2.0.
If you want the scripts to make diagrams of your river network, you will need to install the program GraphViz (not just the Python library, but the standalone software as well). Go to: https://graphviz.org/download/. During the installation, make sure you add GraphViz to the system path.
The script will make these diagrams if you set the variable NETWORK_DIAGRAMS = True
in config.py
.
The major steps are the following, with more detailed instructions below. To simply try the program with the sample data provided, you can skip to step 5. When you are ready to try with your own locations, you will need to download additional data as described in steps 1 and 2.
- Download basin-scale MERIT-Hydro raster data (mghydro.com)
- Download MERIT-Basins vector data (reachhydro.com)
- Create a CSV file with your desired watershed outlet points
- Edit settings in
config.py
- Run
subbasins.py
to delineate watersheds - Review output
- Run again to fix mistakes (repeat steps 4 – 7)
Before you begin downloading the data in steps 1 and 2, determine which files you need based on your region of interest. The data files are organized into continental-scale river basins, or Pfafstetter Level 2 basins. There are 61 of these basins in total. Basins are identified by a 2-digit code, with values from 11 to 91.
You will need two sets of MERIT-Hydro gridded (or raster) data: flow accumulation and flow direction. The authors of the MERIT-Hydro created 5-degree tiles, but this script needs seamless layers that cover entire river basins. I have already done the work of merging these raster data files. You can freely download them here: https://mghydro.com/watersheds/rasters
You will need to update config.py
to tell the script where to find these data files.
Modify these variables:
MERIT_FDIR_DIR
(for flow direction)MERIT_ACCUM_DIR
(for flow accumulation)
Download the shapefiles for unit catchments and river reaches from the creators of MERIT-Basins. Follow the instructions here: https://www.reachhydro.org/home/params/merit-basins
In the folder pfaf_level_02
, download two sets of files:
- unit catchment shapefiles:
cat_pfaf_##_MERIT_Hydro_v07_Basins_v01.shp
- river reach shapefiles:
riv_pfaf_##_MERIT_Hydro_v07_Basins_v01.shp
In these files, ##
is the Pfafstetter Level 2 basin code.
See the figure above to determine which of the 61 level 2 basins you need,
depending on your region of interest.
(Note: Do NOT download the files marked 'bugfix' from MERIT-Basins. These files cause errors for some reason.)
Unzip these files and save them to a folder on your hard drive.
Then, in config.py
, update the variables
CATCHMENTS_DIR
and RIVERS_DIR
to tell the script where to find these data.
The script reads information about your desired watershed outlet points from a plain-text comma-delimited (CSV) file. Edit this file carefully, as the script will not run if this file is not formatted correctly.
The CSV file must contain these 4 required fields or columns.
-
id - required: a unique identifier for your watershed or outlet point, an alphanumeric string. The id may be any length, but shorter is better. The script uses the id as the filename for output, so avoid using any forbidden characters. On Linux, do not use the forward slash /. On Windows, the list of forbidden characters is slightly longer (
\< \> : " / \ | ? \*
). Also, do not use id = 0, since the convention is that 0 is reserved for discharging to the ocean. -
lat - required: latitude in decimal degrees of the watershed outlet. Avoid using a whole number without a decimal in the first row. For example, use 23.0 instead of 23.
-
lng - required: longitude in decimal degrees
-
is_outlet: If the gage is a watershed outlet, enter
true
orTrue
(capitalization does not matter). For intermediate upstream points that will be sub-basin outlets, enterfalse
orFalse
.
All latitude and longitude coordinates should be in decimal degrees (EPSG: 4326, https://spatialreference.org/ref/epsg/4326//).
-
The order of the columns does not matter, but the names must be exactly as shown above.
-
If you are delineating more than one main watershed, put any subbasin outlets immediately after the main outlet, as in the following example:
id | lat | lng | name | is_outlet |
---|---|---|---|---|
foz-tua | 41.217954 | -7.423504 | "Foz Tua, Portugal" | true |
barragem-torga | 41.720 | -7.113 | "Barragem de Torga" | false |
cm-1185 | 41.359 | -7.402 | "CM 1185 Crossing near Martim" | false |
baixo-sabor | 41.22864 | -7.0124 | "Baixo Sabor, Portugal" | true |
algoso | 41.455 | -6.591 | "EN 219 Crossing near Algoso" | false |
In this example, there are two main outlets. The first, "foz-tua," has two subbasin outlets. The second, "baixo-sabor," has one subbasin outlet.
Read through the options and set the variables as you wish. Make sure to set the correct folder paths to where you are storing the input data on your computer.
Once you have downloaded the datasets listed above, and updated config.py
,
you are ready to delineate watersheds. The script takes exactly two arguments:
input_csv
- Input CSV filename, for example outlets.csv
output_prefix
- Output prefix, a string. The output files will start with this string. For example,
if you provide 'shasta', the script will produce shasta_subbasins.shp
, shasta_outlets.shp
, etc.
Run the script from the command line like this:
$ python subbasins.py outlets.csv shasta
This assumes you have put outlets.csv
in the same folder with the scripts,
which is not always convenient. You can provide a full file path as follows
(examples for Windows and Linux):
> python subbasins.py C:\Users\matt\Desktop\outlets.csv test
$ python subbasins.py /home/files/outlets.csv test
Alternatively, you can call the delineation routine from Python, in a script or via the console:
>> from subbasins import delineate
>> delineate('outlets.csv', 'testrun')
The script can output several different geodata formats,
as long as the format is supported by GeoPandas
. Shapefiles are popular,
but I recommend GeoPackage, as it is a more modern and open format.
Feather is another lightweight, portable data format.
To get a full list of available formats, follow the directions
here
(see Supported Drivers).
Automated watershed delineation is often incorrect. The good news is that errors can often be fixed by slightly moving the location of your watershed outlets.
Repeat steps 3 to 6 to create a new outlets CSV file, or modifying your existing file, using revised coordinates. The script will automatically overwrite existing files without any warning, so first make sure to back up anything you want to save.
Optional. One of the slow steps in the script is reading shapefiles and creating a GeoDataFrame.
Once you have done this once, you can save time in the future by storing the GeoDataFrame as a .pkl file.
Enter a file path for the constant PICKLE_DIR
, and the script will automatically save files here.
The script will not search for pickle files if you leave this as a blank string, ''
Note that these files are not any smaller than the original shapefile, so they don't save disk space;
they are just faster to load.
The Python routine I am using to simplify the subbasin polygons (topojson
) is not perfect.
Sometimes, the output will contain weird overlaps and slivers. If appearances matter,
I recommend setting SIMPLIFY = False
and using external software for simplification.
Mapshaper works well. You can use the web version, or you can install and run it from the command line. Note that mapshaper will only accept shapefiles or geojson as input, and not geopackages or feather files.
As an alternative, GIS software like QGIS (free) or ArcGIS (commercial) do the job nicely.
The unit catchments in MERIT-Basins have an average size of around 40 km². If you wish to create
larger subbasins, in config.py
, set the variable CONSOLIDATE = True
.
Then, set a value for MAX_AREA
in km². This sets the upper limit on the size of subbasins.
The script will merge unit catchments such that the overall structure
and connectivity of the drainage network is maintained. This example shows the subbasins
for the Yellowstone River with different values of MAX_AREA
.
By "rediscretizing" the subbasins with this option, you can reduce their number while increasing their size. This means that your hydrologic model will be smaller and simpler and probably run faster! 😊
The simplification routine also appears to
make the subbasin sizes somewhat more homogeneous. The area of MERIT-Basins unit catchments
is highly variable, and highly skewed, with many very small unit catchments.
After consolidation, the distribution
of subbasin areas tends to be more tightly clustered around the mean,
as indicated by a lower coefficient
of variation (standard deviation divided by the mean). Here are the results of a
little experiment in using different values of MAX_AREA
for the Yellowstone River
basin in North America. Statistics are for the subbasin areas in km².
MAX_AREA | count | median | mean | std. dev. | CV | skewness |
---|---|---|---|---|---|---|
None | 687 | 37 | 45 | 36 | 0.80 | 1.60 |
200 | 239 | 140 | 128 | 53 | 0.41 | -1.61 |
500 | 97 | 344 | 315 | 132 | 0.42 | -0.52 |
1,000 | 52 | 641 | 588 | 281 | 0.48 | -0.39 |
2,000 | 24 | 1,370 | 1,270 | 490 | 0.39 | -1.03 |
5,000 | 14 | 2,100 | 2,180 | 1,310 | 0.60 | 0.09 |
The code mostly assumes that the user's inputs make sense, and I have not created extensive error handling for bad inputs, or code to handle all the possible edge cases. For example, the MERIT-Basins dataset has some gaps and slivers of missing data; if your outlet points falls into one of these locations, it may cause problems. In this case, simply nudge the location by changing the latitude and/or longitude slightly.
As another example, if you put two points in your input file that are the same, or very close to one another, the script will fail and the error messages will not be very helpful. Please make sure all of your points have a little space between them!
For comments or questions, please contact the author: Matthew Heberger, [email protected].
To report any bugs, you can create an Issue on this GitHub page.
This code is open source, so if you are motivated to make any modifications, additions, or bug fixes, you can fork the repository and then do a pull request on GitHub.
This code was developed with support from Upstream Tech.