use serde::{Deserialize, Serialize};
use crate::com::Address;
use crate::enums::{CustomerType, Division, EnergyDirection};
use crate::traits::{Bo4eMeta, Bo4eObject};
#[derive(Debug, Clone, Default, PartialEq, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct MarketLocation {
#[serde(flatten)]
pub meta: Bo4eMeta,
#[serde(skip_serializing_if = "Option::is_none", alias = "marktlokationsId")]
pub market_location_id: Option<String>,
#[serde(skip_serializing_if = "Option::is_none", alias = "sparte")]
pub division: Option<Division>,
#[serde(skip_serializing_if = "Option::is_none", alias = "energierichtung")]
pub energy_direction: Option<EnergyDirection>,
#[serde(skip_serializing_if = "Option::is_none", alias = "kundentyp")]
pub customer_type: Option<CustomerType>,
#[serde(skip_serializing_if = "Option::is_none", alias = "adresse")]
pub address: Option<Address>,
#[serde(skip_serializing_if = "Option::is_none", alias = "lieferbeginn")]
pub supply_start: Option<chrono::DateTime<chrono::Utc>>,
#[serde(skip_serializing_if = "Option::is_none", alias = "lieferende")]
pub supply_end: Option<chrono::DateTime<chrono::Utc>>,
#[serde(
skip_serializing_if = "Option::is_none",
alias = "jahresverbrauchsprognose"
)]
pub annual_consumption: Option<f64>,
#[serde(
skip_serializing_if = "Option::is_none",
alias = "netzbetreiberCodenummer"
)]
pub network_operator_code: Option<String>,
#[serde(
skip_serializing_if = "Option::is_none",
alias = "grundversorgerCodenummer"
)]
pub basic_supplier_code: Option<String>,
#[serde(
skip_serializing_if = "Option::is_none",
alias = "messstellenbetreiberCodenummer"
)]
pub metering_operator_code: Option<String>,
#[serde(
skip_serializing_if = "Option::is_none",
alias = "uebertragungsnetzbetreiberCodenummer"
)]
pub transmission_operator_code: Option<String>,
#[serde(skip_serializing_if = "Option::is_none", alias = "netzebene")]
pub grid_level: Option<String>,
#[serde(skip_serializing_if = "Option::is_none", alias = "netzgebiet")]
pub network_area: Option<String>,
#[serde(skip_serializing_if = "Option::is_none", alias = "bilanzierungsgebiet")]
pub balancing_area: Option<String>,
#[serde(
default,
skip_serializing_if = "Vec::is_empty",
alias = "messlokationsIds"
)]
pub metering_location_ids: Vec<String>,
#[serde(
skip_serializing_if = "Option::is_none",
alias = "istSteuerbareRessource"
)]
pub is_controllable_resource: Option<bool>,
}
impl Bo4eObject for MarketLocation {
fn type_name_german() -> &'static str {
"Marktlokation"
}
fn type_name_english() -> &'static str {
"MarketLocation"
}
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_malo_id_format() {
let malo = MarketLocation {
market_location_id: Some("12345678901".to_string()),
division: Some(Division::Electricity),
energy_direction: Some(EnergyDirection::FeedOut),
..Default::default()
};
assert_eq!(malo.market_location_id.as_ref().unwrap().len(), 11);
}
#[test]
fn test_serialize() {
let malo = MarketLocation {
meta: Bo4eMeta::with_type("Marktlokation"),
market_location_id: Some("12345678901".to_string()),
..Default::default()
};
let json = serde_json::to_string(&malo).unwrap();
assert!(json.contains(r#""marketLocationId":"12345678901""#));
}
#[test]
fn test_roundtrip() {
let malo = MarketLocation {
meta: Bo4eMeta::with_type("Marktlokation"),
market_location_id: Some("12345678901".to_string()),
division: Some(Division::Electricity),
energy_direction: Some(EnergyDirection::FeedOut),
annual_consumption: Some(3500.0),
metering_location_ids: vec!["DE00012345".to_string()],
..Default::default()
};
let json = serde_json::to_string(&malo).unwrap();
let parsed: MarketLocation = serde_json::from_str(&json).unwrap();
assert_eq!(malo, parsed);
}
#[test]
fn test_bo4e_object_impl() {
assert_eq!(MarketLocation::type_name_german(), "Marktlokation");
assert_eq!(MarketLocation::type_name_english(), "MarketLocation");
}
}