home-assistant-rest

Crates.iohome-assistant-rest
lib.rshome-assistant-rest
version0.2.0
sourcesrc
created_at2023-03-30 12:40:41.111503
updated_at2023-06-06 00:52:37.095265
descriptionHome Assistant Async REST API Client
homepage
repositoryhttps://github.com/StefanBossbaly/home-assistant-rest/
max_upload_size
id825031
size74,800
Stefan Bossbaly (StefanBossbaly)

documentation

README

Home Assistant REST crates.io

This project provides developers with a standard and ergonomic Rust API for calling the various endpoints in Home Assistant's REST API. It handles the serialization and deserialization of the requests and responses and allows the developer to use the provided structures. Currently this project is still under active development and the API might change at any time.

Example Usage

use home_assistant_rest::{get::StateEnum, Client};

#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
    let base_url = "REPLACE_WITH_HASS_BASE_URL";
    let token = "REPLACE_WITH_ACCESS_TOKEN";

    let client = Client::new(base_url, token)?;
    let api_status = client.get_api_status().await?;

    if api_status.message != "API running." {
        println!("API is NOT running");
    } else {
        println!("API is running, getting status of \"sun.sun\" entity");
        let state_entity = client.get_states_of_entity("sun.sun").await?;

        if let Some(state) = state_entity.state {
            match state {
                StateEnum::Boolean(x) => println!("Value is boolean with value {}", x),
                StateEnum::Decimal(x) => println!("Value is decimal with value {}", x),
                StateEnum::Integer(x) => println!("Value is integer with value {}", x),
                StateEnum::String(x) => println!("Value is string with value \"{}\"", x),
            }
        } else {
            println!("Value was not provided");
        }
    }

    Ok(())
}

API Status

Endpoint Request Type Implemented Tested
/api/ GET
/api/config GET
/api/events GET
/api/services GET
/api/history/period/<timestamp> GET
/api/logbook/<timestamp> GET
/api/states GET
/api/states/<entity_id> GET
/api/error_log GET
/api/camera_proxy/<camera entity_id> GET
/api/calendars GET
/api/calendars/<calendar entity_id> GET
/api/states/<entity_id> POST
/api/events/<event_type> POST
/api/services/<domain>/<service> POST
/api/template POST
/api/config/core/check_config POST
/api/intent/handle POST

Differences between the specification and implementation

Home Assistant provides a API specification that lists in the various different endpoints, their parameters and their output. However when implementing this library I have come across a number of discrepancies between the specification and the my local Home Assistant instance. Here is a list of the differences I have found so far:

  1. For the /api/services endpoint, the services attribute is listed in the example as a list of strings. However testing locally the type for the services attribute is actually a map.
  2. For the /api/camera_proxy/<camera entity_id> endpoint, the only listed parameters are the camera id and the time. However looking at the developer tools panel, it shows that a unique token is also needed in order to retrieve the image.
  3. Serialization of non-string types, especially for the state attribute. I have seen many entities that report their state as either an integer, decimal or boolean but when that state is serialized in the JSON response, the type is a string and not the underlying type. In order to work around this issue, when deserializing the state we attempt to deserialize into the underlying types first if that fails then we default to a string type. This is not ideal as it is costly to parse various different types. In the future I will consider adding an option that allows the user to opt-out of this "feature".

Authors

Stefan Bossbaly

License

This project is licensed under the MIT License - see the LICENSE file for details

Acknowledgments

Commit count: 43

cargo fmt