Source code for landlab.utils.watershed

#! /usr/bin/env python
"""Functions to work with watersheds of model grids."""

import numpy as np

from landlab import FieldError


[docs] def get_watershed_mask(grid, outlet_id): """Get the watershed of an outlet returned as a boolean array. Parameters ---------- grid : RasterModelGrid A landlab RasterModelGrid. outlet_id : integer The id of the outlet node. Returns ------- watershed_mask : boolean ndarray True elements of this array correspond to nodes with flow that is received by the outlet. The length of the array is equal to the grid number of nodes. Examples -------- >>> import numpy as np >>> from landlab import RasterModelGrid >>> from landlab.components import FlowAccumulator >>> from landlab.utils import get_watershed_mask >>> rmg = RasterModelGrid((7, 7)) >>> rmg.at_node["topographic__elevation"] = [ ... [-9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0], ... [-9999.0, 26.0, 0.0, 30.0, 32.0, 34.0, -9999.0], ... [-9999.0, 28.0, 1.0, 25.0, 28.0, 32.0, -9999.0], ... [-9999.0, 30.0, 3.0, 3.0, 11.0, 34.0, -9999.0], ... [-9999.0, 32.0, 11.0, 25.0, 18.0, 38.0, -9999.0], ... [-9999.0, 34.0, 32.0, 34.0, 36.0, 40.0, -9999.0], ... [-9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0], ... ] Only the bottom boundary is set to open. >>> rmg.set_closed_boundaries_at_grid_edges(True, True, True, False) >>> rmg.set_fixed_value_boundaries_at_grid_edges(False, False, False, True) Route flow. >>> fr = FlowAccumulator(rmg, flow_director="D8") >>> fr.run_one_step() >>> get_watershed_mask(rmg, 2).reshape(rmg.shape) array([[False, False, True, False, False, False, False], [False, False, True, False, False, False, False], [False, True, True, True, True, True, False], [False, True, True, True, True, True, False], [False, True, True, True, True, True, False], [False, True, True, True, True, True, False], [False, False, False, False, False, False, False]], dtype=bool) """ if "flow__receiver_node" not in grid.at_node: raise FieldError( "A 'flow__receiver_node' field is required at the " "nodes of the input grid." ) if grid.at_node["flow__receiver_node"].size != grid.size("node"): raise NotImplementedError( "A route-to-multiple flow director has been " "run on this grid. The landlab development team has not " "verified that get_watershed_mask is compatible with " "route-to-multiple methods. Please open a GitHub Issue " "to start this process." ) receiver_at_node = grid.at_node["flow__receiver_node"] upstream_node_order = grid.at_node["flow__upstream_node_order"] # Prepare output. watershed_mask = np.zeros(grid.number_of_nodes, dtype=bool) # loop through all nodes once based on upstream node order. This means we # only need to loop through the nodes once. for node in upstream_node_order: # when the outlet_id is encountered, mark it as true, and set # outlet_found to True. if node == outlet_id: watershed_mask[node] = True # once the outlet is found, set the watershed mask to the value of # the reciever at node, this will paint the watershed in as we move # upstream. if watershed_mask[receiver_at_node[node]]: watershed_mask[node] = True return watershed_mask
[docs] def get_watershed_nodes(grid, outlet_id): """Get the watershed of an outlet returned as a list of nodes. Parameters ---------- grid : RasterModelGrid A landlab RasterModelGrid. outlet_id : integer The id of the outlet node. Returns ------- watershed_nodes : integer ndarray The ids of the nodes that flow to the node with the id, outlet_id. Examples -------- >>> import numpy as np >>> from landlab import RasterModelGrid >>> from landlab.components import FlowAccumulator >>> from landlab.utils import get_watershed_nodes >>> rmg = RasterModelGrid((7, 7)) >>> rmg.at_node["topographic__elevation"] = [ ... [-9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0], ... [-9999.0, 26.0, 0.0, 30.0, 32.0, 34.0, -9999.0], ... [-9999.0, 28.0, 1.0, 25.0, 28.0, 32.0, -9999.0], ... [-9999.0, 30.0, 3.0, 3.0, 11.0, 34.0, -9999.0], ... [-9999.0, 32.0, 11.0, 25.0, 18.0, 38.0, -9999.0], ... [-9999.0, 34.0, 32.0, 34.0, 36.0, 40.0, -9999.0], ... [-9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0], ... ] >>> rmg.set_watershed_boundary_condition_outlet_id( ... 2, rmg.at_node["topographic__elevation"], nodata_value=-9999.0 ... ) Route flow. >>> fr = FlowAccumulator(rmg, flow_director="D8") >>> fr.run_one_step() Get the nodes of two watersheds. >>> mainstem_watershed_nodes = get_watershed_nodes(rmg, 2) >>> tributary_watershed_nodes = get_watershed_nodes(rmg, 24) Given the watershed boundary conditions, the number of mainstem watershed nodes should be equal to the number of core nodes plus the outlet node. >>> len(mainstem_watershed_nodes) == rmg.number_of_core_nodes + 1 True """ ws_mask = get_watershed_mask(grid, outlet_id) ws_nodes = np.where(ws_mask)[0] return ws_nodes
[docs] def get_watershed_masks(grid): """Assign the watershed outlet id to all nodes in the grid. Parameters ---------- grid : RasterModelGrid A landlab RasterModelGrid. Returns ------- watershed_masks : integer ndarray The length of the array is equal to the grid number of nodes. Values of this array are the watershed ids. The value of a watershed id is the node id of the watershed outlet. Examples -------- >>> import numpy as np >>> from landlab import RasterModelGrid >>> from landlab.components import FlowAccumulator >>> from landlab.utils import get_watershed_masks Create a grid with a node spacing of 200 meter. >>> rmg = RasterModelGrid((7, 7), xy_spacing=200) >>> rmg.at_node["topographic__elevation"] = [ ... [-9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0], ... [-9999.0, 26.0, 0.0, 26.0, 30.0, 34.0, -9999.0], ... [-9999.0, 28.0, 1.0, 28.0, 5.0, 32.0, -9999.0], ... [-9999.0, 30.0, 3.0, 30.0, 10.0, 34.0, -9999.0], ... [-9999.0, 32.0, 11.0, 32.0, 15.0, 38.0, -9999.0], ... [-9999.0, 34.0, 32.0, 34.0, 36.0, 40.0, -9999.0], ... [-9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0], ... ] >>> rmg.set_closed_boundaries_at_grid_edges(True, True, True, False) Route flow. >>> fr = FlowAccumulator(rmg, flow_director="D8") >>> fr.run_one_step() Assign mask. >>> mask = get_watershed_masks(rmg) >>> mask.reshape(rmg.shape) array([[ 0, 1, 2, 3, 4, 5, 6], [ 7, 1, 2, 3, 4, 5, 13], [14, 2, 2, 2, 18, 18, 20], [21, 2, 2, 2, 18, 18, 27], [28, 2, 2, 2, 18, 18, 34], [35, 2, 2, 2, 18, 18, 41], [42, 43, 44, 45, 46, 47, 48]]) """ upstream_node_order = grid.at_node["flow__upstream_node_order"] flow__receiver_node = grid.at_node["flow__receiver_node"] watershed_mask = np.arange(grid.size("node"), dtype=int) for node_id in upstream_node_order: watershed_mask[node_id] = watershed_mask[flow__receiver_node[node_id]] return watershed_mask
[docs] def get_watershed_masks_with_area_threshold(grid, critical_area): """Get masks of all of the watersheds with a minimum drainage area size. Parameters ---------- grid : RasterModelGrid A landlab RasterModelGrid. critical_area : integer or float The minimum drainage area of the watersheds to identify. Returns ------- watershed_masks : integer ndarray The length of the array is equal to the grid number of nodes. Values of this array are the watershed ids. The value of a watershed id is the node id of the watershed outlet. Nodes with a value of -1 have only downstream nodes with drainage areas below `critical_area`. Examples -------- >>> import numpy as np >>> from landlab import RasterModelGrid >>> from landlab.components import FlowAccumulator >>> from landlab.utils import get_watershed_masks_with_area_threshold Create a grid with a node spacing of 200 meter. >>> rmg = RasterModelGrid((7, 7), xy_spacing=200) >>> rmg.at_node["topographic__elevation"] = [ ... [-9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0], ... [-9999.0, 26.0, 0.0, 26.0, 30.0, 34.0, -9999.0], ... [-9999.0, 28.0, 1.0, 28.0, 5.0, 32.0, -9999.0], ... [-9999.0, 30.0, 3.0, 30.0, 10.0, 34.0, -9999.0], ... [-9999.0, 32.0, 11.0, 32.0, 15.0, 38.0, -9999.0], ... [-9999.0, 34.0, 32.0, 34.0, 36.0, 40.0, -9999.0], ... [-9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0], ... ] >>> rmg.set_closed_boundaries_at_grid_edges(True, True, True, False) Route flow. >>> fr = FlowAccumulator(rmg, flow_director="D8") >>> fr.run_one_step() Get the masks of watersheds greater than or equal to 80,000 square-meters. >>> critical_area = 80000 >>> mask = get_watershed_masks_with_area_threshold(rmg, critical_area) Verify that all mask null nodes have a drainage area below critical area. >>> null_nodes = np.where(mask == -1)[0] >>> A = rmg.at_node["drainage_area"][null_nodes] >>> below_critical_area_nodes = A < critical_area >>> np.all(below_critical_area_nodes) True """ watershed_masks = get_watershed_masks(grid) area = grid.at_node["drainage_area"] area_of_watersheds = area[watershed_masks] too_small = area_of_watersheds < critical_area watershed_masks[too_small] = -1 return watershed_masks
[docs] def get_watershed_outlet(grid, source_node_id): """Get the downstream-most node (the outlet) of the source node. Parameters ---------- grid : RasterModelGrid A landlab RasterModelGrid. source_node_id : integer The id of the node in which to identify its outlet. Returns ------- outlet_node : integer The id of the node that is the downstream-most node (the outlet) of the source node. Examples -------- >>> import numpy as np >>> from landlab import RasterModelGrid >>> from landlab.components import FlowAccumulator >>> from landlab.utils import get_watershed_outlet >>> rmg = RasterModelGrid((7, 7)) >>> rmg.at_node["topographic__elevation"] = [ ... [-9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0], ... [-9999.0, 26.0, 0.0, 30.0, 32.0, 34.0, -9999.0], ... [-9999.0, 28.0, 1.0, 25.0, 28.0, 32.0, -9999.0], ... [-9999.0, 30.0, 3.0, 3.0, 11.0, 34.0, -9999.0], ... [-9999.0, 32.0, 11.0, 25.0, 18.0, 38.0, -9999.0], ... [-9999.0, 34.0, 32.0, 34.0, 36.0, 40.0, -9999.0], ... [-9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0, -9999.0], ... ] >>> imposed_outlet = 2 >>> rmg.set_watershed_boundary_condition_outlet_id( ... imposed_outlet, rmg.at_node["topographic__elevation"], nodata_value=-9999.0 ... ) Route flow. >>> fr = FlowAccumulator(rmg, flow_director="D8") >>> fr.run_one_step() Get the grid watershed outlet. >>> determined_outlet = get_watershed_outlet(rmg, 40) >>> determined_outlet == imposed_outlet True """ if "flow__receiver_node" not in grid.at_node: raise FieldError( "A 'flow__receiver_node' field is required at the " "nodes of the input grid." ) if grid.at_node["flow__receiver_node"].size != grid.size("node"): raise NotImplementedError( "A route-to-multiple flow director has been " "run on this grid. The landlab development team has not " "verified that get_watershed_outlet is compatible with " "route-to-multiple methods. Please open a GitHub Issue " "to start this process." ) receiver_at_node = grid.at_node["flow__receiver_node"] receiver_node = receiver_at_node[source_node_id] outlet_not_found = True while outlet_not_found: node_is_outlet = receiver_node == source_node_id node_is_boundary = grid.node_is_boundary(receiver_node) node_is_pit = receiver_node == receiver_at_node[receiver_node] if node_is_outlet or node_is_boundary or node_is_pit: outlet_not_found = False outlet_node = receiver_node else: receiver_node = receiver_at_node[receiver_node] return outlet_node