teslatte/src/products.rs

267 lines
8.7 KiB
Rust
Raw Normal View History

2022-07-21 21:00:45 +10:00
use crate::error::TeslatteError;
use crate::powerwall::PowerwallId;
use crate::vehicles::VehicleData;
use crate::{get, Api};
use derive_more::Display;
2022-07-21 21:00:45 +10:00
use serde::{Deserialize, Serialize};
use std::str::FromStr;
2022-07-21 21:00:45 +10:00
#[rustfmt::skip]
impl Api {
get!(products, Vec<Product>, "/products");
2022-07-23 15:08:53 +10:00
}
#[derive(Debug, Clone, Deserialize, Display)]
pub struct EnergySiteId(pub u64);
2022-07-23 15:08:53 +10:00
impl FromStr for EnergySiteId {
type Err = TeslatteError;
2022-07-23 15:08:53 +10:00
fn from_str(s: &str) -> Result<Self, Self::Err> {
Ok(EnergySiteId(s.parse().map_err(|_| {
TeslatteError::DecodeEnergySiteIdError(s.to_string())
})?))
2022-07-23 15:08:53 +10:00
}
2022-07-21 21:00:45 +10:00
}
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct GatewayId(String);
#[derive(Debug, Clone, Deserialize)]
#[serde(untagged)]
pub enum Product {
Vehicle(Box<VehicleData>),
Solar(Box<SolarData>),
Powerwall(Box<PowerwallData>),
2022-07-21 21:00:45 +10:00
}
/// This is assumed from https://tesla-api.timdorr.com/api-basics/products
#[derive(Debug, Clone, Deserialize)]
pub struct SolarData {
pub energy_site_id: EnergySiteId,
2022-07-25 10:00:55 +10:00
pub solar_type: String,
2022-07-21 21:00:45 +10:00
/// Should always be "solar".
pub resource_type: String,
pub id: String,
pub asset_site_id: String,
pub solar_power: i64,
pub sync_grid_alert_enabled: bool,
pub breaker_alert_enabled: bool,
pub components: Components,
}
#[derive(Debug, Clone, Deserialize)]
pub struct PowerwallData {
2022-07-25 10:00:55 +10:00
pub energy_site_id: EnergySiteId,
2022-07-21 21:00:45 +10:00
pub battery_type: String,
/// Should always be "battery".
pub resource_type: String,
pub site_name: String,
pub id: PowerwallId,
pub gateway_id: GatewayId,
pub asset_site_id: String,
pub energy_left: f64,
pub total_pack_energy: i64,
pub percentage_charged: f64,
pub backup_capable: bool,
pub battery_power: i64,
pub sync_grid_alert_enabled: bool,
pub breaker_alert_enabled: bool,
pub components: Components,
}
#[derive(Debug, Clone, Deserialize)]
pub struct Components {
pub battery: bool,
pub battery_type: Option<String>,
pub solar: bool,
pub solar_type: Option<String>,
pub grid: bool,
pub load_meter: bool,
pub market_type: String,
}
#[cfg(test)]
mod tests {
use super::*;
use crate::energy_sites::{CalendarHistoryValues, HistoryKind, HistoryPeriod};
use crate::Values;
use chrono::DateTime;
2022-07-21 21:00:45 +10:00
#[test]
fn energy_match_powerwall() {
let json = r#"
{
"energy_site_id": 1032748243,
"resource_type": "battery",
"site_name": "1 Railway Pde",
"id": "ABC2010-1234",
"gateway_id": "3287423824-QWE",
"asset_site_id": "123ecd-123ecd-12345-12345",
"energy_left": 4394.000000000001,
"total_pack_energy": 13494,
"percentage_charged": 32.562620423892106,
"battery_type": "ac_powerwall",
"backup_capable": true,
"battery_power": -280,
"sync_grid_alert_enabled": true,
"breaker_alert_enabled": false,
"components": {
"battery": true,
"battery_type": "ac_powerwall",
"solar": true,
"solar_type": "pv_panel",
"grid": true,
"load_meter": true,
"market_type": "residential"
}
}
"#;
if let Product::Powerwall(data) = serde_json::from_str(json).unwrap() {
2022-07-21 21:00:45 +10:00
assert_eq!(data.battery_type, "ac_powerwall");
2023-08-29 11:41:40 +10:00
assert!(data.backup_capable);
2022-07-21 21:00:45 +10:00
assert_eq!(data.battery_power, -280);
2023-08-29 11:41:40 +10:00
assert!(data.sync_grid_alert_enabled);
assert!(!data.breaker_alert_enabled);
assert!(data.components.battery);
2022-07-21 21:00:45 +10:00
assert_eq!(
data.components.battery_type,
Some("ac_powerwall".to_string())
);
2023-08-29 11:41:40 +10:00
assert!(data.components.solar);
2022-07-21 21:00:45 +10:00
assert_eq!(data.components.solar_type, Some("pv_panel".to_string()));
2023-08-29 11:41:40 +10:00
assert!(data.components.grid);
assert!(data.components.load_meter);
2022-07-21 21:00:45 +10:00
assert_eq!(data.components.market_type, "residential");
} else {
panic!("Expected PowerwallData");
}
}
#[test]
fn energy_match_vehicle() {
let json = r#"
{
"id": 1111193485934,
"user_id": 2222291283912,
"vehicle_id": 333331238921,
"vin": "T234567890123456789",
"display_name": "My Vehicle",
"option_codes": "ASDF,SDFG,DFGH",
"color": null,
"access_type": "OWNER",
"tokens": [
"asdf1234"
],
"state": "online",
"in_service": false,
"id_s": "932423",
"calendar_enabled": true,
"api_version": 42,
"backseat_token": null,
"backseat_token_updated_at": null,
"vehicle_config": {
"aux_park_lamps": "Eu",
"badge_version": 0,
"can_accept_navigation_requests": true,
"can_actuate_trunks": true,
"car_special_type": "base",
"car_type": "model3",
"charge_port_type": "CCS",
"dashcam_clip_save_supported": true,
"default_charge_to_max": false,
"driver_assist": "TeslaAP3",
"ece_restrictions": false,
"efficiency_package": "M32026",
"eu_vehicle": true,
"exterior_color": "MidnightSilver",
"exterior_trim": "Black",
"exterior_trim_override": "",
"has_air_suspension": false,
"has_ludicrous_mode": false,
"has_seat_cooling": false,
"headlamp_type": "Global",
"interior_trim_type": "Black2",
"key_version": 2,
"motorized_charge_port": true,
"paint_color_override": "255,200,253,0.9,0.3",
"performance_package": "Base",
"plg": true,
"pws": false,
"rear_drive_unit": "T15232Z",
"rear_seat_heaters": 1,
"rear_seat_type": 0,
"rhd": true,
"roof_color": "RoofColorGlass",
"seat_type": null,
"spoiler_type": "None",
"sun_roof_installed": null,
"supports_qr_pairing": false,
"third_row_seats": "None",
"timestamp": 1658390117642,
"trim_badging": "9",
"use_range_badging": true,
"utc_offset": 0,
"webcam_supported": false,
"wheel_type": "StilettoRefresh19"
},
"command_signing": "allowed"
}
"#;
let energy_site: Product = serde_json::from_str(json).unwrap();
if let Product::Vehicle(v) = energy_site {
2022-07-21 21:00:45 +10:00
assert_eq!(v.id.0, 1111193485934);
assert_eq!(v.user_id, 2222291283912);
assert_eq!(v.vehicle_id.0, 333331238921);
assert_eq!(v.vin, "T234567890123456789");
2023-08-29 12:32:19 +10:00
assert_eq!(v.display_name.unwrap(), "My Vehicle");
assert_eq!(v.option_codes.unwrap(), "ASDF,SDFG,DFGH");
2022-07-21 21:00:45 +10:00
assert_eq!(v.color, None);
assert_eq!(v.access_type, "OWNER");
assert_eq!(v.tokens, vec!["asdf1234"]);
assert_eq!(v.state, "online");
2023-08-29 11:41:40 +10:00
assert!(!v.in_service);
assert!(v.calendar_enabled);
2022-07-21 21:00:45 +10:00
assert_eq!(v.api_version, 42);
assert_eq!(v.backseat_token, None);
assert_eq!(v.backseat_token_updated_at, None);
assert_eq!(v.vehicle_config.unwrap().aux_park_lamps, Some("Eu".to_string()));
2022-07-21 21:00:45 +10:00
} else {
panic!("Wrong EnergySite");
}
}
2022-07-24 09:16:34 +10:00
#[test]
fn calendar_history_values() {
let v = CalendarHistoryValues {
site_id: EnergySiteId(123),
period: HistoryPeriod::Month,
kind: HistoryKind::Energy,
start_date: None,
end_date: None,
};
let url = v.format("https://base.com/e/{}/history");
assert_eq!(
url,
"https://base.com/e/123/history?period=month&kind=energy"
);
}
#[test]
fn calendar_history_values_dates() {
let v = CalendarHistoryValues {
site_id: EnergySiteId(123),
period: HistoryPeriod::Month,
kind: HistoryKind::Energy,
start_date: Some(DateTime::parse_from_rfc3339("2020-01-01T00:00:00Z").unwrap()),
end_date: Some(DateTime::parse_from_rfc3339("2020-01-31T23:59:59Z").unwrap()),
};
let url = v.format("https://base.com/e/{}/history");
assert_eq!(
url,
"https://base.com/e/123/history?period=month&kind=energy&start_date=2020-01-01T00:00:00Z&end_date=2020-01-31T23:59:59Z"
);
}
2022-07-21 21:00:45 +10:00
}