esox 0.1.4

Library for NISECI and HFBI calc
Documentation
// SPDX-License-Identifier: GPL-3.0-only
/*
    Copyright (C) 2024-2026 jgabaut, gioninjo

    This program is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, version 3 of the License.

    This program is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with this program.  If not, see <https://www.gnu.org/licenses/>.
*/
#[cfg(feature = "experimental")]
use serde::{de, Deserialize, Deserializer};
use std::fmt;
use std::ops::Deref;

#[derive(Debug)]
pub enum PositiveF32Error {
    NotFinite,
    NotPositive,
}

impl fmt::Display for PositiveF32Error {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        match self {
            PositiveF32Error::NotFinite => write!(f, "value is not finite"),
            PositiveF32Error::NotPositive => write!(f, "value is not positive"),
        }
    }
}

impl std::error::Error for PositiveF32Error {}

#[derive(Debug, Clone, Copy)]
#[repr(transparent)]
pub struct PositiveF32(f32);

impl PositiveF32 {
    pub fn new(value: f32) -> Result<Self, PositiveF32Error> {
        if !value.is_finite() {
            return Err(PositiveF32Error::NotFinite);
        }
        if value <= 0.0 {
            return Err(PositiveF32Error::NotPositive);
        }
        Ok(Self(value))
    }
}

impl Deref for PositiveF32 {
    type Target = f32;
    fn deref(&self) -> &f32 {
        &self.0
    }
}

impl From<PositiveF32> for f32 {
    fn from(v: PositiveF32) -> Self {
        v.0
    }
}

#[cfg(feature = "experimental")]
pub(crate) fn deserialize_positive_f32<'de, D>(deserializer: D) -> Result<f32, D::Error>
where
    D: Deserializer<'de>,
{
    let v = f32::deserialize(deserializer)?;

    if v.is_finite() && v > 0.0 {
        Ok(v)
    } else {
        Err(de::Error::custom("expected a positive, finite f32"))
    }
}