Skip to content

Commit

Permalink
include reprojection for multiple files in hugonnet_to_gdir (#1698)
Browse files Browse the repository at this point in the history
* include reprojection of multiple files in hugonnet_to_gdir

* added to whats-new.rst
  • Loading branch information
pat-schmitt authored Mar 20, 2024
1 parent c45194b commit 8c64afa
Show file tree
Hide file tree
Showing 2 changed files with 65 additions and 4 deletions.
7 changes: 7 additions & 0 deletions docs/whats-new.rst
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,13 @@ Bug fixes
``tasks.gridded_data_var_to_geotiff`` the resulting coordinates where
shifted half a pixel, this is now fixed (:pull:`1682`).
By `Patrick Schmitt <https://github.com/pat-schmitt>`_
- When adding the Hugonnet et al. (2021) dh/dt maps to the glacier directory
and multiple files are needed, previously there was no check to ensure they
all used the same coordinate reference system (CRS). This caused errors
during the file merging process. The bug fix now checks if the files have
different CRS. If they do, all files are converted to a single, common CRS
to make sure they can be merged without issues (:pull:`1698`).
By `Patrick Schmitt <https://github.com/pat-schmitt>`_


v1.6.1 (August 27, 2023)
Expand Down
62 changes: 58 additions & 4 deletions oggm/shop/hugonnet_maps.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@

try:
import rasterio
from rasterio.warp import reproject, Resampling
from rasterio.warp import reproject, Resampling, calculate_default_transform
from rasterio import MemoryFile
try:
# rasterio V > 1.0
Expand Down Expand Up @@ -90,6 +90,7 @@ def hugonnet_to_gdir(gdir, add_error=False):
# A glacier area can cover more than one tile:
if len(flist) == 1:
dem_dss = [rasterio.open(flist[0])] # if one tile, just open it
file_crs = dem_dss[0].crs
dem_data = rasterio.band(dem_dss[0], 1)
if Version(rasterio.__version__) >= Version('1.0'):
src_transform = dem_dss[0].transform
Expand All @@ -98,8 +99,61 @@ def hugonnet_to_gdir(gdir, add_error=False):
nodata = dem_dss[0].meta.get('nodata', None)
else:
dem_dss = [rasterio.open(s) for s in flist] # list of rasters
nodata = dem_dss[0].meta.get('nodata', None)
dem_data, src_transform = merge_tool(dem_dss, nodata=nodata) # merge

# make sure all files have the same crs and reproject if needed;
# defining the target crs to the one most commonly used, minimizing
# the number of files for reprojection
crs_list = np.array([dem_ds.crs.to_string() for dem_ds in dem_dss])
unique_crs, crs_counts = np.unique(crs_list, return_counts=True)
file_crs = rasterio.crs.CRS.from_string(
unique_crs[np.argmax(crs_counts)])

if len(unique_crs) != 1:
# more than one crs, we need to do reprojection
memory_files = []
for i, src in enumerate(dem_dss):
if file_crs != src.crs:
transform, width, height = calculate_default_transform(
src.crs, file_crs, src.width, src.height, *src.bounds)
kwargs = src.meta.copy()
kwargs.update({
'crs': file_crs,
'transform': transform,
'width': width,
'height': height
})

reprojected_array = np.empty(shape=(src.count, height, width),
dtype=src.dtypes[0])
# just for completeness; even the data only has one band
for band in range(1, src.count + 1):
reproject(source=rasterio.band(src, band),
destination=reprojected_array[band - 1],
src_transform=src.transform,
src_crs=src.crs,
dst_transform=transform,
dst_crs=file_crs,
resampling=Resampling.nearest)

memfile = MemoryFile()
with memfile.open(**kwargs) as mem_dst:
mem_dst.write(reprojected_array)
memory_files.append(memfile)
else:
memfile = MemoryFile()
with memfile.open(**src.meta) as mem_src:
mem_src.write(src.read())
memory_files.append(memfile)

with rasterio.Env():
datasets_to_merge = [memfile.open() for memfile in memory_files]
nodata = datasets_to_merge[0].meta.get('nodata', None)
dem_data, src_transform = merge_tool(datasets_to_merge,
nodata=nodata)
else:
# only one single crs occurring, no reprojection needed
nodata = dem_dss[0].meta.get('nodata', None)
dem_data, src_transform = merge_tool(dem_dss, nodata=nodata)

# Set up profile for writing output
with rasterio.open(gdir.get_filepath('dem')) as dem_ds:
Expand All @@ -120,7 +174,7 @@ def hugonnet_to_gdir(gdir, add_error=False):
reproject(
# Source parameters
source=dem_data,
src_crs=dem_dss[0].crs,
src_crs=file_crs,
src_transform=src_transform,
src_nodata=nodata,
# Destination parameters
Expand Down

0 comments on commit 8c64afa

Please sign in to comment.