metar 0.9.3

A fast METAR parsing library in pure Rust
Documentation
use chumsky::prelude::*;

use crate::{traits::Parsable, Data};

use super::CloudType;

#[derive(PartialEq, Eq, Copy, Clone, Debug)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
/// Cloud cover
pub struct CloudLayer {
    /// Cloud density
    pub density: Data<CloudDensity>,
    /// Cloud type
    pub kind: Data<CloudType>,
    /// Cloud height in feet
    pub height: Data<u32>,
}

impl Parsable for CloudLayer {
    fn parser<'src>() -> impl Parser<'src, &'src str, Self, extra::Err<crate::MetarError<'src>>> {
        group((
            Data::parser_inline(3, CloudDensity::parser()),
            Data::parser_inline(
                3,
                text::digits(10)
                    .exactly(3)
                    .to_slice()
                    .map(|d: &str| d.parse().unwrap()),
            ),
            Data::parser_inline(3, CloudType::parser()),
        ))
        .map(|(density, height, kind)| CloudLayer {
            density,
            kind,
            height,
        })
    }
}

/// The density of the cloud cover
#[derive(PartialEq, Eq, Copy, Clone, Debug)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
pub enum CloudDensity {
    /// Few clouds (1/8)
    Few,
    /// Scattered cloud cover (3/8)
    Scattered,
    /// Broken cloud cover (5/8)
    Broken,
    /// Overcast cloud cover (7/8)
    Overcast,
}

impl Parsable for CloudDensity {
    fn parser<'src>() -> impl Parser<'src, &'src str, Self, extra::Err<crate::MetarError<'src>>> {
        choice((
            just("FEW").map(|_| CloudDensity::Few),
            just("SCT").map(|_| CloudDensity::Scattered),
            just("BKN").map(|_| CloudDensity::Broken),
            just("OVC").map(|_| CloudDensity::Overcast),
        ))
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn test_cloud_layers() {
        assert_eq!(
            CloudLayer::parse("BKN300CB").unwrap(),
            CloudLayer {
                density: Data::Known(CloudDensity::Broken),
                height: Data::Known(300),
                kind: Data::Known(CloudType::Cumulonimbus),
            }
        );
        assert_eq!(
            CloudLayer::parse("/////////").unwrap(),
            CloudLayer {
                density: Data::Unknown,
                height: Data::Unknown,
                kind: Data::Unknown,
            }
        );
    }
}