mirror of
https://github.com/esphome/esphome.git
synced 2024-11-24 16:08:10 +01:00
Merge branch 'esphome:dev' into dev
This commit is contained in:
commit
1f11073ea9
139 changed files with 805 additions and 11075 deletions
|
@ -1,7 +1,9 @@
|
||||||
{
|
{
|
||||||
"name": "ESPHome Dev",
|
"name": "ESPHome Dev",
|
||||||
"image": "ghcr.io/esphome/esphome-lint:dev",
|
"image": "ghcr.io/esphome/esphome-lint:dev",
|
||||||
"postCreateCommand": ["script/devcontainer-post-create"],
|
"postCreateCommand": [
|
||||||
|
"script/devcontainer-post-create"
|
||||||
|
],
|
||||||
"containerEnv": {
|
"containerEnv": {
|
||||||
"DEVCONTAINER": "1",
|
"DEVCONTAINER": "1",
|
||||||
"PIP_BREAK_SYSTEM_PACKAGES": "1",
|
"PIP_BREAK_SYSTEM_PACKAGES": "1",
|
||||||
|
@ -27,6 +29,9 @@
|
||||||
"extensions": [
|
"extensions": [
|
||||||
// python
|
// python
|
||||||
"ms-python.python",
|
"ms-python.python",
|
||||||
|
"ms-python.pylint",
|
||||||
|
"ms-python.flake8",
|
||||||
|
"ms-python.black-formatter",
|
||||||
"visualstudioexptteam.vscodeintellicode",
|
"visualstudioexptteam.vscodeintellicode",
|
||||||
// yaml
|
// yaml
|
||||||
"redhat.vscode-yaml",
|
"redhat.vscode-yaml",
|
||||||
|
@ -38,9 +43,21 @@
|
||||||
"settings": {
|
"settings": {
|
||||||
"python.languageServer": "Pylance",
|
"python.languageServer": "Pylance",
|
||||||
"python.pythonPath": "/usr/bin/python3",
|
"python.pythonPath": "/usr/bin/python3",
|
||||||
"python.linting.pylintEnabled": true,
|
"pylint.args": [
|
||||||
"python.linting.enabled": true,
|
"--rcfile=${workspaceFolder}/pyproject.toml"
|
||||||
"python.formatting.provider": "black",
|
],
|
||||||
|
"flake8.args": [
|
||||||
|
"--config=${workspaceFolder}/.flake8"
|
||||||
|
],
|
||||||
|
"black-formatter.args": [
|
||||||
|
"--config",
|
||||||
|
"${workspaceFolder}/pyproject.toml"
|
||||||
|
],
|
||||||
|
"[python]": {
|
||||||
|
// VS will say "Value is not accepted" before building the devcontainer, but the warning
|
||||||
|
// should go away after build is completed.
|
||||||
|
"editor.defaultFormatter": "ms-python.black-formatter"
|
||||||
|
},
|
||||||
"editor.formatOnPaste": false,
|
"editor.formatOnPaste": false,
|
||||||
"editor.formatOnSave": true,
|
"editor.formatOnSave": true,
|
||||||
"editor.formatOnType": true,
|
"editor.formatOnType": true,
|
||||||
|
|
4
.github/actions/build-image/action.yaml
vendored
4
.github/actions/build-image/action.yaml
vendored
|
@ -46,7 +46,7 @@ runs:
|
||||||
|
|
||||||
- name: Build and push to ghcr by digest
|
- name: Build and push to ghcr by digest
|
||||||
id: build-ghcr
|
id: build-ghcr
|
||||||
uses: docker/build-push-action@v6.0.1
|
uses: docker/build-push-action@v6.2.0
|
||||||
with:
|
with:
|
||||||
context: .
|
context: .
|
||||||
file: ./docker/Dockerfile
|
file: ./docker/Dockerfile
|
||||||
|
@ -69,7 +69,7 @@ runs:
|
||||||
|
|
||||||
- name: Build and push to dockerhub by digest
|
- name: Build and push to dockerhub by digest
|
||||||
id: build-dockerhub
|
id: build-dockerhub
|
||||||
uses: docker/build-push-action@v6.0.1
|
uses: docker/build-push-action@v6.2.0
|
||||||
with:
|
with:
|
||||||
context: .
|
context: .
|
||||||
file: ./docker/Dockerfile
|
file: ./docker/Dockerfile
|
||||||
|
|
67
.github/workflows/ci.yml
vendored
67
.github/workflows/ci.yml
vendored
|
@ -248,72 +248,6 @@ jobs:
|
||||||
run: script/ci-suggest-changes
|
run: script/ci-suggest-changes
|
||||||
if: always()
|
if: always()
|
||||||
|
|
||||||
compile-tests-list:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
outputs:
|
|
||||||
matrix: ${{ steps.set-matrix.outputs.matrix }}
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v4.1.7
|
|
||||||
- name: Find all YAML test files
|
|
||||||
id: set-matrix
|
|
||||||
run: echo "matrix=$(ls tests/test*.yaml | jq -R -s -c 'split("\n")[:-1]')" >> $GITHUB_OUTPUT
|
|
||||||
|
|
||||||
validate-tests:
|
|
||||||
name: Validate YAML test ${{ matrix.file }}
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs:
|
|
||||||
- common
|
|
||||||
- compile-tests-list
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
matrix:
|
|
||||||
file: ${{ fromJson(needs.compile-tests-list.outputs.matrix) }}
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v4.1.7
|
|
||||||
- name: Restore Python
|
|
||||||
uses: ./.github/actions/restore-python
|
|
||||||
with:
|
|
||||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
|
||||||
cache-key: ${{ needs.common.outputs.cache-key }}
|
|
||||||
- name: Run esphome config ${{ matrix.file }}
|
|
||||||
run: |
|
|
||||||
. venv/bin/activate
|
|
||||||
esphome config ${{ matrix.file }}
|
|
||||||
|
|
||||||
compile-tests:
|
|
||||||
name: Run YAML test ${{ matrix.file }}
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs:
|
|
||||||
- common
|
|
||||||
- black
|
|
||||||
- ci-custom
|
|
||||||
- clang-format
|
|
||||||
- flake8
|
|
||||||
- pylint
|
|
||||||
- pytest
|
|
||||||
- pyupgrade
|
|
||||||
- compile-tests-list
|
|
||||||
- validate-tests
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
max-parallel: 2
|
|
||||||
matrix:
|
|
||||||
file: ${{ fromJson(needs.compile-tests-list.outputs.matrix) }}
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v4.1.7
|
|
||||||
- name: Restore Python
|
|
||||||
uses: ./.github/actions/restore-python
|
|
||||||
with:
|
|
||||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
|
||||||
cache-key: ${{ needs.common.outputs.cache-key }}
|
|
||||||
- name: Run esphome compile ${{ matrix.file }}
|
|
||||||
run: |
|
|
||||||
. venv/bin/activate
|
|
||||||
esphome compile ${{ matrix.file }}
|
|
||||||
|
|
||||||
clang-tidy:
|
clang-tidy:
|
||||||
name: ${{ matrix.name }}
|
name: ${{ matrix.name }}
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
@ -550,7 +484,6 @@ jobs:
|
||||||
- pylint
|
- pylint
|
||||||
- pytest
|
- pytest
|
||||||
- pyupgrade
|
- pyupgrade
|
||||||
- compile-tests
|
|
||||||
- clang-tidy
|
- clang-tidy
|
||||||
- list-components
|
- list-components
|
||||||
- test-build-components
|
- test-build-components
|
||||||
|
|
|
@ -34,28 +34,32 @@ RUN \
|
||||||
python3-wheel=0.38.4-2 \
|
python3-wheel=0.38.4-2 \
|
||||||
iputils-ping=3:20221126-1 \
|
iputils-ping=3:20221126-1 \
|
||||||
git=1:2.39.2-1.1 \
|
git=1:2.39.2-1.1 \
|
||||||
curl=7.88.1-10+deb12u5 \
|
curl=7.88.1-10+deb12u6 \
|
||||||
openssh-client=1:9.2p1-2+deb12u2 \
|
openssh-client=1:9.2p1-2+deb12u2 \
|
||||||
python3-cffi=1.15.1-5 \
|
python3-cffi=1.15.1-5 \
|
||||||
libcairo2=1.16.0-7 \
|
libcairo2=1.16.0-7 \
|
||||||
libmagic1=1:5.44-3 \
|
libmagic1=1:5.44-3 \
|
||||||
patch=2.7.6-7; \
|
patch=2.7.6-7 \
|
||||||
if [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]; then \
|
&& ( \
|
||||||
|
( \
|
||||||
|
[ "$TARGETARCH$TARGETVARIANT" = "armv7" ] && \
|
||||||
apt-get install -y --no-install-recommends \
|
apt-get install -y --no-install-recommends \
|
||||||
build-essential=12.9 \
|
build-essential=12.9 \
|
||||||
python3-dev=3.11.2-1+b1 \
|
python3-dev=3.11.2-1+b1 \
|
||||||
zlib1g-dev=1:1.2.13.dfsg-1 \
|
zlib1g-dev=1:1.2.13.dfsg-1 \
|
||||||
libjpeg-dev=1:2.1.5-2 \
|
libjpeg-dev=1:2.1.5-2 \
|
||||||
libfreetype-dev=2.12.1+dfsg-5 \
|
libfreetype-dev=2.12.1+dfsg-5+deb12u3 \
|
||||||
libssl-dev=3.0.11-1~deb12u2 \
|
libssl-dev=3.0.13-1~deb12u1 \
|
||||||
libffi-dev=3.4.4-1 \
|
libffi-dev=3.4.4-1 \
|
||||||
libopenjp2-7=2.5.0-2 \
|
libopenjp2-7=2.5.0-2 \
|
||||||
libtiff6=4.5.0-6+deb12u1 \
|
libtiff6=4.5.0-6+deb12u1 \
|
||||||
cargo=0.66.0+ds1-1 \
|
cargo=0.66.0+ds1-1 \
|
||||||
pkg-config=1.8.1-1 \
|
pkg-config=1.8.1-1 \
|
||||||
gcc-arm-linux-gnueabihf=4:12.2.0-3; \
|
gcc-arm-linux-gnueabihf=4:12.2.0-3 \
|
||||||
fi; \
|
) \
|
||||||
rm -rf \
|
|| [ "$TARGETARCH$TARGETVARIANT" != "armv7" ] \
|
||||||
|
) \
|
||||||
|
&& rm -rf \
|
||||||
/tmp/* \
|
/tmp/* \
|
||||||
/var/{cache,log}/* \
|
/var/{cache,log}/* \
|
||||||
/var/lib/apt/lists/*
|
/var/lib/apt/lists/*
|
||||||
|
@ -190,8 +194,8 @@ RUN \
|
||||||
clang-format-13=1:13.0.1-11+b2 \
|
clang-format-13=1:13.0.1-11+b2 \
|
||||||
clang-tidy-14=1:14.0.6-12 \
|
clang-tidy-14=1:14.0.6-12 \
|
||||||
patch=2.7.6-7 \
|
patch=2.7.6-7 \
|
||||||
software-properties-common=0.99.30-4 \
|
software-properties-common=0.99.30-4.1~deb12u1 \
|
||||||
nano=7.2-1 \
|
nano=7.2-1+deb12u1 \
|
||||||
build-essential=12.9 \
|
build-essential=12.9 \
|
||||||
python3-dev=3.11.2-1+b1 \
|
python3-dev=3.11.2-1+b1 \
|
||||||
&& rm -rf \
|
&& rm -rf \
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -4,33 +4,33 @@ from esphome.components import sensor
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
CONF_GAS_RESISTANCE,
|
CONF_GAS_RESISTANCE,
|
||||||
CONF_HUMIDITY,
|
CONF_HUMIDITY,
|
||||||
|
CONF_IAQ_ACCURACY,
|
||||||
CONF_PRESSURE,
|
CONF_PRESSURE,
|
||||||
|
CONF_SAMPLE_RATE,
|
||||||
CONF_TEMPERATURE,
|
CONF_TEMPERATURE,
|
||||||
|
DEVICE_CLASS_ATMOSPHERIC_PRESSURE,
|
||||||
DEVICE_CLASS_CARBON_DIOXIDE,
|
DEVICE_CLASS_CARBON_DIOXIDE,
|
||||||
DEVICE_CLASS_HUMIDITY,
|
DEVICE_CLASS_HUMIDITY,
|
||||||
DEVICE_CLASS_VOLATILE_ORGANIC_COMPOUNDS_PARTS,
|
|
||||||
DEVICE_CLASS_ATMOSPHERIC_PRESSURE,
|
|
||||||
DEVICE_CLASS_TEMPERATURE,
|
DEVICE_CLASS_TEMPERATURE,
|
||||||
|
DEVICE_CLASS_VOLATILE_ORGANIC_COMPOUNDS_PARTS,
|
||||||
|
ICON_GAS_CYLINDER,
|
||||||
|
ICON_GAUGE,
|
||||||
STATE_CLASS_MEASUREMENT,
|
STATE_CLASS_MEASUREMENT,
|
||||||
UNIT_CELSIUS,
|
UNIT_CELSIUS,
|
||||||
UNIT_HECTOPASCAL,
|
UNIT_HECTOPASCAL,
|
||||||
UNIT_OHM,
|
UNIT_OHM,
|
||||||
UNIT_PARTS_PER_MILLION,
|
UNIT_PARTS_PER_MILLION,
|
||||||
UNIT_PERCENT,
|
UNIT_PERCENT,
|
||||||
ICON_GAS_CYLINDER,
|
|
||||||
ICON_GAUGE,
|
|
||||||
)
|
)
|
||||||
from . import (
|
from . import (
|
||||||
BME680BSECComponent,
|
BME680BSECComponent,
|
||||||
CONF_BME680_BSEC_ID,
|
CONF_BME680_BSEC_ID,
|
||||||
CONF_SAMPLE_RATE,
|
|
||||||
SAMPLE_RATE_OPTIONS,
|
SAMPLE_RATE_OPTIONS,
|
||||||
)
|
)
|
||||||
|
|
||||||
DEPENDENCIES = ["bme680_bsec"]
|
DEPENDENCIES = ["bme680_bsec"]
|
||||||
|
|
||||||
CONF_IAQ = "iaq"
|
CONF_IAQ = "iaq"
|
||||||
CONF_IAQ_ACCURACY = "iaq_accuracy"
|
|
||||||
CONF_CO2_EQUIVALENT = "co2_equivalent"
|
CONF_CO2_EQUIVALENT = "co2_equivalent"
|
||||||
CONF_BREATH_VOC_EQUIVALENT = "breath_voc_equivalent"
|
CONF_BREATH_VOC_EQUIVALENT = "breath_voc_equivalent"
|
||||||
UNIT_IAQ = "IAQ"
|
UNIT_IAQ = "IAQ"
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
from esphome.components import text_sensor
|
from esphome.components import text_sensor
|
||||||
|
from esphome.const import CONF_IAQ_ACCURACY
|
||||||
from . import BME680BSECComponent, CONF_BME680_BSEC_ID
|
from . import BME680BSECComponent, CONF_BME680_BSEC_ID
|
||||||
|
|
||||||
DEPENDENCIES = ["bme680_bsec"]
|
DEPENDENCIES = ["bme680_bsec"]
|
||||||
|
|
||||||
CONF_IAQ_ACCURACY = "iaq_accuracy"
|
|
||||||
ICON_ACCURACY = "mdi:checkbox-marked-circle-outline"
|
ICON_ACCURACY = "mdi:checkbox-marked-circle-outline"
|
||||||
|
|
||||||
TYPES = [CONF_IAQ_ACCURACY]
|
TYPES = [CONF_IAQ_ACCURACY]
|
||||||
|
|
|
@ -145,10 +145,8 @@ 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;
|
|
||||||
} else {
|
|
||||||
switch (this->resolution_) {
|
switch (this->resolution_) {
|
||||||
case 9:
|
case 9:
|
||||||
temp &= 0xfff8;
|
temp &= 0xfff8;
|
||||||
|
@ -163,7 +161,6 @@ float DallasTemperatureSensor::get_temp_c_() {
|
||||||
default:
|
default:
|
||||||
break;
|
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{
|
||||||
|
.second = uint8_t(ds1307_.reg.second + 10 * ds1307_.reg.second_10),
|
||||||
.minute = uint8_t(ds1307_.reg.minute + 10u * ds1307_.reg.minute_10),
|
.minute = uint8_t(ds1307_.reg.minute + 10u * ds1307_.reg.minute_10),
|
||||||
.hour = uint8_t(ds1307_.reg.hour + 10u * ds1307_.reg.hour_10),
|
.hour = uint8_t(ds1307_.reg.hour + 10u * ds1307_.reg.hour_10),
|
||||||
.day_of_week = uint8_t(ds1307_.reg.weekday),
|
.day_of_week = uint8_t(ds1307_.reg.weekday),
|
||||||
.day_of_month = uint8_t(ds1307_.reg.day + 10u * ds1307_.reg.day_10),
|
.day_of_month = uint8_t(ds1307_.reg.day + 10u * ds1307_.reg.day_10),
|
||||||
.day_of_year = 1, // ignored by recalc_timestamp_utc(false)
|
.day_of_year = 1, // ignored by recalc_timestamp_utc(false)
|
||||||
.month = uint8_t(ds1307_.reg.month + 10u * ds1307_.reg.month_10),
|
.month = uint8_t(ds1307_.reg.month + 10u * ds1307_.reg.month_10),
|
||||||
.year = uint16_t(ds1307_.reg.year + 10u * ds1307_.reg.year_10 + 2000)};
|
.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):
|
||||||
|
|
|
@ -17,7 +17,6 @@ from esphome.helpers import (
|
||||||
cpp_string_escape,
|
cpp_string_escape,
|
||||||
)
|
)
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
__version__,
|
|
||||||
CONF_FAMILY,
|
CONF_FAMILY,
|
||||||
CONF_FILE,
|
CONF_FILE,
|
||||||
CONF_GLYPHS,
|
CONF_GLYPHS,
|
||||||
|
@ -185,31 +184,6 @@ def get_font_path(value, type) -> Path:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
def download_content(url: str, path: Path) -> None:
|
|
||||||
if not external_files.has_remote_file_changed(url, path):
|
|
||||||
_LOGGER.debug("Remote file has not changed %s", url)
|
|
||||||
return
|
|
||||||
|
|
||||||
_LOGGER.debug(
|
|
||||||
"Remote file has changed, downloading from %s to %s",
|
|
||||||
url,
|
|
||||||
path,
|
|
||||||
)
|
|
||||||
|
|
||||||
try:
|
|
||||||
req = requests.get(
|
|
||||||
url,
|
|
||||||
timeout=external_files.NETWORK_TIMEOUT,
|
|
||||||
headers={"User-agent": f"ESPHome/{__version__} (https://esphome.io)"},
|
|
||||||
)
|
|
||||||
req.raise_for_status()
|
|
||||||
except requests.exceptions.RequestException as e:
|
|
||||||
raise cv.Invalid(f"Could not download from {url}: {e}")
|
|
||||||
|
|
||||||
path.parent.mkdir(parents=True, exist_ok=True)
|
|
||||||
path.write_bytes(req.content)
|
|
||||||
|
|
||||||
|
|
||||||
def download_gfont(value):
|
def download_gfont(value):
|
||||||
name = (
|
name = (
|
||||||
f"{value[CONF_FAMILY]}:ital,wght@{int(value[CONF_ITALIC])},{value[CONF_WEIGHT]}"
|
f"{value[CONF_FAMILY]}:ital,wght@{int(value[CONF_ITALIC])},{value[CONF_WEIGHT]}"
|
||||||
|
@ -236,7 +210,7 @@ def download_gfont(value):
|
||||||
ttf_url = match.group(1)
|
ttf_url = match.group(1)
|
||||||
_LOGGER.debug("download_gfont: ttf_url=%s", ttf_url)
|
_LOGGER.debug("download_gfont: ttf_url=%s", ttf_url)
|
||||||
|
|
||||||
download_content(ttf_url, path)
|
external_files.download_content(ttf_url, path)
|
||||||
return value
|
return value
|
||||||
|
|
||||||
|
|
||||||
|
@ -244,7 +218,7 @@ def download_web_font(value):
|
||||||
url = value[CONF_URL]
|
url = value[CONF_URL]
|
||||||
path = get_font_path(value, TYPE_WEB)
|
path = get_font_path(value, TYPE_WEB)
|
||||||
|
|
||||||
download_content(url, path)
|
external_files.download_content(url, path)
|
||||||
_LOGGER.debug("download_web_font: path=%s", path)
|
_LOGGER.debug("download_web_font: path=%s", path)
|
||||||
return value
|
return value
|
||||||
|
|
||||||
|
|
|
@ -9,6 +9,10 @@ static const char *const TAG = "gpio.one_wire";
|
||||||
|
|
||||||
void GPIOOneWireBus::setup() {
|
void GPIOOneWireBus::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up 1-wire bus...");
|
ESP_LOGCONFIG(TAG, "Setting up 1-wire bus...");
|
||||||
|
this->t_pin_->setup();
|
||||||
|
// clear bus with 480µs high, otherwise initial reset in search might fail
|
||||||
|
this->t_pin_->pin_mode(gpio::FLAG_INPUT | gpio::FLAG_PULLUP);
|
||||||
|
delayMicroseconds(480);
|
||||||
this->search();
|
this->search();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -90,13 +94,15 @@ bool HOT IRAM_ATTR GPIOOneWireBus::read_bit_() {
|
||||||
|
|
||||||
// measure from start value directly, to get best accurate timing no matter
|
// measure from start value directly, to get best accurate timing no matter
|
||||||
// how long pin_mode/delayMicroseconds took
|
// how long pin_mode/delayMicroseconds took
|
||||||
delayMicroseconds(12 - (micros() - start));
|
uint32_t now = micros();
|
||||||
|
if (now - start < 12)
|
||||||
|
delayMicroseconds(12 - (now - start));
|
||||||
|
|
||||||
// sample bus to read bit from peer
|
// sample bus to read bit from peer
|
||||||
bool r = pin_.digital_read();
|
bool r = pin_.digital_read();
|
||||||
|
|
||||||
// read slot is at least 60µs; get as close to 60µs to spend less time with interrupts locked
|
// read slot is at least 60µs; get as close to 60µs to spend less time with interrupts locked
|
||||||
uint32_t now = micros();
|
now = micros();
|
||||||
if (now - start < 60)
|
if (now - start < 60)
|
||||||
delayMicroseconds(60 - (now - start));
|
delayMicroseconds(60 - (now - start));
|
||||||
|
|
||||||
|
|
|
@ -56,7 +56,7 @@ SENSOR_TYPES = {
|
||||||
|
|
||||||
CONFIG_SCHEMA = cv.Schema(
|
CONFIG_SCHEMA = cv.Schema(
|
||||||
{
|
{
|
||||||
cv.Required(CONF_HAIER_ID): cv.use_id(HonClimate),
|
cv.GenerateID(CONF_HAIER_ID): cv.use_id(HonClimate),
|
||||||
}
|
}
|
||||||
).extend({cv.Optional(type): schema for type, schema in SENSOR_TYPES.items()})
|
).extend({cv.Optional(type): schema for type, schema in SENSOR_TYPES.items()})
|
||||||
|
|
||||||
|
@ -64,8 +64,8 @@ CONFIG_SCHEMA = cv.Schema(
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
paren = await cg.get_variable(config[CONF_HAIER_ID])
|
paren = await cg.get_variable(config[CONF_HAIER_ID])
|
||||||
|
|
||||||
for type, _ in SENSOR_TYPES.items():
|
for type_ in SENSOR_TYPES:
|
||||||
if conf := config.get(type):
|
if conf := config.get(type_):
|
||||||
sens = await binary_sensor.new_binary_sensor(conf)
|
sens = await binary_sensor.new_binary_sensor(conf)
|
||||||
binary_sensor_type = getattr(BinarySensorTypeEnum, type.upper())
|
binary_sensor_type = getattr(BinarySensorTypeEnum, type_.upper())
|
||||||
cg.add(paren.set_sub_binary_sensor(binary_sensor_type, sens))
|
cg.add(paren.set_sub_binary_sensor(binary_sensor_type, sens))
|
||||||
|
|
|
@ -21,7 +21,7 @@ ICON_SPRAY_BOTTLE = "mdi:spray-bottle"
|
||||||
|
|
||||||
CONFIG_SCHEMA = cv.Schema(
|
CONFIG_SCHEMA = cv.Schema(
|
||||||
{
|
{
|
||||||
cv.Required(CONF_HAIER_ID): cv.use_id(HonClimate),
|
cv.GenerateID(CONF_HAIER_ID): cv.use_id(HonClimate),
|
||||||
cv.Optional(CONF_SELF_CLEANING): button.button_schema(
|
cv.Optional(CONF_SELF_CLEANING): button.button_schema(
|
||||||
SelfCleaningButton,
|
SelfCleaningButton,
|
||||||
icon=ICON_SPRAY_BOTTLE,
|
icon=ICON_SPRAY_BOTTLE,
|
||||||
|
|
|
@ -183,7 +183,6 @@ BASE_CONFIG_SCHEMA = (
|
||||||
cv.Optional(
|
cv.Optional(
|
||||||
CONF_SUPPORTED_SWING_MODES,
|
CONF_SUPPORTED_SWING_MODES,
|
||||||
default=[
|
default=[
|
||||||
"OFF",
|
|
||||||
"VERTICAL",
|
"VERTICAL",
|
||||||
"HORIZONTAL",
|
"HORIZONTAL",
|
||||||
"BOTH",
|
"BOTH",
|
||||||
|
@ -211,7 +210,7 @@ CONFIG_SCHEMA = cv.All(
|
||||||
): cv.boolean,
|
): cv.boolean,
|
||||||
cv.Optional(
|
cv.Optional(
|
||||||
CONF_SUPPORTED_PRESETS,
|
CONF_SUPPORTED_PRESETS,
|
||||||
default=list(["BOOST", "COMFORT"]), # No AWAY by default
|
default=["BOOST", "COMFORT"], # No AWAY by default
|
||||||
): cv.ensure_list(
|
): cv.ensure_list(
|
||||||
cv.enum(SUPPORTED_CLIMATE_PRESETS_SMARTAIR2_OPTIONS, upper=True)
|
cv.enum(SUPPORTED_CLIMATE_PRESETS_SMARTAIR2_OPTIONS, upper=True)
|
||||||
),
|
),
|
||||||
|
@ -231,7 +230,7 @@ CONFIG_SCHEMA = cv.All(
|
||||||
): cv.int_range(min=PROTOCOL_CONTROL_PACKET_SIZE, max=50),
|
): cv.int_range(min=PROTOCOL_CONTROL_PACKET_SIZE, max=50),
|
||||||
cv.Optional(
|
cv.Optional(
|
||||||
CONF_SUPPORTED_PRESETS,
|
CONF_SUPPORTED_PRESETS,
|
||||||
default=list(["BOOST", "ECO", "SLEEP"]), # No AWAY by default
|
default=["BOOST", "ECO", "SLEEP"], # No AWAY by default
|
||||||
): cv.ensure_list(
|
): cv.ensure_list(
|
||||||
cv.enum(SUPPORTED_CLIMATE_PRESETS_HON_OPTIONS, upper=True)
|
cv.enum(SUPPORTED_CLIMATE_PRESETS_HON_OPTIONS, upper=True)
|
||||||
),
|
),
|
||||||
|
@ -427,11 +426,7 @@ def _final_validate(config):
|
||||||
"No logger component found, logging for Haier protocol is disabled"
|
"No logger component found, logging for Haier protocol is disabled"
|
||||||
)
|
)
|
||||||
cg.add_build_flag("-DHAIER_LOG_LEVEL=0")
|
cg.add_build_flag("-DHAIER_LOG_LEVEL=0")
|
||||||
if (
|
if config.get(CONF_WIFI_SIGNAL) and CONF_WIFI not in full_config:
|
||||||
(CONF_WIFI_SIGNAL in config)
|
|
||||||
and (config[CONF_WIFI_SIGNAL])
|
|
||||||
and CONF_WIFI not in full_config
|
|
||||||
):
|
|
||||||
raise cv.Invalid(
|
raise cv.Invalid(
|
||||||
f"No WiFi configured, if you want to use haier climate without WiFi add {CONF_WIFI_SIGNAL}: false to climate configuration"
|
f"No WiFi configured, if you want to use haier climate without WiFi add {CONF_WIFI_SIGNAL}: false to climate configuration"
|
||||||
)
|
)
|
||||||
|
|
|
@ -137,16 +137,16 @@ SENSOR_TYPES = {
|
||||||
|
|
||||||
CONFIG_SCHEMA = cv.Schema(
|
CONFIG_SCHEMA = cv.Schema(
|
||||||
{
|
{
|
||||||
cv.Required(CONF_HAIER_ID): cv.use_id(HonClimate),
|
cv.GenerateID(CONF_HAIER_ID): cv.use_id(HonClimate),
|
||||||
}
|
}
|
||||||
).extend({cv.Optional(type): schema for type, schema in SENSOR_TYPES.items()})
|
).extend({cv.Optional(type_): schema for type_, schema in SENSOR_TYPES.items()})
|
||||||
|
|
||||||
|
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
paren = await cg.get_variable(config[CONF_HAIER_ID])
|
paren = await cg.get_variable(config[CONF_HAIER_ID])
|
||||||
|
|
||||||
for type, _ in SENSOR_TYPES.items():
|
for type_ in SENSOR_TYPES:
|
||||||
if conf := config.get(type):
|
if conf := config.get(type_):
|
||||||
sens = await sensor.new_sensor(conf)
|
sens = await sensor.new_sensor(conf)
|
||||||
sensor_type = getattr(SensorTypeEnum, type.upper())
|
sensor_type = getattr(SensorTypeEnum, type_.upper())
|
||||||
cg.add(paren.set_sub_sensor(sensor_type, sens))
|
cg.add(paren.set_sub_sensor(sensor_type, sens))
|
||||||
|
|
|
@ -39,7 +39,7 @@ TEXT_SENSOR_TYPES = {
|
||||||
|
|
||||||
CONFIG_SCHEMA = cv.Schema(
|
CONFIG_SCHEMA = cv.Schema(
|
||||||
{
|
{
|
||||||
cv.Required(CONF_HAIER_ID): cv.use_id(HonClimate),
|
cv.GenerateID(CONF_HAIER_ID): cv.use_id(HonClimate),
|
||||||
}
|
}
|
||||||
).extend({cv.Optional(type): schema for type, schema in TEXT_SENSOR_TYPES.items()})
|
).extend({cv.Optional(type): schema for type, schema in TEXT_SENSOR_TYPES.items()})
|
||||||
|
|
||||||
|
@ -47,8 +47,8 @@ CONFIG_SCHEMA = cv.Schema(
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
paren = await cg.get_variable(config[CONF_HAIER_ID])
|
paren = await cg.get_variable(config[CONF_HAIER_ID])
|
||||||
|
|
||||||
for type, _ in TEXT_SENSOR_TYPES.items():
|
for type_ in TEXT_SENSOR_TYPES:
|
||||||
if conf := config.get(type):
|
if conf := config.get(type_):
|
||||||
sens = await text_sensor.new_text_sensor(conf)
|
sens = await text_sensor.new_text_sensor(conf)
|
||||||
text_sensor_type = getattr(TextSensorTypeEnum, type.upper())
|
text_sensor_type = getattr(TextSensorTypeEnum, type_.upper())
|
||||||
cg.add(paren.set_sub_text_sensor(text_sensor_type, sens))
|
cg.add(paren.set_sub_text_sensor(text_sensor_type, sens))
|
||||||
|
|
|
@ -34,6 +34,7 @@ PROTOCOLS = {
|
||||||
"greeyan": Protocol.PROTOCOL_GREEYAN,
|
"greeyan": Protocol.PROTOCOL_GREEYAN,
|
||||||
"greeyac": Protocol.PROTOCOL_GREEYAC,
|
"greeyac": Protocol.PROTOCOL_GREEYAC,
|
||||||
"greeyt": Protocol.PROTOCOL_GREEYT,
|
"greeyt": Protocol.PROTOCOL_GREEYT,
|
||||||
|
"greeyap": Protocol.PROTOCOL_GREEYAP,
|
||||||
"hisense_aud": Protocol.PROTOCOL_HISENSE_AUD,
|
"hisense_aud": Protocol.PROTOCOL_HISENSE_AUD,
|
||||||
"hitachi": Protocol.PROTOCOL_HITACHI,
|
"hitachi": Protocol.PROTOCOL_HITACHI,
|
||||||
"hyundai": Protocol.PROTOCOL_HYUNDAI,
|
"hyundai": Protocol.PROTOCOL_HYUNDAI,
|
||||||
|
@ -61,6 +62,11 @@ PROTOCOLS = {
|
||||||
"toshiba_daiseikai": Protocol.PROTOCOL_TOSHIBA_DAISEIKAI,
|
"toshiba_daiseikai": Protocol.PROTOCOL_TOSHIBA_DAISEIKAI,
|
||||||
"toshiba": Protocol.PROTOCOL_TOSHIBA,
|
"toshiba": Protocol.PROTOCOL_TOSHIBA,
|
||||||
"zhlt01": Protocol.PROTOCOL_ZHLT01,
|
"zhlt01": Protocol.PROTOCOL_ZHLT01,
|
||||||
|
"nibe": Protocol.PROTOCOL_NIBE,
|
||||||
|
"carrier_qlima_1": Protocol.PROTOCOL_QLIMA_1,
|
||||||
|
"carrier_qlima_2": Protocol.PROTOCOL_QLIMA_2,
|
||||||
|
"samsung_aqv12msan": Protocol.PROTOCOL_SAMSUNG_AQV12MSAN,
|
||||||
|
"zhjg01": Protocol.PROTOCOL_ZHJG01,
|
||||||
}
|
}
|
||||||
|
|
||||||
CONF_HORIZONTAL_DEFAULT = "horizontal_default"
|
CONF_HORIZONTAL_DEFAULT = "horizontal_default"
|
||||||
|
@ -116,7 +122,7 @@ def to_code(config):
|
||||||
cg.add(var.set_max_temperature(config[CONF_MAX_TEMPERATURE]))
|
cg.add(var.set_max_temperature(config[CONF_MAX_TEMPERATURE]))
|
||||||
cg.add(var.set_min_temperature(config[CONF_MIN_TEMPERATURE]))
|
cg.add(var.set_min_temperature(config[CONF_MIN_TEMPERATURE]))
|
||||||
|
|
||||||
cg.add_library("tonia/HeatpumpIR", "1.0.23")
|
cg.add_library("tonia/HeatpumpIR", "1.0.26")
|
||||||
|
|
||||||
if CORE.is_esp8266 or CORE.is_esp32:
|
if CORE.is_esp8266 or CORE.is_esp32:
|
||||||
cg.add_library("crankyoldgit/IRremoteESP8266", "2.8.4")
|
cg.add_library("crankyoldgit/IRremoteESP8266", "2.8.6")
|
||||||
|
|
|
@ -28,6 +28,7 @@ const std::map<Protocol, std::function<HeatpumpIR *()>> PROTOCOL_CONSTRUCTOR_MAP
|
||||||
{PROTOCOL_GREEYAN, []() { return new GreeYANHeatpumpIR(); }}, // NOLINT
|
{PROTOCOL_GREEYAN, []() { return new GreeYANHeatpumpIR(); }}, // NOLINT
|
||||||
{PROTOCOL_GREEYAC, []() { return new GreeYACHeatpumpIR(); }}, // NOLINT
|
{PROTOCOL_GREEYAC, []() { return new GreeYACHeatpumpIR(); }}, // NOLINT
|
||||||
{PROTOCOL_GREEYT, []() { return new GreeYTHeatpumpIR(); }}, // NOLINT
|
{PROTOCOL_GREEYT, []() { return new GreeYTHeatpumpIR(); }}, // NOLINT
|
||||||
|
{PROTOCOL_GREEYAP, []() { return new GreeYAPHeatpumpIR(); }}, // NOLINT
|
||||||
{PROTOCOL_HISENSE_AUD, []() { return new HisenseHeatpumpIR(); }}, // NOLINT
|
{PROTOCOL_HISENSE_AUD, []() { return new HisenseHeatpumpIR(); }}, // NOLINT
|
||||||
{PROTOCOL_HITACHI, []() { return new HitachiHeatpumpIR(); }}, // NOLINT
|
{PROTOCOL_HITACHI, []() { return new HitachiHeatpumpIR(); }}, // NOLINT
|
||||||
{PROTOCOL_HYUNDAI, []() { return new HyundaiHeatpumpIR(); }}, // NOLINT
|
{PROTOCOL_HYUNDAI, []() { return new HyundaiHeatpumpIR(); }}, // NOLINT
|
||||||
|
@ -55,6 +56,11 @@ const std::map<Protocol, std::function<HeatpumpIR *()>> PROTOCOL_CONSTRUCTOR_MAP
|
||||||
{PROTOCOL_TOSHIBA_DAISEIKAI, []() { return new ToshibaDaiseikaiHeatpumpIR(); }}, // NOLINT
|
{PROTOCOL_TOSHIBA_DAISEIKAI, []() { return new ToshibaDaiseikaiHeatpumpIR(); }}, // NOLINT
|
||||||
{PROTOCOL_TOSHIBA, []() { return new ToshibaHeatpumpIR(); }}, // NOLINT
|
{PROTOCOL_TOSHIBA, []() { return new ToshibaHeatpumpIR(); }}, // NOLINT
|
||||||
{PROTOCOL_ZHLT01, []() { return new ZHLT01HeatpumpIR(); }}, // NOLINT
|
{PROTOCOL_ZHLT01, []() { return new ZHLT01HeatpumpIR(); }}, // NOLINT
|
||||||
|
{PROTOCOL_NIBE, []() { return new NibeHeatpumpIR(); }}, // NOLINT
|
||||||
|
{PROTOCOL_QLIMA_1, []() { return new Qlima1HeatpumpIR(); }}, // NOLINT
|
||||||
|
{PROTOCOL_QLIMA_2, []() { return new Qlima2HeatpumpIR(); }}, // NOLINT
|
||||||
|
{PROTOCOL_SAMSUNG_AQV12MSAN, []() { return new SamsungAQV12MSANHeatpumpIR(); }}, // NOLINT
|
||||||
|
{PROTOCOL_ZHJG01, []() { return new ZHJG01HeatpumpIR(); }}, // NOLINT
|
||||||
};
|
};
|
||||||
|
|
||||||
void HeatpumpIRClimate::setup() {
|
void HeatpumpIRClimate::setup() {
|
||||||
|
|
|
@ -28,6 +28,7 @@ enum Protocol {
|
||||||
PROTOCOL_GREEYAN,
|
PROTOCOL_GREEYAN,
|
||||||
PROTOCOL_GREEYAC,
|
PROTOCOL_GREEYAC,
|
||||||
PROTOCOL_GREEYT,
|
PROTOCOL_GREEYT,
|
||||||
|
PROTOCOL_GREEYAP,
|
||||||
PROTOCOL_HISENSE_AUD,
|
PROTOCOL_HISENSE_AUD,
|
||||||
PROTOCOL_HITACHI,
|
PROTOCOL_HITACHI,
|
||||||
PROTOCOL_HYUNDAI,
|
PROTOCOL_HYUNDAI,
|
||||||
|
@ -55,6 +56,11 @@ enum Protocol {
|
||||||
PROTOCOL_TOSHIBA_DAISEIKAI,
|
PROTOCOL_TOSHIBA_DAISEIKAI,
|
||||||
PROTOCOL_TOSHIBA,
|
PROTOCOL_TOSHIBA,
|
||||||
PROTOCOL_ZHLT01,
|
PROTOCOL_ZHLT01,
|
||||||
|
PROTOCOL_NIBE,
|
||||||
|
PROTOCOL_QLIMA_1,
|
||||||
|
PROTOCOL_QLIMA_2,
|
||||||
|
PROTOCOL_SAMSUNG_AQV12MSAN,
|
||||||
|
PROTOCOL_ZHJG01,
|
||||||
};
|
};
|
||||||
|
|
||||||
// Simple enum to represent horizontal directios
|
// Simple enum to represent horizontal directios
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -149,11 +149,21 @@ template<typename... Ts> class HttpRequestSendAction : public Action<Ts...> {
|
||||||
}
|
}
|
||||||
response_body.reserve(read_index);
|
response_body.reserve(read_index);
|
||||||
response_body.assign((char *) buf, read_index);
|
response_body.assign((char *) buf, read_index);
|
||||||
|
allocator.deallocate(buf, max_length);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this->response_triggers_.size() == 1) {
|
||||||
|
// 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_) {
|
for (auto *trigger : this->response_triggers_) {
|
||||||
trigger->process(container, response_body);
|
// 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();
|
||||||
}
|
}
|
||||||
|
|
|
@ -90,7 +90,7 @@ std::shared_ptr<HttpContainer> HttpRequestIDF::start(std::string url, std::strin
|
||||||
int write_left = body_len;
|
int write_left = body_len;
|
||||||
int write_index = 0;
|
int write_index = 0;
|
||||||
const char *buf = body.c_str();
|
const char *buf = body.c_str();
|
||||||
while (body_len > 0) {
|
while (write_left > 0) {
|
||||||
int written = esp_http_client_write(client, buf + write_index, write_left);
|
int written = esp_http_client_write(client, buf + write_index, write_left);
|
||||||
if (written < 0) {
|
if (written < 0) {
|
||||||
err = ESP_FAIL;
|
err = ESP_FAIL;
|
||||||
|
|
|
@ -46,7 +46,7 @@ void WatchdogManager::set_timeout_(uint32_t timeout_ms) {
|
||||||
};
|
};
|
||||||
esp_task_wdt_reconfigure(&wdt_config);
|
esp_task_wdt_reconfigure(&wdt_config);
|
||||||
#else
|
#else
|
||||||
esp_task_wdt_init(timeout_ms, true);
|
esp_task_wdt_init(timeout_ms / 1000, true);
|
||||||
#endif // ESP_IDF_VERSION_MAJOR
|
#endif // ESP_IDF_VERSION_MAJOR
|
||||||
#endif // USE_ESP32
|
#endif // USE_ESP32
|
||||||
|
|
||||||
|
|
|
@ -6,7 +6,6 @@ import hashlib
|
||||||
import io
|
import io
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
import re
|
import re
|
||||||
import requests
|
|
||||||
from magic import Magic
|
from magic import Magic
|
||||||
|
|
||||||
from esphome import core
|
from esphome import core
|
||||||
|
@ -15,7 +14,6 @@ from esphome import external_files
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
__version__,
|
|
||||||
CONF_DITHER,
|
CONF_DITHER,
|
||||||
CONF_FILE,
|
CONF_FILE,
|
||||||
CONF_ICON,
|
CONF_ICON,
|
||||||
|
@ -75,31 +73,6 @@ def compute_local_image_path(value: dict) -> Path:
|
||||||
return base_dir / key
|
return base_dir / key
|
||||||
|
|
||||||
|
|
||||||
def download_content(url: str, path: Path) -> None:
|
|
||||||
if not external_files.has_remote_file_changed(url, path):
|
|
||||||
_LOGGER.debug("Remote file has not changed %s", url)
|
|
||||||
return
|
|
||||||
|
|
||||||
_LOGGER.debug(
|
|
||||||
"Remote file has changed, downloading from %s to %s",
|
|
||||||
url,
|
|
||||||
path,
|
|
||||||
)
|
|
||||||
|
|
||||||
try:
|
|
||||||
req = requests.get(
|
|
||||||
url,
|
|
||||||
timeout=IMAGE_DOWNLOAD_TIMEOUT,
|
|
||||||
headers={"User-agent": f"ESPHome/{__version__} (https://esphome.io)"},
|
|
||||||
)
|
|
||||||
req.raise_for_status()
|
|
||||||
except requests.exceptions.RequestException as e:
|
|
||||||
raise cv.Invalid(f"Could not download from {url}: {e}")
|
|
||||||
|
|
||||||
path.parent.mkdir(parents=True, exist_ok=True)
|
|
||||||
path.write_bytes(req.content)
|
|
||||||
|
|
||||||
|
|
||||||
def download_mdi(value):
|
def download_mdi(value):
|
||||||
validate_cairosvg_installed(value)
|
validate_cairosvg_installed(value)
|
||||||
|
|
||||||
|
@ -108,7 +81,7 @@ def download_mdi(value):
|
||||||
|
|
||||||
url = f"https://raw.githubusercontent.com/Templarian/MaterialDesign/master/svg/{mdi_id}.svg"
|
url = f"https://raw.githubusercontent.com/Templarian/MaterialDesign/master/svg/{mdi_id}.svg"
|
||||||
|
|
||||||
download_content(url, path)
|
external_files.download_content(url, path, IMAGE_DOWNLOAD_TIMEOUT)
|
||||||
|
|
||||||
return value
|
return value
|
||||||
|
|
||||||
|
@ -117,7 +90,7 @@ def download_image(value):
|
||||||
url = value[CONF_URL]
|
url = value[CONF_URL]
|
||||||
path = compute_local_image_path(value)
|
path = compute_local_image_path(value)
|
||||||
|
|
||||||
download_content(url, path)
|
external_files.download_content(url, path, IMAGE_DOWNLOAD_TIMEOUT)
|
||||||
|
|
||||||
return value
|
return value
|
||||||
|
|
||||||
|
|
|
@ -57,7 +57,7 @@ optional<uint8_t> ImprovSerialComponent::read_byte_() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
#if defined(CONFIG_ESP_CONSOLE_USB_CDC) && (defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3))
|
#ifdef USE_LOGGER_USB_CDC
|
||||||
case logger::UART_SELECTION_USB_CDC:
|
case logger::UART_SELECTION_USB_CDC:
|
||||||
#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 0, 0)
|
#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 0, 0)
|
||||||
if (esp_usb_console_available_for_read()) {
|
if (esp_usb_console_available_for_read()) {
|
||||||
|
@ -68,15 +68,15 @@ optional<uint8_t> ImprovSerialComponent::read_byte_() {
|
||||||
byte = data;
|
byte = data;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
#endif // USE_ESP32_VARIANT_ESP32S2 || USE_ESP32_VARIANT_ESP32S3
|
#endif // USE_LOGGER_USB_CDC
|
||||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32C6) || defined(USE_ESP32_VARIANT_ESP32S3)
|
#ifdef USE_LOGGER_USB_SERIAL_JTAG
|
||||||
case logger::UART_SELECTION_USB_SERIAL_JTAG: {
|
case logger::UART_SELECTION_USB_SERIAL_JTAG: {
|
||||||
if (usb_serial_jtag_read_bytes((char *) &data, 1, 0)) {
|
if (usb_serial_jtag_read_bytes((char *) &data, 1, 0)) {
|
||||||
byte = data;
|
byte = data;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
#endif // USE_ESP32_VARIANT_ESP32C3 || USE_ESP32_VARIANT_ESP32C6 || USE_ESP32_VARIANT_ESP32S3
|
#endif // USE_LOGGER_USB_SERIAL_JTAG
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -99,19 +99,19 @@ void ImprovSerialComponent::write_data_(std::vector<uint8_t> &data) {
|
||||||
#endif // !USE_ESP32_VARIANT_ESP32C3 && !USE_ESP32_VARIANT_ESP32S2 && !USE_ESP32_VARIANT_ESP32S3
|
#endif // !USE_ESP32_VARIANT_ESP32C3 && !USE_ESP32_VARIANT_ESP32S2 && !USE_ESP32_VARIANT_ESP32S3
|
||||||
uart_write_bytes(this->uart_num_, data.data(), data.size());
|
uart_write_bytes(this->uart_num_, data.data(), data.size());
|
||||||
break;
|
break;
|
||||||
#if defined(CONFIG_ESP_CONSOLE_USB_CDC) && (defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3))
|
#ifdef USE_LOGGER_USB_CDC
|
||||||
case logger::UART_SELECTION_USB_CDC: {
|
case logger::UART_SELECTION_USB_CDC: {
|
||||||
const char *msg = (char *) data.data();
|
const char *msg = (char *) data.data();
|
||||||
esp_usb_console_write_buf(msg, data.size());
|
esp_usb_console_write_buf(msg, data.size());
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
#endif // USE_ESP32_VARIANT_ESP32S2 || USE_ESP32_VARIANT_ESP32S3
|
#endif // USE_LOGGER_USB_CDC
|
||||||
#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32C6) || defined(USE_ESP32_VARIANT_ESP32S3)
|
#ifdef USE_LOGGER_USB_SERIAL_JTAG
|
||||||
case logger::UART_SELECTION_USB_SERIAL_JTAG:
|
case logger::UART_SELECTION_USB_SERIAL_JTAG:
|
||||||
usb_serial_jtag_write_bytes((char *) data.data(), data.size(), 20 / portTICK_PERIOD_MS);
|
usb_serial_jtag_write_bytes((char *) data.data(), data.size(), 20 / portTICK_PERIOD_MS);
|
||||||
usb_serial_jtag_ll_txfifo_flush(); // fixes for issue in IDF 4.4.7
|
usb_serial_jtag_ll_txfifo_flush(); // fixes for issue in IDF 4.4.7
|
||||||
break;
|
break;
|
||||||
#endif // USE_ESP32_VARIANT_ESP32C3 || USE_ESP32_VARIANT_ESP32S3
|
#endif // USE_LOGGER_USB_SERIAL_JTAG
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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_);
|
||||||
|
|
|
@ -74,6 +74,9 @@ def mdns_service(
|
||||||
|
|
||||||
@coroutine_with_priority(55.0)
|
@coroutine_with_priority(55.0)
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
|
if config[CONF_DISABLED] is True:
|
||||||
|
return
|
||||||
|
|
||||||
if CORE.using_arduino:
|
if CORE.using_arduino:
|
||||||
if CORE.is_esp32:
|
if CORE.is_esp32:
|
||||||
cg.add_library("ESPmDNS", None)
|
cg.add_library("ESPmDNS", None)
|
||||||
|
@ -92,9 +95,6 @@ async def to_code(config):
|
||||||
path="components/mdns",
|
path="components/mdns",
|
||||||
)
|
)
|
||||||
|
|
||||||
if config[CONF_DISABLED]:
|
|
||||||
return
|
|
||||||
|
|
||||||
cg.add_define("USE_MDNS")
|
cg.add_define("USE_MDNS")
|
||||||
|
|
||||||
var = cg.new_Pvariable(config[CONF_ID])
|
var = cg.new_Pvariable(config[CONF_ID])
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
#include "mdns_component.h"
|
|
||||||
#include "esphome/core/defines.h"
|
#include "esphome/core/defines.h"
|
||||||
|
#ifdef USE_MDNS
|
||||||
|
#include "mdns_component.h"
|
||||||
#include "esphome/core/version.h"
|
#include "esphome/core/version.h"
|
||||||
#include "esphome/core/application.h"
|
#include "esphome/core/application.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
@ -125,3 +126,4 @@ void MDNSComponent::dump_config() {
|
||||||
|
|
||||||
} // namespace mdns
|
} // namespace mdns
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
#endif
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
#include "esphome/core/defines.h"
|
||||||
|
#ifdef USE_MDNS
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include "esphome/core/component.h"
|
#include "esphome/core/component.h"
|
||||||
|
@ -46,3 +47,4 @@ class MDNSComponent : public Component {
|
||||||
|
|
||||||
} // namespace mdns
|
} // namespace mdns
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
#endif
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
#ifdef USE_ESP32
|
#include "esphome/core/defines.h"
|
||||||
|
#if defined(USE_ESP32) && defined(USE_MDNS)
|
||||||
|
|
||||||
#include <mdns.h>
|
#include <mdns.h>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
#if defined(USE_ESP8266) && defined(USE_ARDUINO)
|
#include "esphome/core/defines.h"
|
||||||
|
#if defined(USE_ESP8266) && defined(USE_ARDUINO) && defined(USE_MDNS)
|
||||||
|
|
||||||
#include <ESP8266mDNS.h>
|
#include <ESP8266mDNS.h>
|
||||||
#include "esphome/components/network/ip_address.h"
|
#include "esphome/components/network/ip_address.h"
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
#ifdef USE_HOST
|
#include "esphome/core/defines.h"
|
||||||
|
#if defined(USE_HOST) && defined(USE_MDNS)
|
||||||
|
|
||||||
#include "esphome/components/network/ip_address.h"
|
#include "esphome/components/network/ip_address.h"
|
||||||
#include "esphome/components/network/util.h"
|
#include "esphome/components/network/util.h"
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
#ifdef USE_LIBRETINY
|
#include "esphome/core/defines.h"
|
||||||
|
#if defined(USE_LIBRETINY) && defined(USE_MDNS)
|
||||||
|
|
||||||
#include "esphome/components/network/ip_address.h"
|
#include "esphome/components/network/ip_address.h"
|
||||||
#include "esphome/components/network/util.h"
|
#include "esphome/components/network/util.h"
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
#ifdef USE_RP2040
|
#include "esphome/core/defines.h"
|
||||||
|
#if defined(USE_RP2040) && defined(USE_MDNS)
|
||||||
|
|
||||||
#include "esphome/components/network/ip_address.h"
|
#include "esphome/components/network/ip_address.h"
|
||||||
#include "esphome/components/network/util.h"
|
#include "esphome/components/network/util.h"
|
||||||
|
|
|
@ -293,4 +293,4 @@ async def to_code(config):
|
||||||
if CONF_HUMIDITY_SETPOINT in config:
|
if CONF_HUMIDITY_SETPOINT in config:
|
||||||
sens = await sensor.new_sensor(config[CONF_HUMIDITY_SETPOINT])
|
sens = await sensor.new_sensor(config[CONF_HUMIDITY_SETPOINT])
|
||||||
cg.add(var.set_humidity_setpoint_sensor(sens))
|
cg.add(var.set_humidity_setpoint_sensor(sens))
|
||||||
cg.add_library("dudanov/MideaUART", "1.1.8")
|
cg.add_library("dudanov/MideaUART", "1.1.9")
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -37,6 +37,7 @@ RAW_ENCODING = {
|
||||||
"NONE": RawEncoding.NONE,
|
"NONE": RawEncoding.NONE,
|
||||||
"HEXBYTES": RawEncoding.HEXBYTES,
|
"HEXBYTES": RawEncoding.HEXBYTES,
|
||||||
"COMMA": RawEncoding.COMMA,
|
"COMMA": RawEncoding.COMMA,
|
||||||
|
"ANSI": RawEncoding.ANSI,
|
||||||
}
|
}
|
||||||
|
|
||||||
CONFIG_SCHEMA = cv.All(
|
CONFIG_SCHEMA = cv.All(
|
||||||
|
@ -49,7 +50,7 @@ CONFIG_SCHEMA = cv.All(
|
||||||
cv.Optional(CONF_REGISTER_TYPE): cv.enum(MODBUS_REGISTER_TYPE),
|
cv.Optional(CONF_REGISTER_TYPE): cv.enum(MODBUS_REGISTER_TYPE),
|
||||||
cv.Optional(CONF_REGISTER_COUNT, default=0): cv.positive_int,
|
cv.Optional(CONF_REGISTER_COUNT, default=0): cv.positive_int,
|
||||||
cv.Optional(CONF_RESPONSE_SIZE, default=2): cv.positive_int,
|
cv.Optional(CONF_RESPONSE_SIZE, default=2): cv.positive_int,
|
||||||
cv.Optional(CONF_RAW_ENCODE, default="NONE"): cv.enum(RAW_ENCODING),
|
cv.Optional(CONF_RAW_ENCODE, default="ANSI"): cv.enum(RAW_ENCODING),
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
validate_modbus_register,
|
validate_modbus_register,
|
||||||
|
|
|
@ -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_) {
|
||||||
|
@ -27,8 +27,11 @@ void ModbusTextSensor::parse_and_publish(const std::vector<uint8_t> &data) {
|
||||||
sprintf(buffer, index != this->offset ? ",%d" : "%d", b);
|
sprintf(buffer, index != this->offset ? ",%d" : "%d", b);
|
||||||
output << buffer;
|
output << buffer;
|
||||||
break;
|
break;
|
||||||
|
case RawEncoding::ANSI:
|
||||||
|
if (b < 0x20)
|
||||||
|
break;
|
||||||
|
// FALLTHROUGH
|
||||||
// Anything else no encoding
|
// Anything else no encoding
|
||||||
case RawEncoding::NONE:
|
|
||||||
default:
|
default:
|
||||||
output << (char) b;
|
output << (char) b;
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -9,7 +9,7 @@
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace modbus_controller {
|
namespace modbus_controller {
|
||||||
|
|
||||||
enum class RawEncoding { NONE = 0, HEXBYTES = 1, COMMA = 2 };
|
enum class RawEncoding { NONE = 0, HEXBYTES = 1, COMMA = 2, ANSI = 3 };
|
||||||
|
|
||||||
class ModbusTextSensor : public Component, public text_sensor::TextSensor, public SensorItem {
|
class ModbusTextSensor : public Component, public text_sensor::TextSensor, public SensorItem {
|
||||||
public:
|
public:
|
||||||
|
|
|
@ -27,7 +27,7 @@ async def to_code(config):
|
||||||
var = await binary_sensor.new_binary_sensor(config)
|
var = await binary_sensor.new_binary_sensor(config)
|
||||||
hub = await cg.get_variable(config[CONF_MPR121_ID])
|
hub = await cg.get_variable(config[CONF_MPR121_ID])
|
||||||
cg.add(var.set_channel(config[CONF_CHANNEL]))
|
cg.add(var.set_channel(config[CONF_CHANNEL]))
|
||||||
cg.register_parented(var, hub)
|
await cg.register_parented(var, hub)
|
||||||
|
|
||||||
if CONF_TOUCH_THRESHOLD in config:
|
if CONF_TOUCH_THRESHOLD in config:
|
||||||
cg.add(var.set_touch_threshold(config[CONF_TOUCH_THRESHOLD]))
|
cg.add(var.set_touch_threshold(config[CONF_TOUCH_THRESHOLD]))
|
||||||
|
|
|
@ -26,6 +26,7 @@ from esphome.const import (
|
||||||
DEVICE_CLASS_BATTERY,
|
DEVICE_CLASS_BATTERY,
|
||||||
DEVICE_CLASS_CARBON_DIOXIDE,
|
DEVICE_CLASS_CARBON_DIOXIDE,
|
||||||
DEVICE_CLASS_CARBON_MONOXIDE,
|
DEVICE_CLASS_CARBON_MONOXIDE,
|
||||||
|
DEVICE_CLASS_CONDUCTIVITY,
|
||||||
DEVICE_CLASS_CURRENT,
|
DEVICE_CLASS_CURRENT,
|
||||||
DEVICE_CLASS_DATA_RATE,
|
DEVICE_CLASS_DATA_RATE,
|
||||||
DEVICE_CLASS_DATA_SIZE,
|
DEVICE_CLASS_DATA_SIZE,
|
||||||
|
@ -82,6 +83,7 @@ DEVICE_CLASSES = [
|
||||||
DEVICE_CLASS_BATTERY,
|
DEVICE_CLASS_BATTERY,
|
||||||
DEVICE_CLASS_CARBON_DIOXIDE,
|
DEVICE_CLASS_CARBON_DIOXIDE,
|
||||||
DEVICE_CLASS_CARBON_MONOXIDE,
|
DEVICE_CLASS_CARBON_MONOXIDE,
|
||||||
|
DEVICE_CLASS_CONDUCTIVITY,
|
||||||
DEVICE_CLASS_CURRENT,
|
DEVICE_CLASS_CURRENT,
|
||||||
DEVICE_CLASS_DATA_RATE,
|
DEVICE_CLASS_DATA_RATE,
|
||||||
DEVICE_CLASS_DATA_SIZE,
|
DEVICE_CLASS_DATA_SIZE,
|
||||||
|
|
|
@ -65,13 +65,10 @@ class QspiAmoLed : public display::DisplayBuffer,
|
||||||
|
|
||||||
void set_reset_pin(GPIOPin *reset_pin) { this->reset_pin_ = reset_pin; }
|
void set_reset_pin(GPIOPin *reset_pin) { this->reset_pin_ = reset_pin; }
|
||||||
void set_enable_pin(GPIOPin *enable_pin) { this->enable_pin_ = enable_pin; }
|
void set_enable_pin(GPIOPin *enable_pin) { this->enable_pin_ = enable_pin; }
|
||||||
void set_width(uint16_t width) { this->width_ = width; }
|
|
||||||
void set_dimensions(uint16_t width, uint16_t height) {
|
void set_dimensions(uint16_t width, uint16_t height) {
|
||||||
this->width_ = width;
|
this->width_ = width;
|
||||||
this->height_ = height;
|
this->height_ = height;
|
||||||
}
|
}
|
||||||
int get_width() override { return this->width_; }
|
|
||||||
int get_height() override { return this->height_; }
|
|
||||||
void set_invert_colors(bool invert_colors) {
|
void set_invert_colors(bool invert_colors) {
|
||||||
this->invert_colors_ = invert_colors;
|
this->invert_colors_ = invert_colors;
|
||||||
this->reset_params_();
|
this->reset_params_();
|
||||||
|
|
|
@ -56,9 +56,7 @@ 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])
|
var = cg.new_Pvariable(config[CONF_ID])
|
||||||
await cg.register_component(var, config)
|
await cg.register_component(var, config)
|
||||||
|
|
||||||
|
@ -72,5 +70,6 @@ async def to_code(config):
|
||||||
config[CONF_BOOT_IS_GOOD_AFTER],
|
config[CONF_BOOT_IS_GOOD_AFTER],
|
||||||
)
|
)
|
||||||
cg.add(RawExpression(f"if ({condition}) return"))
|
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
|
||||||
|
|
|
@ -88,7 +88,7 @@ def validate_parameter_name(value):
|
||||||
raise cv.Invalid(f"Script's parameter name cannot be {CONF_ID}")
|
raise cv.Invalid(f"Script's parameter name cannot be {CONF_ID}")
|
||||||
|
|
||||||
|
|
||||||
ALLOWED_PARAM_TYPE_CHARSET = set("abcdefghijklmnopqrstuvwxyz0123456789_:*&[]")
|
ALLOWED_PARAM_TYPE_CHARSET = set("abcdefghijklmnopqrstuvwxyz0123456789_:*&[]<>")
|
||||||
|
|
||||||
|
|
||||||
def validate_parameter_type(value):
|
def validate_parameter_type(value):
|
||||||
|
|
|
@ -43,6 +43,7 @@ from esphome.const import (
|
||||||
DEVICE_CLASS_BATTERY,
|
DEVICE_CLASS_BATTERY,
|
||||||
DEVICE_CLASS_CARBON_DIOXIDE,
|
DEVICE_CLASS_CARBON_DIOXIDE,
|
||||||
DEVICE_CLASS_CARBON_MONOXIDE,
|
DEVICE_CLASS_CARBON_MONOXIDE,
|
||||||
|
DEVICE_CLASS_CONDUCTIVITY,
|
||||||
DEVICE_CLASS_CURRENT,
|
DEVICE_CLASS_CURRENT,
|
||||||
DEVICE_CLASS_DATA_RATE,
|
DEVICE_CLASS_DATA_RATE,
|
||||||
DEVICE_CLASS_DATA_SIZE,
|
DEVICE_CLASS_DATA_SIZE,
|
||||||
|
@ -103,6 +104,7 @@ DEVICE_CLASSES = [
|
||||||
DEVICE_CLASS_BATTERY,
|
DEVICE_CLASS_BATTERY,
|
||||||
DEVICE_CLASS_CARBON_DIOXIDE,
|
DEVICE_CLASS_CARBON_DIOXIDE,
|
||||||
DEVICE_CLASS_CARBON_MONOXIDE,
|
DEVICE_CLASS_CARBON_MONOXIDE,
|
||||||
|
DEVICE_CLASS_CONDUCTIVITY,
|
||||||
DEVICE_CLASS_CURRENT,
|
DEVICE_CLASS_CURRENT,
|
||||||
DEVICE_CLASS_DATA_RATE,
|
DEVICE_CLASS_DATA_RATE,
|
||||||
DEVICE_CLASS_DATA_SIZE,
|
DEVICE_CLASS_DATA_SIZE,
|
||||||
|
|
|
@ -223,13 +223,19 @@ void Tuya::handle_command_(uint8_t command, uint8_t version, const uint8_t *buff
|
||||||
break;
|
break;
|
||||||
case TuyaCommandType::DATAPOINT_DELIVER:
|
case TuyaCommandType::DATAPOINT_DELIVER:
|
||||||
break;
|
break;
|
||||||
case TuyaCommandType::DATAPOINT_REPORT:
|
case TuyaCommandType::DATAPOINT_REPORT_ASYNC:
|
||||||
|
case TuyaCommandType::DATAPOINT_REPORT_SYNC:
|
||||||
if (this->init_state_ == TuyaInitState::INIT_DATAPOINT) {
|
if (this->init_state_ == TuyaInitState::INIT_DATAPOINT) {
|
||||||
this->init_state_ = TuyaInitState::INIT_DONE;
|
this->init_state_ = TuyaInitState::INIT_DONE;
|
||||||
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_datapoints_(buffer, len);
|
this->handle_datapoints_(buffer, len);
|
||||||
|
|
||||||
|
if (command_type == TuyaCommandType::DATAPOINT_REPORT_SYNC) {
|
||||||
|
this->send_command_(
|
||||||
|
TuyaCommand{.cmd = TuyaCommandType::DATAPOINT_REPORT_ACK, .payload = std::vector<uint8_t>{0x01}});
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case TuyaCommandType::DATAPOINT_QUERY:
|
case TuyaCommandType::DATAPOINT_QUERY:
|
||||||
break;
|
break;
|
||||||
|
@ -423,7 +429,7 @@ void Tuya::send_raw_command_(TuyaCommand command) {
|
||||||
break;
|
break;
|
||||||
case TuyaCommandType::DATAPOINT_DELIVER:
|
case TuyaCommandType::DATAPOINT_DELIVER:
|
||||||
case TuyaCommandType::DATAPOINT_QUERY:
|
case TuyaCommandType::DATAPOINT_QUERY:
|
||||||
this->expected_response_ = TuyaCommandType::DATAPOINT_REPORT;
|
this->expected_response_ = TuyaCommandType::DATAPOINT_REPORT_ASYNC;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -53,10 +53,12 @@ enum class TuyaCommandType : uint8_t {
|
||||||
WIFI_RESET = 0x04,
|
WIFI_RESET = 0x04,
|
||||||
WIFI_SELECT = 0x05,
|
WIFI_SELECT = 0x05,
|
||||||
DATAPOINT_DELIVER = 0x06,
|
DATAPOINT_DELIVER = 0x06,
|
||||||
DATAPOINT_REPORT = 0x07,
|
DATAPOINT_REPORT_ASYNC = 0x07,
|
||||||
DATAPOINT_QUERY = 0x08,
|
DATAPOINT_QUERY = 0x08,
|
||||||
WIFI_TEST = 0x0E,
|
WIFI_TEST = 0x0E,
|
||||||
LOCAL_TIME_QUERY = 0x1C,
|
LOCAL_TIME_QUERY = 0x1C,
|
||||||
|
DATAPOINT_REPORT_SYNC = 0x22,
|
||||||
|
DATAPOINT_REPORT_ACK = 0x23,
|
||||||
WIFI_RSSI = 0x24,
|
WIFI_RSSI = 0x24,
|
||||||
VACUUM_MAP_UPLOAD = 0x28,
|
VACUUM_MAP_UPLOAD = 0x28,
|
||||||
GET_NETWORK_STATUS = 0x2B,
|
GET_NETWORK_STATUS = 0x2B,
|
||||||
|
|
|
@ -96,10 +96,24 @@ void ESP32ArduinoUARTComponent::setup() {
|
||||||
next_uart_num++;
|
next_uart_num++;
|
||||||
} else {
|
} else {
|
||||||
#ifdef USE_LOGGER
|
#ifdef USE_LOGGER
|
||||||
// The logger doesn't use this UART component, instead it targets the UARTs
|
bool logger_uses_hardware_uart = true;
|
||||||
// directly (i.e. Serial/Serial0, Serial1, and Serial2). If the logger is
|
|
||||||
// enabled, skip the UART that it is configured to use.
|
#ifdef USE_LOGGER_USB_CDC
|
||||||
if (logger::global_logger->get_baud_rate() > 0 && logger::global_logger->get_uart() == next_uart_num) {
|
if (logger::global_logger->get_uart() == logger::UART_SELECTION_USB_CDC) {
|
||||||
|
// this is not a hardware UART, ignore it
|
||||||
|
logger_uses_hardware_uart = false;
|
||||||
|
}
|
||||||
|
#endif // USE_LOGGER_USB_CDC
|
||||||
|
|
||||||
|
#ifdef USE_LOGGER_USB_SERIAL_JTAG
|
||||||
|
if (logger::global_logger->get_uart() == logger::UART_SELECTION_USB_SERIAL_JTAG) {
|
||||||
|
// this is not a hardware UART, ignore it
|
||||||
|
logger_uses_hardware_uart = false;
|
||||||
|
}
|
||||||
|
#endif // USE_LOGGER_USB_SERIAL_JTAG
|
||||||
|
|
||||||
|
if (logger_uses_hardware_uart && logger::global_logger->get_baud_rate() > 0 &&
|
||||||
|
logger::global_logger->get_uart() == next_uart_num) {
|
||||||
next_uart_num++;
|
next_uart_num++;
|
||||||
}
|
}
|
||||||
#endif // USE_LOGGER
|
#endif // USE_LOGGER
|
||||||
|
|
|
@ -60,10 +60,30 @@ uart_config_t IDFUARTComponent::get_config_() {
|
||||||
|
|
||||||
void IDFUARTComponent::setup() {
|
void IDFUARTComponent::setup() {
|
||||||
static uint8_t next_uart_num = 0;
|
static uint8_t next_uart_num = 0;
|
||||||
|
|
||||||
#ifdef USE_LOGGER
|
#ifdef USE_LOGGER
|
||||||
if (logger::global_logger->get_uart_num() == next_uart_num)
|
bool logger_uses_hardware_uart = true;
|
||||||
|
|
||||||
|
#ifdef USE_LOGGER_USB_CDC
|
||||||
|
if (logger::global_logger->get_uart() == logger::UART_SELECTION_USB_CDC) {
|
||||||
|
// this is not a hardware UART, ignore it
|
||||||
|
logger_uses_hardware_uart = false;
|
||||||
|
}
|
||||||
|
#endif // USE_LOGGER_USB_CDC
|
||||||
|
|
||||||
|
#ifdef USE_LOGGER_USB_SERIAL_JTAG
|
||||||
|
if (logger::global_logger->get_uart() == logger::UART_SELECTION_USB_SERIAL_JTAG) {
|
||||||
|
// this is not a hardware UART, ignore it
|
||||||
|
logger_uses_hardware_uart = false;
|
||||||
|
}
|
||||||
|
#endif // USE_LOGGER_USB_SERIAL_JTAG
|
||||||
|
|
||||||
|
if (logger_uses_hardware_uart && logger::global_logger->get_baud_rate() > 0 &&
|
||||||
|
logger::global_logger->get_uart_num() == next_uart_num) {
|
||||||
next_uart_num++;
|
next_uart_num++;
|
||||||
#endif
|
}
|
||||||
|
#endif // USE_LOGGER
|
||||||
|
|
||||||
if (next_uart_num >= UART_NUM_MAX) {
|
if (next_uart_num >= UART_NUM_MAX) {
|
||||||
ESP_LOGW(TAG, "Maximum number of UART components created already.");
|
ESP_LOGW(TAG, "Maximum number of UART components created already.");
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
|
|
|
@ -69,7 +69,7 @@ async def setup_update_core_(var, config):
|
||||||
await mqtt.register_mqtt_component(mqtt_, config)
|
await mqtt.register_mqtt_component(mqtt_, config)
|
||||||
|
|
||||||
if web_server_id_config := config.get(CONF_WEB_SERVER_ID):
|
if web_server_id_config := config.get(CONF_WEB_SERVER_ID):
|
||||||
web_server_ = cg.get_variable(web_server_id_config)
|
web_server_ = await cg.get_variable(web_server_id_config)
|
||||||
web_server.add_entity_to_sorting_list(web_server_, var, config)
|
web_server.add_entity_to_sorting_list(web_server_, var, config)
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
from esphome.components import sensor
|
from esphome.components import sensor, time
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
|
CONF_TIME_ID,
|
||||||
|
DEVICE_CLASS_TIMESTAMP,
|
||||||
ENTITY_CATEGORY_DIAGNOSTIC,
|
ENTITY_CATEGORY_DIAGNOSTIC,
|
||||||
STATE_CLASS_TOTAL_INCREASING,
|
STATE_CLASS_TOTAL_INCREASING,
|
||||||
UNIT_SECOND,
|
UNIT_SECOND,
|
||||||
|
@ -10,19 +12,50 @@ from esphome.const import (
|
||||||
)
|
)
|
||||||
|
|
||||||
uptime_ns = cg.esphome_ns.namespace("uptime")
|
uptime_ns = cg.esphome_ns.namespace("uptime")
|
||||||
UptimeSensor = uptime_ns.class_("UptimeSensor", sensor.Sensor, cg.PollingComponent)
|
UptimeSecondsSensor = uptime_ns.class_(
|
||||||
|
"UptimeSecondsSensor", sensor.Sensor, cg.PollingComponent
|
||||||
|
)
|
||||||
|
UptimeTimestampSensor = uptime_ns.class_(
|
||||||
|
"UptimeTimestampSensor", sensor.Sensor, cg.Component
|
||||||
|
)
|
||||||
|
|
||||||
CONFIG_SCHEMA = sensor.sensor_schema(
|
|
||||||
UptimeSensor,
|
CONFIG_SCHEMA = cv.typed_schema(
|
||||||
|
{
|
||||||
|
"seconds": sensor.sensor_schema(
|
||||||
|
UptimeSecondsSensor,
|
||||||
unit_of_measurement=UNIT_SECOND,
|
unit_of_measurement=UNIT_SECOND,
|
||||||
icon=ICON_TIMER,
|
icon=ICON_TIMER,
|
||||||
accuracy_decimals=0,
|
accuracy_decimals=0,
|
||||||
state_class=STATE_CLASS_TOTAL_INCREASING,
|
state_class=STATE_CLASS_TOTAL_INCREASING,
|
||||||
device_class=DEVICE_CLASS_DURATION,
|
device_class=DEVICE_CLASS_DURATION,
|
||||||
entity_category=ENTITY_CATEGORY_DIAGNOSTIC,
|
entity_category=ENTITY_CATEGORY_DIAGNOSTIC,
|
||||||
).extend(cv.polling_component_schema("60s"))
|
).extend(cv.polling_component_schema("60s")),
|
||||||
|
"timestamp": sensor.sensor_schema(
|
||||||
|
UptimeTimestampSensor,
|
||||||
|
icon=ICON_TIMER,
|
||||||
|
accuracy_decimals=0,
|
||||||
|
device_class=DEVICE_CLASS_TIMESTAMP,
|
||||||
|
entity_category=ENTITY_CATEGORY_DIAGNOSTIC,
|
||||||
|
)
|
||||||
|
.extend(
|
||||||
|
cv.Schema(
|
||||||
|
{
|
||||||
|
cv.GenerateID(CONF_TIME_ID): cv.All(
|
||||||
|
cv.requires_component("time"), cv.use_id(time.RealTimeClock)
|
||||||
|
),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.extend(cv.COMPONENT_SCHEMA),
|
||||||
|
},
|
||||||
|
default_type="seconds",
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
var = await sensor.new_sensor(config)
|
var = await sensor.new_sensor(config)
|
||||||
await cg.register_component(var, config)
|
await cg.register_component(var, config)
|
||||||
|
if time_id_config := config.get(CONF_TIME_ID):
|
||||||
|
time_id = await cg.get_variable(time_id_config)
|
||||||
|
cg.add(var.set_time(time_id))
|
||||||
|
|
|
@ -1,14 +1,15 @@
|
||||||
#include "uptime_sensor.h"
|
#include "uptime_seconds_sensor.h"
|
||||||
#include "esphome/core/log.h"
|
|
||||||
#include "esphome/core/helpers.h"
|
|
||||||
#include "esphome/core/hal.h"
|
#include "esphome/core/hal.h"
|
||||||
|
#include "esphome/core/helpers.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace uptime {
|
namespace uptime {
|
||||||
|
|
||||||
static const char *const TAG = "uptime.sensor";
|
static const char *const TAG = "uptime.sensor";
|
||||||
|
|
||||||
void UptimeSensor::update() {
|
void UptimeSecondsSensor::update() {
|
||||||
const uint32_t ms = millis();
|
const uint32_t ms = millis();
|
||||||
const uint64_t ms_mask = (1ULL << 32) - 1ULL;
|
const uint64_t ms_mask = (1ULL << 32) - 1ULL;
|
||||||
const uint32_t last_ms = this->uptime_ & ms_mask;
|
const uint32_t last_ms = this->uptime_ & ms_mask;
|
||||||
|
@ -26,9 +27,12 @@ void UptimeSensor::update() {
|
||||||
const float seconds = float(seconds_int) + (this->uptime_ % 1000ULL) / 1000.0f;
|
const float seconds = float(seconds_int) + (this->uptime_ % 1000ULL) / 1000.0f;
|
||||||
this->publish_state(seconds);
|
this->publish_state(seconds);
|
||||||
}
|
}
|
||||||
std::string UptimeSensor::unique_id() { return get_mac_address() + "-uptime"; }
|
std::string UptimeSecondsSensor::unique_id() { return get_mac_address() + "-uptime"; }
|
||||||
float UptimeSensor::get_setup_priority() const { return setup_priority::HARDWARE; }
|
float UptimeSecondsSensor::get_setup_priority() const { return setup_priority::HARDWARE; }
|
||||||
void UptimeSensor::dump_config() { LOG_SENSOR("", "Uptime Sensor", this); }
|
void UptimeSecondsSensor::dump_config() {
|
||||||
|
LOG_SENSOR("", "Uptime Sensor", this);
|
||||||
|
ESP_LOGCONFIG(TAG, " Type: Seconds");
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace uptime
|
} // namespace uptime
|
||||||
} // namespace esphome
|
} // namespace esphome
|
|
@ -1,12 +1,12 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "esphome/core/component.h"
|
|
||||||
#include "esphome/components/sensor/sensor.h"
|
#include "esphome/components/sensor/sensor.h"
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace uptime {
|
namespace uptime {
|
||||||
|
|
||||||
class UptimeSensor : public sensor::Sensor, public PollingComponent {
|
class UptimeSecondsSensor : public sensor::Sensor, public PollingComponent {
|
||||||
public:
|
public:
|
||||||
void update() override;
|
void update() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
39
esphome/components/uptime/uptime_timestamp_sensor.cpp
Normal file
39
esphome/components/uptime/uptime_timestamp_sensor.cpp
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
#include "uptime_timestamp_sensor.h"
|
||||||
|
|
||||||
|
#ifdef USE_TIME
|
||||||
|
|
||||||
|
#include "esphome/core/hal.h"
|
||||||
|
#include "esphome/core/helpers.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace uptime {
|
||||||
|
|
||||||
|
static const char *const TAG = "uptime.sensor";
|
||||||
|
|
||||||
|
void UptimeTimestampSensor::setup() {
|
||||||
|
this->time_->add_on_time_sync_callback([this]() {
|
||||||
|
if (this->has_state_)
|
||||||
|
return; // No need to update the timestamp if it's already set
|
||||||
|
|
||||||
|
auto now = this->time_->now();
|
||||||
|
const uint32_t ms = millis();
|
||||||
|
if (!now.is_valid())
|
||||||
|
return; // No need to update the timestamp if the time is not valid
|
||||||
|
|
||||||
|
time_t timestamp = now.timestamp;
|
||||||
|
uint32_t seconds = ms / 1000;
|
||||||
|
timestamp -= seconds;
|
||||||
|
this->publish_state(timestamp);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
float UptimeTimestampSensor::get_setup_priority() const { return setup_priority::HARDWARE; }
|
||||||
|
void UptimeTimestampSensor::dump_config() {
|
||||||
|
LOG_SENSOR("", "Uptime Sensor", this);
|
||||||
|
ESP_LOGCONFIG(TAG, " Type: Timestamp");
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace uptime
|
||||||
|
} // namespace esphome
|
||||||
|
|
||||||
|
#endif // USE_TIME
|
30
esphome/components/uptime/uptime_timestamp_sensor.h
Normal file
30
esphome/components/uptime/uptime_timestamp_sensor.h
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/core/defines.h"
|
||||||
|
|
||||||
|
#ifdef USE_TIME
|
||||||
|
|
||||||
|
#include "esphome/components/sensor/sensor.h"
|
||||||
|
#include "esphome/components/time/real_time_clock.h"
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace uptime {
|
||||||
|
|
||||||
|
class UptimeTimestampSensor : public sensor::Sensor, public Component {
|
||||||
|
public:
|
||||||
|
void setup() override;
|
||||||
|
void dump_config() override;
|
||||||
|
|
||||||
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
|
void set_time(time::RealTimeClock *time) { this->time_ = time; }
|
||||||
|
|
||||||
|
protected:
|
||||||
|
time::RealTimeClock *time_;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace uptime
|
||||||
|
} // namespace esphome
|
||||||
|
|
||||||
|
#endif // USE_TIME
|
|
@ -132,7 +132,7 @@ async def to_code(config):
|
||||||
# the '+1' modifier is relative to the device's own address that will
|
# the '+1' modifier is relative to the device's own address that will
|
||||||
# be automatically added to the provided list.
|
# be automatically added to the provided list.
|
||||||
cg.add_build_flag(f"-DCONFIG_WIREGUARD_MAX_SRC_IPS={len(allowed_ips) + 1}")
|
cg.add_build_flag(f"-DCONFIG_WIREGUARD_MAX_SRC_IPS={len(allowed_ips) + 1}")
|
||||||
cg.add_library("droscy/esp_wireguard", "0.4.1")
|
cg.add_library("droscy/esp_wireguard", "0.4.2")
|
||||||
|
|
||||||
await cg.register_component(var, config)
|
await cg.register_component(var, config)
|
||||||
|
|
||||||
|
|
|
@ -355,6 +355,7 @@ CONF_HUMIDITY_SENSOR = "humidity_sensor"
|
||||||
CONF_HYSTERESIS = "hysteresis"
|
CONF_HYSTERESIS = "hysteresis"
|
||||||
CONF_I2C = "i2c"
|
CONF_I2C = "i2c"
|
||||||
CONF_I2C_ID = "i2c_id"
|
CONF_I2C_ID = "i2c_id"
|
||||||
|
CONF_IAQ_ACCURACY = "iaq_accuracy"
|
||||||
CONF_IBEACON_MAJOR = "ibeacon_major"
|
CONF_IBEACON_MAJOR = "ibeacon_major"
|
||||||
CONF_IBEACON_MINOR = "ibeacon_minor"
|
CONF_IBEACON_MINOR = "ibeacon_minor"
|
||||||
CONF_IBEACON_UUID = "ibeacon_uuid"
|
CONF_IBEACON_UUID = "ibeacon_uuid"
|
||||||
|
@ -1071,6 +1072,7 @@ DEVICE_CLASS_BUTTON = "button"
|
||||||
DEVICE_CLASS_CARBON_DIOXIDE = "carbon_dioxide"
|
DEVICE_CLASS_CARBON_DIOXIDE = "carbon_dioxide"
|
||||||
DEVICE_CLASS_CARBON_MONOXIDE = "carbon_monoxide"
|
DEVICE_CLASS_CARBON_MONOXIDE = "carbon_monoxide"
|
||||||
DEVICE_CLASS_COLD = "cold"
|
DEVICE_CLASS_COLD = "cold"
|
||||||
|
DEVICE_CLASS_CONDUCTIVITY = "conductivity"
|
||||||
DEVICE_CLASS_CONNECTIVITY = "connectivity"
|
DEVICE_CLASS_CONNECTIVITY = "connectivity"
|
||||||
DEVICE_CLASS_CURRENT = "current"
|
DEVICE_CLASS_CURRENT = "current"
|
||||||
DEVICE_CLASS_CURTAIN = "curtain"
|
DEVICE_CLASS_CURTAIN = "curtain"
|
||||||
|
|
|
@ -93,7 +93,7 @@ std::string to_string(long double value) { return str_snprintf("%Lf", 32, value)
|
||||||
// Mathematics
|
// Mathematics
|
||||||
|
|
||||||
float lerp(float completion, float start, float end) { return start + (end - start) * completion; }
|
float lerp(float completion, float start, float end) { return start + (end - start) * completion; }
|
||||||
uint8_t crc8(uint8_t *data, uint8_t len) {
|
uint8_t crc8(const uint8_t *data, uint8_t len) {
|
||||||
uint8_t crc = 0;
|
uint8_t crc = 0;
|
||||||
|
|
||||||
while ((len--) != 0u) {
|
while ((len--) != 0u) {
|
||||||
|
|
|
@ -155,7 +155,7 @@ template<typename T, typename U> T remap(U value, U min, U max, T min_out, T max
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Calculate a CRC-8 checksum of \p data with size \p len.
|
/// Calculate a CRC-8 checksum of \p data with size \p len.
|
||||||
uint8_t crc8(uint8_t *data, uint8_t len);
|
uint8_t crc8(const uint8_t *data, uint8_t len);
|
||||||
|
|
||||||
/// Calculate a CRC-16 checksum of \p data with size \p len.
|
/// Calculate a CRC-16 checksum of \p data with size \p len.
|
||||||
uint16_t crc16(const uint8_t *data, uint16_t len, uint16_t crc = 0xffff, uint16_t reverse_poly = 0xa001,
|
uint16_t crc16(const uint8_t *data, uint16_t len, uint16_t crc = 0xffff, uint16_t reverse_poly = 0xa001,
|
||||||
|
|
|
@ -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")
|
||||||
|
|
|
@ -7,6 +7,7 @@ from datetime import datetime
|
||||||
import requests
|
import requests
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
from esphome.core import CORE, TimePeriodSeconds
|
from esphome.core import CORE, TimePeriodSeconds
|
||||||
|
from esphome.const import __version__
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
CODEOWNERS = ["@landonr"]
|
CODEOWNERS = ["@landonr"]
|
||||||
|
@ -75,3 +76,28 @@ def compute_local_file_dir(domain: str) -> Path:
|
||||||
base_directory.mkdir(parents=True, exist_ok=True)
|
base_directory.mkdir(parents=True, exist_ok=True)
|
||||||
|
|
||||||
return base_directory
|
return base_directory
|
||||||
|
|
||||||
|
|
||||||
|
def download_content(url: str, path: Path, timeout=NETWORK_TIMEOUT) -> None:
|
||||||
|
if not has_remote_file_changed(url, path):
|
||||||
|
_LOGGER.debug("Remote file has not changed %s", url)
|
||||||
|
return
|
||||||
|
|
||||||
|
_LOGGER.debug(
|
||||||
|
"Remote file has changed, downloading from %s to %s",
|
||||||
|
url,
|
||||||
|
path,
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
req = requests.get(
|
||||||
|
url,
|
||||||
|
timeout=timeout,
|
||||||
|
headers={"User-agent": f"ESPHome/{__version__} (https://esphome.io)"},
|
||||||
|
)
|
||||||
|
req.raise_for_status()
|
||||||
|
except requests.exceptions.RequestException as e:
|
||||||
|
raise cv.Invalid(f"Could not download from {url}: {e}")
|
||||||
|
|
||||||
|
path.parent.mkdir(parents=True, exist_ok=True)
|
||||||
|
path.write_bytes(req.content)
|
||||||
|
|
|
@ -64,8 +64,8 @@ lib_deps =
|
||||||
freekode/TM1651@1.0.1 ; tm1651
|
freekode/TM1651@1.0.1 ; tm1651
|
||||||
glmnet/Dsmr@0.7 ; dsmr
|
glmnet/Dsmr@0.7 ; dsmr
|
||||||
rweather/Crypto@0.4.0 ; dsmr
|
rweather/Crypto@0.4.0 ; dsmr
|
||||||
dudanov/MideaUART@1.1.8 ; midea
|
dudanov/MideaUART@1.1.9 ; midea
|
||||||
tonia/HeatpumpIR@1.0.23 ; heatpumpir
|
tonia/HeatpumpIR@1.0.26 ; heatpumpir
|
||||||
build_flags =
|
build_flags =
|
||||||
${common.build_flags}
|
${common.build_flags}
|
||||||
-DUSE_ARDUINO
|
-DUSE_ARDUINO
|
||||||
|
@ -93,8 +93,8 @@ lib_deps =
|
||||||
ESP8266HTTPClient ; http_request (Arduino built-in)
|
ESP8266HTTPClient ; http_request (Arduino built-in)
|
||||||
ESP8266mDNS ; mdns (Arduino built-in)
|
ESP8266mDNS ; mdns (Arduino built-in)
|
||||||
DNSServer ; captive_portal (Arduino built-in)
|
DNSServer ; captive_portal (Arduino built-in)
|
||||||
crankyoldgit/IRremoteESP8266@~2.8.4 ; heatpumpir
|
crankyoldgit/IRremoteESP8266@2.8.6 ; heatpumpir
|
||||||
droscy/esp_wireguard@0.4.1 ; wireguard
|
droscy/esp_wireguard@0.4.2 ; wireguard
|
||||||
build_flags =
|
build_flags =
|
||||||
${common:arduino.build_flags}
|
${common:arduino.build_flags}
|
||||||
-Wno-nonnull-compare
|
-Wno-nonnull-compare
|
||||||
|
@ -123,8 +123,8 @@ lib_deps =
|
||||||
ESPmDNS ; mdns (Arduino built-in)
|
ESPmDNS ; mdns (Arduino built-in)
|
||||||
DNSServer ; captive_portal (Arduino built-in)
|
DNSServer ; captive_portal (Arduino built-in)
|
||||||
esphome/ESP32-audioI2S@2.0.7 ; i2s_audio
|
esphome/ESP32-audioI2S@2.0.7 ; i2s_audio
|
||||||
crankyoldgit/IRremoteESP8266@~2.8.4 ; heatpumpir
|
crankyoldgit/IRremoteESP8266@2.8.6 ; heatpumpir
|
||||||
droscy/esp_wireguard@0.4.1 ; wireguard
|
droscy/esp_wireguard@0.4.2 ; wireguard
|
||||||
build_flags =
|
build_flags =
|
||||||
${common:arduino.build_flags}
|
${common:arduino.build_flags}
|
||||||
-DUSE_ESP32
|
-DUSE_ESP32
|
||||||
|
@ -142,7 +142,7 @@ platform_packages =
|
||||||
framework = espidf
|
framework = espidf
|
||||||
lib_deps =
|
lib_deps =
|
||||||
${common:idf.lib_deps}
|
${common:idf.lib_deps}
|
||||||
droscy/esp_wireguard@0.4.1 ; wireguard
|
droscy/esp_wireguard@0.4.2 ; wireguard
|
||||||
build_flags =
|
build_flags =
|
||||||
${common:idf.build_flags}
|
${common:idf.build_flags}
|
||||||
-Wno-nonnull-compare
|
-Wno-nonnull-compare
|
||||||
|
@ -174,7 +174,7 @@ extends = common:arduino
|
||||||
platform = libretiny
|
platform = libretiny
|
||||||
framework = arduino
|
framework = arduino
|
||||||
lib_deps =
|
lib_deps =
|
||||||
droscy/esp_wireguard@0.4.1 ; wireguard
|
droscy/esp_wireguard@0.4.2 ; wireguard
|
||||||
build_flags =
|
build_flags =
|
||||||
${common:arduino.build_flags}
|
${common:arduino.build_flags}
|
||||||
-DUSE_LIBRETINY
|
-DUSE_LIBRETINY
|
||||||
|
|
25
script/extract_automations.py
Executable file
25
script/extract_automations.py
Executable file
|
@ -0,0 +1,25 @@
|
||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import json
|
||||||
|
|
||||||
|
from helpers import git_ls_files
|
||||||
|
|
||||||
|
from esphome.automation import ACTION_REGISTRY, CONDITION_REGISTRY
|
||||||
|
from esphome.pins import PIN_SCHEMA_REGISTRY
|
||||||
|
|
||||||
|
list_components = __import__("list-components")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
files = git_ls_files()
|
||||||
|
files = filter(list_components.filter_component_files, files)
|
||||||
|
|
||||||
|
components = list_components.get_components(files, True)
|
||||||
|
|
||||||
|
dump = {
|
||||||
|
"actions": sorted(list(ACTION_REGISTRY.keys())),
|
||||||
|
"conditions": sorted(list(CONDITION_REGISTRY.keys())),
|
||||||
|
"pin_providers": sorted(list(PIN_SCHEMA_REGISTRY.keys())),
|
||||||
|
}
|
||||||
|
|
||||||
|
print(json.dumps(dump, indent=2))
|
|
@ -50,6 +50,7 @@ def create_components_graph():
|
||||||
{KEY_TARGET_FRAMEWORK: "arduino", KEY_TARGET_PLATFORM: None},
|
{KEY_TARGET_FRAMEWORK: "arduino", KEY_TARGET_PLATFORM: None},
|
||||||
{KEY_TARGET_FRAMEWORK: "esp-idf", KEY_TARGET_PLATFORM: None},
|
{KEY_TARGET_FRAMEWORK: "esp-idf", KEY_TARGET_PLATFORM: None},
|
||||||
{KEY_TARGET_FRAMEWORK: None, KEY_TARGET_PLATFORM: PLATFORM_ESP32},
|
{KEY_TARGET_FRAMEWORK: None, KEY_TARGET_PLATFORM: PLATFORM_ESP32},
|
||||||
|
{KEY_TARGET_FRAMEWORK: None, KEY_TARGET_PLATFORM: PLATFORM_ESP8266},
|
||||||
]
|
]
|
||||||
CORE.data[KEY_CORE] = TARGET_CONFIGURATIONS[0]
|
CORE.data[KEY_CORE] = TARGET_CONFIGURATIONS[0]
|
||||||
|
|
||||||
|
@ -119,10 +120,30 @@ def find_children_of_component(components_graph, component_name, depth=0):
|
||||||
return list(set(children))
|
return list(set(children))
|
||||||
|
|
||||||
|
|
||||||
|
def get_components(files: list[str], get_dependencies: bool = False):
|
||||||
|
components = extract_component_names_array_from_files_array(files)
|
||||||
|
|
||||||
|
if get_dependencies:
|
||||||
|
components_graph = create_components_graph()
|
||||||
|
|
||||||
|
all_components = components.copy()
|
||||||
|
for c in components:
|
||||||
|
all_components.extend(find_children_of_component(components_graph, c))
|
||||||
|
# Remove duplicate values
|
||||||
|
all_changed_components = list(set(all_components))
|
||||||
|
|
||||||
|
return sorted(all_changed_components)
|
||||||
|
|
||||||
|
return sorted(components)
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
parser = argparse.ArgumentParser()
|
parser = argparse.ArgumentParser()
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"-c", "--changed", action="store_true", help="Only run on changed files"
|
"-c",
|
||||||
|
"--changed",
|
||||||
|
action="store_true",
|
||||||
|
help="List all components required for testing based on changes",
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"-b", "--branch", help="Branch to compare changed files against"
|
"-b", "--branch", help="Branch to compare changed files against"
|
||||||
|
@ -140,25 +161,11 @@ def main():
|
||||||
changed = changed_files(args.branch)
|
changed = changed_files(args.branch)
|
||||||
else:
|
else:
|
||||||
changed = changed_files()
|
changed = changed_files()
|
||||||
|
# If any base test file(s) changed, there's no need to filter out components
|
||||||
|
if not any("tests/test_build_components" in file for file in changed):
|
||||||
files = [f for f in files if f in changed]
|
files = [f for f in files if f in changed]
|
||||||
|
|
||||||
components = extract_component_names_array_from_files_array(files)
|
for c in get_components(files, args.changed):
|
||||||
|
|
||||||
if args.changed:
|
|
||||||
components_graph = create_components_graph()
|
|
||||||
|
|
||||||
all_changed_components = components.copy()
|
|
||||||
for c in components:
|
|
||||||
all_changed_components.extend(
|
|
||||||
find_children_of_component(components_graph, c)
|
|
||||||
)
|
|
||||||
# Remove duplicate values
|
|
||||||
all_changed_components = list(set(all_changed_components))
|
|
||||||
|
|
||||||
for c in sorted(all_changed_components):
|
|
||||||
print(c)
|
|
||||||
else:
|
|
||||||
for c in sorted(components):
|
|
||||||
print(c)
|
print(c)
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -54,7 +54,6 @@ climate:
|
||||||
|
|
||||||
sensor:
|
sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
outdoor_temperature:
|
outdoor_temperature:
|
||||||
name: Haier outdoor temperature
|
name: Haier outdoor temperature
|
||||||
humidity:
|
humidity:
|
||||||
|
@ -80,7 +79,6 @@ sensor:
|
||||||
|
|
||||||
binary_sensor:
|
binary_sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
compressor_status:
|
compressor_status:
|
||||||
name: Haier Outdoor Compressor Status
|
name: Haier Outdoor Compressor Status
|
||||||
defrost_status:
|
defrost_status:
|
||||||
|
@ -96,7 +94,6 @@ binary_sensor:
|
||||||
|
|
||||||
button:
|
button:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
self_cleaning:
|
self_cleaning:
|
||||||
name: Haier start self cleaning
|
name: Haier start self cleaning
|
||||||
steri_cleaning:
|
steri_cleaning:
|
||||||
|
@ -104,7 +101,6 @@ button:
|
||||||
|
|
||||||
text_sensor:
|
text_sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
appliance_name:
|
appliance_name:
|
||||||
name: Haier appliance name
|
name: Haier appliance name
|
||||||
cleaning_status:
|
cleaning_status:
|
||||||
|
|
|
@ -54,7 +54,6 @@ climate:
|
||||||
|
|
||||||
sensor:
|
sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
outdoor_temperature:
|
outdoor_temperature:
|
||||||
name: Haier outdoor temperature
|
name: Haier outdoor temperature
|
||||||
humidity:
|
humidity:
|
||||||
|
@ -80,7 +79,6 @@ sensor:
|
||||||
|
|
||||||
binary_sensor:
|
binary_sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
compressor_status:
|
compressor_status:
|
||||||
name: Haier Outdoor Compressor Status
|
name: Haier Outdoor Compressor Status
|
||||||
defrost_status:
|
defrost_status:
|
||||||
|
@ -96,7 +94,6 @@ binary_sensor:
|
||||||
|
|
||||||
button:
|
button:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
self_cleaning:
|
self_cleaning:
|
||||||
name: Haier start self cleaning
|
name: Haier start self cleaning
|
||||||
steri_cleaning:
|
steri_cleaning:
|
||||||
|
@ -104,7 +101,6 @@ button:
|
||||||
|
|
||||||
text_sensor:
|
text_sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
appliance_name:
|
appliance_name:
|
||||||
name: Haier appliance name
|
name: Haier appliance name
|
||||||
cleaning_status:
|
cleaning_status:
|
||||||
|
|
|
@ -54,7 +54,6 @@ climate:
|
||||||
|
|
||||||
sensor:
|
sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
outdoor_temperature:
|
outdoor_temperature:
|
||||||
name: Haier outdoor temperature
|
name: Haier outdoor temperature
|
||||||
humidity:
|
humidity:
|
||||||
|
@ -80,7 +79,6 @@ sensor:
|
||||||
|
|
||||||
binary_sensor:
|
binary_sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
compressor_status:
|
compressor_status:
|
||||||
name: Haier Outdoor Compressor Status
|
name: Haier Outdoor Compressor Status
|
||||||
defrost_status:
|
defrost_status:
|
||||||
|
@ -96,7 +94,6 @@ binary_sensor:
|
||||||
|
|
||||||
button:
|
button:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
self_cleaning:
|
self_cleaning:
|
||||||
name: Haier start self cleaning
|
name: Haier start self cleaning
|
||||||
steri_cleaning:
|
steri_cleaning:
|
||||||
|
@ -104,7 +101,6 @@ button:
|
||||||
|
|
||||||
text_sensor:
|
text_sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
appliance_name:
|
appliance_name:
|
||||||
name: Haier appliance name
|
name: Haier appliance name
|
||||||
cleaning_status:
|
cleaning_status:
|
||||||
|
|
|
@ -54,7 +54,6 @@ climate:
|
||||||
|
|
||||||
sensor:
|
sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
outdoor_temperature:
|
outdoor_temperature:
|
||||||
name: Haier outdoor temperature
|
name: Haier outdoor temperature
|
||||||
humidity:
|
humidity:
|
||||||
|
@ -80,7 +79,6 @@ sensor:
|
||||||
|
|
||||||
binary_sensor:
|
binary_sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
compressor_status:
|
compressor_status:
|
||||||
name: Haier Outdoor Compressor Status
|
name: Haier Outdoor Compressor Status
|
||||||
defrost_status:
|
defrost_status:
|
||||||
|
@ -96,7 +94,6 @@ binary_sensor:
|
||||||
|
|
||||||
button:
|
button:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
self_cleaning:
|
self_cleaning:
|
||||||
name: Haier start self cleaning
|
name: Haier start self cleaning
|
||||||
steri_cleaning:
|
steri_cleaning:
|
||||||
|
@ -104,7 +101,6 @@ button:
|
||||||
|
|
||||||
text_sensor:
|
text_sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
appliance_name:
|
appliance_name:
|
||||||
name: Haier appliance name
|
name: Haier appliance name
|
||||||
cleaning_status:
|
cleaning_status:
|
||||||
|
|
|
@ -54,7 +54,6 @@ climate:
|
||||||
|
|
||||||
sensor:
|
sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
outdoor_temperature:
|
outdoor_temperature:
|
||||||
name: Haier outdoor temperature
|
name: Haier outdoor temperature
|
||||||
humidity:
|
humidity:
|
||||||
|
@ -80,7 +79,6 @@ sensor:
|
||||||
|
|
||||||
binary_sensor:
|
binary_sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
compressor_status:
|
compressor_status:
|
||||||
name: Haier Outdoor Compressor Status
|
name: Haier Outdoor Compressor Status
|
||||||
defrost_status:
|
defrost_status:
|
||||||
|
@ -96,7 +94,6 @@ binary_sensor:
|
||||||
|
|
||||||
button:
|
button:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
self_cleaning:
|
self_cleaning:
|
||||||
name: Haier start self cleaning
|
name: Haier start self cleaning
|
||||||
steri_cleaning:
|
steri_cleaning:
|
||||||
|
@ -104,7 +101,6 @@ button:
|
||||||
|
|
||||||
text_sensor:
|
text_sensor:
|
||||||
- platform: haier
|
- platform: haier
|
||||||
haier_id: haier_ac
|
|
||||||
appliance_name:
|
appliance_name:
|
||||||
name: Haier appliance name
|
name: Haier appliance name
|
||||||
cleaning_status:
|
cleaning_status:
|
||||||
|
|
8
tests/components/logger/common-usb_cdc.yaml
Normal file
8
tests/components/logger/common-usb_cdc.yaml
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
esphome:
|
||||||
|
on_boot:
|
||||||
|
then:
|
||||||
|
- logger.log: Hello world
|
||||||
|
|
||||||
|
logger:
|
||||||
|
level: DEBUG
|
||||||
|
hardware_uart: USB_CDC
|
8
tests/components/logger/common-usb_serial_jtag.yaml
Normal file
8
tests/components/logger/common-usb_serial_jtag.yaml
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
esphome:
|
||||||
|
on_boot:
|
||||||
|
then:
|
||||||
|
- logger.log: Hello world
|
||||||
|
|
||||||
|
logger:
|
||||||
|
level: DEBUG
|
||||||
|
hardware_uart: USB_SERIAL_JTAG
|
1
tests/components/logger/test-usb_cdc.esp32-c3-ard.yaml
Normal file
1
tests/components/logger/test-usb_cdc.esp32-c3-ard.yaml
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<<: !include common-usb_cdc.yaml
|
1
tests/components/logger/test-usb_cdc.esp32-s2-ard.yaml
Normal file
1
tests/components/logger/test-usb_cdc.esp32-s2-ard.yaml
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<<: !include common-usb_cdc.yaml
|
1
tests/components/logger/test-usb_cdc.esp32-s2-idf.yaml
Normal file
1
tests/components/logger/test-usb_cdc.esp32-s2-idf.yaml
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<<: !include common-usb_cdc.yaml
|
1
tests/components/logger/test-usb_cdc.esp32-s3-ard.yaml
Normal file
1
tests/components/logger/test-usb_cdc.esp32-s3-ard.yaml
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<<: !include common-usb_cdc.yaml
|
|
@ -0,0 +1 @@
|
||||||
|
<<: !include common-usb_serial_jtag.yaml
|
|
@ -0,0 +1 @@
|
||||||
|
<<: !include common-usb_serial_jtag.yaml
|
|
@ -1 +1 @@
|
||||||
<<: !include common.yaml
|
<<: !include common-default_uart.yaml
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<<: !include common.yaml
|
<<: !include common-default_uart.yaml
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<<: !include common.yaml
|
<<: !include common-default_uart.yaml
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<<: !include common.yaml
|
<<: !include common-default_uart.yaml
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<<: !include common.yaml
|
<<: !include common-default_uart.yaml
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
<<: !include common.yaml
|
<<: !include common-default_uart.yaml
|
||||||
|
|
6
tests/components/mdns/test-disabled.esp32-idf.yaml
Normal file
6
tests/components/mdns/test-disabled.esp32-idf.yaml
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
wifi:
|
||||||
|
ssid: MySSID
|
||||||
|
password: password1
|
||||||
|
|
||||||
|
mdns:
|
||||||
|
disabled: true
|
1
tests/components/mdns/test-enabled.esp32-ard.yaml
Normal file
1
tests/components/mdns/test-enabled.esp32-ard.yaml
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<<: !include common-enabled.yaml
|
1
tests/components/mdns/test-enabled.esp32-c3-ard.yaml
Normal file
1
tests/components/mdns/test-enabled.esp32-c3-ard.yaml
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<<: !include common-enabled.yaml
|
1
tests/components/mdns/test-enabled.esp32-c3-idf.yaml
Normal file
1
tests/components/mdns/test-enabled.esp32-c3-idf.yaml
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<<: !include common-enabled.yaml
|
1
tests/components/mdns/test-enabled.esp32-idf.yaml
Normal file
1
tests/components/mdns/test-enabled.esp32-idf.yaml
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<<: !include common-enabled.yaml
|
1
tests/components/mdns/test-enabled.esp8266-ard.yaml
Normal file
1
tests/components/mdns/test-enabled.esp8266-ard.yaml
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<<: !include common-enabled.yaml
|
1
tests/components/mdns/test-enabled.rp2040-ard.yaml
Normal file
1
tests/components/mdns/test-enabled.rp2040-ard.yaml
Normal file
|
@ -0,0 +1 @@
|
||||||
|
<<: !include common-enabled.yaml
|
|
@ -1 +0,0 @@
|
||||||
<<: !include common.yaml
|
|
|
@ -1 +0,0 @@
|
||||||
<<: !include common.yaml
|
|
|
@ -1 +0,0 @@
|
||||||
<<: !include common.yaml
|
|
|
@ -1 +0,0 @@
|
||||||
<<: !include common.yaml
|
|
|
@ -1 +0,0 @@
|
||||||
<<: !include common.yaml
|
|
|
@ -1 +0,0 @@
|
||||||
<<: !include common.yaml
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue