mirror of
https://github.com/esphome/esphome.git
synced 2024-11-30 18:54:14 +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
|
||||
bool ListEntitiesIterator::on_binary_sensor(binary_sensor::BinarySensor *binary_sensor) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(
|
||||
this->web_server_->binary_sensor_json(binary_sensor, binary_sensor->state, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(
|
||||
this->web_server_->binary_sensor_json(binary_sensor, binary_sensor->state, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
#ifdef USE_COVER
|
||||
bool ListEntitiesIterator::on_cover(cover::Cover *cover) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->cover_json(cover, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->cover_json(cover, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
#ifdef USE_FAN
|
||||
bool ListEntitiesIterator::on_fan(fan::Fan *fan) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->fan_json(fan, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->fan_json(fan, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
#ifdef USE_LIGHT
|
||||
bool ListEntitiesIterator::on_light(light::LightState *light) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->light_json(light, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->light_json(light, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
#ifdef USE_SENSOR
|
||||
bool ListEntitiesIterator::on_sensor(sensor::Sensor *sensor) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->sensor_json(sensor, sensor->state, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->sensor_json(sensor, sensor->state, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
#ifdef USE_SWITCH
|
||||
bool ListEntitiesIterator::on_switch(switch_::Switch *a_switch) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->switch_json(a_switch, a_switch->state, DETAIL_ALL).c_str(),
|
||||
"state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->switch_json(a_switch, a_switch->state, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
#ifdef USE_BUTTON
|
||||
bool ListEntitiesIterator::on_button(button::Button *button) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->button_json(button, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->button_json(button, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
#ifdef USE_TEXT_SENSOR
|
||||
bool ListEntitiesIterator::on_text_sensor(text_sensor::TextSensor *text_sensor) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(
|
||||
this->web_server_->text_sensor_json(text_sensor, text_sensor->state, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(
|
||||
this->web_server_->text_sensor_json(text_sensor, text_sensor->state, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
#ifdef USE_LOCK
|
||||
bool ListEntitiesIterator::on_lock(lock::Lock *a_lock) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->lock_json(a_lock, a_lock->state, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->lock_json(a_lock, a_lock->state, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef USE_VALVE
|
||||
bool ListEntitiesIterator::on_valve(valve::Valve *valve) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->valve_json(valve, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->valve_json(valve, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef USE_CLIMATE
|
||||
bool ListEntitiesIterator::on_climate(climate::Climate *climate) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->climate_json(climate, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->climate_json(climate, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef USE_NUMBER
|
||||
bool ListEntitiesIterator::on_number(number::Number *number) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->number_json(number, number->state, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->number_json(number, number->state, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef USE_DATETIME_DATE
|
||||
bool ListEntitiesIterator::on_date(datetime::DateEntity *date) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->date_json(date, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->date_json(date, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef USE_DATETIME_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 this->process(this->web_server_->time_json(time, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef USE_DATETIME_DATETIME
|
||||
bool ListEntitiesIterator::on_datetime(datetime::DateTimeEntity *datetime) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->datetime_json(datetime, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->datetime_json(datetime, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef USE_TEXT
|
||||
bool ListEntitiesIterator::on_text(text::Text *text) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->text_json(text, text->state, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->text_json(text, text->state, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef USE_SELECT
|
||||
bool ListEntitiesIterator::on_select(select::Select *select) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->select_json(select, select->state, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->select_json(select, select->state, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef USE_ALARM_CONTROL_PANEL
|
||||
bool ListEntitiesIterator::on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(
|
||||
this->web_server_->alarm_control_panel_json(a_alarm_control_panel, a_alarm_control_panel->get_state(), DETAIL_ALL)
|
||||
.c_str(),
|
||||
"state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(
|
||||
this->web_server_->alarm_control_panel_json(a_alarm_control_panel, a_alarm_control_panel->get_state(), DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
|
||||
|
@ -173,20 +154,27 @@ bool ListEntitiesIterator::on_alarm_control_panel(alarm_control_panel::AlarmCont
|
|||
bool ListEntitiesIterator::on_event(event::Event *event) {
|
||||
// Null event type, since we are just iterating over entities
|
||||
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 this->process(this->web_server_->event_json(event, null_event_type, DETAIL_ALL));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef USE_UPDATE
|
||||
bool ListEntitiesIterator::on_update(update::UpdateEntity *update) {
|
||||
if (this->web_server_->events_.count() == 0)
|
||||
return true;
|
||||
this->web_server_->events_.send(this->web_server_->update_json(update, DETAIL_ALL).c_str(), "state");
|
||||
if (!this->has_connected_client())
|
||||
return true;
|
||||
return this->process(this->web_server_->update_json(update, DETAIL_ALL));
|
||||
}
|
||||
#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 esphome
|
||||
#endif
|
||||
|
|
|
@ -75,6 +75,8 @@ class ListEntitiesIterator : public ComponentIterator {
|
|||
|
||||
protected:
|
||||
WebServer *web_server_;
|
||||
bool has_connected_client();
|
||||
virtual bool process(std::string s);
|
||||
};
|
||||
|
||||
} // 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
|
||||
|
||||
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) \
|
||||
(root)["id"] = sensor; \
|
||||
if (((start_config) == DETAIL_ALL)) { \
|
||||
|
@ -1553,6 +1569,9 @@ bool WebServer::canHandle(AsyncWebServerRequest *request) {
|
|||
if (request->url() == "/")
|
||||
return true;
|
||||
|
||||
if (request->url() == "/states")
|
||||
return true;
|
||||
|
||||
#ifdef USE_WEBSERVER_CSS_INCLUDE
|
||||
if (request->url() == "/0.css")
|
||||
return true;
|
||||
|
@ -1686,6 +1705,11 @@ void WebServer::handleRequest(AsyncWebServerRequest *request) {
|
|||
return;
|
||||
}
|
||||
|
||||
if (request->url() == "/states") {
|
||||
this->handle_states_request(request);
|
||||
return;
|
||||
}
|
||||
|
||||
#ifdef USE_WEBSERVER_CSS_INCLUDE
|
||||
if (request->url() == "/0.css") {
|
||||
this->handle_css_request(request);
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
#pragma once
|
||||
|
||||
#include "list_entities.h"
|
||||
#include "states.h"
|
||||
|
||||
#include "esphome/components/web_server_base/web_server_base.h"
|
||||
#ifdef USE_WEBSERVER
|
||||
|
@ -148,6 +149,9 @@ class WebServer : public Controller, public Component, public AsyncWebHandler {
|
|||
void handle_pna_cors_request(AsyncWebServerRequest *request);
|
||||
#endif
|
||||
|
||||
/// Handle a states request under '/states'.
|
||||
void handle_states_request(AsyncWebServerRequest *request);
|
||||
|
||||
#ifdef USE_SENSOR
|
||||
void on_sensor_update(sensor::Sensor *obj, float state) override;
|
||||
/// Handle a sensor request under '/sensor/<id>'.
|
||||
|
|
Loading…
Reference in a new issue