mirror of
https://github.com/esphome/esphome.git
synced 2025-11-14 05:45:48 +00:00
Compare commits
13 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7b3d6747d5 | ||
|
|
7904d3b157 | ||
|
|
3a48b10757 | ||
|
|
0e50cac399 | ||
|
|
04225d5717 | ||
|
|
86791422f0 | ||
|
|
9c2af6318c | ||
|
|
c747d7d45d | ||
|
|
bbd7c9cf86 | ||
|
|
169fb79c97 | ||
|
|
1579dfeb80 | ||
|
|
d8a6d8594a | ||
|
|
7be071a0e9 |
@@ -60,6 +60,7 @@ from esphome.cpp_types import ( # noqa
|
|||||||
std_ns,
|
std_ns,
|
||||||
std_shared_ptr,
|
std_shared_ptr,
|
||||||
std_string,
|
std_string,
|
||||||
|
std_string_ref,
|
||||||
std_vector,
|
std_vector,
|
||||||
uint8,
|
uint8,
|
||||||
uint16,
|
uint16,
|
||||||
|
|||||||
@@ -145,24 +145,21 @@ bool DallasTemperatureSensor::check_scratch_pad_() {
|
|||||||
float DallasTemperatureSensor::get_temp_c_() {
|
float DallasTemperatureSensor::get_temp_c_() {
|
||||||
int16_t temp = (this->scratch_pad_[1] << 8) | this->scratch_pad_[0];
|
int16_t temp = (this->scratch_pad_[1] << 8) | this->scratch_pad_[0];
|
||||||
if ((this->address_ & 0xff) == DALLAS_MODEL_DS18S20) {
|
if ((this->address_ & 0xff) == DALLAS_MODEL_DS18S20) {
|
||||||
if (this->scratch_pad_[7] != 0x10)
|
return (temp >> 1) + (this->scratch_pad_[7] - this->scratch_pad_[6]) / float(this->scratch_pad_[7]) - 0.25;
|
||||||
ESP_LOGE(TAG, "unexpected COUNT_PER_C value: %u", this->scratch_pad_[7]);
|
}
|
||||||
temp = ((temp & 0xfff7) << 3) + (0x10 - this->scratch_pad_[6]) - 4;
|
switch (this->resolution_) {
|
||||||
} else {
|
case 9:
|
||||||
switch (this->resolution_) {
|
temp &= 0xfff8;
|
||||||
case 9:
|
break;
|
||||||
temp &= 0xfff8;
|
case 10:
|
||||||
break;
|
temp &= 0xfffc;
|
||||||
case 10:
|
break;
|
||||||
temp &= 0xfffc;
|
case 11:
|
||||||
break;
|
temp &= 0xfffe;
|
||||||
case 11:
|
break;
|
||||||
temp &= 0xfffe;
|
case 12:
|
||||||
break;
|
default:
|
||||||
case 12:
|
break;
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return temp / 16.0f;
|
return temp / 16.0f;
|
||||||
|
|||||||
@@ -37,14 +37,18 @@ void DS1307Component::read_time() {
|
|||||||
ESP_LOGW(TAG, "RTC halted, not syncing to system clock.");
|
ESP_LOGW(TAG, "RTC halted, not syncing to system clock.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
ESPTime rtc_time{.second = uint8_t(ds1307_.reg.second + 10 * ds1307_.reg.second_10),
|
ESPTime rtc_time{
|
||||||
.minute = uint8_t(ds1307_.reg.minute + 10u * ds1307_.reg.minute_10),
|
.second = uint8_t(ds1307_.reg.second + 10 * ds1307_.reg.second_10),
|
||||||
.hour = uint8_t(ds1307_.reg.hour + 10u * ds1307_.reg.hour_10),
|
.minute = uint8_t(ds1307_.reg.minute + 10u * ds1307_.reg.minute_10),
|
||||||
.day_of_week = uint8_t(ds1307_.reg.weekday),
|
.hour = uint8_t(ds1307_.reg.hour + 10u * ds1307_.reg.hour_10),
|
||||||
.day_of_month = uint8_t(ds1307_.reg.day + 10u * ds1307_.reg.day_10),
|
.day_of_week = uint8_t(ds1307_.reg.weekday),
|
||||||
.day_of_year = 1, // ignored by recalc_timestamp_utc(false)
|
.day_of_month = uint8_t(ds1307_.reg.day + 10u * ds1307_.reg.day_10),
|
||||||
.month = uint8_t(ds1307_.reg.month + 10u * ds1307_.reg.month_10),
|
.day_of_year = 1, // ignored by recalc_timestamp_utc(false)
|
||||||
.year = uint16_t(ds1307_.reg.year + 10u * ds1307_.reg.year_10 + 2000)};
|
.month = uint8_t(ds1307_.reg.month + 10u * ds1307_.reg.month_10),
|
||||||
|
.year = uint16_t(ds1307_.reg.year + 10u * ds1307_.reg.year_10 + 2000),
|
||||||
|
.is_dst = false, // not used
|
||||||
|
.timestamp = 0 // overwritten by recalc_timestamp_utc(false)
|
||||||
|
};
|
||||||
rtc_time.recalc_timestamp_utc(false);
|
rtc_time.recalc_timestamp_utc(false);
|
||||||
if (!rtc_time.is_valid()) {
|
if (!rtc_time.is_valid()) {
|
||||||
ESP_LOGE(TAG, "Invalid RTC time, not syncing to system clock.");
|
ESP_LOGE(TAG, "Invalid RTC time, not syncing to system clock.");
|
||||||
|
|||||||
@@ -1,10 +1,17 @@
|
|||||||
|
import logging
|
||||||
|
|
||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
|
import esphome.final_validate as fv
|
||||||
from esphome.components.ota import BASE_OTA_SCHEMA, ota_to_code, OTAComponent
|
from esphome.components.ota import BASE_OTA_SCHEMA, ota_to_code, OTAComponent
|
||||||
|
from esphome.config_helpers import merge_config
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
|
CONF_ESPHOME,
|
||||||
CONF_ID,
|
CONF_ID,
|
||||||
CONF_NUM_ATTEMPTS,
|
CONF_NUM_ATTEMPTS,
|
||||||
|
CONF_OTA,
|
||||||
CONF_PASSWORD,
|
CONF_PASSWORD,
|
||||||
|
CONF_PLATFORM,
|
||||||
CONF_PORT,
|
CONF_PORT,
|
||||||
CONF_REBOOT_TIMEOUT,
|
CONF_REBOOT_TIMEOUT,
|
||||||
CONF_SAFE_MODE,
|
CONF_SAFE_MODE,
|
||||||
@@ -12,6 +19,8 @@ from esphome.const import (
|
|||||||
)
|
)
|
||||||
from esphome.core import coroutine_with_priority
|
from esphome.core import coroutine_with_priority
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
CODEOWNERS = ["@esphome/core"]
|
CODEOWNERS = ["@esphome/core"]
|
||||||
AUTO_LOAD = ["md5", "socket"]
|
AUTO_LOAD = ["md5", "socket"]
|
||||||
@@ -21,6 +30,65 @@ esphome = cg.esphome_ns.namespace("esphome")
|
|||||||
ESPHomeOTAComponent = esphome.class_("ESPHomeOTAComponent", OTAComponent)
|
ESPHomeOTAComponent = esphome.class_("ESPHomeOTAComponent", OTAComponent)
|
||||||
|
|
||||||
|
|
||||||
|
def ota_esphome_final_validate(config):
|
||||||
|
full_conf = fv.full_config.get()
|
||||||
|
full_ota_conf = full_conf[CONF_OTA]
|
||||||
|
new_ota_conf = []
|
||||||
|
merged_ota_esphome_configs_by_port = {}
|
||||||
|
ports_with_merged_configs = []
|
||||||
|
for ota_conf in full_ota_conf:
|
||||||
|
if ota_conf.get(CONF_PLATFORM) == CONF_ESPHOME:
|
||||||
|
if (
|
||||||
|
conf_port := ota_conf.get(CONF_PORT)
|
||||||
|
) not in merged_ota_esphome_configs_by_port:
|
||||||
|
merged_ota_esphome_configs_by_port[conf_port] = ota_conf
|
||||||
|
else:
|
||||||
|
if merged_ota_esphome_configs_by_port[conf_port][
|
||||||
|
CONF_VERSION
|
||||||
|
] != ota_conf.get(CONF_VERSION):
|
||||||
|
raise cv.Invalid(
|
||||||
|
f"Found multiple configurations but {CONF_VERSION} is inconsistent"
|
||||||
|
)
|
||||||
|
if (
|
||||||
|
merged_ota_esphome_configs_by_port[conf_port][CONF_ID].is_manual
|
||||||
|
and ota_conf.get(CONF_ID).is_manual
|
||||||
|
):
|
||||||
|
raise cv.Invalid(
|
||||||
|
f"Found multiple configurations but {CONF_ID} is inconsistent"
|
||||||
|
)
|
||||||
|
if (
|
||||||
|
CONF_PASSWORD in merged_ota_esphome_configs_by_port[conf_port]
|
||||||
|
and CONF_PASSWORD in ota_conf
|
||||||
|
and merged_ota_esphome_configs_by_port[conf_port][CONF_PASSWORD]
|
||||||
|
!= ota_conf.get(CONF_PASSWORD)
|
||||||
|
):
|
||||||
|
raise cv.Invalid(
|
||||||
|
f"Found multiple configurations but {CONF_PASSWORD} is inconsistent"
|
||||||
|
)
|
||||||
|
|
||||||
|
ports_with_merged_configs.append(conf_port)
|
||||||
|
merged_ota_esphome_configs_by_port[conf_port] = merge_config(
|
||||||
|
merged_ota_esphome_configs_by_port[conf_port], ota_conf
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
new_ota_conf.append(ota_conf)
|
||||||
|
|
||||||
|
for port_conf in merged_ota_esphome_configs_by_port.values():
|
||||||
|
new_ota_conf.append(port_conf)
|
||||||
|
|
||||||
|
full_conf[CONF_OTA] = new_ota_conf
|
||||||
|
fv.full_config.set(full_conf)
|
||||||
|
|
||||||
|
if len(ports_with_merged_configs) > 0:
|
||||||
|
_LOGGER.warning(
|
||||||
|
"Found and merged multiple configurations for %s %s %s port(s) %s",
|
||||||
|
CONF_OTA,
|
||||||
|
CONF_PLATFORM,
|
||||||
|
CONF_ESPHOME,
|
||||||
|
ports_with_merged_configs,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
CONFIG_SCHEMA = (
|
CONFIG_SCHEMA = (
|
||||||
cv.Schema(
|
cv.Schema(
|
||||||
{
|
{
|
||||||
@@ -50,6 +118,8 @@ CONFIG_SCHEMA = (
|
|||||||
.extend(cv.COMPONENT_SCHEMA)
|
.extend(cv.COMPONENT_SCHEMA)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
FINAL_VALIDATE_SCHEMA = ota_esphome_final_validate
|
||||||
|
|
||||||
|
|
||||||
@coroutine_with_priority(52.0)
|
@coroutine_with_priority(52.0)
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
|
|||||||
@@ -257,7 +257,7 @@ async def http_request_action_to_code(config, action_id, template_arg, args):
|
|||||||
trigger,
|
trigger,
|
||||||
[
|
[
|
||||||
(cg.std_shared_ptr.template(HttpContainer), "response"),
|
(cg.std_shared_ptr.template(HttpContainer), "response"),
|
||||||
(cg.std_string, "body"),
|
(cg.std_string_ref, "body"),
|
||||||
],
|
],
|
||||||
conf,
|
conf,
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -43,10 +43,10 @@ class HttpContainer : public Parented<HttpRequestComponent> {
|
|||||||
bool secure_{false};
|
bool secure_{false};
|
||||||
};
|
};
|
||||||
|
|
||||||
class HttpRequestResponseTrigger : public Trigger<std::shared_ptr<HttpContainer>, std::string> {
|
class HttpRequestResponseTrigger : public Trigger<std::shared_ptr<HttpContainer>, std::string &> {
|
||||||
public:
|
public:
|
||||||
void process(std::shared_ptr<HttpContainer> container, std::string response_body) {
|
void process(std::shared_ptr<HttpContainer> container, std::string &response_body) {
|
||||||
this->trigger(std::move(container), std::move(response_body));
|
this->trigger(std::move(container), response_body);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -153,8 +153,17 @@ template<typename... Ts> class HttpRequestSendAction : public Action<Ts...> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto *trigger : this->response_triggers_) {
|
if (this->response_triggers_.size() == 1) {
|
||||||
trigger->process(container, response_body);
|
// if there is only one trigger, no need to copy the response body
|
||||||
|
this->response_triggers_[0]->process(container, response_body);
|
||||||
|
} else {
|
||||||
|
for (auto *trigger : this->response_triggers_) {
|
||||||
|
// with multiple triggers, pass a copy of the response body to each
|
||||||
|
// one so that modifications made in one trigger are not visible to
|
||||||
|
// the others
|
||||||
|
auto response_body_copy = std::string(response_body);
|
||||||
|
trigger->process(container, response_body_copy);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
container->end();
|
container->end();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -115,12 +115,15 @@ void LEDCOutput::write_state(float state) {
|
|||||||
const uint32_t max_duty = (uint32_t(1) << this->bit_depth_) - 1;
|
const uint32_t max_duty = (uint32_t(1) << this->bit_depth_) - 1;
|
||||||
const float duty_rounded = roundf(state * max_duty);
|
const float duty_rounded = roundf(state * max_duty);
|
||||||
auto duty = static_cast<uint32_t>(duty_rounded);
|
auto duty = static_cast<uint32_t>(duty_rounded);
|
||||||
|
|
||||||
#ifdef USE_ARDUINO
|
#ifdef USE_ARDUINO
|
||||||
ESP_LOGV(TAG, "Setting duty: %u on channel %u", duty, this->channel_);
|
ESP_LOGV(TAG, "Setting duty: %u on channel %u", duty, this->channel_);
|
||||||
ledcWrite(this->channel_, duty);
|
ledcWrite(this->channel_, duty);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ESP_IDF
|
#ifdef USE_ESP_IDF
|
||||||
|
// ensure that 100% on is not 99.975% on
|
||||||
|
if ((duty == max_duty) && (max_duty != 1)) {
|
||||||
|
duty = max_duty + 1;
|
||||||
|
}
|
||||||
auto speed_mode = get_speed_mode(channel_);
|
auto speed_mode = get_speed_mode(channel_);
|
||||||
auto chan_num = static_cast<ledc_channel_t>(channel_ % 8);
|
auto chan_num = static_cast<ledc_channel_t>(channel_ % 8);
|
||||||
int hpoint = ledc_angle_to_htop(this->phase_angle_, this->bit_depth_);
|
int hpoint = ledc_angle_to_htop(this->phase_angle_, this->bit_depth_);
|
||||||
|
|||||||
@@ -116,7 +116,8 @@ void ModbusController::on_modbus_read_registers(uint8_t function_code, uint16_t
|
|||||||
ESP_LOGD(TAG, "Matched register. Address: 0x%02X. Value type: %zu. Register count: %u. Value: %0.1f.",
|
ESP_LOGD(TAG, "Matched register. Address: 0x%02X. Value type: %zu. Register count: %u. Value: %0.1f.",
|
||||||
server_register->address, static_cast<uint8_t>(server_register->value_type),
|
server_register->address, static_cast<uint8_t>(server_register->value_type),
|
||||||
server_register->register_count, value);
|
server_register->register_count, value);
|
||||||
number_to_payload(sixteen_bit_response, value, server_register->value_type);
|
std::vector<uint16_t> payload = float_to_payload(value, server_register->value_type);
|
||||||
|
sixteen_bit_response.insert(sixteen_bit_response.end(), payload.cbegin(), payload.cend());
|
||||||
current_address += server_register->register_count;
|
current_address += server_register->register_count;
|
||||||
found = true;
|
found = true;
|
||||||
break;
|
break;
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ void ModbusTextSensor::parse_and_publish(const std::vector<uint8_t> &data) {
|
|||||||
std::ostringstream output;
|
std::ostringstream output;
|
||||||
uint8_t items_left = this->response_bytes;
|
uint8_t items_left = this->response_bytes;
|
||||||
uint8_t index = this->offset;
|
uint8_t index = this->offset;
|
||||||
char buffer[4];
|
char buffer[5];
|
||||||
while ((items_left > 0) && index < data.size()) {
|
while ((items_left > 0) && index < data.size()) {
|
||||||
uint8_t b = data[index];
|
uint8_t b = data[index];
|
||||||
switch (this->encode_) {
|
switch (this->encode_) {
|
||||||
|
|||||||
@@ -56,21 +56,20 @@ CONFIG_SCHEMA = cv.All(
|
|||||||
|
|
||||||
@coroutine_with_priority(50.0)
|
@coroutine_with_priority(50.0)
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
if config[CONF_DISABLED]:
|
if not config[CONF_DISABLED]:
|
||||||
return
|
var = cg.new_Pvariable(config[CONF_ID])
|
||||||
|
await cg.register_component(var, config)
|
||||||
|
|
||||||
var = cg.new_Pvariable(config[CONF_ID])
|
for conf in config.get(CONF_ON_SAFE_MODE, []):
|
||||||
await cg.register_component(var, config)
|
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
||||||
|
await automation.build_automation(trigger, [], conf)
|
||||||
|
|
||||||
for conf in config.get(CONF_ON_SAFE_MODE, []):
|
condition = var.should_enter_safe_mode(
|
||||||
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
config[CONF_NUM_ATTEMPTS],
|
||||||
await automation.build_automation(trigger, [], conf)
|
config[CONF_REBOOT_TIMEOUT],
|
||||||
|
config[CONF_BOOT_IS_GOOD_AFTER],
|
||||||
|
)
|
||||||
|
cg.add(RawExpression(f"if ({condition}) return"))
|
||||||
|
|
||||||
condition = var.should_enter_safe_mode(
|
|
||||||
config[CONF_NUM_ATTEMPTS],
|
|
||||||
config[CONF_REBOOT_TIMEOUT],
|
|
||||||
config[CONF_BOOT_IS_GOOD_AFTER],
|
|
||||||
)
|
|
||||||
cg.add(RawExpression(f"if ({condition}) return"))
|
|
||||||
CORE.data[CONF_SAFE_MODE] = {}
|
CORE.data[CONF_SAFE_MODE] = {}
|
||||||
CORE.data[CONF_SAFE_MODE][KEY_PAST_SAFE_MODE] = True
|
CORE.data[CONF_SAFE_MODE][KEY_PAST_SAFE_MODE] = True
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
"""Constants used by esphome."""
|
"""Constants used by esphome."""
|
||||||
|
|
||||||
__version__ = "2024.6.2"
|
__version__ = "2024.6.4"
|
||||||
|
|
||||||
ALLOWED_NAME_CHARS = "abcdefghijklmnopqrstuvwxyz0123456789-_"
|
ALLOWED_NAME_CHARS = "abcdefghijklmnopqrstuvwxyz0123456789-_"
|
||||||
VALID_SUBSTITUTIONS_CHARACTERS = (
|
VALID_SUBSTITUTIONS_CHARACTERS = (
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ int_ = global_ns.namespace("int")
|
|||||||
std_ns = global_ns.namespace("std")
|
std_ns = global_ns.namespace("std")
|
||||||
std_shared_ptr = std_ns.class_("shared_ptr")
|
std_shared_ptr = std_ns.class_("shared_ptr")
|
||||||
std_string = std_ns.class_("string")
|
std_string = std_ns.class_("string")
|
||||||
|
std_string_ref = std_ns.namespace("string &")
|
||||||
std_vector = std_ns.class_("vector")
|
std_vector = std_ns.class_("vector")
|
||||||
uint8 = global_ns.namespace("uint8_t")
|
uint8 = global_ns.namespace("uint8_t")
|
||||||
uint16 = global_ns.namespace("uint16_t")
|
uint16 = global_ns.namespace("uint16_t")
|
||||||
|
|||||||
Reference in New Issue
Block a user