From 36820bbed975b42216996ca9bc3479658a83ce5f Mon Sep 17 00:00:00 2001 From: Pete Gadomski Date: Thu, 11 Apr 2024 13:28:09 -0600 Subject: [PATCH] feat: add more proj fields to gdal item creation --- stac/CHANGELOG.md | 2 +- stac/src/extensions/projection.rs | 2 +- stac/src/gdal.rs | 48 +++++++++++++++++++++++++++++-- 3 files changed, 48 insertions(+), 4 deletions(-) diff --git a/stac/CHANGELOG.md b/stac/CHANGELOG.md index 75bb7481..bbeb222d 100644 --- a/stac/CHANGELOG.md +++ b/stac/CHANGELOG.md @@ -10,7 +10,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), - The projection and raster extensions, the `Extension` trait, and the `Fields` trait ([#234](https://github.com/stac-utils/stac-rs/pull/234)) - `stac::item::Builder` ([#237](https://github.com/stac-utils/stac-rs/pull/237)) -- The `gdal` feature ([#232](https://github.com/stac-utils/stac-rs/pull/232)) +- The `gdal` feature ([#232](https://github.com/stac-utils/stac-rs/pull/232), [#240](https://github.com/stac-utils/stac-rs/pull/240)) - `Bounds` ([#232](https://github.com/stac-utils/stac-rs/pull/232)) ### Changed diff --git a/stac/src/extensions/projection.rs b/stac/src/extensions/projection.rs index b21c945e..d01eb45a 100644 --- a/stac/src/extensions/projection.rs +++ b/stac/src/extensions/projection.rs @@ -37,7 +37,7 @@ pub struct Projection { /// Number of pixels in Y and X directions for the default grid #[serde(skip_serializing_if = "Option::is_none")] - pub shape: Option>, + pub shape: Option>, /// The affine transformation coefficients for the default grid #[serde(skip_serializing_if = "Option::is_none")] diff --git a/stac/src/gdal.rs b/stac/src/gdal.rs index 44b8883a..0b783977 100644 --- a/stac/src/gdal.rs +++ b/stac/src/gdal.rs @@ -2,12 +2,16 @@ use crate::{ extensions::{ + projection::Centroid, raster::{Band, Raster, Statistics}, Projection, }, Asset, Bounds, Extensions, Geometry, Item, Result, }; -use gdal::Dataset; +use gdal::{ + spatial_ref::{CoordTransform, SpatialRef}, + Dataset, +}; /// Update an [Item] using GDAL. /// @@ -82,7 +86,17 @@ fn update_asset( let dataset = Dataset::open(&asset.href)?; let mut spatial_resolution = None; let mut projection = Projection::default(); + let (width, height) = dataset.raster_size(); + projection.shape = Some(vec![height, width]); if let Ok(geo_transform) = dataset.geo_transform() { + projection.transform = Some(vec![ + geo_transform[1], + geo_transform[2], + geo_transform[0], + geo_transform[4], + geo_transform[5], + geo_transform[3], + ]); spatial_resolution = Some((geo_transform[1].abs() + geo_transform[5].abs()) / 2.0); let (width, height) = dataset.raster_size(); let width = width as f64; @@ -117,6 +131,17 @@ fn update_asset( .ok() .and_then(|s| serde_json::from_str(&s).ok()); } + if let Some(bbox) = projection.bbox.as_ref() { + let mut x = [(bbox[0] + bbox[2]) / 2.]; + let mut y = [(bbox[1] + bbox[3]) / 2.]; + let coord_transform = CoordTransform::new(&spatial_ref, &SpatialRef::from_epsg(4326)?)?; + coord_transform.transform_coords(&mut x, &mut y, &mut [])?; + let round = |n: f64| (n * 10_000_000.).round() / 10_000_000.; + projection.centroid = Some(Centroid { + lat: round(x[0]), + lon: round(y[0]), + }); + } } let count = dataset.raster_count(); let mut bands = Vec::with_capacity(count.try_into()?); @@ -143,7 +168,7 @@ fn update_asset( #[cfg(test)] mod tests { use crate::{ - extensions::{raster::DataType, Projection, Raster}, + extensions::{projection::Centroid, raster::DataType, Projection, Raster}, item::Builder, Extensions, }; @@ -202,5 +227,24 @@ mod tests { projection.bbox.unwrap(), vec![373185.0, 8019284.949381611, 639014.9492102272, 8286015.0] ); + assert_eq!(projection.shape.unwrap(), vec![2667, 2658]); + assert_eq!( + projection.transform.unwrap(), + vec![ + 100.01126757344893, + 0.0, + 373185.0, + 0.0, + -100.01126757344893, + 8286015.0, + ] + ); + assert_eq!( + projection.centroid.unwrap(), + Centroid { + lon: -56.8079473, + lat: 73.4675736, + } + ) } }