use async_trait::async_trait; use google_home::errors::ErrorCode; use google_home::{ device, traits::{self, OnOff}, types::Type, GoogleHomeDevice, }; use rumqttc::{AsyncClient, Publish}; use serde::Deserialize; use std::time::Duration; use tokio::task::JoinHandle; use tracing::{debug, error, trace, warn}; use crate::config::{CreateDevice, InfoConfig, MqttDeviceConfig}; use crate::devices::Device; use crate::error::CreateDeviceError; use crate::event::EventChannel; use crate::event::OnMqtt; use crate::event::OnPresence; use crate::messages::OnOffMessage; #[derive(Debug, Clone, Deserialize, PartialEq, Eq, Copy)] pub enum OutletType { Outlet, Kettle, Charger, Light, } #[derive(Debug, Clone, Deserialize)] pub struct IkeaOutletConfig { #[serde(flatten)] info: InfoConfig, #[serde(flatten)] mqtt: MqttDeviceConfig, #[serde(default = "default_outlet_type")] outlet_type: OutletType, timeout: Option, // Timeout in seconds } fn default_outlet_type() -> OutletType { OutletType::Outlet } #[derive(Debug)] pub struct IkeaOutlet { identifier: String, info: InfoConfig, mqtt: MqttDeviceConfig, outlet_type: OutletType, timeout: Option, client: AsyncClient, last_known_state: bool, handle: Option>, } impl CreateDevice for IkeaOutlet { type Config = IkeaOutletConfig; fn create( identifier: &str, config: Self::Config, _event_channel: &EventChannel, client: &AsyncClient, _presence_topic: &str, ) -> Result { trace!( id = identifier, name = config.info.name, room = config.info.room, "Setting up IkeaOutlet" ); Ok(Self { identifier: identifier.to_owned(), info: config.info, mqtt: config.mqtt, outlet_type: config.outlet_type, timeout: config.timeout, client: client.clone(), last_known_state: false, handle: None, }) } } async fn set_on(client: AsyncClient, topic: &str, on: bool) { let message = OnOffMessage::new(on); let topic = format!("{}/set", topic); // TODO: Handle potential errors here client .publish( topic.clone(), rumqttc::QoS::AtLeastOnce, false, serde_json::to_string(&message).unwrap(), ) .await .map_err(|err| warn!("Failed to update state on {topic}: {err}")) .ok(); } impl Device for IkeaOutlet { fn get_id(&self) -> &str { &self.identifier } } #[async_trait] impl OnMqtt for IkeaOutlet { fn topics(&self) -> Vec<&str> { vec![&self.mqtt.topic] } async fn on_mqtt(&mut self, message: Publish) { // Update the internal state based on what the device has reported let state = match OnOffMessage::try_from(message) { Ok(state) => state.state(), Err(err) => { error!(id = self.identifier, "Failed to parse message: {err}"); return; } }; // No need to do anything if the state has not changed if state == self.last_known_state { return; } // Abort any timer that is currently running if let Some(handle) = self.handle.take() { handle.abort(); } debug!(id = self.identifier, "Updating state to {state}"); self.last_known_state = state; // If this is a kettle start a timeout for turning it of again if state { let timeout = match self.timeout { Some(timeout) => Duration::from_secs(timeout), None => return, }; // Turn the kettle of after the specified timeout // TODO: Impl Drop for IkeaOutlet that will abort the handle if the IkeaOutlet // get dropped let client = self.client.clone(); let topic = self.mqtt.topic.clone(); let id = self.identifier.clone(); self.handle = Some(tokio::spawn(async move { debug!(id, "Starting timeout ({timeout:?})..."); tokio::time::sleep(timeout).await; debug!(id, "Turning outlet off!"); // TODO: Idealy we would call self.set_on(false), however since we want to do // it after a timeout we have to put it in a seperate task. // I don't think we can really get around calling outside function set_on(client, &topic, false).await; })); } } } #[async_trait] impl OnPresence for IkeaOutlet { async fn on_presence(&mut self, presence: bool) { // Turn off the outlet when we leave the house (Not if it is a battery charger) if !presence && self.outlet_type != OutletType::Charger { debug!(id = self.identifier, "Turning device off"); self.set_on(false).await.ok(); } } } impl GoogleHomeDevice for IkeaOutlet { fn get_device_type(&self) -> Type { match self.outlet_type { OutletType::Outlet => Type::Outlet, OutletType::Kettle => Type::Kettle, OutletType::Light => Type::Light, // Find a better device type for this, ideally would like to use charger, but that needs more work OutletType::Charger => Type::Outlet, // Find a better device type for this, ideally would like to use charger, but that needs more work } } fn get_device_name(&self) -> device::Name { device::Name::new(&self.info.name) } fn get_id(&self) -> &str { Device::get_id(self) } fn is_online(&self) -> bool { true } fn get_room_hint(&self) -> Option<&str> { self.info.room.as_deref() } fn will_report_state(&self) -> bool { // TODO: Implement state reporting false } } #[async_trait] impl traits::OnOff for IkeaOutlet { async fn is_on(&self) -> Result { Ok(self.last_known_state) } async fn set_on(&mut self, on: bool) -> Result<(), ErrorCode> { set_on(self.client.clone(), &self.mqtt.topic, on).await; Ok(()) } }