Report AirFilter humidity
This commit is contained in:
parent
8b191f6013
commit
8333c75a0d
|
@ -17,4 +17,6 @@ pub struct Attributes {
|
||||||
pub command_only_fan_speed: Option<bool>,
|
pub command_only_fan_speed: Option<bool>,
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
pub available_fan_speeds: Option<AvailableSpeeds>,
|
pub available_fan_speeds: Option<AvailableSpeeds>,
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub query_only_humidity_setting: Option<bool>,
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,7 @@ use serde::Serialize;
|
||||||
use crate::errors::{DeviceError, ErrorCode};
|
use crate::errors::{DeviceError, ErrorCode};
|
||||||
use crate::request::execute::CommandType;
|
use crate::request::execute::CommandType;
|
||||||
use crate::response;
|
use crate::response;
|
||||||
use crate::traits::{FanSpeed, OnOff, Scene, Trait};
|
use crate::traits::{FanSpeed, HumiditySetting, OnOff, Scene, Trait};
|
||||||
use crate::types::Type;
|
use crate::types::Type;
|
||||||
|
|
||||||
// TODO: Find a more elegant way to do this
|
// TODO: Find a more elegant way to do this
|
||||||
|
@ -42,7 +42,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
#[impl_cast::device(As: OnOff + Scene + FanSpeed)]
|
#[impl_cast::device(As: OnOff + Scene + FanSpeed + HumiditySetting)]
|
||||||
pub trait GoogleHomeDevice: AsGoogleHomeDevice + Sync + Send + 'static {
|
pub trait GoogleHomeDevice: AsGoogleHomeDevice + Sync + Send + 'static {
|
||||||
fn get_device_type(&self) -> Type;
|
fn get_device_type(&self) -> Type;
|
||||||
fn get_device_name(&self) -> Name;
|
fn get_device_name(&self) -> Name;
|
||||||
|
@ -95,6 +95,12 @@ pub trait GoogleHomeDevice: AsGoogleHomeDevice + Sync + Send + 'static {
|
||||||
device.attributes.available_fan_speeds = Some(fan_speed.available_speeds());
|
device.attributes.available_fan_speeds = Some(fan_speed.available_speeds());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(humidity_setting) = As::<dyn HumiditySetting>::cast(self) {
|
||||||
|
traits.push(Trait::HumiditySetting);
|
||||||
|
device.attributes.query_only_humidity_setting =
|
||||||
|
humidity_setting.query_only_humidity_setting();
|
||||||
|
}
|
||||||
|
|
||||||
device.traits = traits;
|
device.traits = traits;
|
||||||
|
|
||||||
device
|
device
|
||||||
|
@ -120,6 +126,11 @@ pub trait GoogleHomeDevice: AsGoogleHomeDevice + Sync + Send + 'static {
|
||||||
device.state.current_fan_speed_setting = Some(fan_speed.current_speed().await);
|
device.state.current_fan_speed_setting = Some(fan_speed.current_speed().await);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(humidity_setting) = As::<dyn HumiditySetting>::cast(self) {
|
||||||
|
device.state.humidity_ambient_percent =
|
||||||
|
Some(humidity_setting.humidity_ambient_percent().await);
|
||||||
|
}
|
||||||
|
|
||||||
device
|
device
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -36,4 +36,7 @@ pub struct State {
|
||||||
|
|
||||||
#[serde(skip_serializing_if = "Option::is_none")]
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
pub current_fan_speed_setting: Option<String>,
|
pub current_fan_speed_setting: Option<String>,
|
||||||
|
|
||||||
|
#[serde(skip_serializing_if = "Option::is_none")]
|
||||||
|
pub humidity_ambient_percent: Option<f32>,
|
||||||
}
|
}
|
||||||
|
|
|
@ -98,6 +98,7 @@ mod tests {
|
||||||
let state = State {
|
let state = State {
|
||||||
on: Some(true),
|
on: Some(true),
|
||||||
current_fan_speed_setting: None,
|
current_fan_speed_setting: None,
|
||||||
|
humidity_ambient_percent: None,
|
||||||
};
|
};
|
||||||
let mut command = Command::new(Status::Success);
|
let mut command = Command::new(Status::Success);
|
||||||
command.states = Some(States {
|
command.states = Some(States {
|
||||||
|
|
|
@ -11,6 +11,8 @@ pub enum Trait {
|
||||||
Scene,
|
Scene,
|
||||||
#[serde(rename = "action.devices.traits.FanSpeed")]
|
#[serde(rename = "action.devices.traits.FanSpeed")]
|
||||||
FanSpeed,
|
FanSpeed,
|
||||||
|
#[serde(rename = "action.devices.traits.HumiditySetting")]
|
||||||
|
HumiditySetting,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
|
@ -72,3 +74,14 @@ pub trait FanSpeed {
|
||||||
async fn current_speed(&self) -> String;
|
async fn current_speed(&self) -> String;
|
||||||
async fn set_speed(&self, speed: &str) -> Result<(), ErrorCode>;
|
async fn set_speed(&self, speed: &str) -> Result<(), ErrorCode>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
#[impl_cast::device_trait]
|
||||||
|
pub trait HumiditySetting {
|
||||||
|
// TODO: This implementation is not complete, I have only implemented what I need right now
|
||||||
|
fn query_only_humidity_setting(&self) -> Option<bool> {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn humidity_ambient_percent(&self) -> f32;
|
||||||
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use async_trait::async_trait;
|
use async_trait::async_trait;
|
||||||
use google_home::device::Name;
|
use google_home::device::Name;
|
||||||
use google_home::errors::ErrorCode;
|
use google_home::errors::ErrorCode;
|
||||||
use google_home::traits::{AvailableSpeeds, FanSpeed, OnOff, Speed, SpeedValues};
|
use google_home::traits::{AvailableSpeeds, FanSpeed, HumiditySetting, OnOff, Speed, SpeedValues};
|
||||||
use google_home::types::Type;
|
use google_home::types::Type;
|
||||||
use google_home::GoogleHomeDevice;
|
use google_home::GoogleHomeDevice;
|
||||||
use rumqttc::{AsyncClient, Publish};
|
use rumqttc::{AsyncClient, Publish};
|
||||||
|
@ -13,7 +13,7 @@ use crate::device_manager::{ConfigExternal, DeviceConfig};
|
||||||
use crate::devices::Device;
|
use crate::devices::Device;
|
||||||
use crate::error::DeviceConfigError;
|
use crate::error::DeviceConfigError;
|
||||||
use crate::event::OnMqtt;
|
use crate::event::OnMqtt;
|
||||||
use crate::messages::{AirFilterMessage, AirFilterState};
|
use crate::messages::{AirFilterFanState, AirFilterState, SetAirFilterFanState};
|
||||||
|
|
||||||
#[derive(Debug, Deserialize)]
|
#[derive(Debug, Deserialize)]
|
||||||
pub struct AirFilterConfig {
|
pub struct AirFilterConfig {
|
||||||
|
@ -35,7 +35,10 @@ impl DeviceConfig for AirFilterConfig {
|
||||||
info: self.info,
|
info: self.info,
|
||||||
mqtt: self.mqtt,
|
mqtt: self.mqtt,
|
||||||
client: ext.client.clone(),
|
client: ext.client.clone(),
|
||||||
last_known_state: AirFilterState::Off,
|
last_known_state: AirFilterState {
|
||||||
|
state: AirFilterFanState::Off,
|
||||||
|
humidity: 0.0,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
Ok(Box::new(device))
|
Ok(Box::new(device))
|
||||||
|
@ -53,8 +56,8 @@ pub struct AirFilter {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AirFilter {
|
impl AirFilter {
|
||||||
async fn set_speed(&self, state: AirFilterState) {
|
async fn set_speed(&self, state: AirFilterFanState) {
|
||||||
let message = AirFilterMessage::new(state);
|
let message = SetAirFilterFanState::new(state);
|
||||||
|
|
||||||
let topic = format!("{}/set", self.mqtt.topic);
|
let topic = format!("{}/set", self.mqtt.topic);
|
||||||
// TODO: Handle potential errors here
|
// TODO: Handle potential errors here
|
||||||
|
@ -84,8 +87,8 @@ impl OnMqtt for AirFilter {
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn on_mqtt(&mut self, message: Publish) {
|
async fn on_mqtt(&mut self, message: Publish) {
|
||||||
let state = match AirFilterMessage::try_from(message) {
|
let state = match AirFilterState::try_from(message) {
|
||||||
Ok(state) => state.state(),
|
Ok(state) => state,
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
error!(id = self.identifier, "Failed to parse message: {err}");
|
error!(id = self.identifier, "Failed to parse message: {err}");
|
||||||
return;
|
return;
|
||||||
|
@ -131,16 +134,16 @@ impl GoogleHomeDevice for AirFilter {
|
||||||
#[async_trait]
|
#[async_trait]
|
||||||
impl OnOff for AirFilter {
|
impl OnOff for AirFilter {
|
||||||
async fn is_on(&self) -> Result<bool, ErrorCode> {
|
async fn is_on(&self) -> Result<bool, ErrorCode> {
|
||||||
Ok(self.last_known_state != AirFilterState::Off)
|
Ok(self.last_known_state.state != AirFilterFanState::Off)
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn set_on(&mut self, on: bool) -> Result<(), ErrorCode> {
|
async fn set_on(&mut self, on: bool) -> Result<(), ErrorCode> {
|
||||||
debug!("Turning on air filter: {on}");
|
debug!("Turning on air filter: {on}");
|
||||||
|
|
||||||
if on {
|
if on {
|
||||||
self.set_speed(AirFilterState::High).await;
|
self.set_speed(AirFilterFanState::High).await;
|
||||||
} else {
|
} else {
|
||||||
self.set_speed(AirFilterState::Off).await;
|
self.set_speed(AirFilterFanState::Off).await;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -186,11 +189,11 @@ impl FanSpeed for AirFilter {
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn current_speed(&self) -> String {
|
async fn current_speed(&self) -> String {
|
||||||
let speed = match self.last_known_state {
|
let speed = match self.last_known_state.state {
|
||||||
AirFilterState::Off => "off",
|
AirFilterFanState::Off => "off",
|
||||||
AirFilterState::Low => "low",
|
AirFilterFanState::Low => "low",
|
||||||
AirFilterState::Medium => "medium",
|
AirFilterFanState::Medium => "medium",
|
||||||
AirFilterState::High => "high",
|
AirFilterFanState::High => "high",
|
||||||
};
|
};
|
||||||
|
|
||||||
speed.into()
|
speed.into()
|
||||||
|
@ -198,13 +201,13 @@ impl FanSpeed for AirFilter {
|
||||||
|
|
||||||
async fn set_speed(&self, speed: &str) -> Result<(), ErrorCode> {
|
async fn set_speed(&self, speed: &str) -> Result<(), ErrorCode> {
|
||||||
let state = if speed == "off" {
|
let state = if speed == "off" {
|
||||||
AirFilterState::Off
|
AirFilterFanState::Off
|
||||||
} else if speed == "low" {
|
} else if speed == "low" {
|
||||||
AirFilterState::Low
|
AirFilterFanState::Low
|
||||||
} else if speed == "medium" {
|
} else if speed == "medium" {
|
||||||
AirFilterState::Medium
|
AirFilterFanState::Medium
|
||||||
} else if speed == "high" {
|
} else if speed == "high" {
|
||||||
AirFilterState::High
|
AirFilterFanState::High
|
||||||
} else {
|
} else {
|
||||||
return Err(google_home::errors::DeviceError::TransientError.into());
|
return Err(google_home::errors::DeviceError::TransientError.into());
|
||||||
};
|
};
|
||||||
|
@ -214,3 +217,14 @@ impl FanSpeed for AirFilter {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl HumiditySetting for AirFilter {
|
||||||
|
fn query_only_humidity_setting(&self) -> Option<bool> {
|
||||||
|
Some(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn humidity_ambient_percent(&self) -> f32 {
|
||||||
|
self.last_known_state.humidity
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -245,7 +245,7 @@ impl TryFrom<Bytes> for HueMessage {
|
||||||
// TODO: Import this from the air_filter code itself instead of copying
|
// TODO: Import this from the air_filter code itself instead of copying
|
||||||
#[derive(PartialEq, Eq, Debug, Clone, Copy, Deserialize, Serialize)]
|
#[derive(PartialEq, Eq, Debug, Clone, Copy, Deserialize, Serialize)]
|
||||||
#[serde(rename_all = "snake_case")]
|
#[serde(rename_all = "snake_case")]
|
||||||
pub enum AirFilterState {
|
pub enum AirFilterFanState {
|
||||||
Off,
|
Off,
|
||||||
Low,
|
Low,
|
||||||
Medium,
|
Medium,
|
||||||
|
@ -253,21 +253,23 @@ pub enum AirFilterState {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, Deserialize, Serialize)]
|
#[derive(Debug, Clone, Copy, Deserialize, Serialize)]
|
||||||
pub struct AirFilterMessage {
|
pub struct SetAirFilterFanState {
|
||||||
state: AirFilterState,
|
state: AirFilterFanState,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AirFilterMessage {
|
#[derive(PartialEq, Debug, Clone, Copy, Deserialize, Serialize)]
|
||||||
pub fn state(&self) -> AirFilterState {
|
pub struct AirFilterState {
|
||||||
self.state
|
pub state: AirFilterFanState,
|
||||||
|
pub humidity: f32,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new(state: AirFilterState) -> Self {
|
impl SetAirFilterFanState {
|
||||||
|
pub fn new(state: AirFilterFanState) -> Self {
|
||||||
Self { state }
|
Self { state }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TryFrom<Publish> for AirFilterMessage {
|
impl TryFrom<Publish> for AirFilterState {
|
||||||
type Error = ParseError;
|
type Error = ParseError;
|
||||||
|
|
||||||
fn try_from(message: Publish) -> Result<Self, Self::Error> {
|
fn try_from(message: Publish) -> Result<Self, Self::Error> {
|
||||||
|
|
Loading…
Reference in New Issue
Block a user