12 #include "StreamString.h" 29 #ifdef USE_WEBSERVER_LOCAL 30 #if USE_WEBSERVER_VERSION == 2 32 #elif USE_WEBSERVER_VERSION == 3 38 namespace web_server {
40 static const char *
const TAG =
"web_server";
42 #ifdef USE_WEBSERVER_PRIVATE_NETWORK_ACCESS 43 static const char *
const HEADER_PNA_NAME =
"Private-Network-Access-Name";
44 static const char *
const HEADER_PNA_ID =
"Private-Network-Access-ID";
45 static const char *
const HEADER_CORS_REQ_PNA =
"Access-Control-Request-Private-Network";
46 static const char *
const HEADER_CORS_ALLOW_PNA =
"Access-Control-Allow-Private-Network";
52 size_t domain_end = url.find(
'/', 1);
53 if (domain_end == std::string::npos)
55 match.
domain = url.substr(1, domain_end - 1);
60 if (url.length() == domain_end - 1)
62 size_t id_begin = domain_end + 1;
63 size_t id_end = url.find(
'/', id_begin);
65 if (id_end == std::string::npos) {
66 match.
id = url.substr(id_begin, url.length() - id_begin);
69 match.
id = url.substr(id_begin, id_end - id_begin);
70 size_t method_begin = id_end + 1;
71 match.
method = url.substr(method_begin, url.length() - method_begin);
81 [&item](
const DeferredEvent &test) ->
bool {
return test == item; });
93 std::string message = de.message_generator_(
web_server_, de.source_);
94 if (this->try_send(message.c_str(),
"state")) {
116 if (source ==
nullptr)
118 if (event_type ==
nullptr)
120 if (message_generator ==
nullptr)
123 if (0 != strcmp(event_type,
"state_detail_all") && 0 != strcmp(event_type,
"state")) {
124 ESP_LOGE(TAG,
"Can't defer non-state event");
133 std::string message = message_generator(
web_server_, source);
134 if (!this->try_send(message.c_str(),
"state")) {
142 uint32_t reconnect) {
143 this->send(message, event,
id, reconnect);
160 uint32_t reconnect) {
171 ws->
defer([
this, ws, es]() { this->on_client_connect_(ws, es); });
178 es->handleRequest(request);
189 root[
"name"] = group.second.name;
190 root[
"sorting_weight"] = group.second.weight;
209 this->
remove(source);
220 #ifdef USE_WEBSERVER_CSS_INCLUDE 223 #ifdef USE_WEBSERVER_JS_INCLUDE 238 ESP_LOGCONFIG(TAG,
"Setting up web server...");
246 [
this](
int level,
const char *tag,
const char *message) {
267 std::function<void()> fn;
285 ESP_LOGCONFIG(TAG,
"Web Server:");
290 #ifdef USE_WEBSERVER_LOCAL 292 AsyncWebServerResponse *response = request->beginResponse_P(200,
"text/html", INDEX_GZ,
sizeof(INDEX_GZ));
293 response->addHeader(
"Content-Encoding",
"gzip");
294 request->send(response);
296 #elif USE_WEBSERVER_VERSION >= 2 298 AsyncWebServerResponse *response =
301 request->send(response);
305 #ifdef USE_WEBSERVER_PRIVATE_NETWORK_ACCESS 307 AsyncWebServerResponse *response = request->beginResponse(200,
"");
308 response->addHeader(HEADER_CORS_ALLOW_PNA,
"true");
309 response->addHeader(HEADER_PNA_NAME,
App.
get_name().c_str());
311 response->addHeader(HEADER_PNA_ID, mac.c_str());
312 request->send(response);
316 #ifdef USE_WEBSERVER_CSS_INCLUDE 318 AsyncWebServerResponse *response =
320 response->addHeader(
"Content-Encoding",
"gzip");
321 request->send(response);
325 #ifdef USE_WEBSERVER_JS_INCLUDE 327 AsyncWebServerResponse *response =
329 response->addHeader(
"Content-Encoding",
"gzip");
330 request->send(response);
334 #define set_json_id(root, obj, sensor, start_config) \ 335 (root)["id"] = sensor; \ 336 if (((start_config) == DETAIL_ALL)) { \ 337 (root)["name"] = (obj)->get_name(); \ 338 (root)["icon"] = (obj)->get_icon(); \ 339 (root)["entity_category"] = (obj)->get_entity_category(); \ 340 if ((obj)->is_disabled_by_default()) \ 341 (root)["is_disabled_by_default"] = (obj)->is_disabled_by_default(); \ 344 #define set_json_value(root, obj, sensor, value, start_config) \ 345 set_json_id((root), (obj), sensor, start_config); \ 346 (root)["value"] = value; 348 #define set_json_icon_state_value(root, obj, sensor, state, value, start_config) \ 349 set_json_value(root, obj, sensor, value, start_config); \ 350 (root)["state"] = state; 360 if (obj->get_object_id() != match.
id)
362 if (request->method() == HTTP_GET && match.
method.empty()) {
364 auto *param = request->getParam(
"detail");
365 if (param && param->value() ==
"all") {
368 std::string data = this->
sensor_json(obj, obj->state, detail);
369 request->send(200,
"application/json", data.c_str());
384 if (std::isnan(value)) {
391 set_json_icon_state_value(root, obj,
"sensor-" + obj->
get_object_id(),
state, value, start_config);
406 #ifdef USE_TEXT_SENSOR 414 if (obj->get_object_id() != match.
id)
416 if (request->method() == HTTP_GET && match.
method.empty()) {
418 auto *param = request->getParam(
"detail");
419 if (param && param->value() ==
"all") {
423 request->send(200,
"application/json", data.c_str());
440 set_json_icon_state_value(root, obj,
"text_sensor-" + obj->
get_object_id(), value, value, start_config);
461 if (obj->get_object_id() != match.
id)
464 if (request->method() == HTTP_GET && match.
method.empty()) {
466 auto *param = request->getParam(
"detail");
467 if (param && param->value() ==
"all") {
470 std::string data = this->
switch_json(obj, obj->state, detail);
471 request->send(200,
"application/json", data.c_str());
472 }
else if (match.
method ==
"toggle") {
473 this->
schedule_([obj]() { obj->toggle(); });
475 }
else if (match.
method ==
"turn_on") {
476 this->
schedule_([obj]() { obj->turn_on(); });
478 }
else if (match.
method ==
"turn_off") {
479 this->
schedule_([obj]() { obj->turn_off(); });
496 set_json_icon_state_value(root, obj,
"switch-" + obj->
get_object_id(), value ?
"ON" :
"OFF", value, start_config);
513 if (obj->get_object_id() != match.
id)
515 if (request->method() == HTTP_GET && match.
method.empty()) {
517 auto *param = request->getParam(
"detail");
518 if (param && param->value() ==
"all") {
522 request->send(200,
"application/json", data.c_str());
523 }
else if (match.
method ==
"press") {
524 this->
schedule_([obj]() { obj->press(); });
542 set_json_id(root, obj,
"button-" + obj->
get_object_id(), start_config);
555 #ifdef USE_BINARY_SENSOR 563 if (obj->get_object_id() != match.
id)
565 if (request->method() == HTTP_GET && match.
method.empty()) {
567 auto *param = request->getParam(
"detail");
568 if (param && param->value() ==
"all") {
572 request->send(200,
"application/json", data.c_str());
588 set_json_icon_state_value(root, obj,
"binary_sensor-" + obj->
get_object_id(), value ?
"ON" :
"OFF", value,
610 if (obj->get_object_id() != match.
id)
613 if (request->method() == HTTP_GET && match.
method.empty()) {
615 auto *param = request->getParam(
"detail");
616 if (param && param->value() ==
"all") {
619 std::string data = this->
fan_json(obj, detail);
620 request->send(200,
"application/json", data.c_str());
621 }
else if (match.
method ==
"toggle") {
622 this->
schedule_([obj]() { obj->toggle().perform(); });
624 }
else if (match.
method ==
"turn_on" || match.
method ==
"turn_off") {
625 auto call = match.
method ==
"turn_on" ? obj->turn_on() : obj->turn_off();
627 if (request->hasParam(
"speed_level")) {
628 auto speed_level = request->getParam(
"speed_level")->value();
629 auto val = parse_number<int>(speed_level.c_str());
630 if (!
val.has_value()) {
631 ESP_LOGW(TAG,
"Can't convert '%s' to number!", speed_level.c_str());
636 if (request->hasParam(
"oscillation")) {
637 auto speed = request->getParam(
"oscillation")->value();
641 call.set_oscillating(
true);
644 call.set_oscillating(
false);
647 call.set_oscillating(!obj->oscillating);
674 if (traits.supports_speed()) {
675 root[
"speed_level"] = obj->
speed;
676 root[
"speed_count"] = traits.supported_speed_count();
700 if (obj->get_object_id() != match.
id)
703 if (request->method() == HTTP_GET && match.
method.empty()) {
705 auto *param = request->getParam(
"detail");
706 if (param && param->value() ==
"all") {
709 std::string data = this->
light_json(obj, detail);
710 request->send(200,
"application/json", data.c_str());
711 }
else if (match.
method ==
"toggle") {
712 this->
schedule_([obj]() { obj->toggle().perform(); });
714 }
else if (match.
method ==
"turn_on") {
715 auto call = obj->turn_on();
716 if (request->hasParam(
"brightness")) {
717 auto brightness = parse_number<float>(request->getParam(
"brightness")->value().c_str());
718 if (brightness.has_value()) {
719 call.set_brightness(*brightness / 255.0f);
722 if (request->hasParam(
"r")) {
723 auto r = parse_number<float>(request->getParam(
"r")->value().c_str());
725 call.set_red(*r / 255.0f);
728 if (request->hasParam(
"g")) {
729 auto g = parse_number<float>(request->getParam(
"g")->value().c_str());
731 call.set_green(*g / 255.0f);
734 if (request->hasParam(
"b")) {
735 auto b = parse_number<float>(request->getParam(
"b")->value().c_str());
737 call.set_blue(*b / 255.0f);
740 if (request->hasParam(
"white_value")) {
741 auto white_value = parse_number<float>(request->getParam(
"white_value")->value().c_str());
742 if (white_value.has_value()) {
743 call.set_white(*white_value / 255.0f);
746 if (request->hasParam(
"color_temp")) {
747 auto color_temp = parse_number<float>(request->getParam(
"color_temp")->value().c_str());
748 if (color_temp.has_value()) {
749 call.set_color_temperature(*color_temp);
752 if (request->hasParam(
"flash")) {
753 auto flash = parse_number<uint32_t>(request->getParam(
"flash")->value().c_str());
754 if (flash.has_value()) {
755 call.set_flash_length(*flash * 1000);
758 if (request->hasParam(
"transition")) {
759 auto transition = parse_number<uint32_t>(request->getParam(
"transition")->value().c_str());
760 if (transition.has_value()) {
761 call.set_transition_length(*transition * 1000);
764 if (request->hasParam(
"effect")) {
765 const char *effect = request->getParam(
"effect")->value().c_str();
766 call.set_effect(effect);
771 }
else if (match.
method ==
"turn_off") {
772 auto call = obj->turn_off();
773 if (request->hasParam(
"transition")) {
774 auto transition = parse_number<uint32_t>(request->getParam(
"transition")->value().c_str());
775 if (transition.has_value()) {
776 call.set_transition_length(*transition * 1000);
796 set_json_id(root, obj,
"light-" + obj->
get_object_id(), start_config);
801 JsonArray opt = root.createNestedArray(
"effects");
804 opt.add(option->get_name());
825 if (obj->get_object_id() != match.
id)
828 if (request->method() == HTTP_GET && match.
method.empty()) {
830 auto *param = request->getParam(
"detail");
831 if (param && param->value() ==
"all") {
834 std::string data = this->
cover_json(obj, detail);
835 request->send(200,
"application/json", data.c_str());
839 auto call = obj->make_call();
840 if (match.
method ==
"open") {
841 call.set_command_open();
842 }
else if (match.
method ==
"close") {
843 call.set_command_close();
844 }
else if (match.
method ==
"stop") {
845 call.set_command_stop();
846 }
else if (match.
method ==
"toggle") {
847 call.set_command_toggle();
848 }
else if (match.
method !=
"set") {
853 auto traits = obj->get_traits();
854 if ((request->hasParam(
"position") && !traits.get_supports_position()) ||
855 (request->hasParam(
"tilt") && !traits.get_supports_tilt())) {
860 if (request->hasParam(
"position")) {
861 auto position = parse_number<float>(request->getParam(
"position")->value().c_str());
866 if (request->hasParam(
"tilt")) {
867 auto tilt = parse_number<float>(request->getParam(
"tilt")->value().c_str());
868 if (
tilt.has_value()) {
894 root[
"tilt"] = obj->
tilt;
915 if (obj->get_object_id() != match.
id)
918 if (request->method() == HTTP_GET && match.
method.empty()) {
920 auto *param = request->getParam(
"detail");
921 if (param && param->value() ==
"all") {
924 std::string data = this->
number_json(obj, obj->state, detail);
925 request->send(200,
"application/json", data.c_str());
928 if (match.
method !=
"set") {
933 auto call = obj->make_call();
934 if (request->hasParam(
"value")) {
935 auto value = parse_number<float>(request->getParam(
"value")->value().c_str());
936 if (value.has_value())
937 call.set_value(*value);
955 set_json_id(root, obj,
"number-" + obj->
get_object_id(), start_config);
973 if (std::isnan(value)) {
974 root[
"value"] =
"\"NaN\"";
975 root[
"state"] =
"NA";
981 root[
"state"] =
state;
987 #ifdef USE_DATETIME_DATE 995 if (obj->get_object_id() != match.
id)
997 if (request->method() == HTTP_GET && match.
method.empty()) {
999 auto *param = request->getParam(
"detail");
1000 if (param && param->value() ==
"all") {
1003 std::string data = this->
date_json(obj, detail);
1004 request->send(200,
"application/json", data.c_str());
1007 if (match.
method !=
"set") {
1012 auto call = obj->make_call();
1014 if (!request->hasParam(
"value")) {
1019 if (request->hasParam(
"value")) {
1020 std::string value = request->getParam(
"value")->value().c_str();
1021 call.set_date(value);
1039 set_json_id(root, obj,
"date-" + obj->
get_object_id(), start_config);
1041 root[
"value"] = value;
1042 root[
"state"] = value;
1053 #endif // USE_DATETIME_DATE 1055 #ifdef USE_DATETIME_TIME 1063 if (obj->get_object_id() != match.
id)
1065 if (request->method() == HTTP_GET && match.
method.empty()) {
1067 auto *param = request->getParam(
"detail");
1068 if (param && param->value() ==
"all") {
1071 std::string data = this->
time_json(obj, detail);
1072 request->send(200,
"application/json", data.c_str());
1075 if (match.
method !=
"set") {
1080 auto call = obj->make_call();
1082 if (!request->hasParam(
"value")) {
1087 if (request->hasParam(
"value")) {
1088 std::string value = request->getParam(
"value")->value().c_str();
1089 call.set_time(value);
1106 set_json_id(root, obj,
"time-" + obj->
get_object_id(), start_config);
1108 root[
"value"] = value;
1109 root[
"state"] = value;
1120 #endif // USE_DATETIME_TIME 1122 #ifdef USE_DATETIME_DATETIME 1130 if (obj->get_object_id() != match.
id)
1132 if (request->method() == HTTP_GET && match.
method.empty()) {
1134 auto *param = request->getParam(
"detail");
1135 if (param && param->value() ==
"all") {
1139 request->send(200,
"application/json", data.c_str());
1142 if (match.
method !=
"set") {
1147 auto call = obj->make_call();
1149 if (!request->hasParam(
"value")) {
1154 if (request->hasParam(
"value")) {
1155 std::string value = request->getParam(
"value")->value().c_str();
1156 call.set_datetime(value);
1173 set_json_id(root, obj,
"datetime-" + obj->
get_object_id(), start_config);
1176 root[
"value"] = value;
1177 root[
"state"] = value;
1188 #endif // USE_DATETIME_DATETIME 1198 if (obj->get_object_id() != match.
id)
1201 if (request->method() == HTTP_GET && match.
method.empty()) {
1203 auto *param = request->getParam(
"detail");
1204 if (param && param->value() ==
"all") {
1207 std::string data = this->
text_json(obj, obj->state, detail);
1208 request->send(200,
"application/json", data.c_str());
1211 if (match.
method !=
"set") {
1216 auto call = obj->make_call();
1217 if (request->hasParam(
"value")) {
1218 String value = request->getParam(
"value")->value();
1219 call.set_value(value.c_str());
1236 return json::build_json([
this, obj, value, start_config](JsonObject root) {
1237 set_json_id(root, obj,
"text-" + obj->
get_object_id(), start_config);
1242 root[
"state"] =
"********";
1244 root[
"state"] = value;
1246 root[
"value"] = value;
1268 if (obj->get_object_id() != match.
id)
1271 if (request->method() == HTTP_GET && match.
method.empty()) {
1273 auto *param = request->getParam(
"detail");
1274 if (param && param->value() ==
"all") {
1277 std::string data = this->
select_json(obj, obj->state, detail);
1278 request->send(200,
"application/json", data.c_str());
1282 if (match.
method !=
"set") {
1287 auto call = obj->make_call();
1289 if (request->hasParam(
"option")) {
1290 auto option = request->getParam(
"option")->value();
1291 call.set_option(option.c_str());
1307 return json::build_json([
this, obj, value, start_config](JsonObject root) {
1308 set_json_icon_state_value(root, obj,
"select-" + obj->
get_object_id(), value, value, start_config);
1310 JsonArray opt = root.createNestedArray(
"option");
1326 #define PSTR_LOCAL(mode_s) strncpy_P(buf, (PGM_P) ((mode_s)), 15) 1336 if (obj->get_object_id() != match.
id)
1339 if (request->method() == HTTP_GET && match.
method.empty()) {
1341 auto *param = request->getParam(
"detail");
1342 if (param && param->value() ==
"all") {
1346 request->send(200,
"application/json", data.c_str());
1350 if (match.
method !=
"set") {
1355 auto call = obj->make_call();
1357 if (request->hasParam(
"mode")) {
1358 auto mode = request->getParam(
"mode")->value();
1362 if (request->hasParam(
"fan_mode")) {
1363 auto mode = request->getParam(
"fan_mode")->value();
1367 if (request->hasParam(
"swing_mode")) {
1368 auto mode = request->getParam(
"swing_mode")->value();
1372 if (request->hasParam(
"target_temperature_high")) {
1373 auto target_temperature_high = parse_number<float>(request->getParam(
"target_temperature_high")->value().c_str());
1378 if (request->hasParam(
"target_temperature_low")) {
1379 auto target_temperature_low = parse_number<float>(request->getParam(
"target_temperature_low")->value().c_str());
1384 if (request->hasParam(
"target_temperature")) {
1385 auto target_temperature = parse_number<float>(request->getParam(
"target_temperature")->value().c_str());
1404 set_json_id(root, obj,
"climate-" + obj->
get_object_id(), start_config);
1407 int8_t current_accuracy = traits.get_current_temperature_accuracy_decimals();
1411 JsonArray opt = root.createNestedArray(
"modes");
1414 if (!traits.get_supported_custom_fan_modes().empty()) {
1415 JsonArray opt = root.createNestedArray(
"fan_modes");
1420 if (!traits.get_supported_custom_fan_modes().empty()) {
1421 JsonArray opt = root.createNestedArray(
"custom_fan_modes");
1422 for (
auto const &
custom_fan_mode : traits.get_supported_custom_fan_modes())
1425 if (traits.get_supports_swing_modes()) {
1426 JsonArray opt = root.createNestedArray(
"swing_modes");
1427 for (
auto swing_mode : traits.get_supported_swing_modes())
1431 JsonArray opt = root.createNestedArray(
"presets");
1436 JsonArray opt = root.createNestedArray(
"custom_presets");
1437 for (
auto const &
custom_preset : traits.get_supported_custom_presets())
1448 bool has_state =
false;
1452 root[
"step"] = traits.get_visual_target_temperature_step();
1453 if (traits.get_supports_action()) {
1455 root[
"state"] = root[
"action"];
1470 if (traits.get_supports_swing_modes()) {
1473 if (traits.get_supports_current_temperature()) {
1477 root[
"current_temperature"] =
"NA";
1480 if (traits.get_supports_two_point_target_temperature()) {
1490 root[
"state"] = root[
"target_temperature"];
1504 if (obj->get_object_id() != match.
id)
1507 if (request->method() == HTTP_GET && match.
method.empty()) {
1509 auto *param = request->getParam(
"detail");
1510 if (param && param->value() ==
"all") {
1513 std::string data = this->
lock_json(obj, obj->state, detail);
1514 request->send(200,
"application/json", data.c_str());
1515 }
else if (match.
method ==
"lock") {
1516 this->
schedule_([obj]() { obj->lock(); });
1518 }
else if (match.
method ==
"unlock") {
1519 this->
schedule_([obj]() { obj->unlock(); });
1521 }
else if (match.
method ==
"open") {
1522 this->
schedule_([obj]() { obj->open(); });
1538 return json::build_json([
this, obj, value, start_config](JsonObject root) {
1561 if (obj->get_object_id() != match.
id)
1564 if (request->method() == HTTP_GET && match.
method.empty()) {
1566 auto *param = request->getParam(
"detail");
1567 if (param && param->value() ==
"all") {
1570 std::string data = this->
valve_json(obj, detail);
1571 request->send(200,
"application/json", data.c_str());
1575 auto call = obj->make_call();
1576 if (match.
method ==
"open") {
1577 call.set_command_open();
1578 }
else if (match.
method ==
"close") {
1579 call.set_command_close();
1580 }
else if (match.
method ==
"stop") {
1581 call.set_command_stop();
1582 }
else if (match.
method ==
"toggle") {
1583 call.set_command_toggle();
1584 }
else if (match.
method !=
"set") {
1589 auto traits = obj->get_traits();
1590 if (request->hasParam(
"position") && !traits.get_supports_position()) {
1595 if (request->hasParam(
"position")) {
1596 auto position = parse_number<float>(request->getParam(
"position")->value().c_str());
1634 #ifdef USE_ALARM_CONTROL_PANEL 1642 if (obj->get_object_id() != match.
id)
1645 if (request->method() == HTTP_GET && match.
method.empty()) {
1647 auto *param = request->getParam(
"detail");
1648 if (param && param->value() ==
"all") {
1652 request->send(200,
"application/json", data.c_str());
1656 auto call = obj->make_call();
1657 if (request->hasParam(
"code")) {
1658 call.set_code(request->getParam(
"code")->value().c_str());
1661 if (match.
method ==
"disarm") {
1663 }
else if (match.
method ==
"arm_away") {
1665 }
else if (match.
method ==
"arm_home") {
1667 }
else if (match.
method ==
"arm_night") {
1669 }
else if (match.
method ==
"arm_vacation") {
1670 call.arm_vacation();
1695 return json::build_json([
this, obj, value, start_config](JsonObject root) {
1697 set_json_icon_state_value(root, obj,
"alarm-control-panel-" + obj->
get_object_id(),
1718 if (obj->get_object_id() != match.
id)
1721 if (request->method() == HTTP_GET && match.
method.empty()) {
1723 auto *param = request->getParam(
"detail");
1724 if (param && param->value() ==
"all") {
1727 std::string data = this->
event_json(obj,
"", detail);
1728 request->send(200,
"application/json", data.c_str());
1743 return json::build_json([
this, obj, event_type, start_config](JsonObject root) {
1744 set_json_id(root, obj,
"event-" + obj->
get_object_id(), start_config);
1745 if (!event_type.empty()) {
1746 root[
"event_type"] = event_type;
1749 JsonArray event_types = root.createNestedArray(
"event_types");
1751 event_types.add(event_type);
1773 if (obj->get_object_id() != match.
id)
1776 if (request->method() == HTTP_GET && match.
method.empty()) {
1778 auto *param = request->getParam(
"detail");
1779 if (param && param->value() ==
"all") {
1782 std::string data = this->
update_json(obj, detail);
1783 request->send(200,
"application/json", data.c_str());
1787 if (match.
method !=
"install") {
1792 this->
schedule_([obj]()
mutable { obj->perform(); });
1806 set_json_id(root, obj,
"update-" + obj->
get_object_id(), start_config);
1808 switch (obj->
state) {
1810 root[
"state"] =
"NO UPDATE";
1813 root[
"state"] =
"UPDATE AVAILABLE";
1816 root[
"state"] =
"INSTALLING";
1819 root[
"state"] =
"UNKNOWN";
1839 if (request->url() ==
"/")
1843 if (request->url() ==
"/events") {
1848 #ifdef USE_WEBSERVER_CSS_INCLUDE 1849 if (request->url() ==
"/0.css")
1853 #ifdef USE_WEBSERVER_JS_INCLUDE 1854 if (request->url() ==
"/0.js")
1858 #ifdef USE_WEBSERVER_PRIVATE_NETWORK_ACCESS 1859 if (request->method() == HTTP_OPTIONS && request->hasHeader(HEADER_CORS_REQ_PNA)) {
1864 request->addInterestingHeader(HEADER_CORS_REQ_PNA);
1874 if (request->method() == HTTP_GET && match.
domain ==
"sensor")
1879 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"switch")
1884 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"button")
1888 #ifdef USE_BINARY_SENSOR 1889 if (request->method() == HTTP_GET && match.
domain ==
"binary_sensor")
1894 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"fan")
1899 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"light")
1903 #ifdef USE_TEXT_SENSOR 1904 if (request->method() == HTTP_GET && match.
domain ==
"text_sensor")
1909 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"cover")
1914 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"number")
1918 #ifdef USE_DATETIME_DATE 1919 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"date")
1923 #ifdef USE_DATETIME_TIME 1924 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"time")
1928 #ifdef USE_DATETIME_DATETIME 1929 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"datetime")
1934 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"text")
1939 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"select")
1944 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"climate")
1949 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"lock")
1954 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"valve")
1958 #ifdef USE_ALARM_CONTROL_PANEL 1959 if ((request->method() == HTTP_GET || request->method() == HTTP_POST) && match.
domain ==
"alarm_control_panel")
1964 if (request->method() == HTTP_GET && match.
domain ==
"event")
1969 if ((request->method() == HTTP_POST || request->method() == HTTP_GET) && match.
domain ==
"update")
1976 if (request->url() ==
"/") {
1982 if (request->url() ==
"/events") {
1988 #ifdef USE_WEBSERVER_CSS_INCLUDE 1989 if (request->url() ==
"/0.css") {
1995 #ifdef USE_WEBSERVER_JS_INCLUDE 1996 if (request->url() ==
"/0.js") {
2002 #ifdef USE_WEBSERVER_PRIVATE_NETWORK_ACCESS 2003 if (request->method() == HTTP_OPTIONS && request->hasHeader(HEADER_CORS_REQ_PNA)) {
2011 if (match.
domain ==
"sensor") {
2018 if (match.
domain ==
"switch") {
2025 if (match.
domain ==
"button") {
2031 #ifdef USE_BINARY_SENSOR 2032 if (match.
domain ==
"binary_sensor") {
2039 if (match.
domain ==
"fan") {
2046 if (match.
domain ==
"light") {
2052 #ifdef USE_TEXT_SENSOR 2053 if (match.
domain ==
"text_sensor") {
2060 if (match.
domain ==
"cover") {
2067 if (match.
domain ==
"number") {
2073 #ifdef USE_DATETIME_DATE 2074 if (match.
domain ==
"date") {
2080 #ifdef USE_DATETIME_TIME 2081 if (match.
domain ==
"time") {
2087 #ifdef USE_DATETIME_DATETIME 2088 if (match.
domain ==
"datetime") {
2095 if (match.
domain ==
"text") {
2102 if (match.
domain ==
"select") {
2109 if (match.
domain ==
"climate") {
2116 if (match.
domain ==
"lock") {
2124 if (match.
domain ==
"valve") {
2130 #ifdef USE_ALARM_CONTROL_PANEL 2131 if (match.
domain ==
"alarm_control_panel") {
2139 if (match.
domain ==
"update") {
2162 this->
defer(std::move(f));
Base class for all switches.
value_type const & value() const
bool state
The current on/off state of the fan.
const size_t ESPHOME_WEBSERVER_CSS_INCLUDE_SIZE
ClimateSwingMode swing_mode
The active swing mode of the climate device.
float target_temperature_low
const std::vector< datetime::DateTimeEntity * > & get_datetimes()
static std::string date_all_json_generator(WebServer *web_server, void *source)
void handle_pna_cors_request(AsyncWebServerRequest *request)
void handle_number_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a number request under '/number/<id>'.
This class represents the communication layer between the front-end MQTT layer and the hardware outpu...
void add_new_client(WebServer *ws, AsyncWebServerRequest *request)
bool oscillating
The current oscillation state of the fan.
void set_interval(const std::string &name, uint32_t interval, std::function< void()> &&f)
Set an interval function with a unique name.
std::string number_json(number::Number *obj, float value, JsonDetail start_config)
Dump the number state with its value as a JSON string.
std::string sensor_json(sensor::Sensor *obj, float value, JsonDetail start_config)
Dump the sensor state with its value as a JSON string.
void add_on_log_callback(std::function< void(int, const char *, const char *)> &&callback)
Register a callback that will be called for every log message sent.
void on_sensor_update(sensor::Sensor *obj, float state) override
bool is_on() const
Get the binary true/false state of these light color values.
Base class for all cover devices.
std::string value_accuracy_to_string(float value, int8_t accuracy_decimals)
Create a string from a value and an accuracy in decimals.
WebServer(web_server_base::WebServerBase *base)
void handleRequest(AsyncWebServerRequest *request) override
Override the web handler's handleRequest method.
TextMode get_mode() const
ClimatePreset
Enum for all preset modes.
void handle_time_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a time request under '/time/<id>'.
static std::string binary_sensor_all_json_generator(WebServer *web_server, void *source)
const std::vector< climate::Climate * > & get_climates()
static std::string number_state_json_generator(WebServer *web_server, void *source)
void deq_push_back_with_dedup_(void *source, message_generator_t *message_generator)
float target_temperature
The target temperature of the climate device.
SemaphoreHandle_t to_schedule_lock_
This class allows users to create a web server with their ESP nodes.
std::vector< DeferredEvent > deferred_queue_
std::string get_use_address()
Get the active network hostname.
void handle_binary_sensor_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a binary sensor request under '/binary_sensor/<id>'.
std::string select_json(select::Select *obj, const std::string &value, JsonDetail start_config)
Dump the select state with its value as a JSON string.
std::string get_device_class()
Get the device class, using the manual override if set.
const std::vector< update::UpdateEntity * > & get_updates()
const std::vector< alarm_control_panel::AlarmControlPanel * > & get_alarm_control_panels()
bool is_fully_closed() const
Helper method to check if the valve is fully closed. Equivalent to comparing .position against 0...
static std::string select_state_json_generator(WebServer *web_server, void *source)
const char * lock_state_to_string(LockState state)
void handle_select_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a select request under '/select/<id>'.
const std::vector< valve::Valve * > & get_valves()
static std::string binary_sensor_state_json_generator(WebServer *web_server, void *source)
void handle_text_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a text input request under '/text/<id>'.
CoverOperation current_operation
The current operation of the cover (idle, opening, closing).
std::map< EntityBase *, SortingComponents > sorting_entitys_
void deferrable_send_state(void *source, const char *event_type, message_generator_t *message_generator)
float position
The position of the valve from 0.0 (fully closed) to 1.0 (fully open).
void handle_event_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a event request under '/event<id>'.
static std::string alarm_control_panel_state_json_generator(WebServer *web_server, void *source)
DeferredEvent(void *source, message_generator_t *message_generator)
void handle_update_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a update request under '/update/<id>'.
const LogString * climate_mode_to_string(ClimateMode mode)
Convert the given ClimateMode to a human-readable string.
virtual FanTraits get_traits()=0
std::set< std::string > get_event_types() const
void handle_valve_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a valve request under '/valve/<id>/<open/close/stop/set>'.
static std::string select_all_json_generator(WebServer *web_server, void *source)
const UpdateState & state
DeferredUpdateEventSource(WebServer *ws, const String &url)
void try_send_nodefer(const char *message, const char *event=nullptr, uint32_t id=0, uint32_t reconnect=0)
void defer(const std::string &name, std::function< void()> &&f)
Defer a callback to the next loop() call.
const std::vector< event::Event * > & get_events()
bool get_supports_position() const
ClimateMode mode
The active mode of the climate device.
void on_lock_update(lock::Lock *obj) override
static std::string update_all_json_generator(WebServer *web_server, void *source)
static std::string text_sensor_all_json_generator(WebServer *web_server, void *source)
static std::string text_all_json_generator(WebServer *web_server, void *source)
virtual bool assumed_state()
Return whether this switch uses an assumed state - i.e.
std::string latest_version
static std::string alarm_control_panel_all_json_generator(WebServer *web_server, void *source)
std::string current_version
const std::string & get_friendly_name() const
Get the friendly name of this Application set by pre_setup().
void setup() override
Setup the internal web server and register handlers.
float target_temperature_high
The maximum target temperature of the climate device, for climate devices with split target temperatu...
float current_temperature
The current temperature of the climate device, as reported from the integration.
const std::vector< fan::Fan * > & get_fans()
static std::string datetime_all_json_generator(WebServer *web_server, void *source)
void on_binary_sensor_update(binary_sensor::BinarySensor *obj, bool state) override
void handle_light_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a light request under '/light/<id>/</turn_on/turn_off/toggle>'.
bool isRequestHandlerTrivial() override
This web handle is not trivial.
void handle_lock_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a lock request under '/lock/<id>/</lock/unlock/open>'.
static std::string text_state_json_generator(WebServer *web_server, void *source)
static std::string fan_all_json_generator(WebServer *web_server, void *source)
float target_temperature_high
void handle_button_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a button request under '/button/<id>/press'.
int get_max_length() const
Base-class for all text inputs.
void deferrable_send_state(void *source, const char *event_type, message_generator_t *message_generator)
bool supports_oscillation() const
Return if this fan supports oscillation.
void on_light_update(light::LightState *obj) override
virtual ValveTraits get_traits()=0
void on_client_disconnect_(DeferredUpdateEventSource *source)
void on_event(event::Event *obj, const std::string &event_type) override
float tilt
The current tilt value of the cover from 0.0 to 1.0.
const std::vector< datetime::TimeEntity * > & get_times()
std::string get_object_id() const
uint32_t IRAM_ATTR HOT millis()
std::string event_json(event::Event *obj, const std::string &event_type, JsonDetail start_config)
Dump the event details with its value as a JSON string.
ParseOnOffState parse_on_off(const char *str, const char *on, const char *off)
Parse a string that contains either on, off or toggle.
ClimateSwingMode swing_mode
void try_send_nodefer(const char *message, const char *event=nullptr, uint32_t id=0, uint32_t reconnect=0)
const size_t ESPHOME_WEBSERVER_JS_INCLUDE_SIZE
Internal helper struct that is used to parse incoming URLs.
static std::string lock_state_json_generator(WebServer *web_server, void *source)
optional< std::string > custom_fan_mode
The active custom fan mode of the climate device.
virtual CoverTraits get_traits()=0
static std::string event_state_json_generator(WebServer *web_server, void *source)
void handle_sensor_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a sensor request under '/sensor/<id>'.
std::map< uint64_t, SortingGroup > sorting_groups_
const std::vector< lock::Lock * > & get_locks()
uint16_t get_port() const
static std::string switch_state_json_generator(WebServer *web_server, void *source)
void dump_config() override
std::string text_sensor_json(text_sensor::TextSensor *obj, const std::string &value, JsonDetail start_config)
Dump the text sensor state with its value as a JSON string.
static std::string time_all_json_generator(WebServer *web_server, void *source)
const size_t ESPHOME_WEBSERVER_INDEX_HTML_SIZE
std::string domain
The domain of the component, for example "sensor".
std::string text_json(text::Text *obj, const std::string &value, JsonDetail start_config)
Dump the text state with its value as a JSON string.
std::string update_json(update::UpdateEntity *obj, JsonDetail start_config)
Dump the update state with its value as a JSON string.
void on_text_sensor_update(text_sensor::TextSensor *obj, const std::string &state) override
void add_handler(AsyncWebHandler *handler)
DeferredUpdateEventSourceList events_
void set_css_include(const char *css_include)
Set local path to the script that's embedded in the index page.
void on_client_connect_(WebServer *ws, DeferredUpdateEventSource *source)
void on_text_update(text::Text *obj, const std::string &state) override
static std::string switch_all_json_generator(WebServer *web_server, void *source)
const std::vector< button::Button * > & get_buttons()
void handle_switch_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a switch request under '/switch/<id>/</turn_on/turn_off/toggle>'.
const LogString * alarm_control_panel_state_to_string(AlarmControlPanelState state)
Returns a string representation of the state.
static std::string valve_all_json_generator(WebServer *web_server, void *source)
std::vector< std::string > get_options() const
static std::string cover_all_json_generator(WebServer *web_server, void *source)
optional< ClimatePreset > preset
The active preset of the climate device.
const UpdateInfo & update_info
static std::string button_state_json_generator(WebServer *web_server, void *source)
static std::string event_all_json_generator(WebServer *web_server, void *source)
UrlMatch match_url(const std::string &url, bool only_domain=false)
const std::vector< switch_::Switch * > & get_switches()
static std::string date_state_json_generator(WebServer *web_server, void *source)
Base-class for all numbers.
std::string str_sprintf(const char *fmt,...)
const char * cover_operation_to_str(CoverOperation op)
int speed
The current fan speed level.
void handle_alarm_control_panel_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a alarm_control_panel request under '/alarm_control_panel/<id>'.
void handle_css_request(AsyncWebServerRequest *request)
Handle included css request under '/0.css'.
static std::string time_state_json_generator(WebServer *web_server, void *source)
bool valid
Whether this match is valid.
static std::string climate_all_json_generator(WebServer *web_server, void *source)
BedjetMode mode
BedJet operating mode.
bool is_fully_closed() const
Helper method to check if the cover is fully closed. Equivalent to comparing .position against 0...
void on_select_update(select::Select *obj, const std::string &state, size_t index) override
ClimateTraits get_traits()
Get the traits of this climate device with all overrides applied.
std::string get_unit_of_measurement()
Get the unit of measurement, using the manual override if set.
std::string time_json(datetime::TimeEntity *obj, JsonDetail start_config)
Dump the time state with its value as a JSON string.
const std::vector< text_sensor::TextSensor * > & get_text_sensors()
const LogString * climate_preset_to_string(ClimatePreset preset)
Convert the given PresetMode to a human-readable string.
int8_t get_target_temperature_accuracy_decimals() const
const std::vector< sensor::Sensor * > & get_sensors()
Application App
Global storage of Application pointer - only one Application can exist.
static std::string button_all_json_generator(WebServer *web_server, void *source)
const std::vector< binary_sensor::BinarySensor * > & get_binary_sensors()
const std::vector< LightEffect * > & get_effects() const
Get all effects for this light state.
static std::string valve_state_json_generator(WebServer *web_server, void *source)
void handle_datetime_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a datetime request under '/datetime/<id>'.
static std::string update_state_json_generator(WebServer *web_server, void *source)
bool get_supports_position() const
int8_t step_to_accuracy_decimals(float step)
Derive accuracy in decimals from an increment step.
std::string switch_json(switch_::Switch *obj, bool value, JsonDetail start_config)
Dump the switch state with its value as a JSON string.
std::string build_json(const json_build_t &f)
Build a JSON string with the provided json build function.
void begin(bool include_internal=false)
static std::string number_all_json_generator(WebServer *web_server, void *source)
static std::string fan_state_json_generator(WebServer *web_server, void *source)
const std::string & get_name() const
Get the name of this Application set by pre_setup().
std::string light_json(light::LightState *obj, JsonDetail start_config)
Dump the light state as a JSON string.
void add_sorting_group(uint64_t group_id, const std::string &group_name, float weight)
void on_valve_update(valve::Valve *obj) override
static void dump_json(LightState &state, JsonObject root)
Dump the state of a light as JSON.
void add_entity_config(EntityBase *entity, float weight, uint64_t group)
const std::vector< text::Text * > & get_texts()
ClimateMode
Enum for all modes a climate device can be in.
void handle_index_request(AsyncWebServerRequest *request)
Handle an index request under '/'.
std::string valve_json(valve::Valve *obj, JsonDetail start_config)
Dump the valve state as a JSON string.
void on_time_update(datetime::TimeEntity *obj) override
void on_climate_update(climate::Climate *obj) override
const std::vector< cover::Cover * > & get_covers()
float get_setup_priority() const override
MQTT setup priority.
optional< std::string > custom_preset
The active custom preset mode of the climate device.
static std::string cover_state_json_generator(WebServer *web_server, void *source)
const LogString * climate_fan_mode_to_string(ClimateFanMode fan_mode)
Convert the given ClimateFanMode to a human-readable string.
optional< ClimateFanMode > fan_mode
The active fan mode of the climate device.
float position
The position of the cover from 0.0 (fully closed) to 1.0 (fully open).
void handle_fan_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a fan request under '/fan/<id>/</turn_on/turn_off/toggle>'.
std::string id
The id of the device that's being accessed, for example "living_room_fan".
const std::vector< light::LightState * > & get_lights()
void on_date_update(datetime::DateEntity *obj) override
ListEntitiesIterator entities_iterator_
std::string get_comment() const
Get the comment of this Application set by pre_setup().
std::string button_json(button::Button *obj, JsonDetail start_config)
Dump the button details with its value as a JSON string.
void on_cover_update(cover::Cover *obj) override
const char * valve_operation_to_str(ValveOperation op)
void handle_cover_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a cover request under '/cover/<id>/<open/close/stop/set>'.
void on_switch_update(switch_::Switch *obj, bool state) override
const char * css_include_
static std::string light_all_json_generator(WebServer *web_server, void *source)
bool get_supports_tilt() const
void on_alarm_control_panel_update(alarm_control_panel::AlarmControlPanel *obj) override
void setup_controller(bool include_internal=false)
static std::string lock_all_json_generator(WebServer *web_server, void *source)
void on_datetime_update(datetime::DateTimeEntity *obj) override
std::string(WebServer *, void *) message_generator_t
std::string date_json(datetime::DateEntity *obj, JsonDetail start_config)
Dump the date state with its value as a JSON string.
static std::string datetime_state_json_generator(WebServer *web_server, void *source)
std::string get_config_json()
Return the webserver configuration as JSON.
std::string datetime_json(datetime::DateTimeEntity *obj, JsonDetail start_config)
Dump the datetime state with its value as a JSON string.
Base-class for all selects.
void on_fan_update(fan::Fan *obj) override
web_server_base::WebServerBase * base_
Implementation of SPI Controller mode.
static std::string sensor_all_json_generator(WebServer *web_server, void *source)
float get_min_value() const
void on_number_update(number::Number *obj, float state) override
Base class for all valve devices.
std::string fan_json(fan::Fan *obj, JsonDetail start_config)
Dump the fan state as a JSON string.
ValveOperation current_operation
The current operation of the valve (idle, opening, closing).
Base class for all binary_sensor-type classes.
LightColorValues remote_values
The remote color values reported to the frontend.
static std::string text_sensor_state_json_generator(WebServer *web_server, void *source)
LockState
Enum for all states a lock can be in.
static std::string light_state_json_generator(WebServer *web_server, void *source)
NumberMode get_mode() const
void handle_climate_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a climate request under '/climate/<id>'.
int8_t get_accuracy_decimals()
Get the accuracy in decimals, using the manual override if set.
const std::vector< datetime::DateEntity * > & get_dates()
int get_min_length() const
const std::vector< select::Select * > & get_selects()
Base-class for all sensors.
std::string get_mac_address_pretty()
Get the device MAC address as a string, in colon-separated uppercase hex notation.
bool canHandle(AsyncWebServerRequest *request) override
Override the web handler's canHandle method.
AsyncEventSourceResponse AsyncEventSourceClient
std::string alarm_control_panel_json(alarm_control_panel::AlarmControlPanel *obj, alarm_control_panel::AlarmControlPanelState value, JsonDetail start_config)
Dump the alarm_control_panel state with its value as a JSON string.
std::deque< std::function< void()> > to_schedule_
const std::vector< number::Number * > & get_numbers()
float get_max_value() const
const LogString * climate_action_to_string(ClimateAction action)
Convert the given ClimateAction to a human-readable string.
void on_update(update::UpdateEntity *obj) override
void handle_text_sensor_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a text sensor request under '/text_sensor/<id>'.
void schedule_(std::function< void()> &&f)
std::string lock_json(lock::Lock *obj, lock::LockState value, JsonDetail start_config)
Dump the lock state with its value as a JSON string.
std::string get_pattern() const
float target_temperature_low
The minimum target temperature of the climate device, for climate devices with split target temperatu...
std::string climate_json(climate::Climate *obj, JsonDetail start_config)
Dump the climate details.
std::string method
The method that's being called, for example "turn_on".
void handle_date_request(AsyncWebServerRequest *request, const UrlMatch &match)
Handle a date request under '/date/<id>'.
Base class for all locks.
ClimateAction action
The active state of the climate device.
ClimateDevice - This is the base class for all climate integrations.
std::string binary_sensor_json(binary_sensor::BinarySensor *obj, bool value, JsonDetail start_config)
Dump the binary sensor state with its value as a JSON string.
std::string cover_json(cover::Cover *obj, JsonDetail start_config)
Dump the cover state as a JSON string.
void handle_js_request(AsyncWebServerRequest *request)
Handle included js request under '/0.js'.
static std::string sensor_state_json_generator(WebServer *web_server, void *source)
void set_js_include(const char *js_include)
Set local path to the script that's embedded in the index page.
static std::string climate_state_json_generator(WebServer *web_server, void *source)
void process_deferred_queue_()
const LogString * climate_swing_mode_to_string(ClimateSwingMode swing_mode)
Convert the given ClimateSwingMode to a human-readable string.