use chrono::{DateTime, Utc};
use serde::{Deserialize, Serialize};
use crate::enums::Unit;
use crate::traits::{Bo4eMeta, Bo4eObject};
#[derive(Debug, Clone, Default, PartialEq, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct LoadCurveData {
#[serde(flatten)]
pub meta: Bo4eMeta,
#[serde(skip_serializing_if = "Option::is_none")]
pub timestamp: Option<DateTime<Utc>>,
#[serde(skip_serializing_if = "Option::is_none")]
pub power_value: Option<f64>,
#[serde(skip_serializing_if = "Option::is_none")]
pub power_unit: Option<Unit>,
#[serde(skip_serializing_if = "Option::is_none")]
pub energy_value: Option<f64>,
#[serde(skip_serializing_if = "Option::is_none")]
pub energy_unit: Option<Unit>,
#[serde(skip_serializing_if = "Option::is_none")]
pub interval_minutes: Option<i32>,
#[serde(skip_serializing_if = "Option::is_none")]
pub obis_code: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub measurement_location_id: Option<String>,
}
impl Bo4eObject for LoadCurveData {
fn type_name_german() -> &'static str {
"Lastkurvendaten"
}
fn type_name_english() -> &'static str {
"LoadCurveData"
}
fn meta(&self) -> &Bo4eMeta {
&self.meta
}
fn meta_mut(&mut self) -> &mut Bo4eMeta {
&mut self.meta
}
}
#[cfg(test)]
mod tests {
use super::*;
use chrono::TimeZone;
#[test]
fn test_load_curve_data() {
let data = LoadCurveData {
timestamp: Some(Utc.with_ymd_and_hms(2024, 1, 15, 12, 15, 0).unwrap()),
power_value: Some(125.5),
power_unit: Some(Unit::Kilowatt),
interval_minutes: Some(15),
..Default::default()
};
let json = serde_json::to_string(&data).unwrap();
assert!(json.contains("125.5"));
assert!(json.contains("15"));
}
#[test]
fn test_with_energy() {
let data = LoadCurveData {
timestamp: Some(Utc::now()),
power_value: Some(100.0),
power_unit: Some(Unit::Kilowatt),
energy_value: Some(25.0),
energy_unit: Some(Unit::KilowattHour),
interval_minutes: Some(15),
..Default::default()
};
let json = serde_json::to_string(&data).unwrap();
assert!(json.contains("100"));
assert!(json.contains("25"));
}
#[test]
fn test_roundtrip() {
let data = LoadCurveData {
timestamp: Some(Utc::now()),
power_value: Some(999.99),
power_unit: Some(Unit::Kilowatt),
obis_code: Some("1-0:1.4.0".to_string()),
..Default::default()
};
let json = serde_json::to_string(&data).unwrap();
let parsed: LoadCurveData = serde_json::from_str(&json).unwrap();
assert_eq!(data.power_value, parsed.power_value);
assert_eq!(data.obis_code, parsed.obis_code);
}
#[test]
fn test_bo4e_object_impl() {
assert_eq!(LoadCurveData::type_name_german(), "Lastkurvendaten");
assert_eq!(LoadCurveData::type_name_english(), "LoadCurveData");
}
}