Optionally show internal components on the web server (#2627)

Co-authored-by: Jesse Hills <3060199+jesserockz@users.noreply.github.com>
Co-authored-by: Oxan van Leeuwen <oxan@oxanvanleeuwen.nl>
This commit is contained in:
mechanarchy 2021-11-30 02:52:20 +11:00 committed by GitHub
parent adf48246a9
commit 6f07421911
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 67 additions and 78 deletions

View File

@ -12,6 +12,7 @@ from esphome.const import (
CONF_AUTH, CONF_AUTH,
CONF_USERNAME, CONF_USERNAME,
CONF_PASSWORD, CONF_PASSWORD,
CONF_INCLUDE_INTERNAL,
CONF_OTA, CONF_OTA,
) )
from esphome.core import CORE, coroutine_with_priority from esphome.core import CORE, coroutine_with_priority
@ -42,6 +43,7 @@ CONFIG_SCHEMA = cv.Schema(
cv.GenerateID(CONF_WEB_SERVER_BASE_ID): cv.use_id( cv.GenerateID(CONF_WEB_SERVER_BASE_ID): cv.use_id(
web_server_base.WebServerBase web_server_base.WebServerBase
), ),
cv.Optional(CONF_INCLUDE_INTERNAL, default=False): cv.boolean,
cv.Optional(CONF_OTA, default=True): cv.boolean, cv.Optional(CONF_OTA, default=True): cv.boolean,
} }
).extend(cv.COMPONENT_SCHEMA) ).extend(cv.COMPONENT_SCHEMA)
@ -75,3 +77,4 @@ async def to_code(config):
path = CORE.relative_config_path(config[CONF_JS_INCLUDE]) path = CORE.relative_config_path(config[CONF_JS_INCLUDE])
with open(file=path, mode="r", encoding="utf-8") as myfile: with open(file=path, mode="r", encoding="utf-8") as myfile:
cg.add(var.set_js_include(myfile.read())) cg.add(var.set_js_include(myfile.read()))
cg.add(var.set_include_internal(config[CONF_INCLUDE_INTERNAL]))

View File

@ -31,10 +31,10 @@ static const char *const TAG = "web_server";
void write_row(AsyncResponseStream *stream, EntityBase *obj, const std::string &klass, const std::string &action, void write_row(AsyncResponseStream *stream, EntityBase *obj, const std::string &klass, const std::string &action,
const std::function<void(AsyncResponseStream &stream, EntityBase *obj)> &action_func = nullptr) { const std::function<void(AsyncResponseStream &stream, EntityBase *obj)> &action_func = nullptr) {
if (obj->is_internal())
return;
stream->print("<tr class=\""); stream->print("<tr class=\"");
stream->print(klass.c_str()); stream->print(klass.c_str());
if (obj->is_internal())
stream->print(" internal");
stream->print("\" id=\""); stream->print("\" id=\"");
stream->print(klass.c_str()); stream->print(klass.c_str());
stream->print("-"); stream->print("-");
@ -83,7 +83,7 @@ void WebServer::set_js_include(const char *js_include) { this->js_include_ = js_
void WebServer::setup() { void WebServer::setup() {
ESP_LOGCONFIG(TAG, "Setting up web server..."); ESP_LOGCONFIG(TAG, "Setting up web server...");
this->setup_controller(); this->setup_controller(this->include_internal_);
this->base_->init(); this->base_->init();
this->events_.onConnect([this](AsyncEventSourceClient *client) { this->events_.onConnect([this](AsyncEventSourceClient *client) {
@ -92,55 +92,55 @@ void WebServer::setup() {
#ifdef USE_SENSOR #ifdef USE_SENSOR
for (auto *obj : App.get_sensors()) for (auto *obj : App.get_sensors())
if (!obj->is_internal()) if (this->include_internal_ || !obj->is_internal())
client->send(this->sensor_json(obj, obj->state).c_str(), "state"); client->send(this->sensor_json(obj, obj->state).c_str(), "state");
#endif #endif
#ifdef USE_SWITCH #ifdef USE_SWITCH
for (auto *obj : App.get_switches()) for (auto *obj : App.get_switches())
if (!obj->is_internal()) if (this->include_internal_ || !obj->is_internal())
client->send(this->switch_json(obj, obj->state).c_str(), "state"); client->send(this->switch_json(obj, obj->state).c_str(), "state");
#endif #endif
#ifdef USE_BINARY_SENSOR #ifdef USE_BINARY_SENSOR
for (auto *obj : App.get_binary_sensors()) for (auto *obj : App.get_binary_sensors())
if (!obj->is_internal()) if (this->include_internal_ || !obj->is_internal())
client->send(this->binary_sensor_json(obj, obj->state).c_str(), "state"); client->send(this->binary_sensor_json(obj, obj->state).c_str(), "state");
#endif #endif
#ifdef USE_FAN #ifdef USE_FAN
for (auto *obj : App.get_fans()) for (auto *obj : App.get_fans())
if (!obj->is_internal()) if (this->include_internal_ || !obj->is_internal())
client->send(this->fan_json(obj).c_str(), "state"); client->send(this->fan_json(obj).c_str(), "state");
#endif #endif
#ifdef USE_LIGHT #ifdef USE_LIGHT
for (auto *obj : App.get_lights()) for (auto *obj : App.get_lights())
if (!obj->is_internal()) if (this->include_internal_ || !obj->is_internal())
client->send(this->light_json(obj).c_str(), "state"); client->send(this->light_json(obj).c_str(), "state");
#endif #endif
#ifdef USE_TEXT_SENSOR #ifdef USE_TEXT_SENSOR
for (auto *obj : App.get_text_sensors()) for (auto *obj : App.get_text_sensors())
if (!obj->is_internal()) if (this->include_internal_ || !obj->is_internal())
client->send(this->text_sensor_json(obj, obj->state).c_str(), "state"); client->send(this->text_sensor_json(obj, obj->state).c_str(), "state");
#endif #endif
#ifdef USE_COVER #ifdef USE_COVER
for (auto *obj : App.get_covers()) for (auto *obj : App.get_covers())
if (!obj->is_internal()) if (this->include_internal_ || !obj->is_internal())
client->send(this->cover_json(obj).c_str(), "state"); client->send(this->cover_json(obj).c_str(), "state");
#endif #endif
#ifdef USE_NUMBER #ifdef USE_NUMBER
for (auto *obj : App.get_numbers()) for (auto *obj : App.get_numbers())
if (!obj->is_internal()) if (this->include_internal_ || !obj->is_internal())
client->send(this->number_json(obj, obj->state).c_str(), "state"); client->send(this->number_json(obj, obj->state).c_str(), "state");
#endif #endif
#ifdef USE_SELECT #ifdef USE_SELECT
for (auto *obj : App.get_selects()) for (auto *obj : App.get_selects())
if (!obj->is_internal()) if (this->include_internal_ || !obj->is_internal())
client->send(this->select_json(obj, obj->state).c_str(), "state"); client->send(this->select_json(obj, obj->state).c_str(), "state");
#endif #endif
}); });
@ -188,46 +188,55 @@ void WebServer::handle_index_request(AsyncWebServerRequest *request) {
#ifdef USE_SENSOR #ifdef USE_SENSOR
for (auto *obj : App.get_sensors()) for (auto *obj : App.get_sensors())
if (this->include_internal_ || !obj->is_internal())
write_row(stream, obj, "sensor", ""); write_row(stream, obj, "sensor", "");
#endif #endif
#ifdef USE_SWITCH #ifdef USE_SWITCH
for (auto *obj : App.get_switches()) for (auto *obj : App.get_switches())
if (this->include_internal_ || !obj->is_internal())
write_row(stream, obj, "switch", "<button>Toggle</button>"); write_row(stream, obj, "switch", "<button>Toggle</button>");
#endif #endif
#ifdef USE_BINARY_SENSOR #ifdef USE_BINARY_SENSOR
for (auto *obj : App.get_binary_sensors()) for (auto *obj : App.get_binary_sensors())
if (this->include_internal_ || !obj->is_internal())
write_row(stream, obj, "binary_sensor", ""); write_row(stream, obj, "binary_sensor", "");
#endif #endif
#ifdef USE_FAN #ifdef USE_FAN
for (auto *obj : App.get_fans()) for (auto *obj : App.get_fans())
if (this->include_internal_ || !obj->is_internal())
write_row(stream, obj, "fan", "<button>Toggle</button>"); write_row(stream, obj, "fan", "<button>Toggle</button>");
#endif #endif
#ifdef USE_LIGHT #ifdef USE_LIGHT
for (auto *obj : App.get_lights()) for (auto *obj : App.get_lights())
if (this->include_internal_ || !obj->is_internal())
write_row(stream, obj, "light", "<button>Toggle</button>"); write_row(stream, obj, "light", "<button>Toggle</button>");
#endif #endif
#ifdef USE_TEXT_SENSOR #ifdef USE_TEXT_SENSOR
for (auto *obj : App.get_text_sensors()) for (auto *obj : App.get_text_sensors())
if (this->include_internal_ || !obj->is_internal())
write_row(stream, obj, "text_sensor", ""); write_row(stream, obj, "text_sensor", "");
#endif #endif
#ifdef USE_COVER #ifdef USE_COVER
for (auto *obj : App.get_covers()) for (auto *obj : App.get_covers())
if (this->include_internal_ || !obj->is_internal())
write_row(stream, obj, "cover", "<button>Open</button><button>Close</button>"); write_row(stream, obj, "cover", "<button>Open</button><button>Close</button>");
#endif #endif
#ifdef USE_NUMBER #ifdef USE_NUMBER
for (auto *obj : App.get_numbers()) for (auto *obj : App.get_numbers())
if (this->include_internal_ || !obj->is_internal())
write_row(stream, obj, "number", ""); write_row(stream, obj, "number", "");
#endif #endif
#ifdef USE_SELECT #ifdef USE_SELECT
for (auto *obj : App.get_selects()) for (auto *obj : App.get_selects())
if (this->include_internal_ || !obj->is_internal())
write_row(stream, obj, "select", "", [](AsyncResponseStream &stream, EntityBase *obj) { write_row(stream, obj, "select", "", [](AsyncResponseStream &stream, EntityBase *obj) {
select::Select *select = (select::Select *) obj; select::Select *select = (select::Select *) obj;
stream.print("<select>"); stream.print("<select>");
@ -293,8 +302,6 @@ void WebServer::on_sensor_update(sensor::Sensor *obj, float state) {
} }
void WebServer::handle_sensor_request(AsyncWebServerRequest *request, const UrlMatch &match) { void WebServer::handle_sensor_request(AsyncWebServerRequest *request, const UrlMatch &match) {
for (sensor::Sensor *obj : App.get_sensors()) { for (sensor::Sensor *obj : App.get_sensors()) {
if (obj->is_internal())
continue;
if (obj->get_object_id() != match.id) if (obj->get_object_id() != match.id)
continue; continue;
std::string data = this->sensor_json(obj, obj->state); std::string data = this->sensor_json(obj, obj->state);
@ -321,8 +328,6 @@ void WebServer::on_text_sensor_update(text_sensor::TextSensor *obj, const std::s
} }
void WebServer::handle_text_sensor_request(AsyncWebServerRequest *request, const UrlMatch &match) { void WebServer::handle_text_sensor_request(AsyncWebServerRequest *request, const UrlMatch &match) {
for (text_sensor::TextSensor *obj : App.get_text_sensors()) { for (text_sensor::TextSensor *obj : App.get_text_sensors()) {
if (obj->is_internal())
continue;
if (obj->get_object_id() != match.id) if (obj->get_object_id() != match.id)
continue; continue;
std::string data = this->text_sensor_json(obj, obj->state); std::string data = this->text_sensor_json(obj, obj->state);
@ -353,8 +358,6 @@ std::string WebServer::switch_json(switch_::Switch *obj, bool value) {
} }
void WebServer::handle_switch_request(AsyncWebServerRequest *request, const UrlMatch &match) { void WebServer::handle_switch_request(AsyncWebServerRequest *request, const UrlMatch &match) {
for (switch_::Switch *obj : App.get_switches()) { for (switch_::Switch *obj : App.get_switches()) {
if (obj->is_internal())
continue;
if (obj->get_object_id() != match.id) if (obj->get_object_id() != match.id)
continue; continue;
@ -381,8 +384,6 @@ void WebServer::handle_switch_request(AsyncWebServerRequest *request, const UrlM
#ifdef USE_BINARY_SENSOR #ifdef USE_BINARY_SENSOR
void WebServer::on_binary_sensor_update(binary_sensor::BinarySensor *obj, bool state) { void WebServer::on_binary_sensor_update(binary_sensor::BinarySensor *obj, bool state) {
if (obj->is_internal())
return;
this->events_.send(this->binary_sensor_json(obj, state).c_str(), "state"); this->events_.send(this->binary_sensor_json(obj, state).c_str(), "state");
} }
std::string WebServer::binary_sensor_json(binary_sensor::BinarySensor *obj, bool value) { std::string WebServer::binary_sensor_json(binary_sensor::BinarySensor *obj, bool value) {
@ -394,8 +395,6 @@ std::string WebServer::binary_sensor_json(binary_sensor::BinarySensor *obj, bool
} }
void WebServer::handle_binary_sensor_request(AsyncWebServerRequest *request, const UrlMatch &match) { void WebServer::handle_binary_sensor_request(AsyncWebServerRequest *request, const UrlMatch &match) {
for (binary_sensor::BinarySensor *obj : App.get_binary_sensors()) { for (binary_sensor::BinarySensor *obj : App.get_binary_sensors()) {
if (obj->is_internal())
continue;
if (obj->get_object_id() != match.id) if (obj->get_object_id() != match.id)
continue; continue;
std::string data = this->binary_sensor_json(obj, obj->state); std::string data = this->binary_sensor_json(obj, obj->state);
@ -407,11 +406,7 @@ void WebServer::handle_binary_sensor_request(AsyncWebServerRequest *request, con
#endif #endif
#ifdef USE_FAN #ifdef USE_FAN
void WebServer::on_fan_update(fan::FanState *obj) { void WebServer::on_fan_update(fan::FanState *obj) { this->events_.send(this->fan_json(obj).c_str(), "state"); }
if (obj->is_internal())
return;
this->events_.send(this->fan_json(obj).c_str(), "state");
}
std::string WebServer::fan_json(fan::FanState *obj) { std::string WebServer::fan_json(fan::FanState *obj) {
return json::build_json([obj](JsonObject &root) { return json::build_json([obj](JsonObject &root) {
root["id"] = "fan-" + obj->get_object_id(); root["id"] = "fan-" + obj->get_object_id();
@ -442,8 +437,6 @@ std::string WebServer::fan_json(fan::FanState *obj) {
} }
void WebServer::handle_fan_request(AsyncWebServerRequest *request, const UrlMatch &match) { void WebServer::handle_fan_request(AsyncWebServerRequest *request, const UrlMatch &match) {
for (fan::FanState *obj : App.get_fans()) { for (fan::FanState *obj : App.get_fans()) {
if (obj->is_internal())
continue;
if (obj->get_object_id() != match.id) if (obj->get_object_id() != match.id)
continue; continue;
@ -504,15 +497,9 @@ void WebServer::handle_fan_request(AsyncWebServerRequest *request, const UrlMatc
#endif #endif
#ifdef USE_LIGHT #ifdef USE_LIGHT
void WebServer::on_light_update(light::LightState *obj) { void WebServer::on_light_update(light::LightState *obj) { this->events_.send(this->light_json(obj).c_str(), "state"); }
if (obj->is_internal())
return;
this->events_.send(this->light_json(obj).c_str(), "state");
}
void WebServer::handle_light_request(AsyncWebServerRequest *request, const UrlMatch &match) { void WebServer::handle_light_request(AsyncWebServerRequest *request, const UrlMatch &match) {
for (light::LightState *obj : App.get_lights()) { for (light::LightState *obj : App.get_lights()) {
if (obj->is_internal())
continue;
if (obj->get_object_id() != match.id) if (obj->get_object_id() != match.id)
continue; continue;
@ -579,15 +566,9 @@ std::string WebServer::light_json(light::LightState *obj) {
#endif #endif
#ifdef USE_COVER #ifdef USE_COVER
void WebServer::on_cover_update(cover::Cover *obj) { void WebServer::on_cover_update(cover::Cover *obj) { this->events_.send(this->cover_json(obj).c_str(), "state"); }
if (obj->is_internal())
return;
this->events_.send(this->cover_json(obj).c_str(), "state");
}
void WebServer::handle_cover_request(AsyncWebServerRequest *request, const UrlMatch &match) { void WebServer::handle_cover_request(AsyncWebServerRequest *request, const UrlMatch &match) {
for (cover::Cover *obj : App.get_covers()) { for (cover::Cover *obj : App.get_covers()) {
if (obj->is_internal())
continue;
if (obj->get_object_id() != match.id) if (obj->get_object_id() != match.id)
continue; continue;
@ -646,8 +627,6 @@ void WebServer::on_number_update(number::Number *obj, float state) {
} }
void WebServer::handle_number_request(AsyncWebServerRequest *request, const UrlMatch &match) { void WebServer::handle_number_request(AsyncWebServerRequest *request, const UrlMatch &match) {
for (auto *obj : App.get_numbers()) { for (auto *obj : App.get_numbers()) {
if (obj->is_internal())
continue;
if (obj->get_object_id() != match.id) if (obj->get_object_id() != match.id)
continue; continue;
std::string data = this->number_json(obj, obj->state); std::string data = this->number_json(obj, obj->state);
@ -673,8 +652,6 @@ void WebServer::on_select_update(select::Select *obj, const std::string &state)
} }
void WebServer::handle_select_request(AsyncWebServerRequest *request, const UrlMatch &match) { void WebServer::handle_select_request(AsyncWebServerRequest *request, const UrlMatch &match) {
for (auto *obj : App.get_selects()) { for (auto *obj : App.get_selects()) {
if (obj->is_internal())
continue;
if (obj->get_object_id() != match.id) if (obj->get_object_id() != match.id)
continue; continue;

View File

@ -58,6 +58,12 @@ class WebServer : public Controller, public Component, public AsyncWebHandler {
*/ */
void set_js_include(const char *js_include); void set_js_include(const char *js_include);
/** Determine whether internal components should be displayed on the web server.
* Defaults to false.
*
* @param include_internal Whether internal components should be displayed.
*/
void set_include_internal(bool include_internal) { include_internal_ = include_internal; }
/** Set whether or not the webserver should expose the OTA form and handler. /** Set whether or not the webserver should expose the OTA form and handler.
* *
* @param allow_ota. * @param allow_ota.
@ -188,6 +194,7 @@ class WebServer : public Controller, public Component, public AsyncWebHandler {
const char *css_include_{nullptr}; const char *css_include_{nullptr};
const char *js_url_{nullptr}; const char *js_url_{nullptr};
const char *js_include_{nullptr}; const char *js_include_{nullptr};
bool include_internal_{false};
bool allow_ota_{true}; bool allow_ota_{true};
}; };

View File

@ -286,6 +286,7 @@ CONF_ILLUMINANCE = "illuminance"
CONF_IMPEDANCE = "impedance" CONF_IMPEDANCE = "impedance"
CONF_IMPORT_ACTIVE_ENERGY = "import_active_energy" CONF_IMPORT_ACTIVE_ENERGY = "import_active_energy"
CONF_IMPORT_REACTIVE_ENERGY = "import_reactive_energy" CONF_IMPORT_REACTIVE_ENERGY = "import_reactive_energy"
CONF_INCLUDE_INTERNAL = "include_internal"
CONF_INCLUDES = "includes" CONF_INCLUDES = "includes"
CONF_INDEX = "index" CONF_INDEX = "index"
CONF_INDOOR = "indoor" CONF_INDOOR = "indoor"

View File

@ -4,64 +4,64 @@
namespace esphome { namespace esphome {
void Controller::setup_controller() { void Controller::setup_controller(bool include_internal) {
#ifdef USE_BINARY_SENSOR #ifdef USE_BINARY_SENSOR
for (auto *obj : App.get_binary_sensors()) { for (auto *obj : App.get_binary_sensors()) {
if (!obj->is_internal()) if (include_internal || !obj->is_internal())
obj->add_on_state_callback([this, obj](bool state) { this->on_binary_sensor_update(obj, state); }); obj->add_on_state_callback([this, obj](bool state) { this->on_binary_sensor_update(obj, state); });
} }
#endif #endif
#ifdef USE_FAN #ifdef USE_FAN
for (auto *obj : App.get_fans()) { for (auto *obj : App.get_fans()) {
if (!obj->is_internal()) if (include_internal || !obj->is_internal())
obj->add_on_state_callback([this, obj]() { this->on_fan_update(obj); }); obj->add_on_state_callback([this, obj]() { this->on_fan_update(obj); });
} }
#endif #endif
#ifdef USE_LIGHT #ifdef USE_LIGHT
for (auto *obj : App.get_lights()) { for (auto *obj : App.get_lights()) {
if (!obj->is_internal()) if (include_internal || !obj->is_internal())
obj->add_new_remote_values_callback([this, obj]() { this->on_light_update(obj); }); obj->add_new_remote_values_callback([this, obj]() { this->on_light_update(obj); });
} }
#endif #endif
#ifdef USE_SENSOR #ifdef USE_SENSOR
for (auto *obj : App.get_sensors()) { for (auto *obj : App.get_sensors()) {
if (!obj->is_internal()) if (include_internal || !obj->is_internal())
obj->add_on_state_callback([this, obj](float state) { this->on_sensor_update(obj, state); }); obj->add_on_state_callback([this, obj](float state) { this->on_sensor_update(obj, state); });
} }
#endif #endif
#ifdef USE_SWITCH #ifdef USE_SWITCH
for (auto *obj : App.get_switches()) { for (auto *obj : App.get_switches()) {
if (!obj->is_internal()) if (include_internal || !obj->is_internal())
obj->add_on_state_callback([this, obj](bool state) { this->on_switch_update(obj, state); }); obj->add_on_state_callback([this, obj](bool state) { this->on_switch_update(obj, state); });
} }
#endif #endif
#ifdef USE_COVER #ifdef USE_COVER
for (auto *obj : App.get_covers()) { for (auto *obj : App.get_covers()) {
if (!obj->is_internal()) if (include_internal || !obj->is_internal())
obj->add_on_state_callback([this, obj]() { this->on_cover_update(obj); }); obj->add_on_state_callback([this, obj]() { this->on_cover_update(obj); });
} }
#endif #endif
#ifdef USE_TEXT_SENSOR #ifdef USE_TEXT_SENSOR
for (auto *obj : App.get_text_sensors()) { for (auto *obj : App.get_text_sensors()) {
if (!obj->is_internal()) if (include_internal || !obj->is_internal())
obj->add_on_state_callback([this, obj](const std::string &state) { this->on_text_sensor_update(obj, state); }); obj->add_on_state_callback([this, obj](const std::string &state) { this->on_text_sensor_update(obj, state); });
} }
#endif #endif
#ifdef USE_CLIMATE #ifdef USE_CLIMATE
for (auto *obj : App.get_climates()) { for (auto *obj : App.get_climates()) {
if (!obj->is_internal()) if (include_internal || !obj->is_internal())
obj->add_on_state_callback([this, obj]() { this->on_climate_update(obj); }); obj->add_on_state_callback([this, obj]() { this->on_climate_update(obj); });
} }
#endif #endif
#ifdef USE_NUMBER #ifdef USE_NUMBER
for (auto *obj : App.get_numbers()) { for (auto *obj : App.get_numbers()) {
if (!obj->is_internal()) if (include_internal || !obj->is_internal())
obj->add_on_state_callback([this, obj](float state) { this->on_number_update(obj, state); }); obj->add_on_state_callback([this, obj](float state) { this->on_number_update(obj, state); });
} }
#endif #endif
#ifdef USE_SELECT #ifdef USE_SELECT
for (auto *obj : App.get_selects()) { for (auto *obj : App.get_selects()) {
if (!obj->is_internal()) if (include_internal || !obj->is_internal())
obj->add_on_state_callback([this, obj](const std::string &state) { this->on_select_update(obj, state); }); obj->add_on_state_callback([this, obj](const std::string &state) { this->on_select_update(obj, state); });
} }
#endif #endif

View File

@ -36,7 +36,7 @@ namespace esphome {
class Controller { class Controller {
public: public:
void setup_controller(); void setup_controller(bool include_internal = false);
#ifdef USE_BINARY_SENSOR #ifdef USE_BINARY_SENSOR
virtual void on_binary_sensor_update(binary_sensor::BinarySensor *obj, bool state){}; virtual void on_binary_sensor_update(binary_sensor::BinarySensor *obj, bool state){};
#endif #endif

View File

@ -49,6 +49,7 @@ web_server:
auth: auth:
username: admin username: admin
password: admin password: admin
include_internal: true
time: time:
- platform: sntp - platform: sntp