mirror of
https://github.com/esphome/esphome.git
synced 2024-11-26 12:27:13 +01:00
Handle Tuya multi-datapoint messages (#3159)
Co-authored-by: Samuel Sieb <samuel@sieb.net>
This commit is contained in:
parent
e7864a28a1
commit
42984fa72a
@ -199,7 +199,7 @@ void Tuya::handle_command_(uint8_t command, uint8_t version, const uint8_t *buff
|
|||||||
this->set_timeout("datapoint_dump", 1000, [this] { this->dump_config(); });
|
this->set_timeout("datapoint_dump", 1000, [this] { this->dump_config(); });
|
||||||
this->initialized_callback_.call();
|
this->initialized_callback_.call();
|
||||||
}
|
}
|
||||||
this->handle_datapoint_(buffer, len);
|
this->handle_datapoints_(buffer, len);
|
||||||
break;
|
break;
|
||||||
case TuyaCommandType::DATAPOINT_QUERY:
|
case TuyaCommandType::DATAPOINT_QUERY:
|
||||||
break;
|
break;
|
||||||
@ -224,105 +224,110 @@ void Tuya::handle_command_(uint8_t command, uint8_t version, const uint8_t *buff
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Tuya::handle_datapoint_(const uint8_t *buffer, size_t len) {
|
void Tuya::handle_datapoints_(const uint8_t *buffer, size_t len) {
|
||||||
if (len < 2)
|
while (len >= 4) {
|
||||||
return;
|
TuyaDatapoint datapoint{};
|
||||||
|
datapoint.id = buffer[0];
|
||||||
|
datapoint.type = (TuyaDatapointType) buffer[1];
|
||||||
|
datapoint.value_uint = 0;
|
||||||
|
|
||||||
TuyaDatapoint datapoint{};
|
size_t data_size = (buffer[2] << 8) + buffer[3];
|
||||||
datapoint.id = buffer[0];
|
const uint8_t *data = buffer + 4;
|
||||||
datapoint.type = (TuyaDatapointType) buffer[1];
|
size_t data_len = len - 4;
|
||||||
datapoint.value_uint = 0;
|
if (data_size > data_len) {
|
||||||
|
ESP_LOGW(TAG, "Datapoint %u is truncated and cannot be parsed (%zu > %zu)", datapoint.id, data_size, data_len);
|
||||||
// Drop update if datapoint is in ignore_mcu_datapoint_update list
|
|
||||||
for (uint8_t i : this->ignore_mcu_update_on_datapoints_) {
|
|
||||||
if (datapoint.id == i) {
|
|
||||||
ESP_LOGV(TAG, "Datapoint %u found in ignore_mcu_update_on_datapoints list, dropping MCU update", datapoint.id);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
size_t data_size = (buffer[2] << 8) + buffer[3];
|
datapoint.len = data_size;
|
||||||
const uint8_t *data = buffer + 4;
|
|
||||||
size_t data_len = len - 4;
|
|
||||||
if (data_size > data_len) {
|
|
||||||
ESP_LOGW(TAG, "Datapoint %u has extra bytes that will be ignored (%zu > %zu)", datapoint.id, data_size, data_len);
|
|
||||||
} else if (data_size < data_len) {
|
|
||||||
ESP_LOGW(TAG, "Datapoint %u is truncated and cannot be parsed (%zu < %zu)", datapoint.id, data_size, data_len);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
datapoint.len = data_len;
|
|
||||||
|
|
||||||
switch (datapoint.type) {
|
switch (datapoint.type) {
|
||||||
case TuyaDatapointType::RAW:
|
case TuyaDatapointType::RAW:
|
||||||
datapoint.value_raw = std::vector<uint8_t>(data, data + data_len);
|
datapoint.value_raw = std::vector<uint8_t>(data, data + data_size);
|
||||||
ESP_LOGD(TAG, "Datapoint %u update to %s", datapoint.id, format_hex_pretty(datapoint.value_raw).c_str());
|
ESP_LOGD(TAG, "Datapoint %u update to %s", datapoint.id, format_hex_pretty(datapoint.value_raw).c_str());
|
||||||
break;
|
break;
|
||||||
case TuyaDatapointType::BOOLEAN:
|
case TuyaDatapointType::BOOLEAN:
|
||||||
if (data_len != 1) {
|
if (data_size != 1) {
|
||||||
ESP_LOGW(TAG, "Datapoint %u has bad boolean len %zu", datapoint.id, data_len);
|
ESP_LOGW(TAG, "Datapoint %u has bad boolean len %zu", datapoint.id, data_size);
|
||||||
return;
|
|
||||||
}
|
|
||||||
datapoint.value_bool = data[0];
|
|
||||||
ESP_LOGD(TAG, "Datapoint %u update to %s", datapoint.id, ONOFF(datapoint.value_bool));
|
|
||||||
break;
|
|
||||||
case TuyaDatapointType::INTEGER:
|
|
||||||
if (data_len != 4) {
|
|
||||||
ESP_LOGW(TAG, "Datapoint %u has bad integer len %zu", datapoint.id, data_len);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
datapoint.value_uint = encode_uint32(data[0], data[1], data[2], data[3]);
|
|
||||||
ESP_LOGD(TAG, "Datapoint %u update to %d", datapoint.id, datapoint.value_int);
|
|
||||||
break;
|
|
||||||
case TuyaDatapointType::STRING:
|
|
||||||
datapoint.value_string = std::string(reinterpret_cast<const char *>(data), data_len);
|
|
||||||
ESP_LOGD(TAG, "Datapoint %u update to %s", datapoint.id, datapoint.value_string.c_str());
|
|
||||||
break;
|
|
||||||
case TuyaDatapointType::ENUM:
|
|
||||||
if (data_len != 1) {
|
|
||||||
ESP_LOGW(TAG, "Datapoint %u has bad enum len %zu", datapoint.id, data_len);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
datapoint.value_enum = data[0];
|
|
||||||
ESP_LOGD(TAG, "Datapoint %u update to %d", datapoint.id, datapoint.value_enum);
|
|
||||||
break;
|
|
||||||
case TuyaDatapointType::BITMASK:
|
|
||||||
switch (data_len) {
|
|
||||||
case 1:
|
|
||||||
datapoint.value_bitmask = encode_uint32(0, 0, 0, data[0]);
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
datapoint.value_bitmask = encode_uint32(0, 0, data[0], data[1]);
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
datapoint.value_bitmask = encode_uint32(data[0], data[1], data[2], data[3]);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
ESP_LOGW(TAG, "Datapoint %u has bad bitmask len %zu", datapoint.id, data_len);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
ESP_LOGD(TAG, "Datapoint %u update to %#08X", datapoint.id, datapoint.value_bitmask);
|
datapoint.value_bool = data[0];
|
||||||
break;
|
ESP_LOGD(TAG, "Datapoint %u update to %s", datapoint.id, ONOFF(datapoint.value_bool));
|
||||||
default:
|
break;
|
||||||
ESP_LOGW(TAG, "Datapoint %u has unknown type %#02hhX", datapoint.id, static_cast<uint8_t>(datapoint.type));
|
case TuyaDatapointType::INTEGER:
|
||||||
return;
|
if (data_size != 4) {
|
||||||
}
|
ESP_LOGW(TAG, "Datapoint %u has bad integer len %zu", datapoint.id, data_size);
|
||||||
|
return;
|
||||||
// Update internal datapoints
|
}
|
||||||
bool found = false;
|
datapoint.value_uint = encode_uint32(data[0], data[1], data[2], data[3]);
|
||||||
for (auto &other : this->datapoints_) {
|
ESP_LOGD(TAG, "Datapoint %u update to %d", datapoint.id, datapoint.value_int);
|
||||||
if (other.id == datapoint.id) {
|
break;
|
||||||
other = datapoint;
|
case TuyaDatapointType::STRING:
|
||||||
found = true;
|
datapoint.value_string = std::string(reinterpret_cast<const char *>(data), data_size);
|
||||||
|
ESP_LOGD(TAG, "Datapoint %u update to %s", datapoint.id, datapoint.value_string.c_str());
|
||||||
|
break;
|
||||||
|
case TuyaDatapointType::ENUM:
|
||||||
|
if (data_size != 1) {
|
||||||
|
ESP_LOGW(TAG, "Datapoint %u has bad enum len %zu", datapoint.id, data_size);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
datapoint.value_enum = data[0];
|
||||||
|
ESP_LOGD(TAG, "Datapoint %u update to %d", datapoint.id, datapoint.value_enum);
|
||||||
|
break;
|
||||||
|
case TuyaDatapointType::BITMASK:
|
||||||
|
switch (data_size) {
|
||||||
|
case 1:
|
||||||
|
datapoint.value_bitmask = encode_uint32(0, 0, 0, data[0]);
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
datapoint.value_bitmask = encode_uint32(0, 0, data[0], data[1]);
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
datapoint.value_bitmask = encode_uint32(data[0], data[1], data[2], data[3]);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ESP_LOGW(TAG, "Datapoint %u has bad bitmask len %zu", datapoint.id, data_size);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
ESP_LOGD(TAG, "Datapoint %u update to %#08X", datapoint.id, datapoint.value_bitmask);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ESP_LOGW(TAG, "Datapoint %u has unknown type %#02hhX", datapoint.id, static_cast<uint8_t>(datapoint.type));
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
if (!found) {
|
|
||||||
this->datapoints_.push_back(datapoint);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Run through listeners
|
len -= data_size + 4;
|
||||||
for (auto &listener : this->listeners_) {
|
buffer = data + data_size;
|
||||||
if (listener.datapoint_id == datapoint.id)
|
|
||||||
listener.on_datapoint(datapoint);
|
// drop update if datapoint is in ignore_mcu_datapoint_update list
|
||||||
|
bool skip = false;
|
||||||
|
for (auto i : this->ignore_mcu_update_on_datapoints_) {
|
||||||
|
if (datapoint.id == i) {
|
||||||
|
ESP_LOGV(TAG, "Datapoint %u found in ignore_mcu_update_on_datapoints list, dropping MCU update", datapoint.id);
|
||||||
|
skip = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (skip)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
// Update internal datapoints
|
||||||
|
bool found = false;
|
||||||
|
for (auto &other : this->datapoints_) {
|
||||||
|
if (other.id == datapoint.id) {
|
||||||
|
other = datapoint;
|
||||||
|
found = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!found) {
|
||||||
|
this->datapoints_.push_back(datapoint);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run through listeners
|
||||||
|
for (auto &listener : this->listeners_) {
|
||||||
|
if (listener.datapoint_id == datapoint.id)
|
||||||
|
listener.on_datapoint(datapoint);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -101,7 +101,7 @@ class Tuya : public Component, public uart::UARTDevice {
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
void handle_char_(uint8_t c);
|
void handle_char_(uint8_t c);
|
||||||
void handle_datapoint_(const uint8_t *buffer, size_t len);
|
void handle_datapoints_(const uint8_t *buffer, size_t len);
|
||||||
optional<TuyaDatapoint> get_datapoint_(uint8_t datapoint_id);
|
optional<TuyaDatapoint> get_datapoint_(uint8_t datapoint_id);
|
||||||
bool validate_message_();
|
bool validate_message_();
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user