mappers_warp 0.1.0

Very simplistic tool for reprojecting maps, based on GdalWarp
Documentation
use std::fmt::Debug;

use mappers::Projection;
use ndarray::{Array2, s};
use thiserror::Error;

#[derive(Error, Debug)]
pub enum WarperError {
    #[error("Invalid raster dimensions")]
    InvalidRasterDimensions,

    #[error("Ndarray error {0}")]
    NdarrayError(#[from] ndarray::ShapeError),

    #[error("Projection error {0}")]
    ProjectionError(#[from] mappers::ProjectionError),

    #[error("Source raster must fully wrap")]
    SourceRasterTooSmall,

    #[error("Could not correctly convert coordinates")]
    ConversionError,

    #[error("Warping produced non-finite value")]
    WarpingError,

    #[error("Source raster cannot contain nan values")]
    NanError,
}

#[cfg(feature = "io")]
#[derive(Error, Debug)]
pub enum WarperIOError {
    #[error("IO error {0}")]
    IoError(#[from] std::io::Error),

    #[error("Rkyv error {0}")]
    BincodeDecodeError(#[from] rkyv::rancor::Error),

    #[error("Ndarray error {0}")]
    NdarrayError(#[from] ndarray::ShapeError),
}

pub trait XYPair: Debug + Clone + Copy + PartialEq + PartialOrd {}
impl XYPair for GenericXYPair {}
impl XYPair for SourceXYPair {}
impl XYPair for TargetXYPair {}

impl From<GenericXYPair> for SourceXYPair {
    fn from(v: GenericXYPair) -> Self {
        Self { x: v.x, y: v.y }
    }
}

impl From<GenericXYPair> for TargetXYPair {
    fn from(v: GenericXYPair) -> Self {
        Self { x: v.x, y: v.y }
    }
}

#[derive(Debug, Clone, Copy, PartialEq, PartialOrd)]
pub struct GenericXYPair {
    pub x: f64,
    pub y: f64,
}

#[derive(Debug, Clone, Copy, PartialEq, PartialOrd)]
pub struct SourceXYPair {
    pub x: f64,
    pub y: f64,
}

#[derive(Debug, Clone, Copy, PartialEq, PartialOrd)]
pub struct TargetXYPair {
    pub x: f64,
    pub y: f64,
}

/// Floating indexes in the source raster
#[derive(Debug, Clone, Copy, PartialEq, PartialOrd)]
pub struct IXJYPair {
    pub ix: f64,
    pub jy: f64,
}

#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd)]
pub struct IJPair {
    pub i: u32,
    pub j: u32,
}

#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct MinMaxPair<T> {
    pub min: T,
    pub max: T,
}

/// `RasterBounds` follows the meteorological convention on defining raster pixels
/// by midpoints of the grid cells.
///
/// So `min`, `max` here refer to the midpoints of corner
/// raster cell and `spacing` indicates distance between those midpoints
#[derive(Debug, Clone, Copy, PartialEq, PartialOrd)]
pub struct RasterBoundsDefinition<P: Projection> {
    min: GenericXYPair,
    max: GenericXYPair,
    spacing: GenericXYPair,
    shape: IJPair,
    proj: P,
}

impl<P: Projection> RasterBoundsDefinition<P> {
    /// Similarly to the struct: `x_bounds`, `y_bounds` refer to the midpoints of corner
    /// raster cell and `dx`, `dy` indicate distance between those midpoints
    pub fn new(
        x_bounds: (f64, f64),
        y_bounds: (f64, f64),
        dx: f64,
        dy: f64,
        proj: P,
    ) -> Result<Self, WarperError> {
        let (min_x, max_x) = x_bounds;
        let (min_y, max_y) = y_bounds;

        if min_x >= max_x || min_y >= max_y {
            return Err(WarperError::InvalidRasterDimensions);
        }

        let nx = (max_x - min_x) / dx;
        let ny = (max_y - min_y) / dy;

        if nx.fract() != 0.0 || ny.fract() != 0.0 {
            return Err(WarperError::InvalidRasterDimensions);
        }

        let nx = nx as u32 + 1;
        let ny = ny as u32 + 1;

        Ok(Self {
            min: GenericXYPair { x: min_x, y: min_y },
            max: GenericXYPair { x: max_x, y: max_y },
            spacing: GenericXYPair { x: dx, y: dy },
            shape: IJPair { i: nx, j: ny },
            proj,
        })
    }
}

#[derive(Debug, Clone, Copy, PartialEq, PartialOrd)]
pub struct RasterBounds<P: Projection, T: XYPair> {
    pub(crate) min: T,
    pub(crate) max: T,
    pub(crate) spacing: GenericXYPair,
    pub(crate) shape: IJPair,
    pub(crate) proj: P,
}

impl<P: Projection> RasterBounds<P, GenericXYPair> {
    pub(crate) fn cast_xy_pairs<T: From<GenericXYPair> + XYPair>(&self) -> RasterBounds<P, T> {
        RasterBounds {
            min: T::from(self.min),
            max: T::from(self.max),
            spacing: self.spacing,
            shape: self.shape,
            proj: self.proj,
        }
    }
}

impl<P: Projection> From<RasterBoundsDefinition<P>> for RasterBounds<P, GenericXYPair> {
    fn from(def: RasterBoundsDefinition<P>) -> Self {
        Self {
            min: def.min,
            max: def.max,
            spacing: def.spacing,
            shape: def.shape,
            proj: def.proj,
        }
    }
}

impl<P: Projection> From<&RasterBoundsDefinition<P>> for RasterBounds<P, GenericXYPair> {
    fn from(def: &RasterBoundsDefinition<P>) -> Self {
        Self {
            min: def.min,
            max: def.max,
            spacing: def.spacing,
            shape: def.shape,
            proj: def.proj,
        }
    }
}

/// As warper requires the source raster to fully wrap the target raster extent (plus some margin),
/// this function can be used to meet that requirement by adding some padding.
#[must_use]
pub fn raster_constant_pad(raster: &Array2<f64>, padding: usize, value: f64) -> Array2<f64> {
    let (ny, nx) = raster.dim();
    let (ny, nx) = (ny + (2 * padding), nx + (2 * padding));

    let mut padded = Array2::from_elem((ny, nx), value);
    let mut data_region = padded.slice_mut(s![padding..ny - padding, padding..nx - padding]);
    data_region.assign(raster);

    padded.to_owned()
}

#[cfg(test)]
mod tests {
    use ndarray::arr2;

    use crate::raster_constant_pad;

    #[test]
    fn pad() {
        let inner = arr2(&[[1., 2., 3.], [2., 3., 4.], [3., 4., 5.]]);
        let target = arr2(&[
            [0., 0., 0., 0., 0.],
            [0., 1., 2., 3., 0.],
            [0., 2., 3., 4., 0.],
            [0., 3., 4., 5., 0.],
            [0., 0., 0., 0., 0.],
        ]);

        let padded = raster_constant_pad(&inner, 1, 0.);
        assert_eq!(padded, target);
    }
}