use serde::{Deserialize, Serialize};
use crate::com::{Address, Hardware, MeterRegister};
use crate::enums::{Division, MeterSize, MeterType};
use crate::traits::{Bo4eMeta, Bo4eObject};
#[derive(Debug, Clone, Default, PartialEq, Serialize, Deserialize)]
#[cfg_attr(feature = "json-schema", derive(schemars::JsonSchema))]
#[cfg_attr(feature = "json-schema", schemars(rename = "Zaehler"))]
#[serde(rename_all = "camelCase")]
pub struct Meter {
#[serde(flatten)]
pub meta: Bo4eMeta,
#[serde(skip_serializing_if = "Option::is_none", alias = "zaehlernummer")]
#[cfg_attr(feature = "json-schema", schemars(rename = "zaehlernummer"))]
pub meter_number: Option<String>,
#[serde(skip_serializing_if = "Option::is_none", alias = "sparte")]
#[cfg_attr(feature = "json-schema", schemars(rename = "sparte"))]
pub division: Option<Division>,
#[serde(skip_serializing_if = "Option::is_none", alias = "zaehlertyp")]
#[cfg_attr(feature = "json-schema", schemars(rename = "zaehlertyp"))]
pub meter_type: Option<MeterType>,
#[serde(skip_serializing_if = "Option::is_none", alias = "zaehlergroesse")]
#[cfg_attr(feature = "json-schema", schemars(rename = "zaehlergroesse"))]
pub meter_size: Option<MeterSize>,
#[serde(skip_serializing_if = "Option::is_none", alias = "standort")]
#[cfg_attr(feature = "json-schema", schemars(rename = "standort"))]
pub location: Option<Address>,
#[serde(default, skip_serializing_if = "Vec::is_empty", alias = "zaehlwerke")]
#[cfg_attr(feature = "json-schema", schemars(rename = "zaehlwerke"))]
pub registers: Vec<MeterRegister>,
#[serde(
default,
skip_serializing_if = "Vec::is_empty",
alias = "geraeteeigenschaften"
)]
#[cfg_attr(feature = "json-schema", schemars(rename = "geraeteeigenschaften"))]
pub hardware: Vec<Hardware>,
#[serde(skip_serializing_if = "Option::is_none", alias = "marktlokationsId")]
#[cfg_attr(feature = "json-schema", schemars(rename = "marktlokationsId"))]
pub market_location_id: Option<String>,
#[serde(skip_serializing_if = "Option::is_none", alias = "messlokationsId")]
#[cfg_attr(feature = "json-schema", schemars(rename = "messlokationsId"))]
pub metering_location_id: Option<String>,
#[serde(
skip_serializing_if = "Option::is_none",
alias = "eigentumsverhaeltnis"
)]
#[cfg_attr(feature = "json-schema", schemars(rename = "eigentumsverhaeltnis"))]
pub ownership: Option<String>,
#[serde(skip_serializing_if = "Option::is_none", alias = "hersteller")]
#[cfg_attr(feature = "json-schema", schemars(rename = "hersteller"))]
pub manufacturer: Option<String>,
#[serde(skip_serializing_if = "Option::is_none", alias = "herstellungsjahr")]
#[cfg_attr(feature = "json-schema", schemars(rename = "herstellungsjahr"))]
pub manufacturing_year: Option<i32>,
#[serde(skip_serializing_if = "Option::is_none", alias = "einbaudatum")]
#[cfg_attr(feature = "json-schema", schemars(rename = "einbaudatum"))]
pub installation_date: Option<chrono::DateTime<chrono::Utc>>,
#[serde(skip_serializing_if = "Option::is_none", alias = "ausbaudatum")]
#[cfg_attr(feature = "json-schema", schemars(rename = "ausbaudatum"))]
pub removal_date: Option<chrono::DateTime<chrono::Utc>>,
#[serde(skip_serializing_if = "Option::is_none", alias = "eichdatum")]
#[cfg_attr(feature = "json-schema", schemars(rename = "eichdatum"))]
pub calibration_date: Option<chrono::DateTime<chrono::Utc>>,
#[serde(skip_serializing_if = "Option::is_none", alias = "eichablaufdatum")]
#[cfg_attr(feature = "json-schema", schemars(rename = "eichablaufdatum"))]
pub calibration_expiry_date: Option<chrono::DateTime<chrono::Utc>>,
}
impl Bo4eObject for Meter {
fn type_name_german() -> &'static str {
"Zaehler"
}
fn type_name_english() -> &'static str {
"Meter"
}
fn meta(&self) -> &Bo4eMeta {
&self.meta
}
fn meta_mut(&mut self) -> &mut Bo4eMeta {
&mut self.meta
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_meter_creation() {
let meter = Meter {
meter_number: Some("1EMH0012345678".to_string()),
division: Some(Division::Electricity),
..Default::default()
};
assert_eq!(meter.meter_number, Some("1EMH0012345678".to_string()));
assert_eq!(meter.division, Some(Division::Electricity));
}
#[test]
fn test_meter_with_registers() {
let register = MeterRegister {
obis_code: Some("1-0:1.8.0".to_string()),
..Default::default()
};
let meter = Meter {
meter_number: Some("TEST123".to_string()),
registers: vec![register],
..Default::default()
};
assert_eq!(meter.registers.len(), 1);
}
#[test]
fn test_serialize() {
let meter = Meter {
meta: Bo4eMeta::with_type("Zaehler"),
meter_number: Some("1EMH0012345678".to_string()),
division: Some(Division::Electricity),
..Default::default()
};
let json = serde_json::to_string(&meter).unwrap();
assert!(json.contains(r#""_typ":"Zaehler""#));
assert!(json.contains(r#""meterNumber":"1EMH0012345678""#));
}
#[test]
fn test_roundtrip() {
let meter = Meter {
meta: Bo4eMeta::with_type("Zaehler"),
meter_number: Some("TEST123".to_string()),
division: Some(Division::Electricity),
meter_type: Some(MeterType::ModernMeasuringDevice),
meter_size: Some(MeterSize::G4),
manufacturer: Some("Acme Corp".to_string()),
manufacturing_year: Some(2023),
..Default::default()
};
let json = serde_json::to_string(&meter).unwrap();
let parsed: Meter = serde_json::from_str(&json).unwrap();
assert_eq!(meter, parsed);
}
#[test]
fn test_bo4e_object_impl() {
assert_eq!(Meter::type_name_german(), "Zaehler");
assert_eq!(Meter::type_name_english(), "Meter");
}
}