Torrie Fischer 6de1704da2 wip-3
2024-12-19 14:48:14 +01:00

183 lines
6.6 KiB
Rust

use core::str;
use std::{borrow::BorrowMut, collections::{HashMap, LinkedList}, thread::JoinHandle};
use esp_idf_svc::mqtt::client::{EspMqttClient, EventPayload, MqttClientConfiguration};
use log::Log;
use serde_json::{json, Value};
use crate::{events::{Event, EventBus, System as SystemNS}, properties::{NamespaceKey, PropertyID, PropertyKey, Variant}, render::props::Output as OutputNS, scenes::props::Scenes as SceneNS, task::{Environment, Task}};
use crate::platform::props::Board as BoardNS;
struct HADevice {
device_id: String,
scenes: Vec<String>
}
impl HADevice {
fn new(device_id: String, scenes: Vec<String>) -> Self {
HADevice {
device_id,
scenes
}
}
fn topic(&self, key: &str) -> String {
format!("renderbug/{}/{}", self.device_id, key)
}
fn command_topic(&self) -> String {
self.topic("command")
}
fn state_topic(&self) -> String {
self.topic("state")
}
fn device_config_topic(&self) -> String {
format!("homeassistant/device/renderbug-rs-{}/config", self.device_id)
}
fn registration(&self) -> Value {
let components: HashMap<String, Value> = self.scenes.iter().map(|f| {
let scene_id = format!("renderbug_{}_scene_{}", self.device_id, f);
(scene_id.clone(), json!({
"p": "scene",
"unique_id": scene_id,
"payload_on": json!({"namespace": SceneNS::Namespace, "key": SceneNS::Current.key, "value": f}).to_string(),
"name": f
}))
}).collect();
json!({
"stat_t": self.state_topic(),
"cmd_t": self.command_topic(),
"dev": {
"name": format!("Renderbug-rs ESP32 {}", self.device_id),
"mdl": "Renderbug-rs ESP32",
"mf": "Phong Robotics",
"identifiers": [self.device_id]
},
"origin": {
"name": "renderbug-rs"
},
"cmps": components
})
}
}
pub struct MqttTask {
client: Option<EspMqttClient<'static>>,
ha_device: Option<HADevice>
}
impl MqttTask {
pub fn new() -> Self {
MqttTask {
client: None,
ha_device: None
}
}
fn start_mqtt(&mut self, env: &mut Environment) {
log::info!("Starting MQTT");
let chip_id: u64 = env.get_property(BoardNS::ChipID).unwrap();
let mut client_bus = env.bus.clone();
self.client = Some(EspMqttClient::new_cb(
"mqtt://10.0.0.2:1883",
&MqttClientConfiguration {
client_id: Some(&format!("{:X}", chip_id)),
..Default::default()
},
move |evt| {
match evt.payload() {
EventPayload::Received { id, topic, data, details } => {
log::info!("got event id={} topic={:?} data={:?} details={:?}", id, topic, str::from_utf8(data), details);
let props: HashMap<String, Variant> = serde_json::from_slice(data).unwrap();
let ns: String = props.get("namespace").unwrap().to_owned().into();
let key: String = props.get("key").unwrap().to_owned().into();
let value: Variant = props.get("value").unwrap().clone();
let all = client_bus.properties().all_props();
let prop_id = all.iter().find(|f| {
f.key.0 == key && f.namespace.0 == ns
});
client_bus.set_property(SystemNS::Idle, false);
match prop_id {
None => log::warn!("Could not find property {} in {}!", key, ns),
Some(id) => client_bus.set_property(id.clone(), value),
}
},
EventPayload::Connected(_) => {
log::info!("MQTT is online!");
client_bus.set_property(props::MQTT::Online, true);
},
EventPayload::Disconnected => {
log::info!("MQTT is offline!");
client_bus.set_property(props::MQTT::Online, false);
},
_ => ()
}
}
).unwrap());
log::info!("Connected!");
}
}
impl Task for MqttTask {
fn start(&mut self, bus: &mut EventBus) {
bus.set_property(props::MQTT::Online, false);
}
fn on_ready(&mut self, bus: &mut EventBus) {
let chip_id: u64 = bus.get_property(BoardNS::ChipID).unwrap();
let scenes: Vec<String> = bus.get_property(SceneNS::All).unwrap();
log::info!("Setting up scenes: {:?}", scenes);
self.ha_device = Some(HADevice::new(format!("{:X}", chip_id), scenes));
}
fn on_property_change(&mut self, key: PropertyID, value: &Variant, bus: &mut EventBus) {
match (key, value) {
(SystemNS::NetworkOnline, Variant::Boolean(true)) => {
log::info!("Registering with MQTT");
self.start_mqtt(bus);
},
(props::MQTT::Online, Variant::Boolean(true)) => {
if let Some(ref mut client) = self.client {
if let Some(ref device) = self.ha_device {
client.enqueue(
&device.device_config_topic(),
esp_idf_svc::mqtt::client::QoS::AtLeastOnce,
false,
device.registration().to_string().as_bytes()
).unwrap();
log::info!("Subscribing to {}", device.command_topic());
client.subscribe(&device.command_topic(), esp_idf_svc::mqtt::client::QoS::AtLeastOnce).unwrap();
}
}
}
/*(prop_id!(OutputNS::FPS), Variant::UInt(fps)) => {
if let Some(ref mut client) = self.client {
if let Some(ref sensor) = self.ha_device {
client.enqueue(
&sensor.topic("state"),
esp_idf_svc::mqtt::client::QoS::AtLeastOnce,
false,
json!(fps).to_string().as_bytes()
).unwrap();
}
}
},*/
_ => ()
}
}
}
pub mod props {
use crate::property_namespace;
property_namespace!(
MQTT,
Online => true
);
}