Skip to content

Commit

Permalink
Merge branch 'dev' into dev-docker-mac
Browse files Browse the repository at this point in the history
  • Loading branch information
mluck committed Jan 6, 2025
2 parents cfe969f + 5e20284 commit c61bfa4
Show file tree
Hide file tree
Showing 22 changed files with 430 additions and 128 deletions.
3 changes: 2 additions & 1 deletion data/bridges/pull_osm_bridges.py
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,7 @@ def pull_osm_features_by_huc(huc_bridge_file, huc_num, huc_geom):
gdf['railway'] = None

# Create the bridge_type column by combining above information
gdf['HUC'] = huc_num
gdf['bridge_type'] = gdf.apply(
lambda row: (
f"highway-{row['highway']}" if pd.notna(row['highway']) else f"railway-{row['railway']}"
Expand Down Expand Up @@ -187,7 +188,7 @@ def combine_huc_features(output_dir):
section_time = dt.datetime.now(dt.timezone.utc)
logging.info(f" .. started: {section_time.strftime('%m/%d/%Y %H:%M:%S')}")

all_bridges_gdf = all_bridges_gdf_raw[['osmid', 'name', 'bridge_type', 'geometry']]
all_bridges_gdf = all_bridges_gdf_raw[['osmid', 'name', 'bridge_type', 'HUC', 'geometry']]
all_bridges_gdf.to_file(osm_bridge_file, driver="GPKG")

return
Expand Down
1 change: 0 additions & 1 deletion data/create_vrt_file.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,6 @@
import argparse
import logging
import os
import sys
from datetime import datetime

from osgeo import gdal
Expand Down
39 changes: 39 additions & 0 deletions docs/CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,45 @@ Updates Python packages to resolve dependency conflicts that were preventing `Do

<br/><br/>



## v4.5.13.4 - 2024-01-03 - [PR#1382](https://github.com/NOAA-OWP/inundation-mapping/pull/1382)

Cleans up Python files within `delineate_hydros_and_produce_HAND.sh` to improve performance, especially memory management, including removing unused imports, deleting object references when objects are no longer needed, and removing GDAL from the `fim_process_unit_wb.sh` step of FIM pipeline. Contributes to #1351 and #1376.

### Changes
- `data/create_vrt_file.py` and `tools/pixel_counter.py`: Removes unused import
- `src/`
- `accumulate_headwaters.py`, `add_crosswalk.py`, `adjust_thalweg_lateral.py`, `filter_catchments_and_add_attributes.py`, `heal_bridges_osm.py`, `make_rem.py`, `make_stages_and_catchlist.py`, `mitigate_branch_outlet_backpool.py`, `reachID_grid_to_vector_points.py`, `split_flows.py`, `unique_pixel_and_allocation.py`: Deletes objects no longer in use
- `delineate_hydros_and_produce_HAND.sh`, `run_by_branch.sh`, `run_unit_wb.sh` : Updates arguments
- `getRasterInfoNative.py`: Refactors in `rasterio` (removed `gdal`)
- `tools/evaluate_crosswalk.py`: Deletes objects no longer in use

<br/><br/>


## v4.5.13.3 - 2025-01-03 - [PR#1048](https://github.com/NOAA-OWP/inundation-mapping/pull/1048)

This script produces inundation depths and attempts to overcome the catchment boundary issue by interpolating water surface elevations between catchments. Water surface calculations require the hydroconditioned DEM (`dem_thalwegCond_{}.tif`) for computation, however, this file is not in the standard outputs from fim_pipeline.sh. Therefore, users may have to re-run fim_pipeline.sh with dem_thalwegCond_{}.tif removed from all deny lists.

### Additions

- `tools/interpolate_water_surface.py`: New post-inundation processing tool for extending depths beyond catchment limits. The `interpolate_wse()` contains the logic for computing the updated depth raster, but users can also call this module directly to perform inundation, similar to how `inundate_mosaic_wrapper.py` works, but with the new post-processing enhancement.

<br/><br/>


## v4.5.13.2 - 2025-01-03 - [PR#1360](https://github.com/NOAA-OWP/inundation-mapping/pull/1360)

Fixed missing osmid in osm_bridge_centroid.gpkg. Also, HUC column is added to outputs.

### Changes
- `data/bridges/pull_osm_bridges.py`
- `src/aggregate_by_huc.py`

<br/><br/>


## v4.5.13.1 - 2024-12-13 - [PR#1361](https://github.com/NOAA-OWP/inundation-mapping/pull/1361)

This PR was triggered by two dep-bot PR's. One for Tornado, one for aiohttp. Upon further research, these two exist only as dependencies for Jupyter and Jupyterlab which were very out of date. Upgrading Jupyter/JupyterLab took care of the other two.
Expand Down
11 changes: 8 additions & 3 deletions src/accumulate_headwaters.py
Original file line number Diff line number Diff line change
Expand Up @@ -38,9 +38,6 @@ def accumulate_flow(
data = src.read(1)
nodata = src.nodata
profile = src.profile
# transform = src.transform
# crs = src.crs
# latlon = crs.to_epsg() == 4326

# Convert the TauDEM flow direction raster to a pyflwdir flow direction array
temp = data.copy()
Expand All @@ -55,17 +52,23 @@ def accumulate_flow(
temp[data == 8] = 2
temp[data == nodata] = 247

del data

temp = temp.astype(np.uint8)

flw = pyflwdir.from_array(temp, ftype='d8')

del temp

# Read the flow direction raster
with rio.open(headwaters_filename) as src:
headwaters = src.read(1)
nodata = src.nodata

flowaccum = flw.accuflux(headwaters, nodata=nodata, direction='up')

del flw

stream = np.where(flowaccum > 0, flow_accumulation_threshold, 0)

# Write the flow accumulation raster
Expand All @@ -76,6 +79,8 @@ def accumulate_flow(
dst.write(flowaccum, 1)
dst2.write(stream, 1)

del flowaccum, stream


if __name__ == '__main__':
parser = argparse.ArgumentParser()
Expand Down
45 changes: 21 additions & 24 deletions src/add_crosswalk.py
Original file line number Diff line number Diff line change
Expand Up @@ -29,38 +29,21 @@ def add_crosswalk(
output_hydro_table_fileName,
input_huc_fileName,
input_nwmflows_fileName,
input_nwmcatras_fileName,
mannings_n,
input_nwmcat_fileName,
extent,
small_segments_filename,
min_catchment_area,
min_stream_length,
huc_id,
calibration_mode=False,
):
input_catchments = gpd.read_file(input_catchments_fileName, engine="pyogrio", use_arrow=True)
input_flows = gpd.read_file(input_flows_fileName, engine="pyogrio", use_arrow=True)
input_huc = gpd.read_file(input_huc_fileName, engine="pyogrio", use_arrow=True)
input_nwmcat = gpd.read_file(input_nwmcat_fileName, engine="pyogrio", use_arrow=True)
input_nwmflows = gpd.read_file(input_nwmflows_fileName, engine="pyogrio", use_arrow=True)
min_catchment_area = float(min_catchment_area) # 0.25#
min_stream_length = float(min_stream_length) # 0.5#

input_catchments = input_catchments.dissolve(by='HydroID').reset_index()

## crosswalk using stream segment midpoint method
input_nwmcat = gpd.read_file(input_nwmcat_fileName, mask=input_huc, engine="fiona")

# only reduce nwm catchments to mainstems if running mainstems
if extent == 'MS':
input_nwmcat = input_nwmcat.loc[input_nwmcat.mainstem == 1]

input_nwmcat = input_nwmcat.rename(columns={'ID': 'feature_id'})
if input_nwmcat.feature_id.dtype != 'int':
input_nwmcat.feature_id = input_nwmcat.feature_id.astype(int)
input_nwmcat = input_nwmcat.set_index('feature_id')

input_nwmflows = input_nwmflows.rename(columns={'ID': 'feature_id'})
if input_nwmflows.feature_id.dtype != 'int':
input_nwmflows.feature_id = input_nwmflows.feature_id.astype(int)
Expand Down Expand Up @@ -89,6 +72,8 @@ def add_crosswalk(
crosswalk = crosswalk.filter(items=['HydroID', 'feature_id', 'distance'])
crosswalk = crosswalk.merge(input_nwmflows[['order_']], on='feature_id')

del input_nwmflows

if crosswalk.empty:
print("No relevant streams within HUC boundaries.")
sys.exit(FIM_exit_codes.NO_VALID_CROSSWALKS.value)
Expand All @@ -97,6 +82,8 @@ def add_crosswalk(
input_catchments.HydroID = input_catchments.HydroID.astype(int)
output_catchments = input_catchments.merge(crosswalk, on='HydroID')

del input_catchments

if output_catchments.empty:
print("No valid catchments remain.")
sys.exit(FIM_exit_codes.NO_VALID_CROSSWALKS.value)
Expand All @@ -105,6 +92,8 @@ def add_crosswalk(
input_flows.HydroID = input_flows.HydroID.astype(int)
output_flows = input_flows.merge(crosswalk, on='HydroID')

del input_flows

# added for GMS. Consider adding filter_catchments_and_add_attributes.py to run_by_branch.sh
if 'areasqkm' not in output_catchments.columns:
output_catchments['areasqkm'] = output_catchments.geometry.area / (1000**2)
Expand Down Expand Up @@ -276,6 +265,9 @@ def add_crosswalk(
input_src_base['Bathymetry_source'] = pd.NA

output_src = input_src_base.drop(columns=['CatchId']).copy()

del input_src_base

if output_src.HydroID.dtype != 'int':
output_src.HydroID = output_src.HydroID.astype(int)

Expand Down Expand Up @@ -304,6 +296,9 @@ def add_crosswalk(
)
merged_output_src = merged_output_src[['HydroID', 'Stage', 'Discharge (m3s-1)_df2']]
output_src = pd.merge(output_src, merged_output_src, on=['HydroID', 'Stage'], how='left')

del merged_output_src

output_src['Discharge (m3s-1)'] = output_src['Discharge (m3s-1)_df2'].fillna(
output_src['Discharge (m3s-1)']
)
Expand All @@ -322,8 +317,12 @@ def add_crosswalk(
['Discharge (m3s-1)'],
] = src_stage[1]

del sml_segs

output_src = output_src.merge(crosswalk[['HydroID', 'feature_id']], on='HydroID')

del crosswalk

output_crosswalk = output_src[['HydroID', 'feature_id']]
output_crosswalk = output_crosswalk.drop_duplicates(ignore_index=True)

Expand Down Expand Up @@ -386,6 +385,8 @@ def add_crosswalk(
input_huc[FIM_ID] = input_huc[FIM_ID].astype(str)
output_hydro_table = output_hydro_table.merge(input_huc.loc[:, [FIM_ID, 'HUC8']], how='left', on=FIM_ID)

del input_huc

if output_flows.HydroID.dtype != 'str':
output_flows.HydroID = output_flows.HydroID.astype(str)
output_hydro_table = output_hydro_table.merge(
Expand Down Expand Up @@ -428,6 +429,8 @@ def add_crosswalk(
with open(output_src_json_fileName, 'w') as f:
json.dump(output_src_json, f, sort_keys=True, indent=2)

del output_catchments, output_flows, output_src, output_crosswalk, output_hydro_table, output_src_json


if __name__ == '__main__':
parser = argparse.ArgumentParser(
Expand All @@ -452,22 +455,16 @@ def add_crosswalk(
parser.add_argument("-t", "--output-hydro-table-fileName", help="Hydrotable", required=True)
parser.add_argument("-w", "--input-huc-fileName", help="HUC8 boundary", required=True)
parser.add_argument("-b", "--input-nwmflows-fileName", help="Subest NWM burnlines", required=True)
parser.add_argument("-y", "--input-nwmcatras-fileName", help="NWM catchment raster", required=False)
parser.add_argument(
"-m",
"--mannings-n",
help="Mannings n. Accepts single parameter set or list of parameter set in calibration mode. Currently input as csv.",
required=True,
)
parser.add_argument("-u", "--huc-id", help="HUC ID", required=False)
parser.add_argument("-z", "--input-nwmcat-fileName", help="NWM catchment polygon", required=True)
parser.add_argument("-p", "--extent", help="GMS only for now", default="GMS", required=False)
parser.add_argument("-u", "--huc-id", help="HUC ID", required=True)
parser.add_argument(
"-k", "--small-segments-filename", help="output list of short segments", required=True
)
parser.add_argument(
"-c", "--calibration-mode", help="Mannings calibration flag", required=False, action="store_true"
)
parser.add_argument("-e", "--min-catchment-area", help="Minimum catchment area", required=True)
parser.add_argument("-g", "--min-stream-length", help="Minimum stream length", required=True)

Expand Down
4 changes: 4 additions & 0 deletions src/adjust_thalweg_lateral.py
Original file line number Diff line number Diff line change
Expand Up @@ -98,6 +98,8 @@ def minimize_thalweg_elevation(dem_window, zone_min_dict, zone_window, thalweg_w
ndv,
)

del elevation_window, zone_window, cost_window

# --------------------------------------------------------------------------------------------- #

# Specify raster object metadata.
Expand All @@ -124,6 +126,8 @@ def minimize_thalweg_elevation(dem_window, zone_min_dict, zone_window, thalweg_w

dem_lateral_thalweg_adj_object.write(minimized_dem_window, window=window, indexes=1)

del dem_window, zone_window, thalweg_window, minimized_dem_window


if __name__ == '__main__':
# Parse arguments.
Expand Down
3 changes: 2 additions & 1 deletion src/aggregate_by_huc.py
Original file line number Diff line number Diff line change
Expand Up @@ -205,7 +205,7 @@ def aggregate_bridge_pnts(self, branch_path, branch_id):
if not os.path.isfile(bridge_filename):
return

bridge_pnts = gpd.read_file(bridge_filename)
bridge_pnts = gpd.read_file(bridge_filename, dtype=self.bridge_dtypes)
if bridge_pnts.empty:
return
hydrotable_filename = join(branch_path, f'hydroTable_{branch_id}.csv')
Expand Down Expand Up @@ -288,6 +288,7 @@ def agg_function(
(c > 1) & (bridge_pnts.feature_id != bridge_pnts.crossing_feature_id), 'is_backwater'
] = 1
# Write file
bridge_pnts = bridge_pnts.astype(self.bridge_dtypes, errors='ignore')
bridge_pnts.to_file(bridge_pnts_file, index=False, engine='fiona')

# print(f"agg_by_huc for huc id {huc_id} is done")
Expand Down
2 changes: 0 additions & 2 deletions src/delineate_hydros_and_produce_HAND.sh
Original file line number Diff line number Diff line change
Expand Up @@ -243,9 +243,7 @@ python3 $srcDir/add_crosswalk.py \
-w $tempHucDataDir/wbd8_clp.gpkg \
-b $b_arg \
-u $hucNumber \
-y $tempCurrentBranchDataDir/nwm_catchments_proj_subset.tif \
-m $manning_n \
-z $z_arg \
-k $tempCurrentBranchDataDir/small_segments_$current_branch_id.csv \
-e $min_catchment_area \
-g $min_stream_length
Expand Down
6 changes: 6 additions & 0 deletions src/filter_catchments_and_add_attributes.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,10 +28,14 @@ def filter_catchments_and_add_attributes(
# filter segments within huc boundary
select_flows = tuple(map(str, map(int, wbd[wbd.HUC8.str.contains(huc_code)][FIM_ID])))

del wbd

if input_flows.HydroID.dtype != 'str':
input_flows.HydroID = input_flows.HydroID.astype(str)
output_flows = input_flows[input_flows.HydroID.str.startswith(select_flows)].copy()

del input_flows

if output_flows.HydroID.dtype != 'int':
output_flows.HydroID = output_flows.HydroID.astype(int)

Expand Down Expand Up @@ -73,6 +77,8 @@ def filter_catchments_and_add_attributes(
print("There are no flowlines in the HUC after stream order filtering.")
sys.exit(FIM_exit_codes.NO_FLOWLINES_EXIST.value) # will send a 61 back

del input_catchments


if __name__ == '__main__':
# Parse arguments.
Expand Down
Loading

0 comments on commit c61bfa4

Please sign in to comment.