use crate::errors::ValidationError;
use crate::traits::ValueObject;
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
pub enum TemperatureUnit {
Celsius,
Fahrenheit,
Kelvin,
}
impl std::fmt::Display for TemperatureUnit {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
TemperatureUnit::Celsius => write!(f, "°C"),
TemperatureUnit::Fahrenheit => write!(f, "°F"),
TemperatureUnit::Kelvin => write!(f, "K"),
}
}
}
#[derive(Debug, Clone, PartialEq)]
pub struct TemperatureInput {
pub value: f64,
pub unit: TemperatureUnit,
}
#[derive(Debug, Clone, PartialEq)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
pub struct Temperature {
value: f64,
unit: TemperatureUnit,
#[cfg_attr(feature = "serde", serde(skip))]
canonical: String,
}
impl ValueObject for Temperature {
type Input = TemperatureInput;
type Output = str;
type Error = ValidationError;
fn new(input: Self::Input) -> Result<Self, Self::Error> {
if !input.value.is_finite() {
return Err(ValidationError::invalid(
"Temperature",
&input.value.to_string(),
));
}
let min = match input.unit {
TemperatureUnit::Kelvin => 0.0,
TemperatureUnit::Celsius => -273.15,
TemperatureUnit::Fahrenheit => -459.67,
};
if input.value < min {
return Err(ValidationError::invalid(
"Temperature",
&input.value.to_string(),
));
}
let canonical = format!("{} {}", input.value, input.unit);
Ok(Self {
value: input.value,
unit: input.unit,
canonical,
})
}
fn value(&self) -> &Self::Output {
&self.canonical
}
fn into_inner(self) -> Self::Input {
TemperatureInput {
value: self.value,
unit: self.unit,
}
}
}
impl Temperature {
pub fn amount(&self) -> f64 {
self.value
}
pub fn unit(&self) -> &TemperatureUnit {
&self.unit
}
}
impl std::fmt::Display for Temperature {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "{}", self.canonical)
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn accepts_celsius() {
let t = Temperature::new(TemperatureInput {
value: 100.0,
unit: TemperatureUnit::Celsius,
})
.unwrap();
assert_eq!(t.value(), "100 °C");
}
#[test]
fn accepts_absolute_zero_kelvin() {
assert!(
Temperature::new(TemperatureInput {
value: 0.0,
unit: TemperatureUnit::Kelvin
})
.is_ok()
);
}
#[test]
fn accepts_absolute_zero_celsius() {
assert!(
Temperature::new(TemperatureInput {
value: -273.15,
unit: TemperatureUnit::Celsius
})
.is_ok()
);
}
#[test]
fn rejects_below_absolute_zero_kelvin() {
assert!(
Temperature::new(TemperatureInput {
value: -0.01,
unit: TemperatureUnit::Kelvin
})
.is_err()
);
}
#[test]
fn rejects_below_absolute_zero_celsius() {
assert!(
Temperature::new(TemperatureInput {
value: -273.16,
unit: TemperatureUnit::Celsius
})
.is_err()
);
}
#[test]
fn rejects_below_absolute_zero_fahrenheit() {
assert!(
Temperature::new(TemperatureInput {
value: -459.68,
unit: TemperatureUnit::Fahrenheit
})
.is_err()
);
}
#[test]
fn rejects_nan() {
assert!(
Temperature::new(TemperatureInput {
value: f64::NAN,
unit: TemperatureUnit::Celsius
})
.is_err()
);
}
}