mirror of
https://github.com/esphome/esphome.git
synced 2025-02-25 04:22:30 +01:00
add /states REST API call that dumps the current component state in a json array
This commit is contained in:
parent
e819185de1
commit
81e9cabf65
6 changed files with 147 additions and 64 deletions
|
@ -13,159 +13,140 @@ ListEntitiesIterator::ListEntitiesIterator(WebServer *web_server) : web_server_(
|
||||||
|
|
||||||
#ifdef USE_BINARY_SENSOR
|
#ifdef USE_BINARY_SENSOR
|
||||||
bool ListEntitiesIterator::on_binary_sensor(binary_sensor::BinarySensor *binary_sensor) {
|
bool ListEntitiesIterator::on_binary_sensor(binary_sensor::BinarySensor *binary_sensor) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(
|
return this->process(
|
||||||
this->web_server_->binary_sensor_json(binary_sensor, binary_sensor->state, DETAIL_ALL).c_str(), "state");
|
this->web_server_->binary_sensor_json(binary_sensor, binary_sensor->state, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_COVER
|
#ifdef USE_COVER
|
||||||
bool ListEntitiesIterator::on_cover(cover::Cover *cover) {
|
bool ListEntitiesIterator::on_cover(cover::Cover *cover) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->cover_json(cover, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->cover_json(cover, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_FAN
|
#ifdef USE_FAN
|
||||||
bool ListEntitiesIterator::on_fan(fan::Fan *fan) {
|
bool ListEntitiesIterator::on_fan(fan::Fan *fan) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->fan_json(fan, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->fan_json(fan, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LIGHT
|
#ifdef USE_LIGHT
|
||||||
bool ListEntitiesIterator::on_light(light::LightState *light) {
|
bool ListEntitiesIterator::on_light(light::LightState *light) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->light_json(light, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->light_json(light, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SENSOR
|
#ifdef USE_SENSOR
|
||||||
bool ListEntitiesIterator::on_sensor(sensor::Sensor *sensor) {
|
bool ListEntitiesIterator::on_sensor(sensor::Sensor *sensor) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->sensor_json(sensor, sensor->state, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->sensor_json(sensor, sensor->state, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SWITCH
|
#ifdef USE_SWITCH
|
||||||
bool ListEntitiesIterator::on_switch(switch_::Switch *a_switch) {
|
bool ListEntitiesIterator::on_switch(switch_::Switch *a_switch) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->switch_json(a_switch, a_switch->state, DETAIL_ALL).c_str(),
|
return this->process(this->web_server_->switch_json(a_switch, a_switch->state, DETAIL_ALL));
|
||||||
"state");
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BUTTON
|
#ifdef USE_BUTTON
|
||||||
bool ListEntitiesIterator::on_button(button::Button *button) {
|
bool ListEntitiesIterator::on_button(button::Button *button) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->button_json(button, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->button_json(button, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT_SENSOR
|
#ifdef USE_TEXT_SENSOR
|
||||||
bool ListEntitiesIterator::on_text_sensor(text_sensor::TextSensor *text_sensor) {
|
bool ListEntitiesIterator::on_text_sensor(text_sensor::TextSensor *text_sensor) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(
|
return this->process(
|
||||||
this->web_server_->text_sensor_json(text_sensor, text_sensor->state, DETAIL_ALL).c_str(), "state");
|
this->web_server_->text_sensor_json(text_sensor, text_sensor->state, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LOCK
|
#ifdef USE_LOCK
|
||||||
bool ListEntitiesIterator::on_lock(lock::Lock *a_lock) {
|
bool ListEntitiesIterator::on_lock(lock::Lock *a_lock) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->lock_json(a_lock, a_lock->state, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->lock_json(a_lock, a_lock->state, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_VALVE
|
#ifdef USE_VALVE
|
||||||
bool ListEntitiesIterator::on_valve(valve::Valve *valve) {
|
bool ListEntitiesIterator::on_valve(valve::Valve *valve) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->valve_json(valve, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->valve_json(valve, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_CLIMATE
|
#ifdef USE_CLIMATE
|
||||||
bool ListEntitiesIterator::on_climate(climate::Climate *climate) {
|
bool ListEntitiesIterator::on_climate(climate::Climate *climate) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->climate_json(climate, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->climate_json(climate, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_NUMBER
|
#ifdef USE_NUMBER
|
||||||
bool ListEntitiesIterator::on_number(number::Number *number) {
|
bool ListEntitiesIterator::on_number(number::Number *number) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->number_json(number, number->state, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->number_json(number, number->state, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_DATETIME_DATE
|
#ifdef USE_DATETIME_DATE
|
||||||
bool ListEntitiesIterator::on_date(datetime::DateEntity *date) {
|
bool ListEntitiesIterator::on_date(datetime::DateEntity *date) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->date_json(date, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->date_json(date, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_DATETIME_TIME
|
#ifdef USE_DATETIME_TIME
|
||||||
bool ListEntitiesIterator::on_time(datetime::TimeEntity *time) {
|
bool ListEntitiesIterator::on_time(datetime::TimeEntity *time) {
|
||||||
this->web_server_->events_.send(this->web_server_->time_json(time, DETAIL_ALL).c_str(), "state");
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
|
return this->process(this->web_server_->time_json(time, DETAIL_ALL));
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_DATETIME_DATETIME
|
#ifdef USE_DATETIME_DATETIME
|
||||||
bool ListEntitiesIterator::on_datetime(datetime::DateTimeEntity *datetime) {
|
bool ListEntitiesIterator::on_datetime(datetime::DateTimeEntity *datetime) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->datetime_json(datetime, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->datetime_json(datetime, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_TEXT
|
#ifdef USE_TEXT
|
||||||
bool ListEntitiesIterator::on_text(text::Text *text) {
|
bool ListEntitiesIterator::on_text(text::Text *text) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->text_json(text, text->state, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->text_json(text, text->state, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
bool ListEntitiesIterator::on_select(select::Select *select) {
|
bool ListEntitiesIterator::on_select(select::Select *select) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->select_json(select, select->state, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->select_json(select, select->state, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
bool ListEntitiesIterator::on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) {
|
bool ListEntitiesIterator::on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(
|
return this->process(
|
||||||
this->web_server_->alarm_control_panel_json(a_alarm_control_panel, a_alarm_control_panel->get_state(), DETAIL_ALL)
|
this->web_server_->alarm_control_panel_json(a_alarm_control_panel, a_alarm_control_panel->get_state(), DETAIL_ALL));
|
||||||
.c_str(),
|
|
||||||
"state");
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -173,20 +154,27 @@ bool ListEntitiesIterator::on_alarm_control_panel(alarm_control_panel::AlarmCont
|
||||||
bool ListEntitiesIterator::on_event(event::Event *event) {
|
bool ListEntitiesIterator::on_event(event::Event *event) {
|
||||||
// Null event type, since we are just iterating over entities
|
// Null event type, since we are just iterating over entities
|
||||||
const std::string null_event_type = "";
|
const std::string null_event_type = "";
|
||||||
this->web_server_->events_.send(this->web_server_->event_json(event, null_event_type, DETAIL_ALL).c_str(), "state");
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
|
return this->process(this->web_server_->event_json(event, null_event_type, DETAIL_ALL));
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_UPDATE
|
#ifdef USE_UPDATE
|
||||||
bool ListEntitiesIterator::on_update(update::UpdateEntity *update) {
|
bool ListEntitiesIterator::on_update(update::UpdateEntity *update) {
|
||||||
if (this->web_server_->events_.count() == 0)
|
if (!this->has_connected_client())
|
||||||
return true;
|
return true;
|
||||||
this->web_server_->events_.send(this->web_server_->update_json(update, DETAIL_ALL).c_str(), "state");
|
return this->process(this->web_server_->update_json(update, DETAIL_ALL));
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
bool ListEntitiesIterator::has_connected_client() { return this->web_server_->events_.count() > 0; }
|
||||||
|
|
||||||
|
bool ListEntitiesIterator::process(std::string s) {
|
||||||
|
this->web_server_->events_.send(s.c_str(), "state");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace web_server
|
} // namespace web_server
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -75,6 +75,8 @@ class ListEntitiesIterator : public ComponentIterator {
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
WebServer *web_server_;
|
WebServer *web_server_;
|
||||||
|
bool has_connected_client();
|
||||||
|
virtual bool process(std::string s);
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace web_server
|
} // namespace web_server
|
||||||
|
|
35
esphome/components/web_server/states.cpp
Normal file
35
esphome/components/web_server/states.cpp
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
#ifdef USE_ARDUINO
|
||||||
|
|
||||||
|
#include "list_entities.h"
|
||||||
|
#include "esphome/core/application.h"
|
||||||
|
|
||||||
|
#include "web_server.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace web_server {
|
||||||
|
|
||||||
|
StatesIterator::StatesIterator(WebServer *web_server) : ListEntitiesIterator::ListEntitiesIterator(web_server) {}
|
||||||
|
|
||||||
|
bool StatesIterator::has_connected_client() { return true; }
|
||||||
|
|
||||||
|
bool StatesIterator::process(std::string s) {
|
||||||
|
this->str_ = s;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
optional<std::string> StatesIterator::next() {
|
||||||
|
while (this->state_ != IteratorState::MAX) {
|
||||||
|
this->str_ = nullopt;
|
||||||
|
this->advance();
|
||||||
|
if(this->str_.has_value()) {
|
||||||
|
return this->str_;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this->str_ = nullopt;
|
||||||
|
return nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace web_server
|
||||||
|
} // namespace esphome
|
||||||
|
|
||||||
|
#endif // USE_ARDUINO
|
30
esphome/components/web_server/states.h
Normal file
30
esphome/components/web_server/states.h
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#ifdef USE_ARDUINO
|
||||||
|
|
||||||
|
#include "list_entities.h"
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
#include "esphome/core/component_iterator.h"
|
||||||
|
#include "esphome/core/defines.h"
|
||||||
|
#include "esphome/components/web_server_base/web_server_base.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace web_server {
|
||||||
|
|
||||||
|
class WebServer;
|
||||||
|
|
||||||
|
class StatesIterator : public ListEntitiesIterator {
|
||||||
|
public:
|
||||||
|
StatesIterator(WebServer *web_server);
|
||||||
|
optional<std::string> next();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
// WebServer *web_server_;
|
||||||
|
optional<std::string> str_;
|
||||||
|
virtual bool process(std::string s);
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace web_server
|
||||||
|
} // namespace esphome
|
||||||
|
|
||||||
|
#endif // USE_ARDUINO
|
|
@ -199,6 +199,22 @@ void WebServer::handle_js_request(AsyncWebServerRequest *request) {
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
void WebServer::handle_states_request(AsyncWebServerRequest *request) {
|
||||||
|
AsyncResponseStream *stream = request->beginResponseStream("text/json");
|
||||||
|
StatesIterator states_it = StatesIterator(this);
|
||||||
|
states_it.begin();
|
||||||
|
optional<std::string> s;
|
||||||
|
stream->print(F("{"));
|
||||||
|
int i = 0;
|
||||||
|
while((s = states_it.next()) != nullopt) {
|
||||||
|
if (i++)
|
||||||
|
stream->print(F(","));
|
||||||
|
stream->print(s->c_str());
|
||||||
|
}
|
||||||
|
stream->print(F("}\n"));
|
||||||
|
request->send(stream);
|
||||||
|
}
|
||||||
|
|
||||||
#define set_json_id(root, obj, sensor, start_config) \
|
#define set_json_id(root, obj, sensor, start_config) \
|
||||||
(root)["id"] = sensor; \
|
(root)["id"] = sensor; \
|
||||||
if (((start_config) == DETAIL_ALL)) { \
|
if (((start_config) == DETAIL_ALL)) { \
|
||||||
|
@ -1553,6 +1569,9 @@ bool WebServer::canHandle(AsyncWebServerRequest *request) {
|
||||||
if (request->url() == "/")
|
if (request->url() == "/")
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
|
if (request->url() == "/states")
|
||||||
|
return true;
|
||||||
|
|
||||||
#ifdef USE_WEBSERVER_CSS_INCLUDE
|
#ifdef USE_WEBSERVER_CSS_INCLUDE
|
||||||
if (request->url() == "/0.css")
|
if (request->url() == "/0.css")
|
||||||
return true;
|
return true;
|
||||||
|
@ -1686,6 +1705,11 @@ void WebServer::handleRequest(AsyncWebServerRequest *request) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (request->url() == "/states") {
|
||||||
|
this->handle_states_request(request);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef USE_WEBSERVER_CSS_INCLUDE
|
#ifdef USE_WEBSERVER_CSS_INCLUDE
|
||||||
if (request->url() == "/0.css") {
|
if (request->url() == "/0.css") {
|
||||||
this->handle_css_request(request);
|
this->handle_css_request(request);
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "list_entities.h"
|
#include "list_entities.h"
|
||||||
|
#include "states.h"
|
||||||
|
|
||||||
#include "esphome/components/web_server_base/web_server_base.h"
|
#include "esphome/components/web_server_base/web_server_base.h"
|
||||||
#ifdef USE_WEBSERVER
|
#ifdef USE_WEBSERVER
|
||||||
|
@ -148,6 +149,9 @@ class WebServer : public Controller, public Component, public AsyncWebHandler {
|
||||||
void handle_pna_cors_request(AsyncWebServerRequest *request);
|
void handle_pna_cors_request(AsyncWebServerRequest *request);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/// Handle a states request under '/states'.
|
||||||
|
void handle_states_request(AsyncWebServerRequest *request);
|
||||||
|
|
||||||
#ifdef USE_SENSOR
|
#ifdef USE_SENSOR
|
||||||
void on_sensor_update(sensor::Sensor *obj, float state) override;
|
void on_sensor_update(sensor::Sensor *obj, float state) override;
|
||||||
/// Handle a sensor request under '/sensor/<id>'.
|
/// Handle a sensor request under '/sensor/<id>'.
|
||||||
|
|
Loading…
Add table
Reference in a new issue