mirror of
https://github.com/esphome/esphome.git
synced 2024-12-02 11:44:13 +01:00
6682c43dfa
## Description: Move esphome-core codebase into esphome (and a bunch of other refactors). See https://github.com/esphome/feature-requests/issues/97 Yes this is a shit ton of work and no there's no way to automate it :( But it will be worth it 👍 Progress: - Core support (file copy etc): 80% - Base Abstractions (light, switch): ~50% - Integrations: ~10% - Working? Yes, (but only with ported components). Other refactors: - Moves all codegen related stuff into a single class: `esphome.codegen` (imported as `cg`) - Rework coroutine syntax - Move from `component/platform.py` to `domain/component.py` structure as with HA - Move all defaults out of C++ and into config validation. - Remove `make_...` helpers from Application class. Reason: Merge conflicts with every single new integration. - Pointer Variables are stored globally instead of locally in setup(). Reason: stack size limit. Future work: - Rework const.py - Move all `CONF_...` into a conf class (usage `conf.UPDATE_INTERVAL` vs `CONF_UPDATE_INTERVAL`). Reason: Less convoluted import block - Enable loading from `custom_components` folder. **Related issue (if applicable):** https://github.com/esphome/feature-requests/issues/97 **Pull request in [esphome-docs](https://github.com/esphome/esphome-docs) with documentation (if applicable):** esphome/esphome-docs#<esphome-docs PR number goes here> ## Checklist: - [ ] The code change is tested and works locally. - [ ] Tests have been added to verify that the new code works (under `tests/` folder). If user exposed functionality or configuration variables are added/changed: - [ ] Documentation added/updated in [esphomedocs](https://github.com/OttoWinter/esphomedocs).
554 lines
20 KiB
C++
554 lines
20 KiB
C++
#include "mqtt_client.h"
|
|
#include "esphome/core/log.h"
|
|
#include "esphome/core/application.h"
|
|
#include "esphome/core/helpers.h"
|
|
#include "esphome/core/util.h"
|
|
#ifdef USE_LOGGER
|
|
#include "esphome/components/logger/logger.h"
|
|
#endif
|
|
#include "lwip/err.h"
|
|
#include "lwip/dns.h"
|
|
#include "mqtt_component.h"
|
|
|
|
namespace esphome {
|
|
namespace mqtt {
|
|
|
|
static const char *TAG = "mqtt";
|
|
|
|
MQTTClientComponent::MQTTClientComponent() { global_mqtt_client = this; }
|
|
|
|
// Connection
|
|
void MQTTClientComponent::setup() {
|
|
ESP_LOGCONFIG(TAG, "Setting up MQTT...");
|
|
this->mqtt_client_.onMessage([this](char *topic, char *payload, AsyncMqttClientMessageProperties properties,
|
|
size_t len, size_t index, size_t total) {
|
|
std::string payload_s(payload, len);
|
|
std::string topic_s(topic);
|
|
this->on_message(topic_s, payload_s);
|
|
});
|
|
this->mqtt_client_.onDisconnect([this](AsyncMqttClientDisconnectReason reason) {
|
|
this->state_ = MQTT_CLIENT_DISCONNECTED;
|
|
this->disconnect_reason_ = reason;
|
|
});
|
|
#ifdef USE_LOGGER
|
|
if (this->is_log_message_enabled() && logger::global_logger != nullptr) {
|
|
logger::global_logger->add_on_log_callback([this](int level, const char *tag, const char *message) {
|
|
if (level <= this->log_level_ && this->is_connected()) {
|
|
this->publish(this->log_message_.topic, message, strlen(message), this->log_message_.qos,
|
|
this->log_message_.retain);
|
|
}
|
|
});
|
|
}
|
|
#endif
|
|
|
|
this->last_connected_ = millis();
|
|
this->start_dnslookup_();
|
|
}
|
|
void MQTTClientComponent::dump_config() {
|
|
ESP_LOGCONFIG(TAG, "MQTT:");
|
|
ESP_LOGCONFIG(TAG, " Server Address: %s:%u (%s)", this->credentials_.address.c_str(), this->credentials_.port,
|
|
this->ip_.toString().c_str());
|
|
ESP_LOGCONFIG(TAG, " Username: " LOG_SECRET("'%s'"), this->credentials_.username.c_str());
|
|
ESP_LOGCONFIG(TAG, " Client ID: " LOG_SECRET("'%s'"), this->credentials_.client_id.c_str());
|
|
if (!this->discovery_info_.prefix.empty()) {
|
|
ESP_LOGCONFIG(TAG, " Discovery prefix: '%s'", this->discovery_info_.prefix.c_str());
|
|
ESP_LOGCONFIG(TAG, " Discovery retain: %s", YESNO(this->discovery_info_.retain));
|
|
}
|
|
ESP_LOGCONFIG(TAG, " Topic Prefix: '%s'", this->topic_prefix_.c_str());
|
|
if (!this->log_message_.topic.empty()) {
|
|
ESP_LOGCONFIG(TAG, " Log Topic: '%s'", this->log_message_.topic.c_str());
|
|
}
|
|
if (!this->availability_.topic.empty()) {
|
|
ESP_LOGCONFIG(TAG, " Availability: '%s'", this->availability_.topic.c_str());
|
|
}
|
|
}
|
|
bool MQTTClientComponent::can_proceed() { return this->is_connected(); }
|
|
|
|
void MQTTClientComponent::start_dnslookup_() {
|
|
for (auto &subscription : this->subscriptions_) {
|
|
subscription.subscribed = false;
|
|
subscription.resubscribe_timeout = 0;
|
|
}
|
|
|
|
this->status_set_warning();
|
|
this->dns_resolve_error_ = false;
|
|
this->dns_resolved_ = false;
|
|
ip_addr_t addr;
|
|
#ifdef ARDUINO_ARCH_ESP32
|
|
err_t err = dns_gethostbyname_addrtype(this->credentials_.address.c_str(), &addr, this->dns_found_callback, this,
|
|
LWIP_DNS_ADDRTYPE_IPV4);
|
|
#endif
|
|
#ifdef ARDUINO_ARCH_ESP8266
|
|
err_t err = dns_gethostbyname(this->credentials_.address.c_str(), &addr,
|
|
esphome::mqtt::MQTTClientComponent::dns_found_callback, this);
|
|
#endif
|
|
switch (err) {
|
|
case ERR_OK: {
|
|
// Got IP immediately
|
|
this->dns_resolved_ = true;
|
|
#ifdef ARDUINO_ARCH_ESP32
|
|
this->ip_ = IPAddress(addr.u_addr.ip4.addr);
|
|
#endif
|
|
#ifdef ARDUINO_ARCH_ESP8266
|
|
this->ip_ = IPAddress(addr.addr);
|
|
#endif
|
|
this->start_connect_();
|
|
return;
|
|
}
|
|
case ERR_INPROGRESS: {
|
|
// wait for callback
|
|
ESP_LOGD(TAG, "Resolving MQTT broker IP address...");
|
|
break;
|
|
}
|
|
default:
|
|
case ERR_ARG: {
|
|
// error
|
|
#if defined(ARDUINO_ARCH_ESP8266)
|
|
ESP_LOGW(TAG, "Error resolving MQTT broker IP address: %ld", err);
|
|
#else
|
|
ESP_LOGW(TAG, "Error resolving MQTT broker IP address: %d", err);
|
|
#endif
|
|
break;
|
|
}
|
|
}
|
|
|
|
this->state_ = MQTT_CLIENT_RESOLVING_ADDRESS;
|
|
this->connect_begin_ = millis();
|
|
}
|
|
void MQTTClientComponent::check_dnslookup_() {
|
|
if (!this->dns_resolved_ && millis() - this->connect_begin_ > 20000) {
|
|
this->dns_resolve_error_ = true;
|
|
}
|
|
|
|
if (this->dns_resolve_error_) {
|
|
ESP_LOGW(TAG, "Couldn't resolve IP address for '%s'!", this->credentials_.address.c_str());
|
|
this->state_ = MQTT_CLIENT_DISCONNECTED;
|
|
return;
|
|
}
|
|
|
|
if (!this->dns_resolved_) {
|
|
return;
|
|
}
|
|
|
|
ESP_LOGD(TAG, "Resolved broker IP address to %s", this->ip_.toString().c_str());
|
|
this->start_connect_();
|
|
}
|
|
#if defined(ARDUINO_ARCH_ESP8266) && LWIP_VERSION_MAJOR == 1
|
|
void MQTTClientComponent::dns_found_callback(const char *name, ip_addr_t *ipaddr, void *callback_arg) {
|
|
#else
|
|
void MQTTClientComponent::dns_found_callback(const char *name, const ip_addr_t *ipaddr, void *callback_arg) {
|
|
#endif
|
|
auto *a_this = (MQTTClientComponent *) callback_arg;
|
|
if (ipaddr == nullptr) {
|
|
a_this->dns_resolve_error_ = true;
|
|
} else {
|
|
#ifdef ARDUINO_ARCH_ESP32
|
|
a_this->ip_ = IPAddress(ipaddr->u_addr.ip4.addr);
|
|
#endif
|
|
#ifdef ARDUINO_ARCH_ESP8266
|
|
a_this->ip_ = IPAddress(ipaddr->addr);
|
|
#endif
|
|
a_this->dns_resolved_ = true;
|
|
}
|
|
}
|
|
|
|
void MQTTClientComponent::start_connect_() {
|
|
if (!network_is_connected())
|
|
return;
|
|
|
|
ESP_LOGI(TAG, "Connecting to MQTT...");
|
|
// Force disconnect first
|
|
this->mqtt_client_.disconnect(true);
|
|
|
|
this->mqtt_client_.setClientId(this->credentials_.client_id.c_str());
|
|
const char *username = nullptr;
|
|
if (!this->credentials_.username.empty())
|
|
username = this->credentials_.username.c_str();
|
|
const char *password = nullptr;
|
|
if (!this->credentials_.password.empty())
|
|
password = this->credentials_.password.c_str();
|
|
|
|
this->mqtt_client_.setCredentials(username, password);
|
|
|
|
this->mqtt_client_.setServer(this->ip_, this->credentials_.port);
|
|
if (!this->last_will_.topic.empty()) {
|
|
this->mqtt_client_.setWill(this->last_will_.topic.c_str(), this->last_will_.qos, this->last_will_.retain,
|
|
this->last_will_.payload.c_str(), this->last_will_.payload.length());
|
|
}
|
|
|
|
this->mqtt_client_.connect();
|
|
this->state_ = MQTT_CLIENT_CONNECTING;
|
|
this->connect_begin_ = millis();
|
|
}
|
|
bool MQTTClientComponent::is_connected() {
|
|
return this->state_ == MQTT_CLIENT_CONNECTED && this->mqtt_client_.connected();
|
|
}
|
|
|
|
void MQTTClientComponent::check_connected() {
|
|
if (!this->mqtt_client_.connected()) {
|
|
if (millis() - this->connect_begin_ > 60000) {
|
|
this->state_ = MQTT_CLIENT_DISCONNECTED;
|
|
this->start_dnslookup_();
|
|
}
|
|
return;
|
|
}
|
|
|
|
this->state_ = MQTT_CLIENT_CONNECTED;
|
|
this->sent_birth_message_ = false;
|
|
this->status_clear_warning();
|
|
ESP_LOGI(TAG, "MQTT Connected!");
|
|
// MQTT Client needs some time to be fully set up.
|
|
delay(100);
|
|
|
|
this->resubscribe_subscriptions_();
|
|
|
|
for (MQTTComponent *component : this->children_)
|
|
component->schedule_resend_state();
|
|
}
|
|
|
|
void MQTTClientComponent::loop() {
|
|
if (this->disconnect_reason_.has_value()) {
|
|
const char *reason_s = nullptr;
|
|
switch (*this->disconnect_reason_) {
|
|
case AsyncMqttClientDisconnectReason::TCP_DISCONNECTED:
|
|
reason_s = "TCP disconnected";
|
|
break;
|
|
case AsyncMqttClientDisconnectReason::MQTT_UNACCEPTABLE_PROTOCOL_VERSION:
|
|
reason_s = "Unacceptable Protocol Version";
|
|
break;
|
|
case AsyncMqttClientDisconnectReason::MQTT_IDENTIFIER_REJECTED:
|
|
reason_s = "Identifier Rejected";
|
|
break;
|
|
case AsyncMqttClientDisconnectReason::MQTT_SERVER_UNAVAILABLE:
|
|
reason_s = "Server Unavailable";
|
|
break;
|
|
case AsyncMqttClientDisconnectReason::MQTT_MALFORMED_CREDENTIALS:
|
|
reason_s = "Malformed Credentials";
|
|
break;
|
|
case AsyncMqttClientDisconnectReason::MQTT_NOT_AUTHORIZED:
|
|
reason_s = "Not Authorized";
|
|
break;
|
|
case AsyncMqttClientDisconnectReason::ESP8266_NOT_ENOUGH_SPACE:
|
|
reason_s = "Not Enough Space";
|
|
break;
|
|
case AsyncMqttClientDisconnectReason::TLS_BAD_FINGERPRINT:
|
|
reason_s = "TLS Bad Fingerprint";
|
|
break;
|
|
default:
|
|
reason_s = "Unknown";
|
|
break;
|
|
}
|
|
if (!network_is_connected()) {
|
|
reason_s = "WiFi disconnected";
|
|
}
|
|
ESP_LOGW(TAG, "MQTT Disconnected: %s.", reason_s);
|
|
this->disconnect_reason_.reset();
|
|
}
|
|
|
|
const uint32_t now = millis();
|
|
|
|
switch (this->state_) {
|
|
case MQTT_CLIENT_DISCONNECTED:
|
|
if (now - this->connect_begin_ > 5000) {
|
|
this->start_dnslookup_();
|
|
}
|
|
break;
|
|
case MQTT_CLIENT_RESOLVING_ADDRESS:
|
|
this->check_dnslookup_();
|
|
break;
|
|
case MQTT_CLIENT_CONNECTING:
|
|
this->check_connected();
|
|
break;
|
|
case MQTT_CLIENT_CONNECTED:
|
|
if (!this->mqtt_client_.connected()) {
|
|
this->state_ = MQTT_CLIENT_DISCONNECTED;
|
|
ESP_LOGW(TAG, "Lost MQTT Client connection!");
|
|
this->start_dnslookup_();
|
|
} else {
|
|
if (!this->birth_message_.topic.empty() && !this->sent_birth_message_) {
|
|
this->sent_birth_message_ = this->publish(this->birth_message_);
|
|
}
|
|
|
|
this->last_connected_ = now;
|
|
this->resubscribe_subscriptions_();
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (millis() - this->last_connected_ > this->reboot_timeout_ && this->reboot_timeout_ != 0) {
|
|
ESP_LOGE(TAG, "Can't connect to MQTT... Restarting...");
|
|
App.reboot();
|
|
}
|
|
}
|
|
float MQTTClientComponent::get_setup_priority() const { return setup_priority::AFTER_WIFI; }
|
|
|
|
// Subscribe
|
|
bool MQTTClientComponent::subscribe_(const char *topic, uint8_t qos) {
|
|
if (!this->is_connected())
|
|
return false;
|
|
|
|
uint16_t ret = this->mqtt_client_.subscribe(topic, qos);
|
|
yield();
|
|
|
|
if (ret != 0) {
|
|
ESP_LOGV(TAG, "subscribe(topic='%s')", topic);
|
|
} else {
|
|
delay(5);
|
|
ESP_LOGV(TAG, "Subscribe failed for topic='%s'. Will retry later.", topic);
|
|
this->status_momentary_warning("subscribe", 1000);
|
|
}
|
|
return ret != 0;
|
|
}
|
|
void MQTTClientComponent::resubscribe_subscription_(MQTTSubscription *sub) {
|
|
if (sub->subscribed)
|
|
return;
|
|
|
|
const uint32_t now = millis();
|
|
bool do_resub = sub->resubscribe_timeout == 0 || now - sub->resubscribe_timeout > 1000;
|
|
|
|
if (do_resub) {
|
|
sub->subscribed = this->subscribe_(sub->topic.c_str(), sub->qos);
|
|
sub->resubscribe_timeout = now;
|
|
}
|
|
}
|
|
void MQTTClientComponent::resubscribe_subscriptions_() {
|
|
for (auto &subscription : this->subscriptions_) {
|
|
this->resubscribe_subscription_(&subscription);
|
|
}
|
|
}
|
|
|
|
void MQTTClientComponent::subscribe(const std::string &topic, mqtt_callback_t callback, uint8_t qos) {
|
|
MQTTSubscription subscription{
|
|
.topic = topic,
|
|
.qos = qos,
|
|
.callback = std::move(callback),
|
|
.subscribed = false,
|
|
.resubscribe_timeout = 0,
|
|
};
|
|
this->resubscribe_subscription_(&subscription);
|
|
this->subscriptions_.push_back(subscription);
|
|
}
|
|
|
|
void MQTTClientComponent::subscribe_json(const std::string &topic, mqtt_json_callback_t callback, uint8_t qos) {
|
|
auto f = [callback](const std::string &topic, const std::string &payload) {
|
|
json::parse_json(payload, [topic, callback](JsonObject &root) { callback(topic, root); });
|
|
};
|
|
MQTTSubscription subscription{
|
|
.topic = topic,
|
|
.qos = qos,
|
|
.callback = f,
|
|
.subscribed = false,
|
|
.resubscribe_timeout = 0,
|
|
};
|
|
this->resubscribe_subscription_(&subscription);
|
|
this->subscriptions_.push_back(subscription);
|
|
}
|
|
|
|
// Publish
|
|
bool MQTTClientComponent::publish(const std::string &topic, const std::string &payload, uint8_t qos, bool retain) {
|
|
return this->publish(topic, payload.data(), payload.size(), qos, retain);
|
|
}
|
|
|
|
bool MQTTClientComponent::publish(const std::string &topic, const char *payload, size_t payload_length, uint8_t qos,
|
|
bool retain) {
|
|
if (!this->is_connected()) {
|
|
// critical components will re-transmit their messages
|
|
return false;
|
|
}
|
|
bool logging_topic = topic == this->log_message_.topic;
|
|
uint16_t ret = this->mqtt_client_.publish(topic.c_str(), qos, retain, payload, payload_length);
|
|
yield();
|
|
if (ret == 0 && !logging_topic && this->is_connected()) {
|
|
delay(5);
|
|
ret = this->mqtt_client_.publish(topic.c_str(), qos, retain, payload, payload_length);
|
|
yield();
|
|
}
|
|
|
|
if (!logging_topic) {
|
|
if (ret != 0) {
|
|
ESP_LOGV(TAG, "Publish(topic='%s' payload='%s' retain=%d)", topic.c_str(), payload, retain);
|
|
} else {
|
|
ESP_LOGW(TAG, "Publish failed for topic='%s' will retry later..", topic.c_str());
|
|
this->status_momentary_warning("publish", 1000);
|
|
}
|
|
}
|
|
return ret != 0;
|
|
}
|
|
|
|
bool MQTTClientComponent::publish(const MQTTMessage &message) {
|
|
return this->publish(message.topic, message.payload, message.qos, message.retain);
|
|
}
|
|
bool MQTTClientComponent::publish_json(const std::string &topic, const json::json_build_t &f, uint8_t qos,
|
|
bool retain) {
|
|
size_t len;
|
|
const char *message = json::build_json(f, &len);
|
|
return this->publish(topic, message, len, qos, retain);
|
|
}
|
|
|
|
/** Check if the message topic matches the given subscription topic
|
|
*
|
|
* INFO: MQTT spec mandates that topics must not be empty and must be valid NULL-terminated UTF-8 strings.
|
|
*
|
|
* @param message The message topic that was received from the MQTT server. Note: this must not contain
|
|
* wildcard characters as mandated by the MQTT spec.
|
|
* @param subscription The subscription topic we are matching against.
|
|
* @param is_normal Is this a "normal" topic - Does the message topic not begin with a "$".
|
|
* @param past_separator Are we past the first '/' topic separator.
|
|
* @return true if the subscription topic matches the message topic, false otherwise.
|
|
*/
|
|
static bool topic_match(const char *message, const char *subscription, bool is_normal, bool past_separator) {
|
|
// Reached end of both strings at the same time, this means we have a successful match
|
|
if (*message == '\0' && *subscription == '\0')
|
|
return true;
|
|
|
|
// Either the message or the subscribe are at the end. This means they don't match.
|
|
if (*message == '\0' || *subscription == '\0')
|
|
return false;
|
|
|
|
bool do_wildcards = is_normal || past_separator;
|
|
|
|
if (*subscription == '+' && do_wildcards) {
|
|
// single level wildcard
|
|
// consume + from subscription
|
|
subscription++;
|
|
// consume everything from message until '/' found or end of string
|
|
while (*message != '\0' && *message != '/') {
|
|
message++;
|
|
}
|
|
// after this, both pointers will point to a '/' or to the end of the string
|
|
|
|
return topic_match(message, subscription, is_normal, true);
|
|
}
|
|
|
|
if (*subscription == '#' && do_wildcards) {
|
|
// multilevel wildcard - MQTT mandates that this must be at end of subscribe topic
|
|
return true;
|
|
}
|
|
|
|
// this handles '/' and normal characters at the same time.
|
|
if (*message != *subscription)
|
|
return false;
|
|
|
|
past_separator = past_separator || *subscription == '/';
|
|
|
|
// consume characters
|
|
subscription++;
|
|
message++;
|
|
|
|
return topic_match(message, subscription, is_normal, past_separator);
|
|
}
|
|
|
|
static bool topic_match(const char *message, const char *subscription) {
|
|
return topic_match(message, subscription, *message != '\0' && *message != '$', false);
|
|
}
|
|
|
|
void MQTTClientComponent::on_message(const std::string &topic, const std::string &payload) {
|
|
#ifdef ARDUINO_ARCH_ESP8266
|
|
// on ESP8266, this is called in LWiP thread; some components do not like running
|
|
// in an ISR.
|
|
this->defer([this, topic, payload]() {
|
|
#endif
|
|
for (auto &subscription : this->subscriptions_)
|
|
if (topic_match(topic.c_str(), subscription.topic.c_str()))
|
|
subscription.callback(topic, payload);
|
|
#ifdef ARDUINO_ARCH_ESP8266
|
|
});
|
|
#endif
|
|
}
|
|
|
|
// Setters
|
|
void MQTTClientComponent::disable_log_message() { this->log_message_.topic = ""; }
|
|
bool MQTTClientComponent::is_log_message_enabled() const { return !this->log_message_.topic.empty(); }
|
|
void MQTTClientComponent::set_reboot_timeout(uint32_t reboot_timeout) { this->reboot_timeout_ = reboot_timeout; }
|
|
void MQTTClientComponent::register_mqtt_component(MQTTComponent *component) { this->children_.push_back(component); }
|
|
void MQTTClientComponent::set_log_level(int level) { this->log_level_ = level; }
|
|
void MQTTClientComponent::set_keep_alive(uint16_t keep_alive_s) { this->mqtt_client_.setKeepAlive(keep_alive_s); }
|
|
void MQTTClientComponent::set_log_message_template(MQTTMessage &&message) { this->log_message_ = std::move(message); }
|
|
const MQTTDiscoveryInfo &MQTTClientComponent::get_discovery_info() const { return this->discovery_info_; }
|
|
void MQTTClientComponent::set_topic_prefix(std::string topic_prefix) { this->topic_prefix_ = std::move(topic_prefix); }
|
|
const std::string &MQTTClientComponent::get_topic_prefix() const { return this->topic_prefix_; }
|
|
void MQTTClientComponent::disable_birth_message() {
|
|
this->birth_message_.topic = "";
|
|
this->recalculate_availability_();
|
|
}
|
|
void MQTTClientComponent::disable_shutdown_message() {
|
|
this->shutdown_message_.topic = "";
|
|
this->recalculate_availability_();
|
|
}
|
|
bool MQTTClientComponent::is_discovery_enabled() const { return !this->discovery_info_.prefix.empty(); }
|
|
const Availability &MQTTClientComponent::get_availability() { return this->availability_; }
|
|
void MQTTClientComponent::recalculate_availability_() {
|
|
if (this->birth_message_.topic.empty() || this->birth_message_.topic != this->last_will_.topic) {
|
|
this->availability_.topic = "";
|
|
return;
|
|
}
|
|
this->availability_.topic = this->birth_message_.topic;
|
|
this->availability_.payload_available = this->birth_message_.payload;
|
|
this->availability_.payload_not_available = this->last_will_.payload;
|
|
}
|
|
|
|
void MQTTClientComponent::set_last_will(MQTTMessage &&message) {
|
|
this->last_will_ = std::move(message);
|
|
this->recalculate_availability_();
|
|
}
|
|
|
|
void MQTTClientComponent::set_birth_message(MQTTMessage &&message) {
|
|
this->birth_message_ = std::move(message);
|
|
this->recalculate_availability_();
|
|
}
|
|
|
|
void MQTTClientComponent::set_shutdown_message(MQTTMessage &&message) { this->shutdown_message_ = std::move(message); }
|
|
|
|
void MQTTClientComponent::set_discovery_info(std::string &&prefix, bool retain, bool clean) {
|
|
this->discovery_info_.prefix = std::move(prefix);
|
|
this->discovery_info_.retain = retain;
|
|
this->discovery_info_.clean = clean;
|
|
}
|
|
|
|
void MQTTClientComponent::disable_last_will() { this->last_will_.topic = ""; }
|
|
|
|
void MQTTClientComponent::disable_discovery() {
|
|
this->discovery_info_ = MQTTDiscoveryInfo{.prefix = "", .retain = false};
|
|
}
|
|
void MQTTClientComponent::on_shutdown() {
|
|
if (!this->shutdown_message_.topic.empty()) {
|
|
yield();
|
|
this->publish(this->shutdown_message_);
|
|
yield();
|
|
}
|
|
this->mqtt_client_.disconnect(true);
|
|
}
|
|
|
|
#if ASYNC_TCP_SSL_ENABLED
|
|
void MQTTClientComponent::add_ssl_fingerprint(const std::array<uint8_t, SHA1_SIZE> &fingerprint) {
|
|
this->mqtt_client_.setSecure(true);
|
|
this->mqtt_client_.addServerFingerprint(fingerprint.data());
|
|
}
|
|
#endif
|
|
|
|
MQTTClientComponent *global_mqtt_client = nullptr;
|
|
|
|
// MQTTMessageTrigger
|
|
MQTTMessageTrigger::MQTTMessageTrigger(const std::string &topic) : topic_(topic) {}
|
|
void MQTTMessageTrigger::set_qos(uint8_t qos) { this->qos_ = qos; }
|
|
void MQTTMessageTrigger::set_payload(const std::string &payload) { this->payload_ = payload; }
|
|
void MQTTMessageTrigger::setup() {
|
|
global_mqtt_client->subscribe(this->topic_,
|
|
[this](const std::string &topic, const std::string &payload) {
|
|
if (this->payload_.has_value() && payload != *this->payload_) {
|
|
return;
|
|
}
|
|
|
|
this->trigger(payload);
|
|
},
|
|
this->qos_);
|
|
}
|
|
void MQTTMessageTrigger::dump_config() {
|
|
ESP_LOGCONFIG(TAG, "MQTT Message Trigger:");
|
|
ESP_LOGCONFIG(TAG, " Topic: '%s'", this->topic_.c_str());
|
|
ESP_LOGCONFIG(TAG, " QoS: %u", this->qos_);
|
|
}
|
|
float MQTTMessageTrigger::get_setup_priority() const { return setup_priority::AFTER_CONNECTION; }
|
|
|
|
} // namespace mqtt
|
|
} // namespace esphome
|