mirror of
https://github.com/esphome/esphome.git
synced 2025-11-05 09:31:54 +00:00
Compare commits
487 Commits
add_api_st
...
2025.7.0b1
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
6178ab7513 | ||
|
|
267574f24c | ||
|
|
5235c80781 | ||
|
|
0ccc5e340e | ||
|
|
86c6e4da2a | ||
|
|
5c8b330eaa | ||
|
|
4158a5c2a3 | ||
|
|
05c5364490 | ||
|
|
78eb236a4a | ||
|
|
691cc5f7dc | ||
|
|
b3d7f001af | ||
|
|
3f8b691c32 | ||
|
|
a30f01d668 | ||
|
|
4648804db6 | ||
|
|
51377b2625 | ||
|
|
256f9f9943 | ||
|
|
a72905191a | ||
|
|
7150f2806f | ||
|
|
ee8ee4e646 | ||
|
|
fb357b8965 | ||
|
|
c4fac1a2ae | ||
|
|
42a1f6922f | ||
|
|
206659ddb8 | ||
|
|
440de12e3f | ||
|
|
b122112d58 | ||
|
|
fe258e1007 | ||
|
|
3976fd02ea | ||
|
|
e58c793da2 | ||
|
|
90fb3680d4 | ||
|
|
832a787271 | ||
|
|
29747fc730 | ||
|
|
e2de6ee29d | ||
|
|
053feb5e3b | ||
|
|
31f36df4ba | ||
|
|
3ef392d433 | ||
|
|
138ff749f3 | ||
|
|
e88b8d10ec | ||
|
|
8147d117a0 | ||
|
|
c6f7e84256 | ||
|
|
db877e688a | ||
|
|
4e25b6da7b | ||
|
|
83512b88c4 | ||
|
|
fde5f88192 | ||
|
|
2510b5ffb5 | ||
|
|
364b6ca8d0 | ||
|
|
e49b89a051 | ||
|
|
bdd52dbaa4 | ||
|
|
765793505d | ||
|
|
a303f93236 | ||
|
|
492580edc3 | ||
|
|
1368139f4d | ||
|
|
b6fade7339 | ||
|
|
8da322fe9e | ||
|
|
e5a699a004 | ||
|
|
e061b6dc55 | ||
|
|
4673a5b48c | ||
|
|
0bc18a8281 | ||
|
|
20ba035e3b | ||
|
|
f7019a4ed7 | ||
|
|
a1291c2730 | ||
|
|
b0f8922056 | ||
|
|
4e9e48e2e7 | ||
|
|
86e7013f40 | ||
|
|
58b4e7dab2 | ||
|
|
d686257cff | ||
|
|
adb7ccdbc7 | ||
|
|
d00e20ccdf | ||
|
|
25457da97c | ||
|
|
14d7c4bdbd | ||
|
|
eef71a79da | ||
|
|
547c7d6dc8 | ||
|
|
1ef7b2d64f | ||
|
|
107304b274 | ||
|
|
b2b6f41ef3 | ||
|
|
34db02661c | ||
|
|
798eef41b9 | ||
|
|
658e4bac47 | ||
|
|
f5aab154a6 | ||
|
|
5b55e205ef | ||
|
|
4ef5c941c9 | ||
|
|
b9391f2cd4 | ||
|
|
66e090ff5b | ||
|
|
d41298897f | ||
|
|
ba42de536c | ||
|
|
bdc9f5f3b2 | ||
|
|
90f9ab0d3e | ||
|
|
00eb56d8db | ||
|
|
60eac6ea07 | ||
|
|
9b3ece4caf | ||
|
|
289aedcfe2 | ||
|
|
4cdc804c17 | ||
|
|
56a963dfe6 | ||
|
|
f6f0e52d5e | ||
|
|
eba2c82fec | ||
|
|
fae96e279c | ||
|
|
2fb23becec | ||
|
|
095acce3e2 | ||
|
|
5fa9d22c5d | ||
|
|
785b14ac84 | ||
|
|
84ab758b22 | ||
|
|
03566c34ed | ||
|
|
6a096c1d5a | ||
|
|
04a46de237 | ||
|
|
0083abe3b5 | ||
|
|
3470305d9d | ||
|
|
35de36d690 | ||
|
|
16ef5a9377 | ||
|
|
e3ccb9b46c | ||
|
|
8c34b72b62 | ||
|
|
27c745d5a1 | ||
|
|
9a0ba1657e | ||
|
|
db7a420e54 | ||
|
|
e58baab563 | ||
|
|
08c88ba0f2 | ||
|
|
78c8cd4c4e | ||
|
|
98e106e0ae | ||
|
|
0cbb5e6c1c | ||
|
|
8014cbc71e | ||
|
|
aaa7117ec9 | ||
|
|
3930609d8b | ||
|
|
3e553f517b | ||
|
|
af0bb634c6 | ||
|
|
8a9769d4e9 | ||
|
|
d86f319d66 | ||
|
|
9890659f61 | ||
|
|
140ca070a2 | ||
|
|
6a354d7c94 | ||
|
|
7f8dd4b254 | ||
|
|
0b1b8f05e1 | ||
|
|
53e9ffe656 | ||
|
|
2289073a1e | ||
|
|
687cb1cd2b | ||
|
|
e907050a17 | ||
|
|
a4b57c7e44 | ||
|
|
24bbfcdce7 | ||
|
|
d78b720350 | ||
|
|
d592208c74 | ||
|
|
971bbd088c | ||
|
|
b743577ebe | ||
|
|
a4cc6166a0 | ||
|
|
ed9850c4a4 | ||
|
|
ddbcf8549c | ||
|
|
921d0888cd | ||
|
|
21e1f3d103 | ||
|
|
53ab016098 | ||
|
|
0c249a7006 | ||
|
|
86c0fb48a3 | ||
|
|
3f1f99cf37 | ||
|
|
13d4823db6 | ||
|
|
30f61b26ff | ||
|
|
58b7d0b412 | ||
|
|
d37f5b87bd | ||
|
|
3f65cee17c | ||
|
|
094bf19ec4 | ||
|
|
f8d59b5aeb | ||
|
|
e9870c2922 | ||
|
|
50b7349fe0 | ||
|
|
61b3379f48 | ||
|
|
5010a0f5e7 | ||
|
|
52ca8deb10 | ||
|
|
156a9160ba | ||
|
|
68d66c873e | ||
|
|
948aa13fb9 | ||
|
|
9e993ac603 | ||
|
|
9f3f4ead4f | ||
|
|
068aa0ff1e | ||
|
|
e146c0796a | ||
|
|
cceab26bfb | ||
|
|
c0b1f32889 | ||
|
|
837dd46adf | ||
|
|
13512440ac | ||
|
|
7931423e8c | ||
|
|
62f28902c5 | ||
|
|
1f94e4cc14 | ||
|
|
61dfd5541f | ||
|
|
87321ce10b | ||
|
|
4f5aacdb3a | ||
|
|
b182f2d544 | ||
|
|
4fac8e9cd5 | ||
|
|
d94896c0fb | ||
|
|
15c5dd222f | ||
|
|
2930c8e9a8 | ||
|
|
b12b9b97f4 | ||
|
|
09e5aa6011 | ||
|
|
9549304007 | ||
|
|
f7ac32ceda | ||
|
|
92365f133d | ||
|
|
9daa9a6de8 | ||
|
|
23b1e428de | ||
|
|
f029f4f20e | ||
|
|
79e3d2b2d7 | ||
|
|
c74e5e0f04 | ||
|
|
15ef93ccc9 | ||
|
|
e017250445 | ||
|
|
17497eec43 | ||
|
|
6d0c6329ad | ||
|
|
f35be6b5cc | ||
|
|
b18ff48b4a | ||
|
|
7c28134214 | ||
|
|
16860e8a30 | ||
|
|
5362d1a89f | ||
|
|
5531296ee0 | ||
|
|
47db5e26f3 | ||
|
|
cf5197b68a | ||
|
|
9f831e91b3 | ||
|
|
2df0ebd895 | ||
|
|
7ad6dab383 | ||
|
|
612c8d5841 | ||
|
|
a35e476be5 | ||
|
|
87a7157fc4 | ||
|
|
fa34adbf6c | ||
|
|
ac942e0670 | ||
|
|
22e360d479 | ||
|
|
649936200e | ||
|
|
5d6e690c12 | ||
|
|
2f2ecadae7 | ||
|
|
6dfb9eba61 | ||
|
|
24587fe875 | ||
|
|
a1aebe6a2c | ||
|
|
2ad266582f | ||
|
|
1a47164876 | ||
|
|
cd22723623 | ||
|
|
aecaffa2f5 | ||
|
|
87df3596a2 | ||
|
|
41c7852128 | ||
|
|
78ec9856fb | ||
|
|
2a45467bf6 | ||
|
|
7fc5bfd787 | ||
|
|
04f592ba6d | ||
|
|
59889a6286 | ||
|
|
dc5cbd4df8 | ||
|
|
7ab9083d77 | ||
|
|
788803d588 | ||
|
|
cbfd904b9f | ||
|
|
c81dbf9d59 | ||
|
|
ac9c608542 | ||
|
|
a6c20853ca | ||
|
|
4ef0264ed3 | ||
|
|
169db9cc0a | ||
|
|
b693b8ccb1 | ||
|
|
3e98cceb00 | ||
|
|
46d962dcf1 | ||
|
|
7dbad42470 | ||
|
|
eb97781f68 | ||
|
|
4d0f8528d2 | ||
|
|
2c17b2bacc | ||
|
|
30bea20f7a | ||
|
|
d4cb4ef994 | ||
|
|
9c90ca297a | ||
|
|
a9e1a4cef3 | ||
|
|
0ce3621ac0 | ||
|
|
d527398dae | ||
|
|
2e9ac8945d | ||
|
|
40a5638005 | ||
|
|
8ba22183b9 | ||
|
|
2e11e66db4 | ||
|
|
eeb0710ad4 | ||
|
|
43c677ef37 | ||
|
|
95544e489d | ||
|
|
a08d021f77 | ||
|
|
b7b1d17ecb | ||
|
|
aa180b9581 | ||
|
|
57388254c4 | ||
|
|
f16f4e2c4c | ||
|
|
89b70e4352 | ||
|
|
6667336bd8 | ||
|
|
669ef7a0b1 | ||
|
|
c612985930 | ||
|
|
2e534ce41e | ||
|
|
fedb54bb38 | ||
|
|
68f5144084 | ||
|
|
fd3c22945b | ||
|
|
53496a1ecd | ||
|
|
da5cf99549 | ||
|
|
849c858495 | ||
|
|
808f964841 | ||
|
|
16a0f9db97 | ||
|
|
3bc5db4fd7 | ||
|
|
5269523ca1 | ||
|
|
89267b9e06 | ||
|
|
4bc9646e8f | ||
|
|
fd83628c49 | ||
|
|
62abfbec9e | ||
|
|
7cc0008837 | ||
|
|
0bf613bd34 | ||
|
|
43ab63455b | ||
|
|
47e7988c8e | ||
|
|
7ed095e635 | ||
|
|
cb8b0ec62e | ||
|
|
bf161f1eaa | ||
|
|
78c8447d1e | ||
|
|
5ffe50381a | ||
|
|
b08bd0c24a | ||
|
|
738ad8e9d3 | ||
|
|
fa7c42511a | ||
|
|
68ef9cb3dc | ||
|
|
8e176b9c61 | ||
|
|
426be153db | ||
|
|
c4f7c2d259 | ||
|
|
2a81efda0b | ||
|
|
882bfc79c7 | ||
|
|
6bad276589 | ||
|
|
47d8048a62 | ||
|
|
20d7ba5d7c | ||
|
|
e435e72654 | ||
|
|
497d66f7ec | ||
|
|
242b02a416 | ||
|
|
9644a6bb9c | ||
|
|
70d66062d6 | ||
|
|
39f6f9b0dc | ||
|
|
0454dd4e07 | ||
|
|
6f4e76c8f3 | ||
|
|
5cdcf2415d | ||
|
|
1719a2e08b | ||
|
|
5640a9fe73 | ||
|
|
4787e22f61 | ||
|
|
fb12e4e66a | ||
|
|
77740a1044 | ||
|
|
1fdfe7578f | ||
|
|
ebecf7047e | ||
|
|
00e8332bf5 | ||
|
|
5fc1f90822 | ||
|
|
0a1be3d19c | ||
|
|
40db3146b9 | ||
|
|
535c495b33 | ||
|
|
592446e430 | ||
|
|
7a5c9a821a | ||
|
|
c17a3b6fcc | ||
|
|
28d11553e0 | ||
|
|
1dbebe90ba | ||
|
|
06810e8e6a | ||
|
|
bd85ba9b6a | ||
|
|
be58cdda3b | ||
|
|
fcce4a8be6 | ||
|
|
61a558a062 | ||
|
|
59f69ac5ca | ||
|
|
f82ac34784 | ||
|
|
07cf6e723b | ||
|
|
78e3c6333f | ||
|
|
98e2684107 | ||
|
|
cb019fff9a | ||
|
|
4305c44440 | ||
|
|
a1e4143600 | ||
|
|
374c33e8dc | ||
|
|
dcfe7af9d3 | ||
|
|
049c7e00ca | ||
|
|
ee37d2f9c8 | ||
|
|
92ea697119 | ||
|
|
1c488d375f | ||
|
|
1a03b4949f | ||
|
|
731b7808cd | ||
|
|
d9da4cf24d | ||
|
|
666a3ee5e9 | ||
|
|
02469c2d4c | ||
|
|
2a629cae93 | ||
|
|
1f14c316a3 | ||
|
|
dac738a916 | ||
|
|
261b561bb2 | ||
|
|
0228379a2e | ||
|
|
da79215bc3 | ||
|
|
44323dc285 | ||
|
|
a59e1c7011 | ||
|
|
abb4d991ad | ||
|
|
f467c79a20 | ||
|
|
dcf41db878 | ||
|
|
c3c3a27af2 | ||
|
|
052f558131 | ||
|
|
e8aa7cff36 | ||
|
|
3411e45a0a | ||
|
|
a488c8cd5c | ||
|
|
9652b1a556 | ||
|
|
69f2c79ccb | ||
|
|
9d9d210176 | ||
|
|
487e1f871f | ||
|
|
0e27ac281f | ||
|
|
ad37f103fa | ||
|
|
b579bbf03b | ||
|
|
7f4d2534aa | ||
|
|
9e26daeb94 | ||
|
|
da6af184a6 | ||
|
|
4d347f1cc6 | ||
|
|
84e57b8136 | ||
|
|
63882c4a74 | ||
|
|
2ed5611a08 | ||
|
|
1467b704b8 | ||
|
|
94848e4811 | ||
|
|
3174f7ae86 | ||
|
|
962a339a8a | ||
|
|
6a76e6ae4a | ||
|
|
ce4371a80d | ||
|
|
b7ca6e087a | ||
|
|
368a0eea8a | ||
|
|
99c368fe62 | ||
|
|
ff406f8e11 | ||
|
|
b98165e077 | ||
|
|
e2a9cced94 | ||
|
|
cdae06e571 | ||
|
|
c0b05ada1a | ||
|
|
80dddb4cae | ||
|
|
245c89a6c1 | ||
|
|
4d044d4ac9 | ||
|
|
9cc2a04d54 | ||
|
|
50cdec19dd | ||
|
|
6d587278bd | ||
|
|
dde63e7459 | ||
|
|
8894f5030a | ||
|
|
9e862b8b53 | ||
|
|
24d4ada841 | ||
|
|
b1a8887548 | ||
|
|
d19997a056 | ||
|
|
de7591882d | ||
|
|
a00fc75c77 | ||
|
|
80fd827f8b | ||
|
|
1dd3c6de90 | ||
|
|
c8c43f13fd | ||
|
|
518bce50a5 | ||
|
|
4f87bea788 | ||
|
|
8054c9b4f5 | ||
|
|
935e0a365f | ||
|
|
b39a9924d8 | ||
|
|
19ec922e28 | ||
|
|
a225d6881f | ||
|
|
6675e99862 | ||
|
|
8cbe2b41f6 | ||
|
|
6a225cb4c0 | ||
|
|
e62d8bfabe | ||
|
|
d4cea84b1b | ||
|
|
b63f90a6c0 | ||
|
|
4370b6695e | ||
|
|
589f13f0f7 | ||
|
|
367017b352 | ||
|
|
ec26d31499 | ||
|
|
1bbc6db1c3 | ||
|
|
162472bdc2 | ||
|
|
aecac15809 | ||
|
|
6554af21b9 | ||
|
|
8583466c6a | ||
|
|
6666604069 | ||
|
|
13e7aacc9d | ||
|
|
737d502614 | ||
|
|
67dd649d00 | ||
|
|
b2fc51367b | ||
|
|
5771bb4907 | ||
|
|
9ba9674437 | ||
|
|
acb1532e34 | ||
|
|
e2093c34da | ||
|
|
a2e4ad90ba | ||
|
|
32e69c67f2 | ||
|
|
df0b5a187e | ||
|
|
cee0e5379b | ||
|
|
daf2bd7e66 | ||
|
|
4031077f6d | ||
|
|
fd72a64053 | ||
|
|
959a8b91bd | ||
|
|
44f1ff10e6 | ||
|
|
64e4589f4e | ||
|
|
20aba45cbe | ||
|
|
0b1c5b825e | ||
|
|
455624105b | ||
|
|
7ac5746e0d | ||
|
|
12997451f6 | ||
|
|
8c77e40695 | ||
|
|
2ddd91acf2 | ||
|
|
729e49cdc3 | ||
|
|
d64b49cc13 | ||
|
|
cfa8b3b272 | ||
|
|
51981335d5 | ||
|
|
70c5e1bbf1 | ||
|
|
43e88af28a | ||
|
|
ffc66f539f | ||
|
|
c4cb694d77 | ||
|
|
3fb9577ad9 | ||
|
|
34169491ac | ||
|
|
8eac859bab | ||
|
|
d99e3237f9 | ||
|
|
d9a9e0aea3 | ||
|
|
0ce03ae26b | ||
|
|
18653f8f69 | ||
|
|
6e0523109a | ||
|
|
b6fa4f641d | ||
|
|
ca6295d1bd | ||
|
|
18a1d31845 | ||
|
|
c5239a63ab | ||
|
|
1911269dc9 | ||
|
|
04ee1a87e9 | ||
|
|
a8fdb6db4d | ||
|
|
8860c74f0c |
4
.github/actions/build-image/action.yaml
vendored
4
.github/actions/build-image/action.yaml
vendored
@@ -47,7 +47,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.17.0
|
uses: docker/build-push-action@v6.18.0
|
||||||
env:
|
env:
|
||||||
DOCKER_BUILD_SUMMARY: false
|
DOCKER_BUILD_SUMMARY: false
|
||||||
DOCKER_BUILD_RECORD_UPLOAD: false
|
DOCKER_BUILD_RECORD_UPLOAD: false
|
||||||
@@ -73,7 +73,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.17.0
|
uses: docker/build-push-action@v6.18.0
|
||||||
env:
|
env:
|
||||||
DOCKER_BUILD_SUMMARY: false
|
DOCKER_BUILD_SUMMARY: false
|
||||||
DOCKER_BUILD_RECORD_UPLOAD: false
|
DOCKER_BUILD_RECORD_UPLOAD: false
|
||||||
|
|||||||
2
.github/workflows/ci-docker.yml
vendored
2
.github/workflows/ci-docker.yml
vendored
@@ -49,7 +49,7 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
python-version: "3.10"
|
python-version: "3.10"
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
uses: docker/setup-buildx-action@v3.10.0
|
uses: docker/setup-buildx-action@v3.11.1
|
||||||
|
|
||||||
- name: Set TAG
|
- name: Set TAG
|
||||||
run: |
|
run: |
|
||||||
|
|||||||
41
.github/workflows/ci.yml
vendored
41
.github/workflows/ci.yml
vendored
@@ -214,17 +214,51 @@ jobs:
|
|||||||
if: matrix.os == 'windows-latest'
|
if: matrix.os == 'windows-latest'
|
||||||
run: |
|
run: |
|
||||||
./venv/Scripts/activate
|
./venv/Scripts/activate
|
||||||
pytest -vv --cov-report=xml --tb=native -n auto tests
|
pytest -vv --cov-report=xml --tb=native -n auto tests --ignore=tests/integration/
|
||||||
- name: Run pytest
|
- name: Run pytest
|
||||||
if: matrix.os == 'ubuntu-latest' || matrix.os == 'macOS-latest'
|
if: matrix.os == 'ubuntu-latest' || matrix.os == 'macOS-latest'
|
||||||
run: |
|
run: |
|
||||||
. venv/bin/activate
|
. venv/bin/activate
|
||||||
pytest -vv --cov-report=xml --tb=native -n auto tests
|
pytest -vv --cov-report=xml --tb=native -n auto tests --ignore=tests/integration/
|
||||||
- name: Upload coverage to Codecov
|
- name: Upload coverage to Codecov
|
||||||
uses: codecov/codecov-action@v5.4.3
|
uses: codecov/codecov-action@v5.4.3
|
||||||
with:
|
with:
|
||||||
token: ${{ secrets.CODECOV_TOKEN }}
|
token: ${{ secrets.CODECOV_TOKEN }}
|
||||||
|
|
||||||
|
integration-tests:
|
||||||
|
name: Run integration tests
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs:
|
||||||
|
- common
|
||||||
|
steps:
|
||||||
|
- name: Check out code from GitHub
|
||||||
|
uses: actions/checkout@v4.2.2
|
||||||
|
- name: Set up Python 3.13
|
||||||
|
id: python
|
||||||
|
uses: actions/setup-python@v5.6.0
|
||||||
|
with:
|
||||||
|
python-version: "3.13"
|
||||||
|
- name: Restore Python virtual environment
|
||||||
|
id: cache-venv
|
||||||
|
uses: actions/cache@v4.2.3
|
||||||
|
with:
|
||||||
|
path: venv
|
||||||
|
key: ${{ runner.os }}-${{ steps.python.outputs.python-version }}-venv-${{ needs.common.outputs.cache-key }}
|
||||||
|
- name: Create Python virtual environment
|
||||||
|
if: steps.cache-venv.outputs.cache-hit != 'true'
|
||||||
|
run: |
|
||||||
|
python -m venv venv
|
||||||
|
. venv/bin/activate
|
||||||
|
python --version
|
||||||
|
pip install -r requirements.txt -r requirements_test.txt
|
||||||
|
pip install -e .
|
||||||
|
- name: Register matcher
|
||||||
|
run: echo "::add-matcher::.github/workflows/matchers/pytest.json"
|
||||||
|
- name: Run integration tests
|
||||||
|
run: |
|
||||||
|
. venv/bin/activate
|
||||||
|
pytest -vv --no-cov --tb=native -n auto tests/integration/
|
||||||
|
|
||||||
clang-format:
|
clang-format:
|
||||||
name: Check clang-format
|
name: Check clang-format
|
||||||
runs-on: ubuntu-24.04
|
runs-on: ubuntu-24.04
|
||||||
@@ -296,7 +330,7 @@ jobs:
|
|||||||
name: Run script/clang-tidy for ZEPHYR
|
name: Run script/clang-tidy for ZEPHYR
|
||||||
options: --environment nrf52-tidy --grep USE_ZEPHYR
|
options: --environment nrf52-tidy --grep USE_ZEPHYR
|
||||||
pio_cache_key: tidy-zephyr
|
pio_cache_key: tidy-zephyr
|
||||||
ignore_errors: true
|
ignore_errors: false
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
@@ -494,6 +528,7 @@ jobs:
|
|||||||
- flake8
|
- flake8
|
||||||
- pylint
|
- pylint
|
||||||
- pytest
|
- pytest
|
||||||
|
- integration-tests
|
||||||
- pyupgrade
|
- pyupgrade
|
||||||
- clang-tidy
|
- clang-tidy
|
||||||
- list-components
|
- list-components
|
||||||
|
|||||||
23
.github/workflows/lock.yml
vendored
23
.github/workflows/lock.yml
vendored
@@ -1,28 +1,11 @@
|
|||||||
---
|
---
|
||||||
name: Lock
|
name: Lock closed issues and PRs
|
||||||
|
|
||||||
on:
|
on:
|
||||||
schedule:
|
schedule:
|
||||||
- cron: "30 0 * * *"
|
- cron: "30 0 * * *" # Run daily at 00:30 UTC
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
|
|
||||||
permissions:
|
|
||||||
issues: write
|
|
||||||
pull-requests: write
|
|
||||||
|
|
||||||
concurrency:
|
|
||||||
group: lock
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
lock:
|
lock:
|
||||||
runs-on: ubuntu-latest
|
uses: esphome/workflows/.github/workflows/lock.yml@main
|
||||||
steps:
|
|
||||||
- uses: dessant/lock-threads@v5.0.1
|
|
||||||
with:
|
|
||||||
pr-inactive-days: "1"
|
|
||||||
pr-lock-reason: ""
|
|
||||||
exclude-any-pr-labels: keep-open
|
|
||||||
|
|
||||||
issue-inactive-days: "7"
|
|
||||||
issue-lock-reason: ""
|
|
||||||
exclude-any-issue-labels: keep-open
|
|
||||||
|
|||||||
4
.github/workflows/release.yml
vendored
4
.github/workflows/release.yml
vendored
@@ -99,7 +99,7 @@ jobs:
|
|||||||
python-version: "3.10"
|
python-version: "3.10"
|
||||||
|
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
uses: docker/setup-buildx-action@v3.10.0
|
uses: docker/setup-buildx-action@v3.11.1
|
||||||
|
|
||||||
- name: Log in to docker hub
|
- name: Log in to docker hub
|
||||||
uses: docker/login-action@v3.4.0
|
uses: docker/login-action@v3.4.0
|
||||||
@@ -178,7 +178,7 @@ jobs:
|
|||||||
merge-multiple: true
|
merge-multiple: true
|
||||||
|
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
uses: docker/setup-buildx-action@v3.10.0
|
uses: docker/setup-buildx-action@v3.11.1
|
||||||
|
|
||||||
- name: Log in to docker hub
|
- name: Log in to docker hub
|
||||||
if: matrix.registry == 'dockerhub'
|
if: matrix.registry == 'dockerhub'
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
repos:
|
repos:
|
||||||
- repo: https://github.com/astral-sh/ruff-pre-commit
|
- repo: https://github.com/astral-sh/ruff-pre-commit
|
||||||
# Ruff version.
|
# Ruff version.
|
||||||
rev: v0.11.10
|
rev: v0.12.2
|
||||||
hooks:
|
hooks:
|
||||||
# Run the linter.
|
# Run the linter.
|
||||||
- id: ruff
|
- id: ruff
|
||||||
@@ -12,7 +12,7 @@ repos:
|
|||||||
# Run the formatter.
|
# Run the formatter.
|
||||||
- id: ruff-format
|
- id: ruff-format
|
||||||
- repo: https://github.com/PyCQA/flake8
|
- repo: https://github.com/PyCQA/flake8
|
||||||
rev: 7.2.0
|
rev: 7.3.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: flake8
|
- id: flake8
|
||||||
additional_dependencies:
|
additional_dependencies:
|
||||||
|
|||||||
18
CODEOWNERS
18
CODEOWNERS
@@ -87,6 +87,7 @@ esphome/components/bp1658cj/* @Cossid
|
|||||||
esphome/components/bp5758d/* @Cossid
|
esphome/components/bp5758d/* @Cossid
|
||||||
esphome/components/button/* @esphome/core
|
esphome/components/button/* @esphome/core
|
||||||
esphome/components/bytebuffer/* @clydebarrow
|
esphome/components/bytebuffer/* @clydebarrow
|
||||||
|
esphome/components/camera/* @DT-art1 @bdraco
|
||||||
esphome/components/canbus/* @danielschramm @mvturnho
|
esphome/components/canbus/* @danielschramm @mvturnho
|
||||||
esphome/components/cap1188/* @mreditor97
|
esphome/components/cap1188/* @mreditor97
|
||||||
esphome/components/captive_portal/* @OttoWinter
|
esphome/components/captive_portal/* @OttoWinter
|
||||||
@@ -124,6 +125,7 @@ esphome/components/dht/* @OttoWinter
|
|||||||
esphome/components/display_menu_base/* @numo68
|
esphome/components/display_menu_base/* @numo68
|
||||||
esphome/components/dps310/* @kbx81
|
esphome/components/dps310/* @kbx81
|
||||||
esphome/components/ds1307/* @badbadc0ffee
|
esphome/components/ds1307/* @badbadc0ffee
|
||||||
|
esphome/components/ds2484/* @mrk-its
|
||||||
esphome/components/dsmr/* @glmnet @zuidwijk
|
esphome/components/dsmr/* @glmnet @zuidwijk
|
||||||
esphome/components/duty_time/* @dudanov
|
esphome/components/duty_time/* @dudanov
|
||||||
esphome/components/ee895/* @Stock-M
|
esphome/components/ee895/* @Stock-M
|
||||||
@@ -139,16 +141,19 @@ esphome/components/es7210/* @kahrendt
|
|||||||
esphome/components/es7243e/* @kbx81
|
esphome/components/es7243e/* @kbx81
|
||||||
esphome/components/es8156/* @kbx81
|
esphome/components/es8156/* @kbx81
|
||||||
esphome/components/es8311/* @kahrendt @kroimon
|
esphome/components/es8311/* @kahrendt @kroimon
|
||||||
|
esphome/components/es8388/* @P4uLT
|
||||||
esphome/components/esp32/* @esphome/core
|
esphome/components/esp32/* @esphome/core
|
||||||
esphome/components/esp32_ble/* @Rapsssito @jesserockz
|
esphome/components/esp32_ble/* @Rapsssito @jesserockz
|
||||||
esphome/components/esp32_ble_client/* @jesserockz
|
esphome/components/esp32_ble_client/* @jesserockz
|
||||||
esphome/components/esp32_ble_server/* @Rapsssito @clydebarrow @jesserockz
|
esphome/components/esp32_ble_server/* @Rapsssito @clydebarrow @jesserockz
|
||||||
esphome/components/esp32_camera_web_server/* @ayufan
|
esphome/components/esp32_camera_web_server/* @ayufan
|
||||||
esphome/components/esp32_can/* @Sympatron
|
esphome/components/esp32_can/* @Sympatron
|
||||||
|
esphome/components/esp32_hosted/* @swoboda1337
|
||||||
esphome/components/esp32_improv/* @jesserockz
|
esphome/components/esp32_improv/* @jesserockz
|
||||||
esphome/components/esp32_rmt/* @jesserockz
|
esphome/components/esp32_rmt/* @jesserockz
|
||||||
esphome/components/esp32_rmt_led_strip/* @jesserockz
|
esphome/components/esp32_rmt_led_strip/* @jesserockz
|
||||||
esphome/components/esp8266/* @esphome/core
|
esphome/components/esp8266/* @esphome/core
|
||||||
|
esphome/components/esp_ldo/* @clydebarrow
|
||||||
esphome/components/ethernet_info/* @gtjadsonsantos
|
esphome/components/ethernet_info/* @gtjadsonsantos
|
||||||
esphome/components/event/* @nohat
|
esphome/components/event/* @nohat
|
||||||
esphome/components/event_emitter/* @Rapsssito
|
esphome/components/event_emitter/* @Rapsssito
|
||||||
@@ -165,6 +170,7 @@ esphome/components/ft5x06/* @clydebarrow
|
|||||||
esphome/components/ft63x6/* @gpambrozio
|
esphome/components/ft63x6/* @gpambrozio
|
||||||
esphome/components/gcja5/* @gcormier
|
esphome/components/gcja5/* @gcormier
|
||||||
esphome/components/gdk101/* @Szewcson
|
esphome/components/gdk101/* @Szewcson
|
||||||
|
esphome/components/gl_r01_i2c/* @pkejval
|
||||||
esphome/components/globals/* @esphome/core
|
esphome/components/globals/* @esphome/core
|
||||||
esphome/components/gp2y1010au0f/* @zry98
|
esphome/components/gp2y1010au0f/* @zry98
|
||||||
esphome/components/gp8403/* @jesserockz
|
esphome/components/gp8403/* @jesserockz
|
||||||
@@ -234,6 +240,7 @@ esphome/components/kamstrup_kmp/* @cfeenstra1024
|
|||||||
esphome/components/key_collector/* @ssieb
|
esphome/components/key_collector/* @ssieb
|
||||||
esphome/components/key_provider/* @ssieb
|
esphome/components/key_provider/* @ssieb
|
||||||
esphome/components/kuntze/* @ssieb
|
esphome/components/kuntze/* @ssieb
|
||||||
|
esphome/components/lc709203f/* @ilikecake
|
||||||
esphome/components/lcd_menu/* @numo68
|
esphome/components/lcd_menu/* @numo68
|
||||||
esphome/components/ld2410/* @regevbr @sebcaps
|
esphome/components/ld2410/* @regevbr @sebcaps
|
||||||
esphome/components/ld2420/* @descipher
|
esphome/components/ld2420/* @descipher
|
||||||
@@ -244,9 +251,11 @@ esphome/components/libretiny_pwm/* @kuba2k2
|
|||||||
esphome/components/light/* @esphome/core
|
esphome/components/light/* @esphome/core
|
||||||
esphome/components/lightwaverf/* @max246
|
esphome/components/lightwaverf/* @max246
|
||||||
esphome/components/lilygo_t5_47/touchscreen/* @jesserockz
|
esphome/components/lilygo_t5_47/touchscreen/* @jesserockz
|
||||||
|
esphome/components/ln882x/* @lamauny
|
||||||
esphome/components/lock/* @esphome/core
|
esphome/components/lock/* @esphome/core
|
||||||
esphome/components/logger/* @esphome/core
|
esphome/components/logger/* @esphome/core
|
||||||
esphome/components/logger/select/* @clydebarrow
|
esphome/components/logger/select/* @clydebarrow
|
||||||
|
esphome/components/lps22/* @nagisa
|
||||||
esphome/components/ltr390/* @latonita @sjtrny
|
esphome/components/ltr390/* @latonita @sjtrny
|
||||||
esphome/components/ltr501/* @latonita
|
esphome/components/ltr501/* @latonita
|
||||||
esphome/components/ltr_als_ps/* @latonita
|
esphome/components/ltr_als_ps/* @latonita
|
||||||
@@ -319,6 +328,8 @@ esphome/components/number/* @esphome/core
|
|||||||
esphome/components/one_wire/* @ssieb
|
esphome/components/one_wire/* @ssieb
|
||||||
esphome/components/online_image/* @clydebarrow @guillempages
|
esphome/components/online_image/* @clydebarrow @guillempages
|
||||||
esphome/components/opentherm/* @olegtarasov
|
esphome/components/opentherm/* @olegtarasov
|
||||||
|
esphome/components/openthread/* @mrene
|
||||||
|
esphome/components/opt3001/* @ccutrer
|
||||||
esphome/components/ota/* @esphome/core
|
esphome/components/ota/* @esphome/core
|
||||||
esphome/components/output/* @esphome/core
|
esphome/components/output/* @esphome/core
|
||||||
esphome/components/packet_transport/* @clydebarrow
|
esphome/components/packet_transport/* @clydebarrow
|
||||||
@@ -326,6 +337,7 @@ esphome/components/pca6416a/* @Mat931
|
|||||||
esphome/components/pca9554/* @clydebarrow @hwstar
|
esphome/components/pca9554/* @clydebarrow @hwstar
|
||||||
esphome/components/pcf85063/* @brogon
|
esphome/components/pcf85063/* @brogon
|
||||||
esphome/components/pcf8563/* @KoenBreeman
|
esphome/components/pcf8563/* @KoenBreeman
|
||||||
|
esphome/components/pi4ioe5v6408/* @jesserockz
|
||||||
esphome/components/pid/* @OttoWinter
|
esphome/components/pid/* @OttoWinter
|
||||||
esphome/components/pipsolar/* @andreashergert1984
|
esphome/components/pipsolar/* @andreashergert1984
|
||||||
esphome/components/pm1006/* @habbie
|
esphome/components/pm1006/* @habbie
|
||||||
@@ -432,6 +444,8 @@ esphome/components/sun/* @OttoWinter
|
|||||||
esphome/components/sun_gtil2/* @Mat931
|
esphome/components/sun_gtil2/* @Mat931
|
||||||
esphome/components/switch/* @esphome/core
|
esphome/components/switch/* @esphome/core
|
||||||
esphome/components/switch/binary_sensor/* @ssieb
|
esphome/components/switch/binary_sensor/* @ssieb
|
||||||
|
esphome/components/sx126x/* @swoboda1337
|
||||||
|
esphome/components/sx127x/* @swoboda1337
|
||||||
esphome/components/syslog/* @clydebarrow
|
esphome/components/syslog/* @clydebarrow
|
||||||
esphome/components/t6615/* @tylermenezes
|
esphome/components/t6615/* @tylermenezes
|
||||||
esphome/components/tc74/* @sethgirvan
|
esphome/components/tc74/* @sethgirvan
|
||||||
@@ -486,10 +500,11 @@ esphome/components/vbus/* @ssieb
|
|||||||
esphome/components/veml3235/* @kbx81
|
esphome/components/veml3235/* @kbx81
|
||||||
esphome/components/veml7700/* @latonita
|
esphome/components/veml7700/* @latonita
|
||||||
esphome/components/version/* @esphome/core
|
esphome/components/version/* @esphome/core
|
||||||
esphome/components/voice_assistant/* @jesserockz
|
esphome/components/voice_assistant/* @jesserockz @kahrendt
|
||||||
esphome/components/wake_on_lan/* @clydebarrow @willwill2will54
|
esphome/components/wake_on_lan/* @clydebarrow @willwill2will54
|
||||||
esphome/components/watchdog/* @oarcher
|
esphome/components/watchdog/* @oarcher
|
||||||
esphome/components/waveshare_epaper/* @clydebarrow
|
esphome/components/waveshare_epaper/* @clydebarrow
|
||||||
|
esphome/components/web_server/ota/* @esphome/core
|
||||||
esphome/components/web_server_base/* @OttoWinter
|
esphome/components/web_server_base/* @OttoWinter
|
||||||
esphome/components/web_server_idf/* @dentra
|
esphome/components/web_server_idf/* @dentra
|
||||||
esphome/components/weikai/* @DrCoolZic
|
esphome/components/weikai/* @DrCoolZic
|
||||||
@@ -516,6 +531,7 @@ esphome/components/xiaomi_lywsd03mmc/* @ahpohl
|
|||||||
esphome/components/xiaomi_mhoc303/* @drug123
|
esphome/components/xiaomi_mhoc303/* @drug123
|
||||||
esphome/components/xiaomi_mhoc401/* @vevsvevs
|
esphome/components/xiaomi_mhoc401/* @vevsvevs
|
||||||
esphome/components/xiaomi_rtcgq02lm/* @jesserockz
|
esphome/components/xiaomi_rtcgq02lm/* @jesserockz
|
||||||
|
esphome/components/xiaomi_xmwsdj04mmc/* @medusalix
|
||||||
esphome/components/xl9535/* @mreditor97
|
esphome/components/xl9535/* @mreditor97
|
||||||
esphome/components/xpt2046/touchscreen/* @nielsnl68 @numo68
|
esphome/components/xpt2046/touchscreen/* @nielsnl68 @numo68
|
||||||
esphome/components/xxtea/* @clydebarrow
|
esphome/components/xxtea/* @clydebarrow
|
||||||
|
|||||||
2
Doxyfile
2
Doxyfile
@@ -48,7 +48,7 @@ PROJECT_NAME = ESPHome
|
|||||||
# could be handy for archiving the generated documentation or if some version
|
# could be handy for archiving the generated documentation or if some version
|
||||||
# control system is used.
|
# control system is used.
|
||||||
|
|
||||||
PROJECT_NUMBER = 2025.6.0-dev
|
PROJECT_NUMBER = 2025.7.0b1
|
||||||
|
|
||||||
# Using the PROJECT_BRIEF tag one can provide an optional one line description
|
# Using the PROJECT_BRIEF tag one can provide an optional one line description
|
||||||
# for a project that appears at the top of each page and should give viewer a
|
# for a project that appears at the top of each page and should give viewer a
|
||||||
|
|||||||
@@ -34,11 +34,9 @@ from esphome.const import (
|
|||||||
CONF_PORT,
|
CONF_PORT,
|
||||||
CONF_SUBSTITUTIONS,
|
CONF_SUBSTITUTIONS,
|
||||||
CONF_TOPIC,
|
CONF_TOPIC,
|
||||||
PLATFORM_BK72XX,
|
|
||||||
PLATFORM_ESP32,
|
PLATFORM_ESP32,
|
||||||
PLATFORM_ESP8266,
|
PLATFORM_ESP8266,
|
||||||
PLATFORM_RP2040,
|
PLATFORM_RP2040,
|
||||||
PLATFORM_RTL87XX,
|
|
||||||
SECRETS_FILES,
|
SECRETS_FILES,
|
||||||
)
|
)
|
||||||
from esphome.core import CORE, EsphomeError, coroutine
|
from esphome.core import CORE, EsphomeError, coroutine
|
||||||
@@ -134,6 +132,7 @@ def get_port_type(port):
|
|||||||
|
|
||||||
|
|
||||||
def run_miniterm(config, port, args):
|
def run_miniterm(config, port, args):
|
||||||
|
from aioesphomeapi import LogParser
|
||||||
import serial
|
import serial
|
||||||
|
|
||||||
from esphome import platformio_api
|
from esphome import platformio_api
|
||||||
@@ -158,6 +157,7 @@ def run_miniterm(config, port, args):
|
|||||||
ser.dtr = False
|
ser.dtr = False
|
||||||
ser.rts = False
|
ser.rts = False
|
||||||
|
|
||||||
|
parser = LogParser()
|
||||||
tries = 0
|
tries = 0
|
||||||
while tries < 5:
|
while tries < 5:
|
||||||
try:
|
try:
|
||||||
@@ -174,8 +174,7 @@ def run_miniterm(config, port, args):
|
|||||||
.decode("utf8", "backslashreplace")
|
.decode("utf8", "backslashreplace")
|
||||||
)
|
)
|
||||||
time_str = datetime.now().time().strftime("[%H:%M:%S]")
|
time_str = datetime.now().time().strftime("[%H:%M:%S]")
|
||||||
message = time_str + line
|
safe_print(parser.parse_line(line, time_str))
|
||||||
safe_print(message)
|
|
||||||
|
|
||||||
backtrace_state = platformio_api.process_stacktrace(
|
backtrace_state = platformio_api.process_stacktrace(
|
||||||
config, line, backtrace_state=backtrace_state
|
config, line, backtrace_state=backtrace_state
|
||||||
@@ -353,7 +352,7 @@ def upload_program(config, args, host):
|
|||||||
if CORE.target_platform in (PLATFORM_RP2040):
|
if CORE.target_platform in (PLATFORM_RP2040):
|
||||||
return upload_using_platformio(config, args.device)
|
return upload_using_platformio(config, args.device)
|
||||||
|
|
||||||
if CORE.target_platform in (PLATFORM_BK72XX, PLATFORM_RTL87XX):
|
if CORE.is_libretiny:
|
||||||
return upload_using_platformio(config, host)
|
return upload_using_platformio(config, host)
|
||||||
|
|
||||||
return 1 # Unknown target platform
|
return 1 # Unknown target platform
|
||||||
@@ -593,15 +592,20 @@ def command_update_all(args):
|
|||||||
middle_text = f" {middle_text} "
|
middle_text = f" {middle_text} "
|
||||||
width = len(click.unstyle(middle_text))
|
width = len(click.unstyle(middle_text))
|
||||||
half_line = "=" * ((twidth - width) // 2)
|
half_line = "=" * ((twidth - width) // 2)
|
||||||
click.echo(f"{half_line}{middle_text}{half_line}")
|
safe_print(f"{half_line}{middle_text}{half_line}")
|
||||||
|
|
||||||
for f in files:
|
for f in files:
|
||||||
print(f"Updating {color(AnsiFore.CYAN, f)}")
|
safe_print(f"Updating {color(AnsiFore.CYAN, f)}")
|
||||||
print("-" * twidth)
|
safe_print("-" * twidth)
|
||||||
print()
|
safe_print()
|
||||||
rc = run_external_process(
|
if CORE.dashboard:
|
||||||
"esphome", "--dashboard", "run", f, "--no-logs", "--device", "OTA"
|
rc = run_external_process(
|
||||||
)
|
"esphome", "--dashboard", "run", f, "--no-logs", "--device", "OTA"
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
rc = run_external_process(
|
||||||
|
"esphome", "run", f, "--no-logs", "--device", "OTA"
|
||||||
|
)
|
||||||
if rc == 0:
|
if rc == 0:
|
||||||
print_bar(f"[{color(AnsiFore.BOLD_GREEN, 'SUCCESS')}] {f}")
|
print_bar(f"[{color(AnsiFore.BOLD_GREEN, 'SUCCESS')}] {f}")
|
||||||
success[f] = True
|
success[f] = True
|
||||||
@@ -609,17 +613,17 @@ def command_update_all(args):
|
|||||||
print_bar(f"[{color(AnsiFore.BOLD_RED, 'ERROR')}] {f}")
|
print_bar(f"[{color(AnsiFore.BOLD_RED, 'ERROR')}] {f}")
|
||||||
success[f] = False
|
success[f] = False
|
||||||
|
|
||||||
print()
|
safe_print()
|
||||||
print()
|
safe_print()
|
||||||
print()
|
safe_print()
|
||||||
|
|
||||||
print_bar(f"[{color(AnsiFore.BOLD_WHITE, 'SUMMARY')}]")
|
print_bar(f"[{color(AnsiFore.BOLD_WHITE, 'SUMMARY')}]")
|
||||||
failed = 0
|
failed = 0
|
||||||
for f in files:
|
for f in files:
|
||||||
if success[f]:
|
if success[f]:
|
||||||
print(f" - {f}: {color(AnsiFore.GREEN, 'SUCCESS')}")
|
safe_print(f" - {f}: {color(AnsiFore.GREEN, 'SUCCESS')}")
|
||||||
else:
|
else:
|
||||||
print(f" - {f}: {color(AnsiFore.BOLD_RED, 'FAILED')}")
|
safe_print(f" - {f}: {color(AnsiFore.BOLD_RED, 'FAILED')}")
|
||||||
failed += 1
|
failed += 1
|
||||||
return failed
|
return failed
|
||||||
|
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ from esphome.cpp_generator import ( # noqa: F401
|
|||||||
TemplateArguments,
|
TemplateArguments,
|
||||||
add,
|
add,
|
||||||
add_build_flag,
|
add_build_flag,
|
||||||
|
add_build_unflag,
|
||||||
add_define,
|
add_define,
|
||||||
add_global,
|
add_global,
|
||||||
add_library,
|
add_library,
|
||||||
@@ -34,6 +35,7 @@ from esphome.cpp_generator import ( # noqa: F401
|
|||||||
process_lambda,
|
process_lambda,
|
||||||
progmem_array,
|
progmem_array,
|
||||||
safe_exp,
|
safe_exp,
|
||||||
|
set_cpp_standard,
|
||||||
statement,
|
statement,
|
||||||
static_const_array,
|
static_const_array,
|
||||||
templatable,
|
templatable,
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ namespace a4988 {
|
|||||||
static const char *const TAG = "a4988.stepper";
|
static const char *const TAG = "a4988.stepper";
|
||||||
|
|
||||||
void A4988::setup() {
|
void A4988::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up A4988...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
if (this->sleep_pin_ != nullptr) {
|
if (this->sleep_pin_ != nullptr) {
|
||||||
this->sleep_pin_->setup();
|
this->sleep_pin_->setup();
|
||||||
this->sleep_pin_->digital_write(false);
|
this->sleep_pin_->digital_write(false);
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ namespace absolute_humidity {
|
|||||||
static const char *const TAG = "absolute_humidity.sensor";
|
static const char *const TAG = "absolute_humidity.sensor";
|
||||||
|
|
||||||
void AbsoluteHumidityComponent::setup() {
|
void AbsoluteHumidityComponent::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up absolute humidity '%s'...", this->get_name().c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'", this->get_name().c_str());
|
||||||
|
|
||||||
ESP_LOGD(TAG, " Added callback for temperature '%s'", this->temperature_sensor_->get_name().c_str());
|
ESP_LOGD(TAG, " Added callback for temperature '%s'", this->temperature_sensor_->get_name().c_str());
|
||||||
this->temperature_sensor_->add_on_state_callback([this](float state) { this->temperature_callback_(state); });
|
this->temperature_sensor_->add_on_state_callback([this](float state) { this->temperature_callback_(state); });
|
||||||
@@ -40,9 +40,11 @@ void AbsoluteHumidityComponent::dump_config() {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGCONFIG(TAG, "Sources");
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Temperature: '%s'", this->temperature_sensor_->get_name().c_str());
|
"Sources\n"
|
||||||
ESP_LOGCONFIG(TAG, " Relative Humidity: '%s'", this->humidity_sensor_->get_name().c_str());
|
" Temperature: '%s'\n"
|
||||||
|
" Relative Humidity: '%s'",
|
||||||
|
this->temperature_sensor_->get_name().c_str(), this->humidity_sensor_->get_name().c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
float AbsoluteHumidityComponent::get_setup_priority() const { return setup_priority::DATA; }
|
float AbsoluteHumidityComponent::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|||||||
@@ -4,6 +4,7 @@
|
|||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
|
#include <numbers>
|
||||||
|
|
||||||
#ifdef USE_ESP8266
|
#ifdef USE_ESP8266
|
||||||
#include <core_esp8266_waveform.h>
|
#include <core_esp8266_waveform.h>
|
||||||
@@ -193,18 +194,17 @@ void AcDimmer::setup() {
|
|||||||
setTimer1Callback(&timer_interrupt);
|
setTimer1Callback(&timer_interrupt);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ESP32
|
#ifdef USE_ESP32
|
||||||
// 80 Divider -> 1 count=1µs
|
// timer frequency of 1mhz
|
||||||
dimmer_timer = timerBegin(0, 80, true);
|
dimmer_timer = timerBegin(1000000);
|
||||||
timerAttachInterrupt(dimmer_timer, &AcDimmerDataStore::s_timer_intr, true);
|
timerAttachInterrupt(dimmer_timer, &AcDimmerDataStore::s_timer_intr);
|
||||||
// For ESP32, we can't use dynamic interval calculation because the timerX functions
|
// For ESP32, we can't use dynamic interval calculation because the timerX functions
|
||||||
// are not callable from ISR (placed in flash storage).
|
// are not callable from ISR (placed in flash storage).
|
||||||
// Here we just use an interrupt firing every 50 µs.
|
// Here we just use an interrupt firing every 50 µs.
|
||||||
timerAlarmWrite(dimmer_timer, 50, true);
|
timerAlarm(dimmer_timer, 50, true, 0);
|
||||||
timerAlarmEnable(dimmer_timer);
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
void AcDimmer::write_state(float state) {
|
void AcDimmer::write_state(float state) {
|
||||||
state = std::acos(1 - (2 * state)) / 3.14159; // RMS power compensation
|
state = std::acos(1 - (2 * state)) / std::numbers::pi; // RMS power compensation
|
||||||
auto new_value = static_cast<uint16_t>(roundf(state * 65535));
|
auto new_value = static_cast<uint16_t>(roundf(state * 65535));
|
||||||
if (new_value != 0 && this->store_.value == 0)
|
if (new_value != 0 && this->store_.value == 0)
|
||||||
this->store_.init_cycle = this->init_with_half_cycle_;
|
this->store_.init_cycle = this->init_with_half_cycle_;
|
||||||
@@ -214,8 +214,10 @@ void AcDimmer::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "AcDimmer:");
|
ESP_LOGCONFIG(TAG, "AcDimmer:");
|
||||||
LOG_PIN(" Output Pin: ", this->gate_pin_);
|
LOG_PIN(" Output Pin: ", this->gate_pin_);
|
||||||
LOG_PIN(" Zero-Cross Pin: ", this->zero_cross_pin_);
|
LOG_PIN(" Zero-Cross Pin: ", this->zero_cross_pin_);
|
||||||
ESP_LOGCONFIG(TAG, " Min Power: %.1f%%", this->store_.min_power / 10.0f);
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Init with half cycle: %s", YESNO(this->init_with_half_cycle_));
|
" Min Power: %.1f%%\n"
|
||||||
|
" Init with half cycle: %s",
|
||||||
|
this->store_.min_power / 10.0f, YESNO(this->init_with_half_cycle_));
|
||||||
if (method_ == DIM_METHOD_LEADING_PULSE) {
|
if (method_ == DIM_METHOD_LEADING_PULSE) {
|
||||||
ESP_LOGCONFIG(TAG, " Method: leading pulse");
|
ESP_LOGCONFIG(TAG, " Method: leading pulse");
|
||||||
} else if (method_ == DIM_METHOD_LEADING) {
|
} else if (method_ == DIM_METHOD_LEADING) {
|
||||||
|
|||||||
@@ -10,8 +10,15 @@ from esphome.components.esp32.const import (
|
|||||||
VARIANT_ESP32S2,
|
VARIANT_ESP32S2,
|
||||||
VARIANT_ESP32S3,
|
VARIANT_ESP32S3,
|
||||||
)
|
)
|
||||||
|
from esphome.config_helpers import filter_source_files_from_platform
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
from esphome.const import CONF_ANALOG, CONF_INPUT, CONF_NUMBER, PLATFORM_ESP8266
|
from esphome.const import (
|
||||||
|
CONF_ANALOG,
|
||||||
|
CONF_INPUT,
|
||||||
|
CONF_NUMBER,
|
||||||
|
PLATFORM_ESP8266,
|
||||||
|
PlatformFramework,
|
||||||
|
)
|
||||||
from esphome.core import CORE
|
from esphome.core import CORE
|
||||||
|
|
||||||
CODEOWNERS = ["@esphome/core"]
|
CODEOWNERS = ["@esphome/core"]
|
||||||
@@ -229,3 +236,20 @@ def validate_adc_pin(value):
|
|||||||
)(value)
|
)(value)
|
||||||
|
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
|
FILTER_SOURCE_FILES = filter_source_files_from_platform(
|
||||||
|
{
|
||||||
|
"adc_sensor_esp32.cpp": {
|
||||||
|
PlatformFramework.ESP32_ARDUINO,
|
||||||
|
PlatformFramework.ESP32_IDF,
|
||||||
|
},
|
||||||
|
"adc_sensor_esp8266.cpp": {PlatformFramework.ESP8266_ARDUINO},
|
||||||
|
"adc_sensor_rp2040.cpp": {PlatformFramework.RP2040_ARDUINO},
|
||||||
|
"adc_sensor_libretiny.cpp": {
|
||||||
|
PlatformFramework.BK72XX_ARDUINO,
|
||||||
|
PlatformFramework.RTL87XX_ARDUINO,
|
||||||
|
PlatformFramework.LN882X_ARDUINO,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|||||||
@@ -15,8 +15,7 @@ namespace adc {
|
|||||||
|
|
||||||
#ifdef USE_ESP32
|
#ifdef USE_ESP32
|
||||||
// clang-format off
|
// clang-format off
|
||||||
#if (ESP_IDF_VERSION_MAJOR == 4 && ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(4, 4, 7)) || \
|
#if (ESP_IDF_VERSION_MAJOR == 5 && \
|
||||||
(ESP_IDF_VERSION_MAJOR == 5 && \
|
|
||||||
((ESP_IDF_VERSION_MINOR == 0 && ESP_IDF_VERSION_PATCH >= 5) || \
|
((ESP_IDF_VERSION_MINOR == 0 && ESP_IDF_VERSION_PATCH >= 5) || \
|
||||||
(ESP_IDF_VERSION_MINOR == 1 && ESP_IDF_VERSION_PATCH >= 3) || \
|
(ESP_IDF_VERSION_MINOR == 1 && ESP_IDF_VERSION_PATCH >= 3) || \
|
||||||
(ESP_IDF_VERSION_MINOR >= 2)) \
|
(ESP_IDF_VERSION_MINOR >= 2)) \
|
||||||
@@ -28,19 +27,24 @@ static const adc_atten_t ADC_ATTEN_DB_12_COMPAT = ADC_ATTEN_DB_11;
|
|||||||
#endif
|
#endif
|
||||||
#endif // USE_ESP32
|
#endif // USE_ESP32
|
||||||
|
|
||||||
enum class SamplingMode : uint8_t { AVG = 0, MIN = 1, MAX = 2 };
|
enum class SamplingMode : uint8_t {
|
||||||
|
AVG = 0,
|
||||||
|
MIN = 1,
|
||||||
|
MAX = 2,
|
||||||
|
};
|
||||||
|
|
||||||
const LogString *sampling_mode_to_str(SamplingMode mode);
|
const LogString *sampling_mode_to_str(SamplingMode mode);
|
||||||
|
|
||||||
class Aggregator {
|
class Aggregator {
|
||||||
public:
|
public:
|
||||||
|
Aggregator(SamplingMode mode);
|
||||||
void add_sample(uint32_t value);
|
void add_sample(uint32_t value);
|
||||||
uint32_t aggregate();
|
uint32_t aggregate();
|
||||||
Aggregator(SamplingMode mode);
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
SamplingMode mode_{SamplingMode::AVG};
|
|
||||||
uint32_t aggr_{0};
|
uint32_t aggr_{0};
|
||||||
uint32_t samples_{0};
|
uint32_t samples_{0};
|
||||||
|
SamplingMode mode_{SamplingMode::AVG};
|
||||||
};
|
};
|
||||||
|
|
||||||
class ADCSensor : public sensor::Sensor, public PollingComponent, public voltage_sampler::VoltageSampler {
|
class ADCSensor : public sensor::Sensor, public PollingComponent, public voltage_sampler::VoltageSampler {
|
||||||
@@ -81,9 +85,9 @@ class ADCSensor : public sensor::Sensor, public PollingComponent, public voltage
|
|||||||
#endif // USE_RP2040
|
#endif // USE_RP2040
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
InternalGPIOPin *pin_;
|
|
||||||
bool output_raw_{false};
|
|
||||||
uint8_t sample_count_{1};
|
uint8_t sample_count_{1};
|
||||||
|
bool output_raw_{false};
|
||||||
|
InternalGPIOPin *pin_;
|
||||||
SamplingMode sampling_mode_{SamplingMode::AVG};
|
SamplingMode sampling_mode_{SamplingMode::AVG};
|
||||||
|
|
||||||
#ifdef USE_RP2040
|
#ifdef USE_RP2040
|
||||||
@@ -95,11 +99,7 @@ class ADCSensor : public sensor::Sensor, public PollingComponent, public voltage
|
|||||||
adc1_channel_t channel1_{ADC1_CHANNEL_MAX};
|
adc1_channel_t channel1_{ADC1_CHANNEL_MAX};
|
||||||
adc2_channel_t channel2_{ADC2_CHANNEL_MAX};
|
adc2_channel_t channel2_{ADC2_CHANNEL_MAX};
|
||||||
bool autorange_{false};
|
bool autorange_{false};
|
||||||
#if ESP_IDF_VERSION_MAJOR >= 5
|
|
||||||
esp_adc_cal_characteristics_t cal_characteristics_[SOC_ADC_ATTEN_NUM] = {};
|
esp_adc_cal_characteristics_t cal_characteristics_[SOC_ADC_ATTEN_NUM] = {};
|
||||||
#else
|
|
||||||
esp_adc_cal_characteristics_t cal_characteristics_[ADC_ATTEN_MAX] = {};
|
|
||||||
#endif // ESP_IDF_VERSION_MAJOR
|
|
||||||
#endif // USE_ESP32
|
#endif // USE_ESP32
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -61,7 +61,7 @@ uint32_t Aggregator::aggregate() {
|
|||||||
|
|
||||||
void ADCSensor::update() {
|
void ADCSensor::update() {
|
||||||
float value_v = this->sample();
|
float value_v = this->sample();
|
||||||
ESP_LOGV(TAG, "'%s': Got voltage=%.4fV", this->get_name().c_str(), value_v);
|
ESP_LOGV(TAG, "'%s': Voltage=%.4fV", this->get_name().c_str(), value_v);
|
||||||
this->publish_state(value_v);
|
this->publish_state(value_v);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -22,7 +22,7 @@ static const int ADC_MAX = (1 << SOC_ADC_RTC_MAX_BITWIDTH) - 1;
|
|||||||
static const int ADC_HALF = (1 << SOC_ADC_RTC_MAX_BITWIDTH) >> 1;
|
static const int ADC_HALF = (1 << SOC_ADC_RTC_MAX_BITWIDTH) >> 1;
|
||||||
|
|
||||||
void ADCSensor::setup() {
|
void ADCSensor::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ADC '%s'...", this->get_name().c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'", this->get_name().c_str());
|
||||||
|
|
||||||
if (this->channel1_ != ADC1_CHANNEL_MAX) {
|
if (this->channel1_ != ADC1_CHANNEL_MAX) {
|
||||||
adc1_config_width(ADC_WIDTH_MAX_SOC_BITS);
|
adc1_config_width(ADC_WIDTH_MAX_SOC_BITS);
|
||||||
@@ -55,30 +55,40 @@ void ADCSensor::setup() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ADCSensor::dump_config() {
|
void ADCSensor::dump_config() {
|
||||||
|
static const char *const ATTEN_AUTO_STR = "auto";
|
||||||
|
static const char *const ATTEN_0DB_STR = "0 db";
|
||||||
|
static const char *const ATTEN_2_5DB_STR = "2.5 db";
|
||||||
|
static const char *const ATTEN_6DB_STR = "6 db";
|
||||||
|
static const char *const ATTEN_12DB_STR = "12 db";
|
||||||
|
const char *atten_str = ATTEN_AUTO_STR;
|
||||||
|
|
||||||
LOG_SENSOR("", "ADC Sensor", this);
|
LOG_SENSOR("", "ADC Sensor", this);
|
||||||
LOG_PIN(" Pin: ", this->pin_);
|
LOG_PIN(" Pin: ", this->pin_);
|
||||||
if (this->autorange_) {
|
|
||||||
ESP_LOGCONFIG(TAG, " Attenuation: auto");
|
if (!this->autorange_) {
|
||||||
} else {
|
|
||||||
switch (this->attenuation_) {
|
switch (this->attenuation_) {
|
||||||
case ADC_ATTEN_DB_0:
|
case ADC_ATTEN_DB_0:
|
||||||
ESP_LOGCONFIG(TAG, " Attenuation: 0db");
|
atten_str = ATTEN_0DB_STR;
|
||||||
break;
|
break;
|
||||||
case ADC_ATTEN_DB_2_5:
|
case ADC_ATTEN_DB_2_5:
|
||||||
ESP_LOGCONFIG(TAG, " Attenuation: 2.5db");
|
atten_str = ATTEN_2_5DB_STR;
|
||||||
break;
|
break;
|
||||||
case ADC_ATTEN_DB_6:
|
case ADC_ATTEN_DB_6:
|
||||||
ESP_LOGCONFIG(TAG, " Attenuation: 6db");
|
atten_str = ATTEN_6DB_STR;
|
||||||
break;
|
break;
|
||||||
case ADC_ATTEN_DB_12_COMPAT:
|
case ADC_ATTEN_DB_12_COMPAT:
|
||||||
ESP_LOGCONFIG(TAG, " Attenuation: 12db");
|
atten_str = ATTEN_12DB_STR;
|
||||||
break;
|
break;
|
||||||
default: // This is to satisfy the unused ADC_ATTEN_MAX
|
default: // This is to satisfy the unused ADC_ATTEN_MAX
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ESP_LOGCONFIG(TAG, " Samples: %i", this->sample_count_);
|
|
||||||
ESP_LOGCONFIG(TAG, " Sampling mode: %s", LOG_STR_ARG(sampling_mode_to_str(this->sampling_mode_)));
|
ESP_LOGCONFIG(TAG,
|
||||||
|
" Attenuation: %s\n"
|
||||||
|
" Samples: %i\n"
|
||||||
|
" Sampling mode: %s",
|
||||||
|
atten_str, this->sample_count_, LOG_STR_ARG(sampling_mode_to_str(this->sampling_mode_)));
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ namespace adc {
|
|||||||
static const char *const TAG = "adc.esp8266";
|
static const char *const TAG = "adc.esp8266";
|
||||||
|
|
||||||
void ADCSensor::setup() {
|
void ADCSensor::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ADC '%s'...", this->get_name().c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'", this->get_name().c_str());
|
||||||
#ifndef USE_ADC_SENSOR_VCC
|
#ifndef USE_ADC_SENSOR_VCC
|
||||||
this->pin_->setup();
|
this->pin_->setup();
|
||||||
#endif
|
#endif
|
||||||
@@ -30,8 +30,10 @@ void ADCSensor::dump_config() {
|
|||||||
#else
|
#else
|
||||||
LOG_PIN(" Pin: ", this->pin_);
|
LOG_PIN(" Pin: ", this->pin_);
|
||||||
#endif // USE_ADC_SENSOR_VCC
|
#endif // USE_ADC_SENSOR_VCC
|
||||||
ESP_LOGCONFIG(TAG, " Samples: %i", this->sample_count_);
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Sampling mode: %s", LOG_STR_ARG(sampling_mode_to_str(this->sampling_mode_)));
|
" Samples: %i\n"
|
||||||
|
" Sampling mode: %s",
|
||||||
|
this->sample_count_, LOG_STR_ARG(sampling_mode_to_str(this->sampling_mode_)));
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ namespace adc {
|
|||||||
static const char *const TAG = "adc.libretiny";
|
static const char *const TAG = "adc.libretiny";
|
||||||
|
|
||||||
void ADCSensor::setup() {
|
void ADCSensor::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ADC '%s'...", this->get_name().c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'", this->get_name().c_str());
|
||||||
#ifndef USE_ADC_SENSOR_VCC
|
#ifndef USE_ADC_SENSOR_VCC
|
||||||
this->pin_->setup();
|
this->pin_->setup();
|
||||||
#endif // !USE_ADC_SENSOR_VCC
|
#endif // !USE_ADC_SENSOR_VCC
|
||||||
@@ -22,8 +22,10 @@ void ADCSensor::dump_config() {
|
|||||||
#else // USE_ADC_SENSOR_VCC
|
#else // USE_ADC_SENSOR_VCC
|
||||||
LOG_PIN(" Pin: ", this->pin_);
|
LOG_PIN(" Pin: ", this->pin_);
|
||||||
#endif // USE_ADC_SENSOR_VCC
|
#endif // USE_ADC_SENSOR_VCC
|
||||||
ESP_LOGCONFIG(TAG, " Samples: %i", this->sample_count_);
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Sampling mode: %s", LOG_STR_ARG(sampling_mode_to_str(this->sampling_mode_)));
|
" Samples: %i\n"
|
||||||
|
" Sampling mode: %s",
|
||||||
|
this->sample_count_, LOG_STR_ARG(sampling_mode_to_str(this->sampling_mode_)));
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ namespace adc {
|
|||||||
static const char *const TAG = "adc.rp2040";
|
static const char *const TAG = "adc.rp2040";
|
||||||
|
|
||||||
void ADCSensor::setup() {
|
void ADCSensor::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ADC '%s'...", this->get_name().c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'", this->get_name().c_str());
|
||||||
static bool initialized = false;
|
static bool initialized = false;
|
||||||
if (!initialized) {
|
if (!initialized) {
|
||||||
adc_init();
|
adc_init();
|
||||||
@@ -33,8 +33,10 @@ void ADCSensor::dump_config() {
|
|||||||
LOG_PIN(" Pin: ", this->pin_);
|
LOG_PIN(" Pin: ", this->pin_);
|
||||||
#endif // USE_ADC_SENSOR_VCC
|
#endif // USE_ADC_SENSOR_VCC
|
||||||
}
|
}
|
||||||
ESP_LOGCONFIG(TAG, " Samples: %i", this->sample_count_);
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Sampling mode: %s", LOG_STR_ARG(sampling_mode_to_str(this->sampling_mode_)));
|
" Samples: %i\n"
|
||||||
|
" Sampling mode: %s",
|
||||||
|
this->sample_count_, LOG_STR_ARG(sampling_mode_to_str(this->sampling_mode_)));
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ static const char *const TAG = "adc128s102";
|
|||||||
float ADC128S102::get_setup_priority() const { return setup_priority::HARDWARE; }
|
float ADC128S102::get_setup_priority() const { return setup_priority::HARDWARE; }
|
||||||
|
|
||||||
void ADC128S102::setup() {
|
void ADC128S102::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up adc128s102");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->spi_setup();
|
this->spi_setup();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -177,11 +177,14 @@ void ADE7880::dump_config() {
|
|||||||
LOG_SENSOR(" ", "Power Factor", this->channel_a_->power_factor);
|
LOG_SENSOR(" ", "Power Factor", this->channel_a_->power_factor);
|
||||||
LOG_SENSOR(" ", "Forward Active Energy", this->channel_a_->forward_active_energy);
|
LOG_SENSOR(" ", "Forward Active Energy", this->channel_a_->forward_active_energy);
|
||||||
LOG_SENSOR(" ", "Reverse Active Energy", this->channel_a_->reverse_active_energy);
|
LOG_SENSOR(" ", "Reverse Active Energy", this->channel_a_->reverse_active_energy);
|
||||||
ESP_LOGCONFIG(TAG, " Calibration:");
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Current: %" PRId32, this->channel_a_->current_gain_calibration);
|
" Calibration:\n"
|
||||||
ESP_LOGCONFIG(TAG, " Voltage: %" PRId32, this->channel_a_->voltage_gain_calibration);
|
" Current: %" PRId32 "\n"
|
||||||
ESP_LOGCONFIG(TAG, " Power: %" PRId32, this->channel_a_->power_gain_calibration);
|
" Voltage: %" PRId32 "\n"
|
||||||
ESP_LOGCONFIG(TAG, " Phase Angle: %u", this->channel_a_->phase_angle_calibration);
|
" Power: %" PRId32 "\n"
|
||||||
|
" Phase Angle: %u",
|
||||||
|
this->channel_a_->current_gain_calibration, this->channel_a_->voltage_gain_calibration,
|
||||||
|
this->channel_a_->power_gain_calibration, this->channel_a_->phase_angle_calibration);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->channel_b_ != nullptr) {
|
if (this->channel_b_ != nullptr) {
|
||||||
@@ -193,11 +196,14 @@ void ADE7880::dump_config() {
|
|||||||
LOG_SENSOR(" ", "Power Factor", this->channel_b_->power_factor);
|
LOG_SENSOR(" ", "Power Factor", this->channel_b_->power_factor);
|
||||||
LOG_SENSOR(" ", "Forward Active Energy", this->channel_b_->forward_active_energy);
|
LOG_SENSOR(" ", "Forward Active Energy", this->channel_b_->forward_active_energy);
|
||||||
LOG_SENSOR(" ", "Reverse Active Energy", this->channel_b_->reverse_active_energy);
|
LOG_SENSOR(" ", "Reverse Active Energy", this->channel_b_->reverse_active_energy);
|
||||||
ESP_LOGCONFIG(TAG, " Calibration:");
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Current: %" PRId32, this->channel_b_->current_gain_calibration);
|
" Calibration:\n"
|
||||||
ESP_LOGCONFIG(TAG, " Voltage: %" PRId32, this->channel_b_->voltage_gain_calibration);
|
" Current: %" PRId32 "\n"
|
||||||
ESP_LOGCONFIG(TAG, " Power: %" PRId32, this->channel_b_->power_gain_calibration);
|
" Voltage: %" PRId32 "\n"
|
||||||
ESP_LOGCONFIG(TAG, " Phase Angle: %u", this->channel_b_->phase_angle_calibration);
|
" Power: %" PRId32 "\n"
|
||||||
|
" Phase Angle: %u",
|
||||||
|
this->channel_b_->current_gain_calibration, this->channel_b_->voltage_gain_calibration,
|
||||||
|
this->channel_b_->power_gain_calibration, this->channel_b_->phase_angle_calibration);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->channel_c_ != nullptr) {
|
if (this->channel_c_ != nullptr) {
|
||||||
@@ -209,18 +215,23 @@ void ADE7880::dump_config() {
|
|||||||
LOG_SENSOR(" ", "Power Factor", this->channel_c_->power_factor);
|
LOG_SENSOR(" ", "Power Factor", this->channel_c_->power_factor);
|
||||||
LOG_SENSOR(" ", "Forward Active Energy", this->channel_c_->forward_active_energy);
|
LOG_SENSOR(" ", "Forward Active Energy", this->channel_c_->forward_active_energy);
|
||||||
LOG_SENSOR(" ", "Reverse Active Energy", this->channel_c_->reverse_active_energy);
|
LOG_SENSOR(" ", "Reverse Active Energy", this->channel_c_->reverse_active_energy);
|
||||||
ESP_LOGCONFIG(TAG, " Calibration:");
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Current: %" PRId32, this->channel_c_->current_gain_calibration);
|
" Calibration:\n"
|
||||||
ESP_LOGCONFIG(TAG, " Voltage: %" PRId32, this->channel_c_->voltage_gain_calibration);
|
" Current: %" PRId32 "\n"
|
||||||
ESP_LOGCONFIG(TAG, " Power: %" PRId32, this->channel_c_->power_gain_calibration);
|
" Voltage: %" PRId32 "\n"
|
||||||
ESP_LOGCONFIG(TAG, " Phase Angle: %u", this->channel_c_->phase_angle_calibration);
|
" Power: %" PRId32 "\n"
|
||||||
|
" Phase Angle: %u",
|
||||||
|
this->channel_c_->current_gain_calibration, this->channel_c_->voltage_gain_calibration,
|
||||||
|
this->channel_c_->power_gain_calibration, this->channel_c_->phase_angle_calibration);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->channel_n_ != nullptr) {
|
if (this->channel_n_ != nullptr) {
|
||||||
ESP_LOGCONFIG(TAG, " Neutral:");
|
ESP_LOGCONFIG(TAG, " Neutral:");
|
||||||
LOG_SENSOR(" ", "Current", this->channel_n_->current);
|
LOG_SENSOR(" ", "Current", this->channel_n_->current);
|
||||||
ESP_LOGCONFIG(TAG, " Calibration:");
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Current: %" PRId32, this->channel_n_->current_gain_calibration);
|
" Calibration:\n"
|
||||||
|
" Current: %" PRId32,
|
||||||
|
this->channel_n_->current_gain_calibration);
|
||||||
}
|
}
|
||||||
|
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
|
|||||||
@@ -85,8 +85,6 @@ class ADE7880 : public i2c::I2CDevice, public PollingComponent {
|
|||||||
|
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
ADE7880Store store_{};
|
ADE7880Store store_{};
|
||||||
InternalGPIOPin *irq0_pin_{nullptr};
|
InternalGPIOPin *irq0_pin_{nullptr};
|
||||||
|
|||||||
@@ -58,15 +58,18 @@ void ADE7953::dump_config() {
|
|||||||
LOG_SENSOR(" ", "Active Power B Sensor", this->active_power_b_sensor_);
|
LOG_SENSOR(" ", "Active Power B Sensor", this->active_power_b_sensor_);
|
||||||
LOG_SENSOR(" ", "Rective Power A Sensor", this->reactive_power_a_sensor_);
|
LOG_SENSOR(" ", "Rective Power A Sensor", this->reactive_power_a_sensor_);
|
||||||
LOG_SENSOR(" ", "Reactive Power B Sensor", this->reactive_power_b_sensor_);
|
LOG_SENSOR(" ", "Reactive Power B Sensor", this->reactive_power_b_sensor_);
|
||||||
ESP_LOGCONFIG(TAG, " USE_ACC_ENERGY_REGS: %d", this->use_acc_energy_regs_);
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " PGA_V_8: 0x%X", pga_v_);
|
" USE_ACC_ENERGY_REGS: %d\n"
|
||||||
ESP_LOGCONFIG(TAG, " PGA_IA_8: 0x%X", pga_ia_);
|
" PGA_V_8: 0x%X\n"
|
||||||
ESP_LOGCONFIG(TAG, " PGA_IB_8: 0x%X", pga_ib_);
|
" PGA_IA_8: 0x%X\n"
|
||||||
ESP_LOGCONFIG(TAG, " VGAIN_32: 0x%08jX", (uintmax_t) vgain_);
|
" PGA_IB_8: 0x%X\n"
|
||||||
ESP_LOGCONFIG(TAG, " AIGAIN_32: 0x%08jX", (uintmax_t) aigain_);
|
" VGAIN_32: 0x%08jX\n"
|
||||||
ESP_LOGCONFIG(TAG, " BIGAIN_32: 0x%08jX", (uintmax_t) bigain_);
|
" AIGAIN_32: 0x%08jX\n"
|
||||||
ESP_LOGCONFIG(TAG, " AWGAIN_32: 0x%08jX", (uintmax_t) awgain_);
|
" BIGAIN_32: 0x%08jX\n"
|
||||||
ESP_LOGCONFIG(TAG, " BWGAIN_32: 0x%08jX", (uintmax_t) bwgain_);
|
" AWGAIN_32: 0x%08jX\n"
|
||||||
|
" BWGAIN_32: 0x%08jX",
|
||||||
|
this->use_acc_energy_regs_, pga_v_, pga_ia_, pga_ib_, (uintmax_t) vgain_, (uintmax_t) aigain_,
|
||||||
|
(uintmax_t) bigain_, (uintmax_t) awgain_, (uintmax_t) bwgain_);
|
||||||
}
|
}
|
||||||
|
|
||||||
#define ADE_PUBLISH_(name, val, factor) \
|
#define ADE_PUBLISH_(name, val, factor) \
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
#include "ade7953_i2c.h"
|
#include "ade7953_i2c.h"
|
||||||
#include "esphome/core/log.h"
|
|
||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace ade7953_i2c {
|
namespace ade7953_i2c {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
#include "ade7953_spi.h"
|
#include "ade7953_spi.h"
|
||||||
#include "esphome/core/log.h"
|
|
||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace ade7953_spi {
|
namespace ade7953_spi {
|
||||||
|
|||||||
@@ -10,15 +10,13 @@ static const uint8_t ADS1115_REGISTER_CONVERSION = 0x00;
|
|||||||
static const uint8_t ADS1115_REGISTER_CONFIG = 0x01;
|
static const uint8_t ADS1115_REGISTER_CONFIG = 0x01;
|
||||||
|
|
||||||
void ADS1115Component::setup() {
|
void ADS1115Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ADS1115...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint16_t value;
|
uint16_t value;
|
||||||
if (!this->read_byte_16(ADS1115_REGISTER_CONVERSION, &value)) {
|
if (!this->read_byte_16(ADS1115_REGISTER_CONVERSION, &value)) {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGCONFIG(TAG, "Configuring ADS1115...");
|
|
||||||
|
|
||||||
uint16_t config = 0;
|
uint16_t config = 0;
|
||||||
// Clear single-shot bit
|
// Clear single-shot bit
|
||||||
// 0b0xxxxxxxxxxxxxxx
|
// 0b0xxxxxxxxxxxxxxx
|
||||||
@@ -68,10 +66,10 @@ void ADS1115Component::setup() {
|
|||||||
this->prev_config_ = config;
|
this->prev_config_ = config;
|
||||||
}
|
}
|
||||||
void ADS1115Component::dump_config() {
|
void ADS1115Component::dump_config() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ADS1115...");
|
ESP_LOGCONFIG(TAG, "ADS1115:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with ADS1115 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
float ADS1115Component::request_measurement(ADS1115Multiplexer multiplexer, ADS1115Gain gain,
|
float ADS1115Component::request_measurement(ADS1115Multiplexer multiplexer, ADS1115Gain gain,
|
||||||
|
|||||||
@@ -49,7 +49,6 @@ class ADS1115Component : public Component, public i2c::I2CDevice {
|
|||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
/// HARDWARE_LATE setup priority
|
/// HARDWARE_LATE setup priority
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
void set_continuous_mode(bool continuous_mode) { continuous_mode_ = continuous_mode; }
|
void set_continuous_mode(bool continuous_mode) { continuous_mode_ = continuous_mode; }
|
||||||
|
|
||||||
/// Helper method to request a measurement from a sensor.
|
/// Helper method to request a measurement from a sensor.
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
#include "ads1118.h"
|
#include "ads1118.h"
|
||||||
|
#include "esphome/core/helpers.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
@@ -8,7 +9,7 @@ static const char *const TAG = "ads1118";
|
|||||||
static const uint8_t ADS1118_DATA_RATE_860_SPS = 0b111;
|
static const uint8_t ADS1118_DATA_RATE_860_SPS = 0b111;
|
||||||
|
|
||||||
void ADS1118::setup() {
|
void ADS1118::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ads1118");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->spi_setup();
|
this->spi_setup();
|
||||||
|
|
||||||
this->config_ = 0;
|
this->config_ = 0;
|
||||||
|
|||||||
@@ -34,7 +34,6 @@ class ADS1118 : public Component,
|
|||||||
ADS1118() = default;
|
ADS1118() = default;
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
/// Helper method to request a measurement from a sensor.
|
/// Helper method to request a measurement from a sensor.
|
||||||
float request_measurement(ADS1118Multiplexer multiplexer, ADS1118Gain gain, bool temperature_mode);
|
float request_measurement(ADS1118Multiplexer multiplexer, ADS1118Gain gain, bool temperature_mode);
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
#include "ags10.h"
|
#include "ags10.h"
|
||||||
|
#include "esphome/core/helpers.h"
|
||||||
|
|
||||||
#include <cinttypes>
|
#include <cinttypes>
|
||||||
|
|
||||||
@@ -23,7 +24,7 @@ static const uint16_t ZP_CURRENT = 0x0000;
|
|||||||
static const uint16_t ZP_DEFAULT = 0xFFFF;
|
static const uint16_t ZP_DEFAULT = 0xFFFF;
|
||||||
|
|
||||||
void AGS10Component::setup() {
|
void AGS10Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ags10...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
auto version = this->read_version_();
|
auto version = this->read_version_();
|
||||||
if (version) {
|
if (version) {
|
||||||
@@ -65,7 +66,7 @@ void AGS10Component::dump_config() {
|
|||||||
case NONE:
|
case NONE:
|
||||||
break;
|
break;
|
||||||
case COMMUNICATION_FAILED:
|
case COMMUNICATION_FAILED:
|
||||||
ESP_LOGE(TAG, "Communication with AGS10 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
break;
|
break;
|
||||||
case CRC_CHECK_FAILED:
|
case CRC_CHECK_FAILED:
|
||||||
ESP_LOGE(TAG, "The crc check failed");
|
ESP_LOGE(TAG, "The crc check failed");
|
||||||
|
|||||||
@@ -31,8 +31,6 @@ class AGS10Component : public PollingComponent, public i2c::I2CDevice {
|
|||||||
|
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Modifies target address of AGS10.
|
* Modifies target address of AGS10.
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -13,8 +13,9 @@
|
|||||||
// results making successive requests; the current implementation makes 3 attempts with a delay of 30ms each time.
|
// results making successive requests; the current implementation makes 3 attempts with a delay of 30ms each time.
|
||||||
|
|
||||||
#include "aht10.h"
|
#include "aht10.h"
|
||||||
#include "esphome/core/log.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 aht10 {
|
namespace aht10 {
|
||||||
@@ -34,57 +35,59 @@ static const uint8_t AHT10_INIT_ATTEMPTS = 10;
|
|||||||
|
|
||||||
static const uint8_t AHT10_STATUS_BUSY = 0x80;
|
static const uint8_t AHT10_STATUS_BUSY = 0x80;
|
||||||
|
|
||||||
|
static const float AHT10_DIVISOR = 1048576.0f; // 2^20, used for temperature and humidity calculations
|
||||||
|
|
||||||
void AHT10Component::setup() {
|
void AHT10Component::setup() {
|
||||||
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
if (this->write(AHT10_SOFTRESET_CMD, sizeof(AHT10_SOFTRESET_CMD)) != i2c::ERROR_OK) {
|
if (this->write(AHT10_SOFTRESET_CMD, sizeof(AHT10_SOFTRESET_CMD)) != i2c::ERROR_OK) {
|
||||||
ESP_LOGE(TAG, "Reset AHT10 failed!");
|
ESP_LOGE(TAG, "Reset failed");
|
||||||
}
|
}
|
||||||
delay(AHT10_SOFTRESET_DELAY);
|
delay(AHT10_SOFTRESET_DELAY);
|
||||||
|
|
||||||
i2c::ErrorCode error_code = i2c::ERROR_INVALID_ARGUMENT;
|
i2c::ErrorCode error_code = i2c::ERROR_INVALID_ARGUMENT;
|
||||||
switch (this->variant_) {
|
switch (this->variant_) {
|
||||||
case AHT10Variant::AHT20:
|
case AHT10Variant::AHT20:
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AHT20");
|
|
||||||
error_code = this->write(AHT20_INITIALIZE_CMD, sizeof(AHT20_INITIALIZE_CMD));
|
error_code = this->write(AHT20_INITIALIZE_CMD, sizeof(AHT20_INITIALIZE_CMD));
|
||||||
break;
|
break;
|
||||||
case AHT10Variant::AHT10:
|
case AHT10Variant::AHT10:
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AHT10");
|
|
||||||
error_code = this->write(AHT10_INITIALIZE_CMD, sizeof(AHT10_INITIALIZE_CMD));
|
error_code = this->write(AHT10_INITIALIZE_CMD, sizeof(AHT10_INITIALIZE_CMD));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (error_code != i2c::ERROR_OK) {
|
if (error_code != i2c::ERROR_OK) {
|
||||||
ESP_LOGE(TAG, "Communication with AHT10 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
uint8_t cal_attempts = 0;
|
||||||
uint8_t data = AHT10_STATUS_BUSY;
|
uint8_t data = AHT10_STATUS_BUSY;
|
||||||
int cal_attempts = 0;
|
|
||||||
while (data & AHT10_STATUS_BUSY) {
|
while (data & AHT10_STATUS_BUSY) {
|
||||||
delay(AHT10_DEFAULT_DELAY);
|
delay(AHT10_DEFAULT_DELAY);
|
||||||
if (this->read(&data, 1) != i2c::ERROR_OK) {
|
if (this->read(&data, 1) != i2c::ERROR_OK) {
|
||||||
ESP_LOGE(TAG, "Communication with AHT10 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
++cal_attempts;
|
++cal_attempts;
|
||||||
if (cal_attempts > AHT10_INIT_ATTEMPTS) {
|
if (cal_attempts > AHT10_INIT_ATTEMPTS) {
|
||||||
ESP_LOGE(TAG, "AHT10 initialization timed out!");
|
ESP_LOGE(TAG, "Initialization timed out");
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ((data & 0x68) != 0x08) { // Bit[6:5] = 0b00, NORMAL mode and Bit[3] = 0b1, CALIBRATED
|
if ((data & 0x68) != 0x08) { // Bit[6:5] = 0b00, NORMAL mode and Bit[3] = 0b1, CALIBRATED
|
||||||
ESP_LOGE(TAG, "AHT10 initialization failed!");
|
ESP_LOGE(TAG, "Initialization failed");
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGV(TAG, "AHT10 initialization");
|
ESP_LOGV(TAG, "Initialization complete");
|
||||||
}
|
}
|
||||||
|
|
||||||
void AHT10Component::restart_read_() {
|
void AHT10Component::restart_read_() {
|
||||||
if (this->read_count_ == AHT10_ATTEMPTS) {
|
if (this->read_count_ == AHT10_ATTEMPTS) {
|
||||||
this->read_count_ = 0;
|
this->read_count_ = 0;
|
||||||
this->status_set_error("Measurements reading timed-out!");
|
this->status_set_error("Reading timed out");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this->read_count_++;
|
this->read_count_++;
|
||||||
@@ -97,24 +100,24 @@ void AHT10Component::read_data_() {
|
|||||||
ESP_LOGD(TAG, "Read attempt %d at %ums", this->read_count_, (unsigned) (millis() - this->start_time_));
|
ESP_LOGD(TAG, "Read attempt %d at %ums", this->read_count_, (unsigned) (millis() - this->start_time_));
|
||||||
}
|
}
|
||||||
if (this->read(data, 6) != i2c::ERROR_OK) {
|
if (this->read(data, 6) != i2c::ERROR_OK) {
|
||||||
this->status_set_warning("AHT10 read failed, retrying soon");
|
this->status_set_warning("Read failed, will retry");
|
||||||
this->restart_read_();
|
this->restart_read_();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((data[0] & 0x80) == 0x80) { // Bit[7] = 0b1, device is busy
|
if ((data[0] & 0x80) == 0x80) { // Bit[7] = 0b1, device is busy
|
||||||
ESP_LOGD(TAG, "AHT10 is busy, waiting...");
|
ESP_LOGD(TAG, "Device busy, will retry");
|
||||||
this->restart_read_();
|
this->restart_read_();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (data[1] == 0x0 && data[2] == 0x0 && (data[3] >> 4) == 0x0) {
|
if (data[1] == 0x0 && data[2] == 0x0 && (data[3] >> 4) == 0x0) {
|
||||||
// Unrealistic humidity (0x0)
|
// Invalid humidity (0x0)
|
||||||
if (this->humidity_sensor_ == nullptr) {
|
if (this->humidity_sensor_ == nullptr) {
|
||||||
ESP_LOGV(TAG, "ATH10 Unrealistic humidity (0x0), but humidity is not required");
|
ESP_LOGV(TAG, "Invalid humidity (reading not required)");
|
||||||
} else {
|
} else {
|
||||||
ESP_LOGD(TAG, "ATH10 Unrealistic humidity (0x0), retrying...");
|
ESP_LOGD(TAG, "Invalid humidity, retrying");
|
||||||
if (this->write(AHT10_MEASURE_CMD, sizeof(AHT10_MEASURE_CMD)) != i2c::ERROR_OK) {
|
if (this->write(AHT10_MEASURE_CMD, sizeof(AHT10_MEASURE_CMD)) != i2c::ERROR_OK) {
|
||||||
this->status_set_warning("Communication with AHT10 failed!");
|
this->status_set_warning(ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
this->restart_read_();
|
this->restart_read_();
|
||||||
return;
|
return;
|
||||||
@@ -123,22 +126,17 @@ void AHT10Component::read_data_() {
|
|||||||
if (this->read_count_ > 1) {
|
if (this->read_count_ > 1) {
|
||||||
ESP_LOGD(TAG, "Success at %ums", (unsigned) (millis() - this->start_time_));
|
ESP_LOGD(TAG, "Success at %ums", (unsigned) (millis() - this->start_time_));
|
||||||
}
|
}
|
||||||
uint32_t raw_temperature = ((data[3] & 0x0F) << 16) | (data[4] << 8) | data[5];
|
uint32_t raw_temperature = encode_uint24(data[3] & 0xF, data[4], data[5]);
|
||||||
uint32_t raw_humidity = ((data[1] << 16) | (data[2] << 8) | data[3]) >> 4;
|
uint32_t raw_humidity = encode_uint24(data[1], data[2], data[3]) >> 4;
|
||||||
|
|
||||||
if (this->temperature_sensor_ != nullptr) {
|
if (this->temperature_sensor_ != nullptr) {
|
||||||
float temperature = ((200.0f * (float) raw_temperature) / 1048576.0f) - 50.0f;
|
float temperature = ((200.0f * static_cast<float>(raw_temperature)) / AHT10_DIVISOR) - 50.0f;
|
||||||
this->temperature_sensor_->publish_state(temperature);
|
this->temperature_sensor_->publish_state(temperature);
|
||||||
}
|
}
|
||||||
if (this->humidity_sensor_ != nullptr) {
|
if (this->humidity_sensor_ != nullptr) {
|
||||||
float humidity;
|
float humidity = raw_humidity == 0 ? NAN : static_cast<float>(raw_humidity) * 100.0f / AHT10_DIVISOR;
|
||||||
if (raw_humidity == 0) { // unrealistic value
|
|
||||||
humidity = NAN;
|
|
||||||
} else {
|
|
||||||
humidity = (float) raw_humidity * 100.0f / 1048576.0f;
|
|
||||||
}
|
|
||||||
if (std::isnan(humidity)) {
|
if (std::isnan(humidity)) {
|
||||||
ESP_LOGW(TAG, "Invalid humidity! Sensor reported 0%% Hum");
|
ESP_LOGW(TAG, "Invalid humidity reading (0%%), ");
|
||||||
}
|
}
|
||||||
this->humidity_sensor_->publish_state(humidity);
|
this->humidity_sensor_->publish_state(humidity);
|
||||||
}
|
}
|
||||||
@@ -150,7 +148,7 @@ void AHT10Component::update() {
|
|||||||
return;
|
return;
|
||||||
this->start_time_ = millis();
|
this->start_time_ = millis();
|
||||||
if (this->write(AHT10_MEASURE_CMD, sizeof(AHT10_MEASURE_CMD)) != i2c::ERROR_OK) {
|
if (this->write(AHT10_MEASURE_CMD, sizeof(AHT10_MEASURE_CMD)) != i2c::ERROR_OK) {
|
||||||
this->status_set_warning("Communication with AHT10 failed!");
|
this->status_set_warning(ESP_LOG_MSG_COMM_FAIL);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this->restart_read_();
|
this->restart_read_();
|
||||||
@@ -162,7 +160,7 @@ void AHT10Component::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "AHT10:");
|
ESP_LOGCONFIG(TAG, "AHT10:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with AHT10 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ static const char *const TAG = "aic3204";
|
|||||||
}
|
}
|
||||||
|
|
||||||
void AIC3204::setup() {
|
void AIC3204::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AIC3204...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
// Set register page to 0
|
// Set register page to 0
|
||||||
ERROR_CHECK(this->write_byte(AIC3204_PAGE_CTRL, 0x00), "Set page 0 failed");
|
ERROR_CHECK(this->write_byte(AIC3204_PAGE_CTRL, 0x00), "Set page 0 failed");
|
||||||
@@ -113,7 +113,7 @@ void AIC3204::dump_config() {
|
|||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
|
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with AIC3204 failed");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -66,7 +66,6 @@ class AIC3204 : public audio_dac::AudioDac, public Component, public i2c::I2CDev
|
|||||||
public:
|
public:
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
|
|
||||||
bool set_mute_off() override;
|
bool set_mute_off() override;
|
||||||
bool set_mute_on() override;
|
bool set_mute_on() override;
|
||||||
|
|||||||
@@ -14,8 +14,8 @@ from esphome.const import (
|
|||||||
CONF_WEB_SERVER,
|
CONF_WEB_SERVER,
|
||||||
)
|
)
|
||||||
from esphome.core import CORE, coroutine_with_priority
|
from esphome.core import CORE, coroutine_with_priority
|
||||||
|
from esphome.core.entity_helpers import entity_duplicate_validator, setup_entity
|
||||||
from esphome.cpp_generator import MockObjClass
|
from esphome.cpp_generator import MockObjClass
|
||||||
from esphome.cpp_helpers import setup_entity
|
|
||||||
|
|
||||||
CODEOWNERS = ["@grahambrown11", "@hwstar"]
|
CODEOWNERS = ["@grahambrown11", "@hwstar"]
|
||||||
IS_PLATFORM_COMPONENT = True
|
IS_PLATFORM_COMPONENT = True
|
||||||
@@ -149,6 +149,9 @@ _ALARM_CONTROL_PANEL_SCHEMA = (
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
_ALARM_CONTROL_PANEL_SCHEMA.add_extra(entity_duplicate_validator("alarm_control_panel"))
|
||||||
|
|
||||||
|
|
||||||
def alarm_control_panel_schema(
|
def alarm_control_panel_schema(
|
||||||
class_: MockObjClass,
|
class_: MockObjClass,
|
||||||
*,
|
*,
|
||||||
@@ -190,7 +193,7 @@ ALARM_CONTROL_PANEL_CONDITION_SCHEMA = maybe_simple_id(
|
|||||||
|
|
||||||
|
|
||||||
async def setup_alarm_control_panel_core_(var, config):
|
async def setup_alarm_control_panel_core_(var, config):
|
||||||
await setup_entity(var, config)
|
await setup_entity(var, config, "alarm_control_panel")
|
||||||
for conf in config.get(CONF_ON_STATE, []):
|
for conf in config.get(CONF_ON_STATE, []):
|
||||||
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
||||||
await automation.build_automation(trigger, [], conf)
|
await automation.build_automation(trigger, [], conf)
|
||||||
@@ -235,6 +238,7 @@ async def register_alarm_control_panel(var, config):
|
|||||||
if not CORE.has_id(config[CONF_ID]):
|
if not CORE.has_id(config[CONF_ID]):
|
||||||
var = cg.Pvariable(config[CONF_ID], var)
|
var = cg.Pvariable(config[CONF_ID], var)
|
||||||
cg.add(cg.App.register_alarm_control_panel(var))
|
cg.add(cg.App.register_alarm_control_panel(var))
|
||||||
|
CORE.register_platform_component("alarm_control_panel", var)
|
||||||
await setup_alarm_control_panel_core_(var, config)
|
await setup_alarm_control_panel_core_(var, config)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -41,7 +41,6 @@ class Alpha3 : public esphome::ble_client::BLEClientNode, public PollingComponen
|
|||||||
void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
||||||
esp_ble_gattc_cb_param_t *param) override;
|
esp_ble_gattc_cb_param_t *param) override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
void set_flow_sensor(sensor::Sensor *sensor) { this->flow_sensor_ = sensor; }
|
void set_flow_sensor(sensor::Sensor *sensor) { this->flow_sensor_ = sensor; }
|
||||||
void set_head_sensor(sensor::Sensor *sensor) { this->head_sensor_ = sensor; }
|
void set_head_sensor(sensor::Sensor *sensor) { this->head_sensor_ = sensor; }
|
||||||
void set_power_sensor(sensor::Sensor *sensor) { this->power_sensor_ = sensor; }
|
void set_power_sensor(sensor::Sensor *sensor) { this->power_sensor_ = sensor; }
|
||||||
|
|||||||
@@ -90,7 +90,7 @@ bool AM2315C::convert_(uint8_t *data, float &humidity, float &temperature) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void AM2315C::setup() {
|
void AM2315C::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AM2315C...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
// get status
|
// get status
|
||||||
uint8_t status = 0;
|
uint8_t status = 0;
|
||||||
@@ -188,7 +188,7 @@ void AM2315C::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "AM2315C:");
|
ESP_LOGCONFIG(TAG, "AM2315C:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with AM2315C failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ void AM2320Component::update() {
|
|||||||
this->status_clear_warning();
|
this->status_clear_warning();
|
||||||
}
|
}
|
||||||
void AM2320Component::setup() {
|
void AM2320Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AM2320...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint8_t data[8];
|
uint8_t data[8];
|
||||||
data[0] = 0;
|
data[0] = 0;
|
||||||
data[1] = 4;
|
data[1] = 4;
|
||||||
@@ -47,7 +47,7 @@ void AM2320Component::dump_config() {
|
|||||||
ESP_LOGD(TAG, "AM2320:");
|
ESP_LOGD(TAG, "AM2320:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with AM2320 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "esphome/core/log.h"
|
|
||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace am43 {
|
namespace am43 {
|
||||||
|
|||||||
@@ -12,8 +12,10 @@ using namespace esphome::cover;
|
|||||||
|
|
||||||
void Am43Component::dump_config() {
|
void Am43Component::dump_config() {
|
||||||
LOG_COVER("", "AM43 Cover", this);
|
LOG_COVER("", "AM43 Cover", this);
|
||||||
ESP_LOGCONFIG(TAG, " Device Pin: %d", this->pin_);
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Invert Position: %d", (int) this->invert_position_);
|
" Device Pin: %d\n"
|
||||||
|
" Invert Position: %d",
|
||||||
|
this->pin_, (int) this->invert_position_);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Am43Component::setup() {
|
void Am43Component::setup() {
|
||||||
|
|||||||
@@ -22,7 +22,6 @@ class Am43Component : public cover::Cover, public esphome::ble_client::BLEClient
|
|||||||
void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
||||||
esp_ble_gattc_cb_param_t *param) override;
|
esp_ble_gattc_cb_param_t *param) override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
cover::CoverTraits get_traits() override;
|
cover::CoverTraits get_traits() override;
|
||||||
void set_pin(uint16_t pin) { this->pin_ = pin; }
|
void set_pin(uint16_t pin) { this->pin_ = pin; }
|
||||||
void set_invert_position(bool invert_position) { this->invert_position_ = invert_position; }
|
void set_invert_position(bool invert_position) { this->invert_position_ = invert_position; }
|
||||||
|
|||||||
@@ -22,7 +22,6 @@ class Am43 : public esphome::ble_client::BLEClientNode, public PollingComponent
|
|||||||
void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
||||||
esp_ble_gattc_cb_param_t *param) override;
|
esp_ble_gattc_cb_param_t *param) override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
void set_battery(sensor::Sensor *battery) { battery_ = battery; }
|
void set_battery(sensor::Sensor *battery) { battery_ = battery; }
|
||||||
void set_illuminance(sensor::Sensor *illuminance) { illuminance_ = illuminance; }
|
void set_illuminance(sensor::Sensor *illuminance) { illuminance_ = illuminance; }
|
||||||
|
|
||||||
|
|||||||
@@ -34,8 +34,10 @@ void AnalogThresholdBinarySensor::set_sensor(sensor::Sensor *analog_sensor) {
|
|||||||
void AnalogThresholdBinarySensor::dump_config() {
|
void AnalogThresholdBinarySensor::dump_config() {
|
||||||
LOG_BINARY_SENSOR("", "Analog Threshold Binary Sensor", this);
|
LOG_BINARY_SENSOR("", "Analog Threshold Binary Sensor", this);
|
||||||
LOG_SENSOR(" ", "Sensor", this->sensor_);
|
LOG_SENSOR(" ", "Sensor", this->sensor_);
|
||||||
ESP_LOGCONFIG(TAG, " Upper threshold: %.11f", this->upper_threshold_.value());
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Lower threshold: %.11f", this->lower_threshold_.value());
|
" Upper threshold: %.11f\n"
|
||||||
|
" Lower threshold: %.11f",
|
||||||
|
this->upper_threshold_.value(), this->lower_threshold_.value());
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace analog_threshold
|
} // namespace analog_threshold
|
||||||
|
|||||||
@@ -12,8 +12,6 @@ class AnalogThresholdBinarySensor : public Component, public binary_sensor::Bina
|
|||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
void setup() override;
|
void setup() override;
|
||||||
|
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
|
|
||||||
void set_sensor(sensor::Sensor *analog_sensor);
|
void set_sensor(sensor::Sensor *analog_sensor);
|
||||||
template<typename T> void set_upper_threshold(T upper_threshold) { this->upper_threshold_ = upper_threshold; }
|
template<typename T> void set_upper_threshold(T upper_threshold) { this->upper_threshold_ = upper_threshold; }
|
||||||
template<typename T> void set_lower_threshold(T lower_threshold) { this->lower_threshold_ = lower_threshold; }
|
template<typename T> void set_lower_threshold(T lower_threshold) { this->lower_threshold_ = lower_threshold; }
|
||||||
|
|||||||
@@ -17,7 +17,11 @@ void Anova::setup() {
|
|||||||
this->current_request_ = 0;
|
this->current_request_ = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Anova::loop() {}
|
void Anova::loop() {
|
||||||
|
// Parent BLEClientNode has a loop() method, but this component uses
|
||||||
|
// polling via update() and BLE callbacks so loop isn't needed
|
||||||
|
this->disable_loop();
|
||||||
|
}
|
||||||
|
|
||||||
void Anova::control(const ClimateCall &call) {
|
void Anova::control(const ClimateCall &call) {
|
||||||
if (call.get_mode().has_value()) {
|
if (call.get_mode().has_value()) {
|
||||||
|
|||||||
@@ -26,7 +26,6 @@ class Anova : public climate::Climate, public esphome::ble_client::BLEClientNode
|
|||||||
void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
||||||
esp_ble_gattc_cb_param_t *param) override;
|
esp_ble_gattc_cb_param_t *param) override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
climate::ClimateTraits traits() override {
|
climate::ClimateTraits traits() override {
|
||||||
auto traits = climate::ClimateTraits();
|
auto traits = climate::ClimateTraits();
|
||||||
traits.set_supports_current_temperature(true);
|
traits.set_supports_current_temperature(true);
|
||||||
|
|||||||
@@ -54,7 +54,7 @@ enum { // APDS9306 registers
|
|||||||
}
|
}
|
||||||
|
|
||||||
void APDS9306::setup() {
|
void APDS9306::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up APDS9306...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
uint8_t id;
|
uint8_t id;
|
||||||
if (!this->read_byte(APDS9306_PART_ID, &id)) { // Part ID register
|
if (!this->read_byte(APDS9306_PART_ID, &id)) { // Part ID register
|
||||||
@@ -97,7 +97,7 @@ void APDS9306::dump_config() {
|
|||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
switch (this->error_code_) {
|
switch (this->error_code_) {
|
||||||
case COMMUNICATION_FAILED:
|
case COMMUNICATION_FAILED:
|
||||||
ESP_LOGE(TAG, "Communication with APDS9306 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
break;
|
break;
|
||||||
case WRONG_ID:
|
case WRONG_ID:
|
||||||
ESP_LOGE(TAG, "APDS9306 has invalid id!");
|
ESP_LOGE(TAG, "APDS9306 has invalid id!");
|
||||||
@@ -108,9 +108,12 @@ void APDS9306::dump_config() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGCONFIG(TAG, " Gain: %u", AMBIENT_LIGHT_GAIN_VALUES[this->gain_]);
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Measurement rate: %u", MEASUREMENT_RATE_VALUES[this->measurement_rate_]);
|
" Gain: %u\n"
|
||||||
ESP_LOGCONFIG(TAG, " Measurement Resolution/Bit width: %d", MEASUREMENT_BIT_WIDTH_VALUES[this->bit_width_]);
|
" Measurement rate: %u\n"
|
||||||
|
" Measurement Resolution/Bit width: %d",
|
||||||
|
AMBIENT_LIGHT_GAIN_VALUES[this->gain_], MEASUREMENT_RATE_VALUES[this->measurement_rate_],
|
||||||
|
MEASUREMENT_BIT_WIDTH_VALUES[this->bit_width_]);
|
||||||
|
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ static const char *const TAG = "apds9960";
|
|||||||
#define APDS9960_WRITE_BYTE(reg, value) APDS9960_ERROR_CHECK(this->write_byte(reg, value));
|
#define APDS9960_WRITE_BYTE(reg, value) APDS9960_ERROR_CHECK(this->write_byte(reg, value));
|
||||||
|
|
||||||
void APDS9960::setup() {
|
void APDS9960::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up APDS9960...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint8_t id;
|
uint8_t id;
|
||||||
if (!this->read_byte(0x92, &id)) { // ID register
|
if (!this->read_byte(0x92, &id)) { // ID register
|
||||||
this->error_code_ = COMMUNICATION_FAILED;
|
this->error_code_ = COMMUNICATION_FAILED;
|
||||||
@@ -23,7 +23,7 @@ void APDS9960::setup() {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (id != 0xAB && id != 0x9C && id != 0xA8) { // APDS9960 all should have one of these IDs
|
if (id != 0xAB && id != 0x9C && id != 0xA8 && id != 0x9E) { // APDS9960 all should have one of these IDs
|
||||||
this->error_code_ = WRONG_ID;
|
this->error_code_ = WRONG_ID;
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
@@ -141,7 +141,7 @@ void APDS9960::dump_config() {
|
|||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
switch (this->error_code_) {
|
switch (this->error_code_) {
|
||||||
case COMMUNICATION_FAILED:
|
case COMMUNICATION_FAILED:
|
||||||
ESP_LOGE(TAG, "Communication with APDS9960 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
break;
|
break;
|
||||||
case WRONG_ID:
|
case WRONG_ID:
|
||||||
ESP_LOGE(TAG, "APDS9960 has invalid id!");
|
ESP_LOGE(TAG, "APDS9960 has invalid id!");
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ import base64
|
|||||||
from esphome import automation
|
from esphome import automation
|
||||||
from esphome.automation import Condition
|
from esphome.automation import Condition
|
||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
|
from esphome.config_helpers import get_logger_level
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
CONF_ACTION,
|
CONF_ACTION,
|
||||||
@@ -49,6 +50,7 @@ SERVICE_ARG_NATIVE_TYPES = {
|
|||||||
"string[]": cg.std_vector.template(cg.std_string),
|
"string[]": cg.std_vector.template(cg.std_string),
|
||||||
}
|
}
|
||||||
CONF_ENCRYPTION = "encryption"
|
CONF_ENCRYPTION = "encryption"
|
||||||
|
CONF_BATCH_DELAY = "batch_delay"
|
||||||
|
|
||||||
|
|
||||||
def validate_encryption_key(value):
|
def validate_encryption_key(value):
|
||||||
@@ -109,6 +111,10 @@ CONFIG_SCHEMA = cv.All(
|
|||||||
): ACTIONS_SCHEMA,
|
): ACTIONS_SCHEMA,
|
||||||
cv.Exclusive(CONF_ACTIONS, group_of_exclusion=CONF_ACTIONS): ACTIONS_SCHEMA,
|
cv.Exclusive(CONF_ACTIONS, group_of_exclusion=CONF_ACTIONS): ACTIONS_SCHEMA,
|
||||||
cv.Optional(CONF_ENCRYPTION): _encryption_schema,
|
cv.Optional(CONF_ENCRYPTION): _encryption_schema,
|
||||||
|
cv.Optional(CONF_BATCH_DELAY, default="100ms"): cv.All(
|
||||||
|
cv.positive_time_period_milliseconds,
|
||||||
|
cv.Range(max=cv.TimePeriod(milliseconds=65535)),
|
||||||
|
),
|
||||||
cv.Optional(CONF_ON_CLIENT_CONNECTED): automation.validate_automation(
|
cv.Optional(CONF_ON_CLIENT_CONNECTED): automation.validate_automation(
|
||||||
single=True
|
single=True
|
||||||
),
|
),
|
||||||
@@ -127,26 +133,32 @@ async def to_code(config):
|
|||||||
await cg.register_component(var, config)
|
await cg.register_component(var, config)
|
||||||
|
|
||||||
cg.add(var.set_port(config[CONF_PORT]))
|
cg.add(var.set_port(config[CONF_PORT]))
|
||||||
cg.add(var.set_password(config[CONF_PASSWORD]))
|
if config[CONF_PASSWORD]:
|
||||||
|
cg.add_define("USE_API_PASSWORD")
|
||||||
|
cg.add(var.set_password(config[CONF_PASSWORD]))
|
||||||
cg.add(var.set_reboot_timeout(config[CONF_REBOOT_TIMEOUT]))
|
cg.add(var.set_reboot_timeout(config[CONF_REBOOT_TIMEOUT]))
|
||||||
|
cg.add(var.set_batch_delay(config[CONF_BATCH_DELAY]))
|
||||||
|
|
||||||
for conf in config.get(CONF_ACTIONS, []):
|
if actions := config.get(CONF_ACTIONS, []):
|
||||||
template_args = []
|
cg.add_define("USE_API_YAML_SERVICES")
|
||||||
func_args = []
|
for conf in actions:
|
||||||
service_arg_names = []
|
template_args = []
|
||||||
for name, var_ in conf[CONF_VARIABLES].items():
|
func_args = []
|
||||||
native = SERVICE_ARG_NATIVE_TYPES[var_]
|
service_arg_names = []
|
||||||
template_args.append(native)
|
for name, var_ in conf[CONF_VARIABLES].items():
|
||||||
func_args.append((native, name))
|
native = SERVICE_ARG_NATIVE_TYPES[var_]
|
||||||
service_arg_names.append(name)
|
template_args.append(native)
|
||||||
templ = cg.TemplateArguments(*template_args)
|
func_args.append((native, name))
|
||||||
trigger = cg.new_Pvariable(
|
service_arg_names.append(name)
|
||||||
conf[CONF_TRIGGER_ID], templ, conf[CONF_ACTION], service_arg_names
|
templ = cg.TemplateArguments(*template_args)
|
||||||
)
|
trigger = cg.new_Pvariable(
|
||||||
cg.add(var.register_user_service(trigger))
|
conf[CONF_TRIGGER_ID], templ, conf[CONF_ACTION], service_arg_names
|
||||||
await automation.build_automation(trigger, func_args, conf)
|
)
|
||||||
|
cg.add(var.register_user_service(trigger))
|
||||||
|
await automation.build_automation(trigger, func_args, conf)
|
||||||
|
|
||||||
if CONF_ON_CLIENT_CONNECTED in config:
|
if CONF_ON_CLIENT_CONNECTED in config:
|
||||||
|
cg.add_define("USE_API_CLIENT_CONNECTED_TRIGGER")
|
||||||
await automation.build_automation(
|
await automation.build_automation(
|
||||||
var.get_client_connected_trigger(),
|
var.get_client_connected_trigger(),
|
||||||
[(cg.std_string, "client_info"), (cg.std_string, "client_address")],
|
[(cg.std_string, "client_info"), (cg.std_string, "client_address")],
|
||||||
@@ -154,6 +166,7 @@ async def to_code(config):
|
|||||||
)
|
)
|
||||||
|
|
||||||
if CONF_ON_CLIENT_DISCONNECTED in config:
|
if CONF_ON_CLIENT_DISCONNECTED in config:
|
||||||
|
cg.add_define("USE_API_CLIENT_DISCONNECTED_TRIGGER")
|
||||||
await automation.build_automation(
|
await automation.build_automation(
|
||||||
var.get_client_disconnected_trigger(),
|
var.get_client_disconnected_trigger(),
|
||||||
[(cg.std_string, "client_info"), (cg.std_string, "client_address")],
|
[(cg.std_string, "client_info"), (cg.std_string, "client_address")],
|
||||||
@@ -172,7 +185,7 @@ async def to_code(config):
|
|||||||
# and plaintext disabled. Only a factory reset can remove it.
|
# and plaintext disabled. Only a factory reset can remove it.
|
||||||
cg.add_define("USE_API_PLAINTEXT")
|
cg.add_define("USE_API_PLAINTEXT")
|
||||||
cg.add_define("USE_API_NOISE")
|
cg.add_define("USE_API_NOISE")
|
||||||
cg.add_library("esphome/noise-c", "0.1.6")
|
cg.add_library("esphome/noise-c", "0.1.10")
|
||||||
else:
|
else:
|
||||||
cg.add_define("USE_API_PLAINTEXT")
|
cg.add_define("USE_API_PLAINTEXT")
|
||||||
|
|
||||||
@@ -301,3 +314,17 @@ async def homeassistant_tag_scanned_to_code(config, action_id, template_arg, arg
|
|||||||
@automation.register_condition("api.connected", APIConnectedCondition, {})
|
@automation.register_condition("api.connected", APIConnectedCondition, {})
|
||||||
async def api_connected_to_code(config, condition_id, template_arg, args):
|
async def api_connected_to_code(config, condition_id, template_arg, args):
|
||||||
return cg.new_Pvariable(condition_id, template_arg)
|
return cg.new_Pvariable(condition_id, template_arg)
|
||||||
|
|
||||||
|
|
||||||
|
def FILTER_SOURCE_FILES() -> list[str]:
|
||||||
|
"""Filter out api_pb2_dump.cpp when proto message dumping is not enabled."""
|
||||||
|
# api_pb2_dump.cpp is only needed when HAS_PROTO_MESSAGE_DUMP is defined
|
||||||
|
# This is a particularly large file that still needs to be opened and read
|
||||||
|
# all the way to the end even when ifdef'd out
|
||||||
|
#
|
||||||
|
# HAS_PROTO_MESSAGE_DUMP is defined when ESPHOME_LOG_HAS_VERY_VERBOSE is set,
|
||||||
|
# which happens when the logger level is VERY_VERBOSE
|
||||||
|
if get_logger_level() != "VERY_VERBOSE":
|
||||||
|
return ["api_pb2_dump.cpp"]
|
||||||
|
|
||||||
|
return []
|
||||||
|
|||||||
@@ -188,6 +188,17 @@ message DeviceInfoRequest {
|
|||||||
// Empty
|
// Empty
|
||||||
}
|
}
|
||||||
|
|
||||||
|
message AreaInfo {
|
||||||
|
uint32 area_id = 1;
|
||||||
|
string name = 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
message DeviceInfo {
|
||||||
|
uint32 device_id = 1;
|
||||||
|
string name = 2;
|
||||||
|
uint32 area_id = 3;
|
||||||
|
}
|
||||||
|
|
||||||
message DeviceInfoResponse {
|
message DeviceInfoResponse {
|
||||||
option (id) = 10;
|
option (id) = 10;
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
@@ -236,6 +247,12 @@ message DeviceInfoResponse {
|
|||||||
|
|
||||||
// Supports receiving and saving api encryption key
|
// Supports receiving and saving api encryption key
|
||||||
bool api_encryption_supported = 19;
|
bool api_encryption_supported = 19;
|
||||||
|
|
||||||
|
repeated DeviceInfo devices = 20;
|
||||||
|
repeated AreaInfo areas = 21;
|
||||||
|
|
||||||
|
// Top-level area info to phase out suggested_area
|
||||||
|
AreaInfo area = 22;
|
||||||
}
|
}
|
||||||
|
|
||||||
message ListEntitiesRequest {
|
message ListEntitiesRequest {
|
||||||
@@ -266,6 +283,7 @@ enum EntityCategory {
|
|||||||
// ==================== BINARY SENSOR ====================
|
// ==================== BINARY SENSOR ====================
|
||||||
message ListEntitiesBinarySensorResponse {
|
message ListEntitiesBinarySensorResponse {
|
||||||
option (id) = 12;
|
option (id) = 12;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_BINARY_SENSOR";
|
option (ifdef) = "USE_BINARY_SENSOR";
|
||||||
|
|
||||||
@@ -279,9 +297,11 @@ message ListEntitiesBinarySensorResponse {
|
|||||||
bool disabled_by_default = 7;
|
bool disabled_by_default = 7;
|
||||||
string icon = 8;
|
string icon = 8;
|
||||||
EntityCategory entity_category = 9;
|
EntityCategory entity_category = 9;
|
||||||
|
uint32 device_id = 10;
|
||||||
}
|
}
|
||||||
message BinarySensorStateResponse {
|
message BinarySensorStateResponse {
|
||||||
option (id) = 21;
|
option (id) = 21;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_BINARY_SENSOR";
|
option (ifdef) = "USE_BINARY_SENSOR";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -291,11 +311,13 @@ message BinarySensorStateResponse {
|
|||||||
// If the binary sensor does not have a valid state yet.
|
// If the binary sensor does not have a valid state yet.
|
||||||
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
||||||
bool missing_state = 3;
|
bool missing_state = 3;
|
||||||
|
uint32 device_id = 4;
|
||||||
}
|
}
|
||||||
|
|
||||||
// ==================== COVER ====================
|
// ==================== COVER ====================
|
||||||
message ListEntitiesCoverResponse {
|
message ListEntitiesCoverResponse {
|
||||||
option (id) = 13;
|
option (id) = 13;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_COVER";
|
option (ifdef) = "USE_COVER";
|
||||||
|
|
||||||
@@ -312,6 +334,7 @@ message ListEntitiesCoverResponse {
|
|||||||
string icon = 10;
|
string icon = 10;
|
||||||
EntityCategory entity_category = 11;
|
EntityCategory entity_category = 11;
|
||||||
bool supports_stop = 12;
|
bool supports_stop = 12;
|
||||||
|
uint32 device_id = 13;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum LegacyCoverState {
|
enum LegacyCoverState {
|
||||||
@@ -325,6 +348,7 @@ enum CoverOperation {
|
|||||||
}
|
}
|
||||||
message CoverStateResponse {
|
message CoverStateResponse {
|
||||||
option (id) = 22;
|
option (id) = 22;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_COVER";
|
option (ifdef) = "USE_COVER";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -337,6 +361,7 @@ message CoverStateResponse {
|
|||||||
float position = 3;
|
float position = 3;
|
||||||
float tilt = 4;
|
float tilt = 4;
|
||||||
CoverOperation current_operation = 5;
|
CoverOperation current_operation = 5;
|
||||||
|
uint32 device_id = 6;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum LegacyCoverCommand {
|
enum LegacyCoverCommand {
|
||||||
@@ -367,6 +392,7 @@ message CoverCommandRequest {
|
|||||||
// ==================== FAN ====================
|
// ==================== FAN ====================
|
||||||
message ListEntitiesFanResponse {
|
message ListEntitiesFanResponse {
|
||||||
option (id) = 14;
|
option (id) = 14;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_FAN";
|
option (ifdef) = "USE_FAN";
|
||||||
|
|
||||||
@@ -383,6 +409,7 @@ message ListEntitiesFanResponse {
|
|||||||
string icon = 10;
|
string icon = 10;
|
||||||
EntityCategory entity_category = 11;
|
EntityCategory entity_category = 11;
|
||||||
repeated string supported_preset_modes = 12;
|
repeated string supported_preset_modes = 12;
|
||||||
|
uint32 device_id = 13;
|
||||||
}
|
}
|
||||||
enum FanSpeed {
|
enum FanSpeed {
|
||||||
FAN_SPEED_LOW = 0;
|
FAN_SPEED_LOW = 0;
|
||||||
@@ -395,6 +422,7 @@ enum FanDirection {
|
|||||||
}
|
}
|
||||||
message FanStateResponse {
|
message FanStateResponse {
|
||||||
option (id) = 23;
|
option (id) = 23;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_FAN";
|
option (ifdef) = "USE_FAN";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -406,6 +434,7 @@ message FanStateResponse {
|
|||||||
FanDirection direction = 5;
|
FanDirection direction = 5;
|
||||||
int32 speed_level = 6;
|
int32 speed_level = 6;
|
||||||
string preset_mode = 7;
|
string preset_mode = 7;
|
||||||
|
uint32 device_id = 8;
|
||||||
}
|
}
|
||||||
message FanCommandRequest {
|
message FanCommandRequest {
|
||||||
option (id) = 31;
|
option (id) = 31;
|
||||||
@@ -444,6 +473,7 @@ enum ColorMode {
|
|||||||
}
|
}
|
||||||
message ListEntitiesLightResponse {
|
message ListEntitiesLightResponse {
|
||||||
option (id) = 15;
|
option (id) = 15;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_LIGHT";
|
option (ifdef) = "USE_LIGHT";
|
||||||
|
|
||||||
@@ -464,9 +494,11 @@ message ListEntitiesLightResponse {
|
|||||||
bool disabled_by_default = 13;
|
bool disabled_by_default = 13;
|
||||||
string icon = 14;
|
string icon = 14;
|
||||||
EntityCategory entity_category = 15;
|
EntityCategory entity_category = 15;
|
||||||
|
uint32 device_id = 16;
|
||||||
}
|
}
|
||||||
message LightStateResponse {
|
message LightStateResponse {
|
||||||
option (id) = 24;
|
option (id) = 24;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_LIGHT";
|
option (ifdef) = "USE_LIGHT";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -484,6 +516,7 @@ message LightStateResponse {
|
|||||||
float cold_white = 12;
|
float cold_white = 12;
|
||||||
float warm_white = 13;
|
float warm_white = 13;
|
||||||
string effect = 9;
|
string effect = 9;
|
||||||
|
uint32 device_id = 14;
|
||||||
}
|
}
|
||||||
message LightCommandRequest {
|
message LightCommandRequest {
|
||||||
option (id) = 32;
|
option (id) = 32;
|
||||||
@@ -536,6 +569,7 @@ enum SensorLastResetType {
|
|||||||
|
|
||||||
message ListEntitiesSensorResponse {
|
message ListEntitiesSensorResponse {
|
||||||
option (id) = 16;
|
option (id) = 16;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_SENSOR";
|
option (ifdef) = "USE_SENSOR";
|
||||||
|
|
||||||
@@ -554,9 +588,11 @@ message ListEntitiesSensorResponse {
|
|||||||
SensorLastResetType legacy_last_reset_type = 11;
|
SensorLastResetType legacy_last_reset_type = 11;
|
||||||
bool disabled_by_default = 12;
|
bool disabled_by_default = 12;
|
||||||
EntityCategory entity_category = 13;
|
EntityCategory entity_category = 13;
|
||||||
|
uint32 device_id = 14;
|
||||||
}
|
}
|
||||||
message SensorStateResponse {
|
message SensorStateResponse {
|
||||||
option (id) = 25;
|
option (id) = 25;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_SENSOR";
|
option (ifdef) = "USE_SENSOR";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -566,11 +602,13 @@ message SensorStateResponse {
|
|||||||
// If the sensor does not have a valid state yet.
|
// If the sensor does not have a valid state yet.
|
||||||
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
||||||
bool missing_state = 3;
|
bool missing_state = 3;
|
||||||
|
uint32 device_id = 4;
|
||||||
}
|
}
|
||||||
|
|
||||||
// ==================== SWITCH ====================
|
// ==================== SWITCH ====================
|
||||||
message ListEntitiesSwitchResponse {
|
message ListEntitiesSwitchResponse {
|
||||||
option (id) = 17;
|
option (id) = 17;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_SWITCH";
|
option (ifdef) = "USE_SWITCH";
|
||||||
|
|
||||||
@@ -584,15 +622,18 @@ message ListEntitiesSwitchResponse {
|
|||||||
bool disabled_by_default = 7;
|
bool disabled_by_default = 7;
|
||||||
EntityCategory entity_category = 8;
|
EntityCategory entity_category = 8;
|
||||||
string device_class = 9;
|
string device_class = 9;
|
||||||
|
uint32 device_id = 10;
|
||||||
}
|
}
|
||||||
message SwitchStateResponse {
|
message SwitchStateResponse {
|
||||||
option (id) = 26;
|
option (id) = 26;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_SWITCH";
|
option (ifdef) = "USE_SWITCH";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
|
|
||||||
fixed32 key = 1;
|
fixed32 key = 1;
|
||||||
bool state = 2;
|
bool state = 2;
|
||||||
|
uint32 device_id = 3;
|
||||||
}
|
}
|
||||||
message SwitchCommandRequest {
|
message SwitchCommandRequest {
|
||||||
option (id) = 33;
|
option (id) = 33;
|
||||||
@@ -607,6 +648,7 @@ message SwitchCommandRequest {
|
|||||||
// ==================== TEXT SENSOR ====================
|
// ==================== TEXT SENSOR ====================
|
||||||
message ListEntitiesTextSensorResponse {
|
message ListEntitiesTextSensorResponse {
|
||||||
option (id) = 18;
|
option (id) = 18;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_TEXT_SENSOR";
|
option (ifdef) = "USE_TEXT_SENSOR";
|
||||||
|
|
||||||
@@ -619,9 +661,11 @@ message ListEntitiesTextSensorResponse {
|
|||||||
bool disabled_by_default = 6;
|
bool disabled_by_default = 6;
|
||||||
EntityCategory entity_category = 7;
|
EntityCategory entity_category = 7;
|
||||||
string device_class = 8;
|
string device_class = 8;
|
||||||
|
uint32 device_id = 9;
|
||||||
}
|
}
|
||||||
message TextSensorStateResponse {
|
message TextSensorStateResponse {
|
||||||
option (id) = 27;
|
option (id) = 27;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_TEXT_SENSOR";
|
option (ifdef) = "USE_TEXT_SENSOR";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -631,6 +675,7 @@ message TextSensorStateResponse {
|
|||||||
// If the text sensor does not have a valid state yet.
|
// If the text sensor does not have a valid state yet.
|
||||||
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
||||||
bool missing_state = 3;
|
bool missing_state = 3;
|
||||||
|
uint32 device_id = 4;
|
||||||
}
|
}
|
||||||
|
|
||||||
// ==================== SUBSCRIBE LOGS ====================
|
// ==================== SUBSCRIBE LOGS ====================
|
||||||
@@ -789,8 +834,9 @@ message ExecuteServiceRequest {
|
|||||||
// ==================== CAMERA ====================
|
// ==================== CAMERA ====================
|
||||||
message ListEntitiesCameraResponse {
|
message ListEntitiesCameraResponse {
|
||||||
option (id) = 43;
|
option (id) = 43;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_ESP32_CAMERA";
|
option (ifdef) = "USE_CAMERA";
|
||||||
|
|
||||||
string object_id = 1;
|
string object_id = 1;
|
||||||
fixed32 key = 2;
|
fixed32 key = 2;
|
||||||
@@ -799,12 +845,13 @@ message ListEntitiesCameraResponse {
|
|||||||
bool disabled_by_default = 5;
|
bool disabled_by_default = 5;
|
||||||
string icon = 6;
|
string icon = 6;
|
||||||
EntityCategory entity_category = 7;
|
EntityCategory entity_category = 7;
|
||||||
|
uint32 device_id = 8;
|
||||||
}
|
}
|
||||||
|
|
||||||
message CameraImageResponse {
|
message CameraImageResponse {
|
||||||
option (id) = 44;
|
option (id) = 44;
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_ESP32_CAMERA";
|
option (ifdef) = "USE_CAMERA";
|
||||||
|
|
||||||
fixed32 key = 1;
|
fixed32 key = 1;
|
||||||
bytes data = 2;
|
bytes data = 2;
|
||||||
@@ -813,7 +860,7 @@ message CameraImageResponse {
|
|||||||
message CameraImageRequest {
|
message CameraImageRequest {
|
||||||
option (id) = 45;
|
option (id) = 45;
|
||||||
option (source) = SOURCE_CLIENT;
|
option (source) = SOURCE_CLIENT;
|
||||||
option (ifdef) = "USE_ESP32_CAMERA";
|
option (ifdef) = "USE_CAMERA";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
|
|
||||||
bool single = 1;
|
bool single = 1;
|
||||||
@@ -869,6 +916,7 @@ enum ClimatePreset {
|
|||||||
}
|
}
|
||||||
message ListEntitiesClimateResponse {
|
message ListEntitiesClimateResponse {
|
||||||
option (id) = 46;
|
option (id) = 46;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_CLIMATE";
|
option (ifdef) = "USE_CLIMATE";
|
||||||
|
|
||||||
@@ -900,9 +948,11 @@ message ListEntitiesClimateResponse {
|
|||||||
bool supports_target_humidity = 23;
|
bool supports_target_humidity = 23;
|
||||||
float visual_min_humidity = 24;
|
float visual_min_humidity = 24;
|
||||||
float visual_max_humidity = 25;
|
float visual_max_humidity = 25;
|
||||||
|
uint32 device_id = 26;
|
||||||
}
|
}
|
||||||
message ClimateStateResponse {
|
message ClimateStateResponse {
|
||||||
option (id) = 47;
|
option (id) = 47;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_CLIMATE";
|
option (ifdef) = "USE_CLIMATE";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -923,6 +973,7 @@ message ClimateStateResponse {
|
|||||||
string custom_preset = 13;
|
string custom_preset = 13;
|
||||||
float current_humidity = 14;
|
float current_humidity = 14;
|
||||||
float target_humidity = 15;
|
float target_humidity = 15;
|
||||||
|
uint32 device_id = 16;
|
||||||
}
|
}
|
||||||
message ClimateCommandRequest {
|
message ClimateCommandRequest {
|
||||||
option (id) = 48;
|
option (id) = 48;
|
||||||
@@ -964,6 +1015,7 @@ enum NumberMode {
|
|||||||
}
|
}
|
||||||
message ListEntitiesNumberResponse {
|
message ListEntitiesNumberResponse {
|
||||||
option (id) = 49;
|
option (id) = 49;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_NUMBER";
|
option (ifdef) = "USE_NUMBER";
|
||||||
|
|
||||||
@@ -981,9 +1033,11 @@ message ListEntitiesNumberResponse {
|
|||||||
string unit_of_measurement = 11;
|
string unit_of_measurement = 11;
|
||||||
NumberMode mode = 12;
|
NumberMode mode = 12;
|
||||||
string device_class = 13;
|
string device_class = 13;
|
||||||
|
uint32 device_id = 14;
|
||||||
}
|
}
|
||||||
message NumberStateResponse {
|
message NumberStateResponse {
|
||||||
option (id) = 50;
|
option (id) = 50;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_NUMBER";
|
option (ifdef) = "USE_NUMBER";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -993,6 +1047,7 @@ message NumberStateResponse {
|
|||||||
// If the number does not have a valid state yet.
|
// If the number does not have a valid state yet.
|
||||||
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
||||||
bool missing_state = 3;
|
bool missing_state = 3;
|
||||||
|
uint32 device_id = 4;
|
||||||
}
|
}
|
||||||
message NumberCommandRequest {
|
message NumberCommandRequest {
|
||||||
option (id) = 51;
|
option (id) = 51;
|
||||||
@@ -1007,6 +1062,7 @@ message NumberCommandRequest {
|
|||||||
// ==================== SELECT ====================
|
// ==================== SELECT ====================
|
||||||
message ListEntitiesSelectResponse {
|
message ListEntitiesSelectResponse {
|
||||||
option (id) = 52;
|
option (id) = 52;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_SELECT";
|
option (ifdef) = "USE_SELECT";
|
||||||
|
|
||||||
@@ -1019,9 +1075,11 @@ message ListEntitiesSelectResponse {
|
|||||||
repeated string options = 6;
|
repeated string options = 6;
|
||||||
bool disabled_by_default = 7;
|
bool disabled_by_default = 7;
|
||||||
EntityCategory entity_category = 8;
|
EntityCategory entity_category = 8;
|
||||||
|
uint32 device_id = 9;
|
||||||
}
|
}
|
||||||
message SelectStateResponse {
|
message SelectStateResponse {
|
||||||
option (id) = 53;
|
option (id) = 53;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_SELECT";
|
option (ifdef) = "USE_SELECT";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -1031,6 +1089,7 @@ message SelectStateResponse {
|
|||||||
// If the select does not have a valid state yet.
|
// If the select does not have a valid state yet.
|
||||||
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
||||||
bool missing_state = 3;
|
bool missing_state = 3;
|
||||||
|
uint32 device_id = 4;
|
||||||
}
|
}
|
||||||
message SelectCommandRequest {
|
message SelectCommandRequest {
|
||||||
option (id) = 54;
|
option (id) = 54;
|
||||||
@@ -1045,6 +1104,7 @@ message SelectCommandRequest {
|
|||||||
// ==================== SIREN ====================
|
// ==================== SIREN ====================
|
||||||
message ListEntitiesSirenResponse {
|
message ListEntitiesSirenResponse {
|
||||||
option (id) = 55;
|
option (id) = 55;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_SIREN";
|
option (ifdef) = "USE_SIREN";
|
||||||
|
|
||||||
@@ -1059,15 +1119,18 @@ message ListEntitiesSirenResponse {
|
|||||||
bool supports_duration = 8;
|
bool supports_duration = 8;
|
||||||
bool supports_volume = 9;
|
bool supports_volume = 9;
|
||||||
EntityCategory entity_category = 10;
|
EntityCategory entity_category = 10;
|
||||||
|
uint32 device_id = 11;
|
||||||
}
|
}
|
||||||
message SirenStateResponse {
|
message SirenStateResponse {
|
||||||
option (id) = 56;
|
option (id) = 56;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_SIREN";
|
option (ifdef) = "USE_SIREN";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
|
|
||||||
fixed32 key = 1;
|
fixed32 key = 1;
|
||||||
bool state = 2;
|
bool state = 2;
|
||||||
|
uint32 device_id = 3;
|
||||||
}
|
}
|
||||||
message SirenCommandRequest {
|
message SirenCommandRequest {
|
||||||
option (id) = 57;
|
option (id) = 57;
|
||||||
@@ -1102,6 +1165,7 @@ enum LockCommand {
|
|||||||
}
|
}
|
||||||
message ListEntitiesLockResponse {
|
message ListEntitiesLockResponse {
|
||||||
option (id) = 58;
|
option (id) = 58;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_LOCK";
|
option (ifdef) = "USE_LOCK";
|
||||||
|
|
||||||
@@ -1120,14 +1184,17 @@ message ListEntitiesLockResponse {
|
|||||||
|
|
||||||
// Not yet implemented:
|
// Not yet implemented:
|
||||||
string code_format = 11;
|
string code_format = 11;
|
||||||
|
uint32 device_id = 12;
|
||||||
}
|
}
|
||||||
message LockStateResponse {
|
message LockStateResponse {
|
||||||
option (id) = 59;
|
option (id) = 59;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_LOCK";
|
option (ifdef) = "USE_LOCK";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
fixed32 key = 1;
|
fixed32 key = 1;
|
||||||
LockState state = 2;
|
LockState state = 2;
|
||||||
|
uint32 device_id = 3;
|
||||||
}
|
}
|
||||||
message LockCommandRequest {
|
message LockCommandRequest {
|
||||||
option (id) = 60;
|
option (id) = 60;
|
||||||
@@ -1145,6 +1212,7 @@ message LockCommandRequest {
|
|||||||
// ==================== BUTTON ====================
|
// ==================== BUTTON ====================
|
||||||
message ListEntitiesButtonResponse {
|
message ListEntitiesButtonResponse {
|
||||||
option (id) = 61;
|
option (id) = 61;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_BUTTON";
|
option (ifdef) = "USE_BUTTON";
|
||||||
|
|
||||||
@@ -1157,6 +1225,7 @@ message ListEntitiesButtonResponse {
|
|||||||
bool disabled_by_default = 6;
|
bool disabled_by_default = 6;
|
||||||
EntityCategory entity_category = 7;
|
EntityCategory entity_category = 7;
|
||||||
string device_class = 8;
|
string device_class = 8;
|
||||||
|
uint32 device_id = 9;
|
||||||
}
|
}
|
||||||
message ButtonCommandRequest {
|
message ButtonCommandRequest {
|
||||||
option (id) = 62;
|
option (id) = 62;
|
||||||
@@ -1196,6 +1265,7 @@ message MediaPlayerSupportedFormat {
|
|||||||
}
|
}
|
||||||
message ListEntitiesMediaPlayerResponse {
|
message ListEntitiesMediaPlayerResponse {
|
||||||
option (id) = 63;
|
option (id) = 63;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_MEDIA_PLAYER";
|
option (ifdef) = "USE_MEDIA_PLAYER";
|
||||||
|
|
||||||
@@ -1211,9 +1281,12 @@ message ListEntitiesMediaPlayerResponse {
|
|||||||
bool supports_pause = 8;
|
bool supports_pause = 8;
|
||||||
|
|
||||||
repeated MediaPlayerSupportedFormat supported_formats = 9;
|
repeated MediaPlayerSupportedFormat supported_formats = 9;
|
||||||
|
|
||||||
|
uint32 device_id = 10;
|
||||||
}
|
}
|
||||||
message MediaPlayerStateResponse {
|
message MediaPlayerStateResponse {
|
||||||
option (id) = 64;
|
option (id) = 64;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_MEDIA_PLAYER";
|
option (ifdef) = "USE_MEDIA_PLAYER";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -1221,6 +1294,7 @@ message MediaPlayerStateResponse {
|
|||||||
MediaPlayerState state = 2;
|
MediaPlayerState state = 2;
|
||||||
float volume = 3;
|
float volume = 3;
|
||||||
bool muted = 4;
|
bool muted = 4;
|
||||||
|
uint32 device_id = 5;
|
||||||
}
|
}
|
||||||
message MediaPlayerCommandRequest {
|
message MediaPlayerCommandRequest {
|
||||||
option (id) = 65;
|
option (id) = 65;
|
||||||
@@ -1615,6 +1689,7 @@ enum VoiceAssistantEvent {
|
|||||||
VOICE_ASSISTANT_STT_VAD_END = 12;
|
VOICE_ASSISTANT_STT_VAD_END = 12;
|
||||||
VOICE_ASSISTANT_TTS_STREAM_START = 98;
|
VOICE_ASSISTANT_TTS_STREAM_START = 98;
|
||||||
VOICE_ASSISTANT_TTS_STREAM_END = 99;
|
VOICE_ASSISTANT_TTS_STREAM_END = 99;
|
||||||
|
VOICE_ASSISTANT_INTENT_PROGRESS = 100;
|
||||||
}
|
}
|
||||||
|
|
||||||
message VoiceAssistantEventData {
|
message VoiceAssistantEventData {
|
||||||
@@ -1735,6 +1810,7 @@ enum AlarmControlPanelStateCommand {
|
|||||||
|
|
||||||
message ListEntitiesAlarmControlPanelResponse {
|
message ListEntitiesAlarmControlPanelResponse {
|
||||||
option (id) = 94;
|
option (id) = 94;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_ALARM_CONTROL_PANEL";
|
option (ifdef) = "USE_ALARM_CONTROL_PANEL";
|
||||||
|
|
||||||
@@ -1748,15 +1824,18 @@ message ListEntitiesAlarmControlPanelResponse {
|
|||||||
uint32 supported_features = 8;
|
uint32 supported_features = 8;
|
||||||
bool requires_code = 9;
|
bool requires_code = 9;
|
||||||
bool requires_code_to_arm = 10;
|
bool requires_code_to_arm = 10;
|
||||||
|
uint32 device_id = 11;
|
||||||
}
|
}
|
||||||
|
|
||||||
message AlarmControlPanelStateResponse {
|
message AlarmControlPanelStateResponse {
|
||||||
option (id) = 95;
|
option (id) = 95;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_ALARM_CONTROL_PANEL";
|
option (ifdef) = "USE_ALARM_CONTROL_PANEL";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
fixed32 key = 1;
|
fixed32 key = 1;
|
||||||
AlarmControlPanelState state = 2;
|
AlarmControlPanelState state = 2;
|
||||||
|
uint32 device_id = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
message AlarmControlPanelCommandRequest {
|
message AlarmControlPanelCommandRequest {
|
||||||
@@ -1776,6 +1855,7 @@ enum TextMode {
|
|||||||
}
|
}
|
||||||
message ListEntitiesTextResponse {
|
message ListEntitiesTextResponse {
|
||||||
option (id) = 97;
|
option (id) = 97;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_TEXT";
|
option (ifdef) = "USE_TEXT";
|
||||||
|
|
||||||
@@ -1791,9 +1871,11 @@ message ListEntitiesTextResponse {
|
|||||||
uint32 max_length = 9;
|
uint32 max_length = 9;
|
||||||
string pattern = 10;
|
string pattern = 10;
|
||||||
TextMode mode = 11;
|
TextMode mode = 11;
|
||||||
|
uint32 device_id = 12;
|
||||||
}
|
}
|
||||||
message TextStateResponse {
|
message TextStateResponse {
|
||||||
option (id) = 98;
|
option (id) = 98;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_TEXT";
|
option (ifdef) = "USE_TEXT";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -1803,6 +1885,7 @@ message TextStateResponse {
|
|||||||
// If the Text does not have a valid state yet.
|
// If the Text does not have a valid state yet.
|
||||||
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
||||||
bool missing_state = 3;
|
bool missing_state = 3;
|
||||||
|
uint32 device_id = 4;
|
||||||
}
|
}
|
||||||
message TextCommandRequest {
|
message TextCommandRequest {
|
||||||
option (id) = 99;
|
option (id) = 99;
|
||||||
@@ -1818,6 +1901,7 @@ message TextCommandRequest {
|
|||||||
// ==================== DATETIME DATE ====================
|
// ==================== DATETIME DATE ====================
|
||||||
message ListEntitiesDateResponse {
|
message ListEntitiesDateResponse {
|
||||||
option (id) = 100;
|
option (id) = 100;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_DATETIME_DATE";
|
option (ifdef) = "USE_DATETIME_DATE";
|
||||||
|
|
||||||
@@ -1829,9 +1913,11 @@ message ListEntitiesDateResponse {
|
|||||||
string icon = 5;
|
string icon = 5;
|
||||||
bool disabled_by_default = 6;
|
bool disabled_by_default = 6;
|
||||||
EntityCategory entity_category = 7;
|
EntityCategory entity_category = 7;
|
||||||
|
uint32 device_id = 8;
|
||||||
}
|
}
|
||||||
message DateStateResponse {
|
message DateStateResponse {
|
||||||
option (id) = 101;
|
option (id) = 101;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_DATETIME_DATE";
|
option (ifdef) = "USE_DATETIME_DATE";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -1843,6 +1929,7 @@ message DateStateResponse {
|
|||||||
uint32 year = 3;
|
uint32 year = 3;
|
||||||
uint32 month = 4;
|
uint32 month = 4;
|
||||||
uint32 day = 5;
|
uint32 day = 5;
|
||||||
|
uint32 device_id = 6;
|
||||||
}
|
}
|
||||||
message DateCommandRequest {
|
message DateCommandRequest {
|
||||||
option (id) = 102;
|
option (id) = 102;
|
||||||
@@ -1859,6 +1946,7 @@ message DateCommandRequest {
|
|||||||
// ==================== DATETIME TIME ====================
|
// ==================== DATETIME TIME ====================
|
||||||
message ListEntitiesTimeResponse {
|
message ListEntitiesTimeResponse {
|
||||||
option (id) = 103;
|
option (id) = 103;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_DATETIME_TIME";
|
option (ifdef) = "USE_DATETIME_TIME";
|
||||||
|
|
||||||
@@ -1870,9 +1958,11 @@ message ListEntitiesTimeResponse {
|
|||||||
string icon = 5;
|
string icon = 5;
|
||||||
bool disabled_by_default = 6;
|
bool disabled_by_default = 6;
|
||||||
EntityCategory entity_category = 7;
|
EntityCategory entity_category = 7;
|
||||||
|
uint32 device_id = 8;
|
||||||
}
|
}
|
||||||
message TimeStateResponse {
|
message TimeStateResponse {
|
||||||
option (id) = 104;
|
option (id) = 104;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_DATETIME_TIME";
|
option (ifdef) = "USE_DATETIME_TIME";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -1884,6 +1974,7 @@ message TimeStateResponse {
|
|||||||
uint32 hour = 3;
|
uint32 hour = 3;
|
||||||
uint32 minute = 4;
|
uint32 minute = 4;
|
||||||
uint32 second = 5;
|
uint32 second = 5;
|
||||||
|
uint32 device_id = 6;
|
||||||
}
|
}
|
||||||
message TimeCommandRequest {
|
message TimeCommandRequest {
|
||||||
option (id) = 105;
|
option (id) = 105;
|
||||||
@@ -1900,6 +1991,7 @@ message TimeCommandRequest {
|
|||||||
// ==================== EVENT ====================
|
// ==================== EVENT ====================
|
||||||
message ListEntitiesEventResponse {
|
message ListEntitiesEventResponse {
|
||||||
option (id) = 107;
|
option (id) = 107;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_EVENT";
|
option (ifdef) = "USE_EVENT";
|
||||||
|
|
||||||
@@ -1914,19 +2006,23 @@ message ListEntitiesEventResponse {
|
|||||||
string device_class = 8;
|
string device_class = 8;
|
||||||
|
|
||||||
repeated string event_types = 9;
|
repeated string event_types = 9;
|
||||||
|
uint32 device_id = 10;
|
||||||
}
|
}
|
||||||
message EventResponse {
|
message EventResponse {
|
||||||
option (id) = 108;
|
option (id) = 108;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_EVENT";
|
option (ifdef) = "USE_EVENT";
|
||||||
|
|
||||||
fixed32 key = 1;
|
fixed32 key = 1;
|
||||||
string event_type = 2;
|
string event_type = 2;
|
||||||
|
uint32 device_id = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// ==================== VALVE ====================
|
// ==================== VALVE ====================
|
||||||
message ListEntitiesValveResponse {
|
message ListEntitiesValveResponse {
|
||||||
option (id) = 109;
|
option (id) = 109;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_VALVE";
|
option (ifdef) = "USE_VALVE";
|
||||||
|
|
||||||
@@ -1943,6 +2039,7 @@ message ListEntitiesValveResponse {
|
|||||||
bool assumed_state = 9;
|
bool assumed_state = 9;
|
||||||
bool supports_position = 10;
|
bool supports_position = 10;
|
||||||
bool supports_stop = 11;
|
bool supports_stop = 11;
|
||||||
|
uint32 device_id = 12;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum ValveOperation {
|
enum ValveOperation {
|
||||||
@@ -1952,6 +2049,7 @@ enum ValveOperation {
|
|||||||
}
|
}
|
||||||
message ValveStateResponse {
|
message ValveStateResponse {
|
||||||
option (id) = 110;
|
option (id) = 110;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_VALVE";
|
option (ifdef) = "USE_VALVE";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -1959,6 +2057,7 @@ message ValveStateResponse {
|
|||||||
fixed32 key = 1;
|
fixed32 key = 1;
|
||||||
float position = 2;
|
float position = 2;
|
||||||
ValveOperation current_operation = 3;
|
ValveOperation current_operation = 3;
|
||||||
|
uint32 device_id = 4;
|
||||||
}
|
}
|
||||||
|
|
||||||
message ValveCommandRequest {
|
message ValveCommandRequest {
|
||||||
@@ -1976,6 +2075,7 @@ message ValveCommandRequest {
|
|||||||
// ==================== DATETIME DATETIME ====================
|
// ==================== DATETIME DATETIME ====================
|
||||||
message ListEntitiesDateTimeResponse {
|
message ListEntitiesDateTimeResponse {
|
||||||
option (id) = 112;
|
option (id) = 112;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_DATETIME_DATETIME";
|
option (ifdef) = "USE_DATETIME_DATETIME";
|
||||||
|
|
||||||
@@ -1987,9 +2087,11 @@ message ListEntitiesDateTimeResponse {
|
|||||||
string icon = 5;
|
string icon = 5;
|
||||||
bool disabled_by_default = 6;
|
bool disabled_by_default = 6;
|
||||||
EntityCategory entity_category = 7;
|
EntityCategory entity_category = 7;
|
||||||
|
uint32 device_id = 8;
|
||||||
}
|
}
|
||||||
message DateTimeStateResponse {
|
message DateTimeStateResponse {
|
||||||
option (id) = 113;
|
option (id) = 113;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_DATETIME_DATETIME";
|
option (ifdef) = "USE_DATETIME_DATETIME";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -1999,6 +2101,7 @@ message DateTimeStateResponse {
|
|||||||
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
// Equivalent to `!obj->has_state()` - inverse logic to make state packets smaller
|
||||||
bool missing_state = 2;
|
bool missing_state = 2;
|
||||||
fixed32 epoch_seconds = 3;
|
fixed32 epoch_seconds = 3;
|
||||||
|
uint32 device_id = 4;
|
||||||
}
|
}
|
||||||
message DateTimeCommandRequest {
|
message DateTimeCommandRequest {
|
||||||
option (id) = 114;
|
option (id) = 114;
|
||||||
@@ -2013,6 +2116,7 @@ message DateTimeCommandRequest {
|
|||||||
// ==================== UPDATE ====================
|
// ==================== UPDATE ====================
|
||||||
message ListEntitiesUpdateResponse {
|
message ListEntitiesUpdateResponse {
|
||||||
option (id) = 116;
|
option (id) = 116;
|
||||||
|
option (base_class) = "InfoResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_UPDATE";
|
option (ifdef) = "USE_UPDATE";
|
||||||
|
|
||||||
@@ -2025,9 +2129,11 @@ message ListEntitiesUpdateResponse {
|
|||||||
bool disabled_by_default = 6;
|
bool disabled_by_default = 6;
|
||||||
EntityCategory entity_category = 7;
|
EntityCategory entity_category = 7;
|
||||||
string device_class = 8;
|
string device_class = 8;
|
||||||
|
uint32 device_id = 9;
|
||||||
}
|
}
|
||||||
message UpdateStateResponse {
|
message UpdateStateResponse {
|
||||||
option (id) = 117;
|
option (id) = 117;
|
||||||
|
option (base_class) = "StateResponseProtoMessage";
|
||||||
option (source) = SOURCE_SERVER;
|
option (source) = SOURCE_SERVER;
|
||||||
option (ifdef) = "USE_UPDATE";
|
option (ifdef) = "USE_UPDATE";
|
||||||
option (no_delay) = true;
|
option (no_delay) = true;
|
||||||
@@ -2042,6 +2148,7 @@ message UpdateStateResponse {
|
|||||||
string title = 8;
|
string title = 8;
|
||||||
string release_summary = 9;
|
string release_summary = 9;
|
||||||
string release_url = 10;
|
string release_url = 10;
|
||||||
|
uint32 device_id = 11;
|
||||||
}
|
}
|
||||||
enum UpdateCommand {
|
enum UpdateCommand {
|
||||||
UPDATE_COMMAND_NONE = 0;
|
UPDATE_COMMAND_NONE = 0;
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -11,56 +11,20 @@
|
|||||||
#include "esphome/core/entity_base.h"
|
#include "esphome/core/entity_base.h"
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
#include <functional>
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace api {
|
namespace api {
|
||||||
|
|
||||||
// Keepalive timeout in milliseconds
|
// Keepalive timeout in milliseconds
|
||||||
static constexpr uint32_t KEEPALIVE_TIMEOUT_MS = 60000;
|
static constexpr uint32_t KEEPALIVE_TIMEOUT_MS = 60000;
|
||||||
|
// Maximum number of entities to process in a single batch during initial state/info sending
|
||||||
using send_message_t = bool (APIConnection::*)(void *);
|
static constexpr size_t MAX_INITIAL_PER_BATCH = 20;
|
||||||
|
|
||||||
/*
|
|
||||||
This class holds a pointer to the source component that wants to publish a message, and a pointer to a function that
|
|
||||||
will lazily publish that message. The two pointers allow dedup in the deferred queue if multiple publishes for the
|
|
||||||
same component are backed up, and take up only 8 bytes of memory. The entry in the deferred queue (a std::vector) is
|
|
||||||
the DeferredMessage instance itself (not a pointer to one elsewhere in heap) so still only 8 bytes per entry. Even
|
|
||||||
100 backed up messages (you'd have to have at least 100 sensors publishing because of dedup) would take up only 0.8
|
|
||||||
kB.
|
|
||||||
*/
|
|
||||||
class DeferredMessageQueue {
|
|
||||||
struct DeferredMessage {
|
|
||||||
friend class DeferredMessageQueue;
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void *source_;
|
|
||||||
send_message_t send_message_;
|
|
||||||
|
|
||||||
public:
|
|
||||||
DeferredMessage(void *source, send_message_t send_message) : source_(source), send_message_(send_message) {}
|
|
||||||
bool operator==(const DeferredMessage &test) const {
|
|
||||||
return (source_ == test.source_ && send_message_ == test.send_message_);
|
|
||||||
}
|
|
||||||
} __attribute__((packed));
|
|
||||||
|
|
||||||
protected:
|
|
||||||
// vector is used very specifically for its zero memory overhead even though items are popped from the front (memory
|
|
||||||
// footprint is more important than speed here)
|
|
||||||
std::vector<DeferredMessage> deferred_queue_;
|
|
||||||
APIConnection *api_connection_;
|
|
||||||
|
|
||||||
// helper for allowing only unique entries in the queue
|
|
||||||
void dmq_push_back_with_dedup_(void *source, send_message_t send_message);
|
|
||||||
|
|
||||||
public:
|
|
||||||
DeferredMessageQueue(APIConnection *api_connection) : api_connection_(api_connection) {}
|
|
||||||
void process_queue();
|
|
||||||
void defer(void *source, send_message_t send_message);
|
|
||||||
bool empty() const { return deferred_queue_.empty(); }
|
|
||||||
};
|
|
||||||
|
|
||||||
class APIConnection : public APIServerConnection {
|
class APIConnection : public APIServerConnection {
|
||||||
public:
|
public:
|
||||||
|
friend class APIServer;
|
||||||
|
friend class ListEntitiesIterator;
|
||||||
APIConnection(std::unique_ptr<socket::Socket> socket, APIServer *parent);
|
APIConnection(std::unique_ptr<socket::Socket> socket, APIServer *parent);
|
||||||
virtual ~APIConnection();
|
virtual ~APIConnection();
|
||||||
|
|
||||||
@@ -68,225 +32,86 @@ class APIConnection : public APIServerConnection {
|
|||||||
void loop();
|
void loop();
|
||||||
|
|
||||||
bool send_list_info_done() {
|
bool send_list_info_done() {
|
||||||
ListEntitiesDoneResponse resp;
|
return this->schedule_message_(nullptr, &APIConnection::try_send_list_info_done,
|
||||||
return this->send_list_entities_done_response(resp);
|
ListEntitiesDoneResponse::MESSAGE_TYPE);
|
||||||
}
|
}
|
||||||
#ifdef USE_BINARY_SENSOR
|
#ifdef USE_BINARY_SENSOR
|
||||||
bool send_binary_sensor_state(binary_sensor::BinarySensor *binary_sensor, bool state);
|
bool send_binary_sensor_state(binary_sensor::BinarySensor *binary_sensor);
|
||||||
void send_binary_sensor_info(binary_sensor::BinarySensor *binary_sensor);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_binary_sensor_state_(binary_sensor::BinarySensor *binary_sensor);
|
|
||||||
bool try_send_binary_sensor_state_(binary_sensor::BinarySensor *binary_sensor, bool state);
|
|
||||||
bool try_send_binary_sensor_info_(binary_sensor::BinarySensor *binary_sensor);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_COVER
|
#ifdef USE_COVER
|
||||||
bool send_cover_state(cover::Cover *cover);
|
bool send_cover_state(cover::Cover *cover);
|
||||||
void send_cover_info(cover::Cover *cover);
|
|
||||||
void cover_command(const CoverCommandRequest &msg) override;
|
void cover_command(const CoverCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_cover_state_(cover::Cover *cover);
|
|
||||||
bool try_send_cover_info_(cover::Cover *cover);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_FAN
|
#ifdef USE_FAN
|
||||||
bool send_fan_state(fan::Fan *fan);
|
bool send_fan_state(fan::Fan *fan);
|
||||||
void send_fan_info(fan::Fan *fan);
|
|
||||||
void fan_command(const FanCommandRequest &msg) override;
|
void fan_command(const FanCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_fan_state_(fan::Fan *fan);
|
|
||||||
bool try_send_fan_info_(fan::Fan *fan);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LIGHT
|
#ifdef USE_LIGHT
|
||||||
bool send_light_state(light::LightState *light);
|
bool send_light_state(light::LightState *light);
|
||||||
void send_light_info(light::LightState *light);
|
|
||||||
void light_command(const LightCommandRequest &msg) override;
|
void light_command(const LightCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_light_state_(light::LightState *light);
|
|
||||||
bool try_send_light_info_(light::LightState *light);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SENSOR
|
#ifdef USE_SENSOR
|
||||||
bool send_sensor_state(sensor::Sensor *sensor, float state);
|
bool send_sensor_state(sensor::Sensor *sensor);
|
||||||
void send_sensor_info(sensor::Sensor *sensor);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_sensor_state_(sensor::Sensor *sensor);
|
|
||||||
bool try_send_sensor_state_(sensor::Sensor *sensor, float state);
|
|
||||||
bool try_send_sensor_info_(sensor::Sensor *sensor);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SWITCH
|
#ifdef USE_SWITCH
|
||||||
bool send_switch_state(switch_::Switch *a_switch, bool state);
|
bool send_switch_state(switch_::Switch *a_switch);
|
||||||
void send_switch_info(switch_::Switch *a_switch);
|
|
||||||
void switch_command(const SwitchCommandRequest &msg) override;
|
void switch_command(const SwitchCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_switch_state_(switch_::Switch *a_switch);
|
|
||||||
bool try_send_switch_state_(switch_::Switch *a_switch, bool state);
|
|
||||||
bool try_send_switch_info_(switch_::Switch *a_switch);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT_SENSOR
|
#ifdef USE_TEXT_SENSOR
|
||||||
bool send_text_sensor_state(text_sensor::TextSensor *text_sensor, std::string state);
|
bool send_text_sensor_state(text_sensor::TextSensor *text_sensor);
|
||||||
void send_text_sensor_info(text_sensor::TextSensor *text_sensor);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_text_sensor_state_(text_sensor::TextSensor *text_sensor);
|
|
||||||
bool try_send_text_sensor_state_(text_sensor::TextSensor *text_sensor, std::string state);
|
|
||||||
bool try_send_text_sensor_info_(text_sensor::TextSensor *text_sensor);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ESP32_CAMERA
|
#ifdef USE_CAMERA
|
||||||
void set_camera_state(std::shared_ptr<esp32_camera::CameraImage> image);
|
void set_camera_state(std::shared_ptr<camera::CameraImage> image);
|
||||||
void send_camera_info(esp32_camera::ESP32Camera *camera);
|
|
||||||
void camera_image(const CameraImageRequest &msg) override;
|
void camera_image(const CameraImageRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_camera_info_(esp32_camera::ESP32Camera *camera);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_CLIMATE
|
#ifdef USE_CLIMATE
|
||||||
bool send_climate_state(climate::Climate *climate);
|
bool send_climate_state(climate::Climate *climate);
|
||||||
void send_climate_info(climate::Climate *climate);
|
|
||||||
void climate_command(const ClimateCommandRequest &msg) override;
|
void climate_command(const ClimateCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_climate_state_(climate::Climate *climate);
|
|
||||||
bool try_send_climate_info_(climate::Climate *climate);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_NUMBER
|
#ifdef USE_NUMBER
|
||||||
bool send_number_state(number::Number *number, float state);
|
bool send_number_state(number::Number *number);
|
||||||
void send_number_info(number::Number *number);
|
|
||||||
void number_command(const NumberCommandRequest &msg) override;
|
void number_command(const NumberCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_number_state_(number::Number *number);
|
|
||||||
bool try_send_number_state_(number::Number *number, float state);
|
|
||||||
bool try_send_number_info_(number::Number *number);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATE
|
#ifdef USE_DATETIME_DATE
|
||||||
bool send_date_state(datetime::DateEntity *date);
|
bool send_date_state(datetime::DateEntity *date);
|
||||||
void send_date_info(datetime::DateEntity *date);
|
|
||||||
void date_command(const DateCommandRequest &msg) override;
|
void date_command(const DateCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_date_state_(datetime::DateEntity *date);
|
|
||||||
bool try_send_date_info_(datetime::DateEntity *date);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_TIME
|
#ifdef USE_DATETIME_TIME
|
||||||
bool send_time_state(datetime::TimeEntity *time);
|
bool send_time_state(datetime::TimeEntity *time);
|
||||||
void send_time_info(datetime::TimeEntity *time);
|
|
||||||
void time_command(const TimeCommandRequest &msg) override;
|
void time_command(const TimeCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_time_state_(datetime::TimeEntity *time);
|
|
||||||
bool try_send_time_info_(datetime::TimeEntity *time);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATETIME
|
#ifdef USE_DATETIME_DATETIME
|
||||||
bool send_datetime_state(datetime::DateTimeEntity *datetime);
|
bool send_datetime_state(datetime::DateTimeEntity *datetime);
|
||||||
void send_datetime_info(datetime::DateTimeEntity *datetime);
|
|
||||||
void datetime_command(const DateTimeCommandRequest &msg) override;
|
void datetime_command(const DateTimeCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_datetime_state_(datetime::DateTimeEntity *datetime);
|
|
||||||
bool try_send_datetime_info_(datetime::DateTimeEntity *datetime);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT
|
#ifdef USE_TEXT
|
||||||
bool send_text_state(text::Text *text, std::string state);
|
bool send_text_state(text::Text *text);
|
||||||
void send_text_info(text::Text *text);
|
|
||||||
void text_command(const TextCommandRequest &msg) override;
|
void text_command(const TextCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_text_state_(text::Text *text);
|
|
||||||
bool try_send_text_state_(text::Text *text, std::string state);
|
|
||||||
bool try_send_text_info_(text::Text *text);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
bool send_select_state(select::Select *select, std::string state);
|
bool send_select_state(select::Select *select);
|
||||||
void send_select_info(select::Select *select);
|
|
||||||
void select_command(const SelectCommandRequest &msg) override;
|
void select_command(const SelectCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_select_state_(select::Select *select);
|
|
||||||
bool try_send_select_state_(select::Select *select, std::string state);
|
|
||||||
bool try_send_select_info_(select::Select *select);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BUTTON
|
#ifdef USE_BUTTON
|
||||||
void send_button_info(button::Button *button);
|
|
||||||
void button_command(const ButtonCommandRequest &msg) override;
|
void button_command(const ButtonCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_button_info_(button::Button *button);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LOCK
|
#ifdef USE_LOCK
|
||||||
bool send_lock_state(lock::Lock *a_lock, lock::LockState state);
|
bool send_lock_state(lock::Lock *a_lock);
|
||||||
void send_lock_info(lock::Lock *a_lock);
|
|
||||||
void lock_command(const LockCommandRequest &msg) override;
|
void lock_command(const LockCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_lock_state_(lock::Lock *a_lock);
|
|
||||||
bool try_send_lock_state_(lock::Lock *a_lock, lock::LockState state);
|
|
||||||
bool try_send_lock_info_(lock::Lock *a_lock);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VALVE
|
#ifdef USE_VALVE
|
||||||
bool send_valve_state(valve::Valve *valve);
|
bool send_valve_state(valve::Valve *valve);
|
||||||
void send_valve_info(valve::Valve *valve);
|
|
||||||
void valve_command(const ValveCommandRequest &msg) override;
|
void valve_command(const ValveCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_valve_state_(valve::Valve *valve);
|
|
||||||
bool try_send_valve_info_(valve::Valve *valve);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
bool send_media_player_state(media_player::MediaPlayer *media_player);
|
bool send_media_player_state(media_player::MediaPlayer *media_player);
|
||||||
void send_media_player_info(media_player::MediaPlayer *media_player);
|
|
||||||
void media_player_command(const MediaPlayerCommandRequest &msg) override;
|
void media_player_command(const MediaPlayerCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_media_player_state_(media_player::MediaPlayer *media_player);
|
|
||||||
bool try_send_media_player_info_(media_player::MediaPlayer *media_player);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
bool try_send_log_message(int level, const char *tag, const char *line);
|
bool try_send_log_message(int level, const char *tag, const char *line, size_t message_len);
|
||||||
void send_homeassistant_service_call(const HomeassistantServiceResponse &call) {
|
void send_homeassistant_service_call(const HomeassistantServiceResponse &call) {
|
||||||
if (!this->service_call_subscription_)
|
if (!this->flags_.service_call_subscription)
|
||||||
return;
|
return;
|
||||||
this->send_homeassistant_service_response(call);
|
this->send_message(call);
|
||||||
}
|
}
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
void subscribe_bluetooth_le_advertisements(const SubscribeBluetoothLEAdvertisementsRequest &msg) override;
|
void subscribe_bluetooth_le_advertisements(const SubscribeBluetoothLEAdvertisementsRequest &msg) override;
|
||||||
@@ -308,7 +133,7 @@ class APIConnection : public APIServerConnection {
|
|||||||
#ifdef USE_HOMEASSISTANT_TIME
|
#ifdef USE_HOMEASSISTANT_TIME
|
||||||
void send_time_request() {
|
void send_time_request() {
|
||||||
GetTimeRequest req;
|
GetTimeRequest req;
|
||||||
this->send_get_time_request(req);
|
this->send_message(req);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@@ -326,45 +151,22 @@ class APIConnection : public APIServerConnection {
|
|||||||
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
bool send_alarm_control_panel_state(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel);
|
bool send_alarm_control_panel_state(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel);
|
||||||
void send_alarm_control_panel_info(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel);
|
|
||||||
void alarm_control_panel_command(const AlarmControlPanelCommandRequest &msg) override;
|
void alarm_control_panel_command(const AlarmControlPanelCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_alarm_control_panel_state_(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel);
|
|
||||||
bool try_send_alarm_control_panel_info_(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_EVENT
|
#ifdef USE_EVENT
|
||||||
void send_event(event::Event *event, std::string event_type);
|
void send_event(event::Event *event, const std::string &event_type);
|
||||||
void send_event_info(event::Event *event);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_event_(event::Event *event);
|
|
||||||
bool try_send_event_(event::Event *event, std::string event_type);
|
|
||||||
bool try_send_event_info_(event::Event *event);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_UPDATE
|
#ifdef USE_UPDATE
|
||||||
bool send_update_state(update::UpdateEntity *update);
|
bool send_update_state(update::UpdateEntity *update);
|
||||||
void send_update_info(update::UpdateEntity *update);
|
|
||||||
void update_command(const UpdateCommandRequest &msg) override;
|
void update_command(const UpdateCommandRequest &msg) override;
|
||||||
|
|
||||||
protected:
|
|
||||||
bool try_send_update_state_(update::UpdateEntity *update);
|
|
||||||
bool try_send_update_info_(update::UpdateEntity *update);
|
|
||||||
|
|
||||||
public:
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
void on_disconnect_response(const DisconnectResponse &value) override;
|
void on_disconnect_response(const DisconnectResponse &value) override;
|
||||||
void on_ping_response(const PingResponse &value) override {
|
void on_ping_response(const PingResponse &value) override {
|
||||||
// we initiated ping
|
// we initiated ping
|
||||||
this->ping_retries_ = 0;
|
this->flags_.sent_ping = false;
|
||||||
this->sent_ping_ = false;
|
|
||||||
}
|
}
|
||||||
void on_home_assistant_state_response(const HomeAssistantStateResponse &msg) override;
|
void on_home_assistant_state_response(const HomeAssistantStateResponse &msg) override;
|
||||||
#ifdef USE_HOMEASSISTANT_TIME
|
#ifdef USE_HOMEASSISTANT_TIME
|
||||||
@@ -377,16 +179,16 @@ class APIConnection : public APIServerConnection {
|
|||||||
DeviceInfoResponse device_info(const DeviceInfoRequest &msg) override;
|
DeviceInfoResponse device_info(const DeviceInfoRequest &msg) override;
|
||||||
void list_entities(const ListEntitiesRequest &msg) override { this->list_entities_iterator_.begin(); }
|
void list_entities(const ListEntitiesRequest &msg) override { this->list_entities_iterator_.begin(); }
|
||||||
void subscribe_states(const SubscribeStatesRequest &msg) override {
|
void subscribe_states(const SubscribeStatesRequest &msg) override {
|
||||||
this->state_subscription_ = true;
|
this->flags_.state_subscription = true;
|
||||||
this->initial_state_iterator_.begin();
|
this->initial_state_iterator_.begin();
|
||||||
}
|
}
|
||||||
void subscribe_logs(const SubscribeLogsRequest &msg) override {
|
void subscribe_logs(const SubscribeLogsRequest &msg) override {
|
||||||
this->log_subscription_ = msg.level;
|
this->flags_.log_subscription = msg.level;
|
||||||
if (msg.dump_config)
|
if (msg.dump_config)
|
||||||
App.schedule_dump_config();
|
App.schedule_dump_config();
|
||||||
}
|
}
|
||||||
void subscribe_homeassistant_services(const SubscribeHomeassistantServicesRequest &msg) override {
|
void subscribe_homeassistant_services(const SubscribeHomeassistantServicesRequest &msg) override {
|
||||||
this->service_call_subscription_ = true;
|
this->flags_.service_call_subscription = true;
|
||||||
}
|
}
|
||||||
void subscribe_home_assistant_states(const SubscribeHomeAssistantStatesRequest &msg) override;
|
void subscribe_home_assistant_states(const SubscribeHomeAssistantStatesRequest &msg) override;
|
||||||
GetTimeResponse get_time(const GetTimeRequest &msg) override {
|
GetTimeResponse get_time(const GetTimeRequest &msg) override {
|
||||||
@@ -398,111 +200,79 @@ class APIConnection : public APIServerConnection {
|
|||||||
NoiseEncryptionSetKeyResponse noise_encryption_set_key(const NoiseEncryptionSetKeyRequest &msg) override;
|
NoiseEncryptionSetKeyResponse noise_encryption_set_key(const NoiseEncryptionSetKeyRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
bool is_authenticated() override { return this->connection_state_ == ConnectionState::AUTHENTICATED; }
|
bool is_authenticated() override {
|
||||||
|
return static_cast<ConnectionState>(this->flags_.connection_state) == ConnectionState::AUTHENTICATED;
|
||||||
|
}
|
||||||
bool is_connection_setup() override {
|
bool is_connection_setup() override {
|
||||||
return this->connection_state_ == ConnectionState ::CONNECTED || this->is_authenticated();
|
return static_cast<ConnectionState>(this->flags_.connection_state) == ConnectionState::CONNECTED ||
|
||||||
|
this->is_authenticated();
|
||||||
}
|
}
|
||||||
void on_fatal_error() override;
|
void on_fatal_error() override;
|
||||||
void on_unauthenticated_access() override;
|
void on_unauthenticated_access() override;
|
||||||
void on_no_setup_connection() override;
|
void on_no_setup_connection() override;
|
||||||
ProtoWriteBuffer create_buffer(uint32_t reserve_size) override {
|
ProtoWriteBuffer create_buffer(uint32_t reserve_size) override {
|
||||||
// FIXME: ensure no recursive writes can happen
|
// FIXME: ensure no recursive writes can happen
|
||||||
this->proto_write_buffer_.clear();
|
|
||||||
// Get header padding size - used for both reserve and insert
|
// Get header padding size - used for both reserve and insert
|
||||||
uint8_t header_padding = this->helper_->frame_header_padding();
|
uint8_t header_padding = this->helper_->frame_header_padding();
|
||||||
|
|
||||||
|
// Get shared buffer from parent server
|
||||||
|
std::vector<uint8_t> &shared_buf = this->parent_->get_shared_buffer_ref();
|
||||||
|
shared_buf.clear();
|
||||||
// Reserve space for header padding + message + footer
|
// Reserve space for header padding + message + footer
|
||||||
// - Header padding: space for protocol headers (7 bytes for Noise, 6 for Plaintext)
|
// - Header padding: space for protocol headers (7 bytes for Noise, 6 for Plaintext)
|
||||||
// - Footer: space for MAC (16 bytes for Noise, 0 for Plaintext)
|
// - Footer: space for MAC (16 bytes for Noise, 0 for Plaintext)
|
||||||
this->proto_write_buffer_.reserve(reserve_size + header_padding + this->helper_->frame_footer_size());
|
shared_buf.reserve(reserve_size + header_padding + this->helper_->frame_footer_size());
|
||||||
// Insert header padding bytes so message encoding starts at the correct position
|
// Resize to add header padding so message encoding starts at the correct position
|
||||||
this->proto_write_buffer_.insert(this->proto_write_buffer_.begin(), header_padding, 0);
|
shared_buf.resize(header_padding);
|
||||||
return {&this->proto_write_buffer_};
|
return {&shared_buf};
|
||||||
}
|
}
|
||||||
bool try_to_clear_buffer(bool log_out_of_space);
|
|
||||||
bool send_buffer(ProtoWriteBuffer buffer, uint32_t message_type) override;
|
|
||||||
|
|
||||||
std::string get_client_combined_info() const { return this->client_combined_info_; }
|
// Prepare buffer for next message in batch
|
||||||
|
ProtoWriteBuffer prepare_message_buffer(uint16_t message_size, bool is_first_message) {
|
||||||
|
// Get reference to shared buffer (it maintains state between batch messages)
|
||||||
|
std::vector<uint8_t> &shared_buf = this->parent_->get_shared_buffer_ref();
|
||||||
|
|
||||||
|
if (is_first_message) {
|
||||||
|
shared_buf.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t current_size = shared_buf.size();
|
||||||
|
|
||||||
|
// Calculate padding to add:
|
||||||
|
// - First message: just header padding
|
||||||
|
// - Subsequent messages: footer for previous message + header padding for this message
|
||||||
|
size_t padding_to_add = is_first_message
|
||||||
|
? this->helper_->frame_header_padding()
|
||||||
|
: this->helper_->frame_header_padding() + this->helper_->frame_footer_size();
|
||||||
|
|
||||||
|
// Reserve space for padding + message
|
||||||
|
shared_buf.reserve(current_size + padding_to_add + message_size);
|
||||||
|
|
||||||
|
// Resize to add the padding bytes
|
||||||
|
shared_buf.resize(current_size + padding_to_add);
|
||||||
|
|
||||||
|
return {&shared_buf};
|
||||||
|
}
|
||||||
|
|
||||||
|
bool try_to_clear_buffer(bool log_out_of_space);
|
||||||
|
bool send_buffer(ProtoWriteBuffer buffer, uint16_t message_type) override;
|
||||||
|
|
||||||
|
std::string get_client_combined_info() const {
|
||||||
|
if (this->client_info_ == this->client_peername_) {
|
||||||
|
// Before Hello message, both are the same (just IP:port)
|
||||||
|
return this->client_info_;
|
||||||
|
}
|
||||||
|
return this->client_info_ + " (" + this->client_peername_ + ")";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Buffer allocator methods for batch processing
|
||||||
|
ProtoWriteBuffer allocate_single_message_buffer(uint16_t size);
|
||||||
|
ProtoWriteBuffer allocate_batch_message_buffer(uint16_t size);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
friend APIServer;
|
// Helper function to fill common entity info fields
|
||||||
|
static void fill_entity_info_base(esphome::EntityBase *entity, InfoResponseProtoMessage &response) {
|
||||||
/**
|
|
||||||
* Generic send entity state method to reduce code duplication.
|
|
||||||
* Only attempts to build and send the message if the transmit buffer is available.
|
|
||||||
*
|
|
||||||
* This is the base version for entities that use their current state.
|
|
||||||
*
|
|
||||||
* @param entity The entity to send state for
|
|
||||||
* @param try_send_func The function that tries to send the state
|
|
||||||
* @return True on success or message deferred, false if subscription check failed
|
|
||||||
*/
|
|
||||||
bool send_state_(esphome::EntityBase *entity, send_message_t try_send_func) {
|
|
||||||
if (!this->state_subscription_)
|
|
||||||
return false;
|
|
||||||
if (this->try_to_clear_buffer(true) && (this->*try_send_func)(entity)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
this->deferred_message_queue_.defer(entity, try_send_func);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Send entity state method that handles explicit state values.
|
|
||||||
* Only attempts to build and send the message if the transmit buffer is available.
|
|
||||||
*
|
|
||||||
* This method accepts a state parameter to be used instead of the entity's current state.
|
|
||||||
* It attempts to send the state with the provided value first, and if that fails due to buffer constraints,
|
|
||||||
* it defers the entity for later processing using the entity-only function.
|
|
||||||
*
|
|
||||||
* @tparam EntityT The entity type
|
|
||||||
* @tparam StateT Type of the state parameter
|
|
||||||
* @tparam Args Additional argument types (if any)
|
|
||||||
* @param entity The entity to send state for
|
|
||||||
* @param try_send_entity_func The function that tries to send the state with entity pointer only
|
|
||||||
* @param try_send_state_func The function that tries to send the state with entity and state parameters
|
|
||||||
* @param state The state value to send
|
|
||||||
* @param args Additional arguments to pass to the try_send_state_func
|
|
||||||
* @return True on success or message deferred, false if subscription check failed
|
|
||||||
*/
|
|
||||||
template<typename EntityT, typename StateT, typename... Args>
|
|
||||||
bool send_state_with_value_(EntityT *entity, bool (APIConnection::*try_send_entity_func)(EntityT *),
|
|
||||||
bool (APIConnection::*try_send_state_func)(EntityT *, StateT, Args...), StateT state,
|
|
||||||
Args... args) {
|
|
||||||
if (!this->state_subscription_)
|
|
||||||
return false;
|
|
||||||
if (this->try_to_clear_buffer(true) && (this->*try_send_state_func)(entity, state, args...)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
this->deferred_message_queue_.defer(entity, reinterpret_cast<send_message_t>(try_send_entity_func));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generic send entity info method to reduce code duplication.
|
|
||||||
* Only attempts to build and send the message if the transmit buffer is available.
|
|
||||||
*
|
|
||||||
* @param entity The entity to send info for
|
|
||||||
* @param try_send_func The function that tries to send the info
|
|
||||||
*/
|
|
||||||
void send_info_(esphome::EntityBase *entity, send_message_t try_send_func) {
|
|
||||||
if (this->try_to_clear_buffer(true) && (this->*try_send_func)(entity)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this->deferred_message_queue_.defer(entity, try_send_func);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generic function for generating entity info response messages.
|
|
||||||
* This is used to reduce duplication in the try_send_*_info functions.
|
|
||||||
*
|
|
||||||
* @param entity The entity to generate info for
|
|
||||||
* @param response The response object
|
|
||||||
* @param send_response_func Function pointer to send the response
|
|
||||||
* @return True if the message was sent successfully
|
|
||||||
*/
|
|
||||||
template<typename ResponseT>
|
|
||||||
bool try_send_entity_info_(esphome::EntityBase *entity, ResponseT &response,
|
|
||||||
bool (APIServerConnectionBase::*send_response_func)(const ResponseT &)) {
|
|
||||||
// Set common fields that are shared by all entity types
|
// Set common fields that are shared by all entity types
|
||||||
response.key = entity->get_object_id_hash();
|
response.key = entity->get_object_id_hash();
|
||||||
response.object_id = entity->get_object_id();
|
response.object_id = entity->get_object_id();
|
||||||
@@ -514,48 +284,416 @@ class APIConnection : public APIServerConnection {
|
|||||||
response.icon = entity->get_icon();
|
response.icon = entity->get_icon();
|
||||||
response.disabled_by_default = entity->is_disabled_by_default();
|
response.disabled_by_default = entity->is_disabled_by_default();
|
||||||
response.entity_category = static_cast<enums::EntityCategory>(entity->get_entity_category());
|
response.entity_category = static_cast<enums::EntityCategory>(entity->get_entity_category());
|
||||||
|
#ifdef USE_DEVICES
|
||||||
// Send the response using the provided send method
|
response.device_id = entity->get_device_id();
|
||||||
return (this->*send_response_func)(response);
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
bool send_(const void *buf, size_t len, bool force);
|
// Helper function to fill common entity state fields
|
||||||
|
static void fill_entity_state_base(esphome::EntityBase *entity, StateResponseProtoMessage &response) {
|
||||||
|
response.key = entity->get_object_id_hash();
|
||||||
|
#ifdef USE_DEVICES
|
||||||
|
response.device_id = entity->get_device_id();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
enum class ConnectionState {
|
// Non-template helper to encode any ProtoMessage
|
||||||
WAITING_FOR_HELLO,
|
static uint16_t encode_message_to_buffer(ProtoMessage &msg, uint16_t message_type, APIConnection *conn,
|
||||||
CONNECTED,
|
uint32_t remaining_size, bool is_single);
|
||||||
AUTHENTICATED,
|
|
||||||
} connection_state_{ConnectionState::WAITING_FOR_HELLO};
|
|
||||||
|
|
||||||
bool remove_{false};
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
|
// Helper to check voice assistant validity and connection ownership
|
||||||
// Buffer used to encode proto messages
|
inline bool check_voice_assistant_api_connection_() const;
|
||||||
// Re-use to prevent allocations
|
|
||||||
std::vector<uint8_t> proto_write_buffer_;
|
|
||||||
std::unique_ptr<APIFrameHelper> helper_;
|
|
||||||
|
|
||||||
std::string client_info_;
|
|
||||||
std::string client_peername_;
|
|
||||||
std::string client_combined_info_;
|
|
||||||
uint32_t client_api_version_major_{0};
|
|
||||||
uint32_t client_api_version_minor_{0};
|
|
||||||
#ifdef USE_ESP32_CAMERA
|
|
||||||
esp32_camera::CameraImageReader image_reader_;
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
bool state_subscription_{false};
|
// Helper method to process multiple entities from an iterator in a batch
|
||||||
int log_subscription_{ESPHOME_LOG_LEVEL_NONE};
|
template<typename Iterator> void process_iterator_batch_(Iterator &iterator) {
|
||||||
uint32_t last_traffic_;
|
size_t initial_size = this->deferred_batch_.size();
|
||||||
uint32_t next_ping_retry_{0};
|
while (!iterator.completed() && (this->deferred_batch_.size() - initial_size) < MAX_INITIAL_PER_BATCH) {
|
||||||
uint8_t ping_retries_{0};
|
iterator.advance();
|
||||||
bool sent_ping_{false};
|
}
|
||||||
bool service_call_subscription_{false};
|
|
||||||
bool next_close_ = false;
|
// If the batch is full, process it immediately
|
||||||
|
// Note: iterator.advance() already calls schedule_batch_() via schedule_message_()
|
||||||
|
if (this->deferred_batch_.size() >= MAX_INITIAL_PER_BATCH) {
|
||||||
|
this->process_batch_();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef USE_BINARY_SENSOR
|
||||||
|
static uint16_t try_send_binary_sensor_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_binary_sensor_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_COVER
|
||||||
|
static uint16_t try_send_cover_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_cover_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_FAN
|
||||||
|
static uint16_t try_send_fan_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
static uint16_t try_send_fan_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_LIGHT
|
||||||
|
static uint16_t try_send_light_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_light_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_SENSOR
|
||||||
|
static uint16_t try_send_sensor_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_sensor_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_SWITCH
|
||||||
|
static uint16_t try_send_switch_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_switch_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_TEXT_SENSOR
|
||||||
|
static uint16_t try_send_text_sensor_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_text_sensor_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_CLIMATE
|
||||||
|
static uint16_t try_send_climate_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_climate_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_NUMBER
|
||||||
|
static uint16_t try_send_number_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_number_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_DATETIME_DATE
|
||||||
|
static uint16_t try_send_date_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
static uint16_t try_send_date_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_DATETIME_TIME
|
||||||
|
static uint16_t try_send_time_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
static uint16_t try_send_time_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_DATETIME_DATETIME
|
||||||
|
static uint16_t try_send_datetime_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_datetime_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_TEXT
|
||||||
|
static uint16_t try_send_text_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
static uint16_t try_send_text_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_SELECT
|
||||||
|
static uint16_t try_send_select_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_select_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_BUTTON
|
||||||
|
static uint16_t try_send_button_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_LOCK
|
||||||
|
static uint16_t try_send_lock_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
static uint16_t try_send_lock_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_VALVE
|
||||||
|
static uint16_t try_send_valve_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_valve_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_MEDIA_PLAYER
|
||||||
|
static uint16_t try_send_media_player_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_media_player_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
|
static uint16_t try_send_alarm_control_panel_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_alarm_control_panel_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_EVENT
|
||||||
|
static uint16_t try_send_event_response(event::Event *event, const std::string &event_type, APIConnection *conn,
|
||||||
|
uint32_t remaining_size, bool is_single);
|
||||||
|
static uint16_t try_send_event_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_UPDATE
|
||||||
|
static uint16_t try_send_update_state(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
static uint16_t try_send_update_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
#ifdef USE_CAMERA
|
||||||
|
static uint16_t try_send_camera_info(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Method for ListEntitiesDone batching
|
||||||
|
static uint16_t try_send_list_info_done(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
|
||||||
|
// Method for DisconnectRequest batching
|
||||||
|
static uint16_t try_send_disconnect_request(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
|
||||||
|
// Helper function to get estimated message size for buffer pre-allocation
|
||||||
|
static uint16_t get_estimated_message_size(uint16_t message_type);
|
||||||
|
|
||||||
|
// Batch message method for ping requests
|
||||||
|
static uint16_t try_send_ping_request(EntityBase *entity, APIConnection *conn, uint32_t remaining_size,
|
||||||
|
bool is_single);
|
||||||
|
|
||||||
|
// === Optimal member ordering for 32-bit systems ===
|
||||||
|
|
||||||
|
// Group 1: Pointers (4 bytes each on 32-bit)
|
||||||
|
std::unique_ptr<APIFrameHelper> helper_;
|
||||||
APIServer *parent_;
|
APIServer *parent_;
|
||||||
DeferredMessageQueue deferred_message_queue_;
|
|
||||||
|
// Group 2: Larger objects (must be 4-byte aligned)
|
||||||
|
// These contain vectors/pointers internally, so putting them early ensures good alignment
|
||||||
InitialStateIterator initial_state_iterator_;
|
InitialStateIterator initial_state_iterator_;
|
||||||
ListEntitiesIterator list_entities_iterator_;
|
ListEntitiesIterator list_entities_iterator_;
|
||||||
|
#ifdef USE_CAMERA
|
||||||
|
std::unique_ptr<camera::CameraImageReader> image_reader_;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Group 3: Strings (12 bytes each on 32-bit, 4-byte aligned)
|
||||||
|
std::string client_info_;
|
||||||
|
std::string client_peername_;
|
||||||
|
|
||||||
|
// Group 4: 4-byte types
|
||||||
|
uint32_t last_traffic_;
|
||||||
int state_subs_at_ = -1;
|
int state_subs_at_ = -1;
|
||||||
|
|
||||||
|
// Function pointer type for message encoding
|
||||||
|
using MessageCreatorPtr = uint16_t (*)(EntityBase *, APIConnection *, uint32_t remaining_size, bool is_single);
|
||||||
|
|
||||||
|
class MessageCreator {
|
||||||
|
public:
|
||||||
|
// Constructor for function pointer
|
||||||
|
MessageCreator(MessageCreatorPtr ptr) { data_.function_ptr = ptr; }
|
||||||
|
|
||||||
|
// Constructor for string state capture
|
||||||
|
explicit MessageCreator(const std::string &str_value) { data_.string_ptr = new std::string(str_value); }
|
||||||
|
|
||||||
|
// No destructor - cleanup must be called explicitly with message_type
|
||||||
|
|
||||||
|
// Delete copy operations - MessageCreator should only be moved
|
||||||
|
MessageCreator(const MessageCreator &other) = delete;
|
||||||
|
MessageCreator &operator=(const MessageCreator &other) = delete;
|
||||||
|
|
||||||
|
// Move constructor
|
||||||
|
MessageCreator(MessageCreator &&other) noexcept : data_(other.data_) { other.data_.function_ptr = nullptr; }
|
||||||
|
|
||||||
|
// Move assignment
|
||||||
|
MessageCreator &operator=(MessageCreator &&other) noexcept {
|
||||||
|
if (this != &other) {
|
||||||
|
// IMPORTANT: Caller must ensure cleanup() was called if this contains a string!
|
||||||
|
// In our usage, this happens in add_item() deduplication and vector::erase()
|
||||||
|
data_ = other.data_;
|
||||||
|
other.data_.function_ptr = nullptr;
|
||||||
|
}
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call operator - uses message_type to determine union type
|
||||||
|
uint16_t operator()(EntityBase *entity, APIConnection *conn, uint32_t remaining_size, bool is_single,
|
||||||
|
uint16_t message_type) const;
|
||||||
|
|
||||||
|
// Manual cleanup method - must be called before destruction for string types
|
||||||
|
void cleanup(uint16_t message_type) {
|
||||||
|
#ifdef USE_EVENT
|
||||||
|
if (message_type == EventResponse::MESSAGE_TYPE && data_.string_ptr != nullptr) {
|
||||||
|
delete data_.string_ptr;
|
||||||
|
data_.string_ptr = nullptr;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
union Data {
|
||||||
|
MessageCreatorPtr function_ptr;
|
||||||
|
std::string *string_ptr;
|
||||||
|
} data_; // 4 bytes on 32-bit, 8 bytes on 64-bit - same as before
|
||||||
|
};
|
||||||
|
|
||||||
|
// Generic batching mechanism for both state updates and entity info
|
||||||
|
struct DeferredBatch {
|
||||||
|
struct BatchItem {
|
||||||
|
EntityBase *entity; // Entity pointer
|
||||||
|
MessageCreator creator; // Function that creates the message when needed
|
||||||
|
uint16_t message_type; // Message type for overhead calculation
|
||||||
|
|
||||||
|
// Constructor for creating BatchItem
|
||||||
|
BatchItem(EntityBase *entity, MessageCreator creator, uint16_t message_type)
|
||||||
|
: entity(entity), creator(std::move(creator)), message_type(message_type) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
std::vector<BatchItem> items;
|
||||||
|
uint32_t batch_start_time{0};
|
||||||
|
|
||||||
|
private:
|
||||||
|
// Helper to cleanup items from the beginning
|
||||||
|
void cleanup_items_(size_t count) {
|
||||||
|
for (size_t i = 0; i < count; i++) {
|
||||||
|
items[i].creator.cleanup(items[i].message_type);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
DeferredBatch() {
|
||||||
|
// Pre-allocate capacity for typical batch sizes to avoid reallocation
|
||||||
|
items.reserve(8);
|
||||||
|
}
|
||||||
|
|
||||||
|
~DeferredBatch() {
|
||||||
|
// Ensure cleanup of any remaining items
|
||||||
|
clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add item to the batch
|
||||||
|
void add_item(EntityBase *entity, MessageCreator creator, uint16_t message_type);
|
||||||
|
// Add item to the front of the batch (for high priority messages like ping)
|
||||||
|
void add_item_front(EntityBase *entity, MessageCreator creator, uint16_t message_type);
|
||||||
|
|
||||||
|
// Clear all items with proper cleanup
|
||||||
|
void clear() {
|
||||||
|
cleanup_items_(items.size());
|
||||||
|
items.clear();
|
||||||
|
batch_start_time = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove processed items from the front with proper cleanup
|
||||||
|
void remove_front(size_t count) {
|
||||||
|
cleanup_items_(count);
|
||||||
|
items.erase(items.begin(), items.begin() + count);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool empty() const { return items.empty(); }
|
||||||
|
size_t size() const { return items.size(); }
|
||||||
|
const BatchItem &operator[](size_t index) const { return items[index]; }
|
||||||
|
};
|
||||||
|
|
||||||
|
// DeferredBatch here (16 bytes, 4-byte aligned)
|
||||||
|
DeferredBatch deferred_batch_;
|
||||||
|
|
||||||
|
// ConnectionState enum for type safety
|
||||||
|
enum class ConnectionState : uint8_t {
|
||||||
|
WAITING_FOR_HELLO = 0,
|
||||||
|
CONNECTED = 1,
|
||||||
|
AUTHENTICATED = 2,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Group 5: Pack all small members together to minimize padding
|
||||||
|
// This group starts at a 4-byte boundary after DeferredBatch
|
||||||
|
struct APIFlags {
|
||||||
|
// Connection state only needs 2 bits (3 states)
|
||||||
|
uint8_t connection_state : 2;
|
||||||
|
// Log subscription needs 3 bits (log levels 0-7)
|
||||||
|
uint8_t log_subscription : 3;
|
||||||
|
// Boolean flags (1 bit each)
|
||||||
|
uint8_t remove : 1;
|
||||||
|
uint8_t state_subscription : 1;
|
||||||
|
uint8_t sent_ping : 1;
|
||||||
|
|
||||||
|
uint8_t service_call_subscription : 1;
|
||||||
|
uint8_t next_close : 1;
|
||||||
|
uint8_t batch_scheduled : 1;
|
||||||
|
uint8_t batch_first_message : 1; // For batch buffer allocation
|
||||||
|
uint8_t should_try_send_immediately : 1; // True after initial states are sent
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
uint8_t log_only_mode : 1;
|
||||||
|
#endif
|
||||||
|
} flags_{}; // 2 bytes total
|
||||||
|
|
||||||
|
// 2-byte types immediately after flags_ (no padding between them)
|
||||||
|
uint16_t client_api_version_major_{0};
|
||||||
|
uint16_t client_api_version_minor_{0};
|
||||||
|
// Total: 2 (flags) + 2 + 2 = 6 bytes, then 2 bytes padding to next 4-byte boundary
|
||||||
|
|
||||||
|
uint32_t get_batch_delay_ms_() const;
|
||||||
|
// Message will use 8 more bytes than the minimum size, and typical
|
||||||
|
// MTU is 1500. Sometimes users will see as low as 1460 MTU.
|
||||||
|
// If its IPv6 the header is 40 bytes, and if its IPv4
|
||||||
|
// the header is 20 bytes. So we have 1460 - 40 = 1420 bytes
|
||||||
|
// available for the payload. But we also need to add the size of
|
||||||
|
// the protobuf overhead, which is 8 bytes.
|
||||||
|
//
|
||||||
|
// To be safe we pick 1390 bytes as the maximum size
|
||||||
|
// to send in one go. This is the maximum size of a single packet
|
||||||
|
// that can be sent over the network.
|
||||||
|
// This is to avoid fragmentation of the packet.
|
||||||
|
static constexpr size_t MAX_PACKET_SIZE = 1390; // MTU
|
||||||
|
|
||||||
|
bool schedule_batch_();
|
||||||
|
void process_batch_();
|
||||||
|
void clear_batch_() {
|
||||||
|
this->deferred_batch_.clear();
|
||||||
|
this->flags_.batch_scheduled = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
// Helper to log a proto message from a MessageCreator object
|
||||||
|
void log_proto_message_(EntityBase *entity, const MessageCreator &creator, uint16_t message_type) {
|
||||||
|
this->flags_.log_only_mode = true;
|
||||||
|
creator(entity, this, MAX_PACKET_SIZE, true, message_type);
|
||||||
|
this->flags_.log_only_mode = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void log_batch_item_(const DeferredBatch::BatchItem &item) {
|
||||||
|
// Use the helper to log the message
|
||||||
|
this->log_proto_message_(item.entity, item.creator, item.message_type);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Helper method to send a message either immediately or via batching
|
||||||
|
bool send_message_smart_(EntityBase *entity, MessageCreatorPtr creator, uint16_t message_type) {
|
||||||
|
// Try to send immediately if:
|
||||||
|
// 1. We should try to send immediately (should_try_send_immediately = true)
|
||||||
|
// 2. Batch delay is 0 (user has opted in to immediate sending)
|
||||||
|
// 3. Buffer has space available
|
||||||
|
if (this->flags_.should_try_send_immediately && this->get_batch_delay_ms_() == 0 &&
|
||||||
|
this->helper_->can_write_without_blocking()) {
|
||||||
|
// Now actually encode and send
|
||||||
|
if (creator(entity, this, MAX_PACKET_SIZE, true) &&
|
||||||
|
this->send_buffer(ProtoWriteBuffer{&this->parent_->get_shared_buffer_ref()}, message_type)) {
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
// Log the message in verbose mode
|
||||||
|
this->log_proto_message_(entity, MessageCreator(creator), message_type);
|
||||||
|
#endif
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If immediate send failed, fall through to batching
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fall back to scheduled batching
|
||||||
|
return this->schedule_message_(entity, creator, message_type);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper function to schedule a deferred message with known message type
|
||||||
|
bool schedule_message_(EntityBase *entity, MessageCreator creator, uint16_t message_type) {
|
||||||
|
this->deferred_batch_.add_item(entity, std::move(creator), message_type);
|
||||||
|
return this->schedule_batch_();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Overload for function pointers (for info messages and current state reads)
|
||||||
|
bool schedule_message_(EntityBase *entity, MessageCreatorPtr function_ptr, uint16_t message_type) {
|
||||||
|
return schedule_message_(entity, MessageCreator(function_ptr), message_type);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper function to schedule a high priority message at the front of the batch
|
||||||
|
bool schedule_message_front_(EntityBase *entity, MessageCreatorPtr function_ptr, uint16_t message_type) {
|
||||||
|
this->deferred_batch_.add_item_front(entity, MessageCreator(function_ptr), message_type);
|
||||||
|
return this->schedule_batch_();
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
#include "api_frame_helper.h"
|
#include "api_frame_helper.h"
|
||||||
#ifdef USE_API
|
#ifdef USE_API
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/application.h"
|
||||||
#include "esphome/core/hal.h"
|
#include "esphome/core/hal.h"
|
||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
#include "esphome/core/application.h"
|
#include "esphome/core/log.h"
|
||||||
#include "proto.h"
|
#include "proto.h"
|
||||||
#include "api_pb2_size.h"
|
#include "api_pb2_size.h"
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
@@ -66,6 +66,17 @@ const char *api_error_to_str(APIError err) {
|
|||||||
return "UNKNOWN";
|
return "UNKNOWN";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Default implementation for loop - handles sending buffered data
|
||||||
|
APIError APIFrameHelper::loop() {
|
||||||
|
if (!this->tx_buf_.empty()) {
|
||||||
|
APIError err = try_send_tx_buf_();
|
||||||
|
if (err != APIError::OK && err != APIError::WOULD_BLOCK) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return APIError::OK; // Convert WOULD_BLOCK to OK to avoid connection termination
|
||||||
|
}
|
||||||
|
|
||||||
// Helper method to buffer data from IOVs
|
// Helper method to buffer data from IOVs
|
||||||
void APIFrameHelper::buffer_data_from_iov_(const struct iovec *iov, int iovcnt, uint16_t total_write_len) {
|
void APIFrameHelper::buffer_data_from_iov_(const struct iovec *iov, int iovcnt, uint16_t total_write_len) {
|
||||||
SendBuffer buffer;
|
SendBuffer buffer;
|
||||||
@@ -214,6 +225,22 @@ APIError APIFrameHelper::init_common_() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#define HELPER_LOG(msg, ...) ESP_LOGVV(TAG, "%s: " msg, this->info_.c_str(), ##__VA_ARGS__)
|
#define HELPER_LOG(msg, ...) ESP_LOGVV(TAG, "%s: " msg, this->info_.c_str(), ##__VA_ARGS__)
|
||||||
|
|
||||||
|
APIError APIFrameHelper::handle_socket_read_result_(ssize_t received) {
|
||||||
|
if (received == -1) {
|
||||||
|
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
||||||
|
return APIError::WOULD_BLOCK;
|
||||||
|
}
|
||||||
|
state_ = State::FAILED;
|
||||||
|
HELPER_LOG("Socket read failed with errno %d", errno);
|
||||||
|
return APIError::SOCKET_READ_FAILED;
|
||||||
|
} else if (received == 0) {
|
||||||
|
state_ = State::FAILED;
|
||||||
|
HELPER_LOG("Connection closed");
|
||||||
|
return APIError::CONNECTION_CLOSED;
|
||||||
|
}
|
||||||
|
return APIError::OK;
|
||||||
|
}
|
||||||
// uncomment to log raw packets
|
// uncomment to log raw packets
|
||||||
//#define HELPER_LOG_PACKETS
|
//#define HELPER_LOG_PACKETS
|
||||||
|
|
||||||
@@ -274,17 +301,21 @@ APIError APINoiseFrameHelper::init() {
|
|||||||
}
|
}
|
||||||
/// Run through handshake messages (if in that phase)
|
/// Run through handshake messages (if in that phase)
|
||||||
APIError APINoiseFrameHelper::loop() {
|
APIError APINoiseFrameHelper::loop() {
|
||||||
APIError err = state_action_();
|
// During handshake phase, process as many actions as possible until we can't progress
|
||||||
if (err != APIError::OK && err != APIError::WOULD_BLOCK) {
|
// socket_->ready() stays true until next main loop, but state_action() will return
|
||||||
return err;
|
// WOULD_BLOCK when no more data is available to read
|
||||||
}
|
while (state_ != State::DATA && this->socket_->ready()) {
|
||||||
if (!this->tx_buf_.empty()) {
|
APIError err = state_action_();
|
||||||
err = try_send_tx_buf_();
|
|
||||||
if (err != APIError::OK && err != APIError::WOULD_BLOCK) {
|
if (err != APIError::OK && err != APIError::WOULD_BLOCK) {
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
if (err == APIError::WOULD_BLOCK) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return APIError::OK; // Convert WOULD_BLOCK to OK to avoid connection termination
|
|
||||||
|
// Use base class implementation for buffer sending
|
||||||
|
return APIFrameHelper::loop();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Read a packet into the rx_buf_. If successful, stores frame data in the frame parameter
|
/** Read a packet into the rx_buf_. If successful, stores frame data in the frame parameter
|
||||||
@@ -312,17 +343,9 @@ APIError APINoiseFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
// no header information yet
|
// no header information yet
|
||||||
uint8_t to_read = 3 - rx_header_buf_len_;
|
uint8_t to_read = 3 - rx_header_buf_len_;
|
||||||
ssize_t received = this->socket_->read(&rx_header_buf_[rx_header_buf_len_], to_read);
|
ssize_t received = this->socket_->read(&rx_header_buf_[rx_header_buf_len_], to_read);
|
||||||
if (received == -1) {
|
APIError err = handle_socket_read_result_(received);
|
||||||
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
if (err != APIError::OK) {
|
||||||
return APIError::WOULD_BLOCK;
|
return err;
|
||||||
}
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Socket read failed with errno %d", errno);
|
|
||||||
return APIError::SOCKET_READ_FAILED;
|
|
||||||
} else if (received == 0) {
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Connection closed");
|
|
||||||
return APIError::CONNECTION_CLOSED;
|
|
||||||
}
|
}
|
||||||
rx_header_buf_len_ += static_cast<uint8_t>(received);
|
rx_header_buf_len_ += static_cast<uint8_t>(received);
|
||||||
if (static_cast<uint8_t>(received) != to_read) {
|
if (static_cast<uint8_t>(received) != to_read) {
|
||||||
@@ -330,17 +353,15 @@ APIError APINoiseFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
return APIError::WOULD_BLOCK;
|
return APIError::WOULD_BLOCK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (rx_header_buf_[0] != 0x01) {
|
||||||
|
state_ = State::FAILED;
|
||||||
|
HELPER_LOG("Bad indicator byte %u", rx_header_buf_[0]);
|
||||||
|
return APIError::BAD_INDICATOR;
|
||||||
|
}
|
||||||
// header reading done
|
// header reading done
|
||||||
}
|
}
|
||||||
|
|
||||||
// read body
|
// read body
|
||||||
uint8_t indicator = rx_header_buf_[0];
|
|
||||||
if (indicator != 0x01) {
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Bad indicator byte %u", indicator);
|
|
||||||
return APIError::BAD_INDICATOR;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint16_t msg_size = (((uint16_t) rx_header_buf_[1]) << 8) | rx_header_buf_[2];
|
uint16_t msg_size = (((uint16_t) rx_header_buf_[1]) << 8) | rx_header_buf_[2];
|
||||||
|
|
||||||
if (state_ != State::DATA && msg_size > 128) {
|
if (state_ != State::DATA && msg_size > 128) {
|
||||||
@@ -359,17 +380,9 @@ APIError APINoiseFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
// more data to read
|
// more data to read
|
||||||
uint16_t to_read = msg_size - rx_buf_len_;
|
uint16_t to_read = msg_size - rx_buf_len_;
|
||||||
ssize_t received = this->socket_->read(&rx_buf_[rx_buf_len_], to_read);
|
ssize_t received = this->socket_->read(&rx_buf_[rx_buf_len_], to_read);
|
||||||
if (received == -1) {
|
APIError err = handle_socket_read_result_(received);
|
||||||
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
if (err != APIError::OK) {
|
||||||
return APIError::WOULD_BLOCK;
|
return err;
|
||||||
}
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Socket read failed with errno %d", errno);
|
|
||||||
return APIError::SOCKET_READ_FAILED;
|
|
||||||
} else if (received == 0) {
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Connection closed");
|
|
||||||
return APIError::CONNECTION_CLOSED;
|
|
||||||
}
|
}
|
||||||
rx_buf_len_ += static_cast<uint16_t>(received);
|
rx_buf_len_ += static_cast<uint16_t>(received);
|
||||||
if (static_cast<uint16_t>(received) != to_read) {
|
if (static_cast<uint16_t>(received) != to_read) {
|
||||||
@@ -586,10 +599,6 @@ APIError APINoiseFrameHelper::read_packet(ReadPacketBuffer *buffer) {
|
|||||||
return APIError::BAD_DATA_PACKET;
|
return APIError::BAD_DATA_PACKET;
|
||||||
}
|
}
|
||||||
|
|
||||||
// uint16_t type;
|
|
||||||
// uint16_t data_len;
|
|
||||||
// uint8_t *data;
|
|
||||||
// uint8_t *padding; zero or more bytes to fill up the rest of the packet
|
|
||||||
uint16_t type = (((uint16_t) msg_data[0]) << 8) | msg_data[1];
|
uint16_t type = (((uint16_t) msg_data[0]) << 8) | msg_data[1];
|
||||||
uint16_t data_len = (((uint16_t) msg_data[2]) << 8) | msg_data[3];
|
uint16_t data_len = (((uint16_t) msg_data[2]) << 8) | msg_data[3];
|
||||||
if (data_len > msg_size - 4) {
|
if (data_len > msg_size - 4) {
|
||||||
@@ -605,9 +614,15 @@ APIError APINoiseFrameHelper::read_packet(ReadPacketBuffer *buffer) {
|
|||||||
return APIError::OK;
|
return APIError::OK;
|
||||||
}
|
}
|
||||||
APIError APINoiseFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) {
|
APIError APINoiseFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) {
|
||||||
int err;
|
// Resize to include MAC space (required for Noise encryption)
|
||||||
APIError aerr;
|
buffer.get_buffer()->resize(buffer.get_buffer()->size() + frame_footer_size_);
|
||||||
aerr = state_action_();
|
PacketInfo packet{type, 0,
|
||||||
|
static_cast<uint16_t>(buffer.get_buffer()->size() - frame_header_padding_ - frame_footer_size_)};
|
||||||
|
return write_protobuf_packets(buffer, std::span<const PacketInfo>(&packet, 1));
|
||||||
|
}
|
||||||
|
|
||||||
|
APIError APINoiseFrameHelper::write_protobuf_packets(ProtoWriteBuffer buffer, std::span<const PacketInfo> packets) {
|
||||||
|
APIError aerr = state_action_();
|
||||||
if (aerr != APIError::OK) {
|
if (aerr != APIError::OK) {
|
||||||
return aerr;
|
return aerr;
|
||||||
}
|
}
|
||||||
@@ -616,56 +631,62 @@ APIError APINoiseFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuf
|
|||||||
return APIError::WOULD_BLOCK;
|
return APIError::WOULD_BLOCK;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<uint8_t> *raw_buffer = buffer.get_buffer();
|
if (packets.empty()) {
|
||||||
// Message data starts after padding
|
return APIError::OK;
|
||||||
uint16_t payload_len = raw_buffer->size() - frame_header_padding_;
|
|
||||||
uint16_t padding = 0;
|
|
||||||
uint16_t msg_len = 4 + payload_len + padding;
|
|
||||||
|
|
||||||
// We need to resize to include MAC space, but we already reserved it in create_buffer
|
|
||||||
raw_buffer->resize(raw_buffer->size() + frame_footer_size_);
|
|
||||||
|
|
||||||
// Write the noise header in the padded area
|
|
||||||
// Buffer layout:
|
|
||||||
// [0] - 0x01 indicator byte
|
|
||||||
// [1-2] - Size of encrypted payload (filled after encryption)
|
|
||||||
// [3-4] - Message type (encrypted)
|
|
||||||
// [5-6] - Payload length (encrypted)
|
|
||||||
// [7...] - Actual payload data (encrypted)
|
|
||||||
uint8_t *buf_start = raw_buffer->data();
|
|
||||||
buf_start[0] = 0x01; // indicator
|
|
||||||
// buf_start[1], buf_start[2] to be set later after encryption
|
|
||||||
const uint8_t msg_offset = 3;
|
|
||||||
buf_start[msg_offset + 0] = (uint8_t) (type >> 8); // type high byte
|
|
||||||
buf_start[msg_offset + 1] = (uint8_t) type; // type low byte
|
|
||||||
buf_start[msg_offset + 2] = (uint8_t) (payload_len >> 8); // data_len high byte
|
|
||||||
buf_start[msg_offset + 3] = (uint8_t) payload_len; // data_len low byte
|
|
||||||
// payload data is already in the buffer starting at position 7
|
|
||||||
|
|
||||||
NoiseBuffer mbuf;
|
|
||||||
noise_buffer_init(mbuf);
|
|
||||||
// The capacity parameter should be msg_len + frame_footer_size_ (MAC length) to allow space for encryption
|
|
||||||
noise_buffer_set_inout(mbuf, buf_start + msg_offset, msg_len, msg_len + frame_footer_size_);
|
|
||||||
err = noise_cipherstate_encrypt(send_cipher_, &mbuf);
|
|
||||||
if (err != 0) {
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("noise_cipherstate_encrypt failed: %s", noise_err_to_str(err).c_str());
|
|
||||||
return APIError::CIPHERSTATE_ENCRYPT_FAILED;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
uint16_t total_len = 3 + mbuf.size;
|
std::vector<uint8_t> *raw_buffer = buffer.get_buffer();
|
||||||
buf_start[1] = (uint8_t) (mbuf.size >> 8);
|
uint8_t *buffer_data = raw_buffer->data(); // Cache buffer pointer
|
||||||
buf_start[2] = (uint8_t) mbuf.size;
|
|
||||||
|
|
||||||
struct iovec iov;
|
this->reusable_iovs_.clear();
|
||||||
// Point iov_base to the beginning of the buffer (no unused padding in Noise)
|
this->reusable_iovs_.reserve(packets.size());
|
||||||
// We send the entire frame: indicator + size + encrypted(type + data_len + payload + MAC)
|
|
||||||
iov.iov_base = buf_start;
|
|
||||||
iov.iov_len = total_len;
|
|
||||||
|
|
||||||
// write raw to not have two packets sent if NAGLE disabled
|
// We need to encrypt each packet in place
|
||||||
return this->write_raw_(&iov, 1);
|
for (const auto &packet : packets) {
|
||||||
|
// The buffer already has padding at offset
|
||||||
|
uint8_t *buf_start = buffer_data + packet.offset;
|
||||||
|
|
||||||
|
// Write noise header
|
||||||
|
buf_start[0] = 0x01; // indicator
|
||||||
|
// buf_start[1], buf_start[2] to be set after encryption
|
||||||
|
|
||||||
|
// Write message header (to be encrypted)
|
||||||
|
const uint8_t msg_offset = 3;
|
||||||
|
buf_start[msg_offset] = static_cast<uint8_t>(packet.message_type >> 8); // type high byte
|
||||||
|
buf_start[msg_offset + 1] = static_cast<uint8_t>(packet.message_type); // type low byte
|
||||||
|
buf_start[msg_offset + 2] = static_cast<uint8_t>(packet.payload_size >> 8); // data_len high byte
|
||||||
|
buf_start[msg_offset + 3] = static_cast<uint8_t>(packet.payload_size); // data_len low byte
|
||||||
|
// payload data is already in the buffer starting at offset + 7
|
||||||
|
|
||||||
|
// Make sure we have space for MAC
|
||||||
|
// The buffer should already have been sized appropriately
|
||||||
|
|
||||||
|
// Encrypt the message in place
|
||||||
|
NoiseBuffer mbuf;
|
||||||
|
noise_buffer_init(mbuf);
|
||||||
|
noise_buffer_set_inout(mbuf, buf_start + msg_offset, 4 + packet.payload_size,
|
||||||
|
4 + packet.payload_size + frame_footer_size_);
|
||||||
|
|
||||||
|
int err = noise_cipherstate_encrypt(send_cipher_, &mbuf);
|
||||||
|
if (err != 0) {
|
||||||
|
state_ = State::FAILED;
|
||||||
|
HELPER_LOG("noise_cipherstate_encrypt failed: %s", noise_err_to_str(err).c_str());
|
||||||
|
return APIError::CIPHERSTATE_ENCRYPT_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fill in the encrypted size
|
||||||
|
buf_start[1] = static_cast<uint8_t>(mbuf.size >> 8);
|
||||||
|
buf_start[2] = static_cast<uint8_t>(mbuf.size);
|
||||||
|
|
||||||
|
// Add iovec for this encrypted packet
|
||||||
|
this->reusable_iovs_.push_back(
|
||||||
|
{buf_start, static_cast<size_t>(3 + mbuf.size)}); // indicator + size + encrypted data
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send all encrypted packets in one writev call
|
||||||
|
return this->write_raw_(this->reusable_iovs_.data(), this->reusable_iovs_.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
APIError APINoiseFrameHelper::write_frame_(const uint8_t *data, uint16_t len) {
|
APIError APINoiseFrameHelper::write_frame_(const uint8_t *data, uint16_t len) {
|
||||||
uint8_t header[3];
|
uint8_t header[3];
|
||||||
header[0] = 0x01; // indicator
|
header[0] = 0x01; // indicator
|
||||||
@@ -780,7 +801,7 @@ extern "C" {
|
|||||||
// declare how noise generates random bytes (here with a good HWRNG based on the RF system)
|
// declare how noise generates random bytes (here with a good HWRNG based on the RF system)
|
||||||
void noise_rand_bytes(void *output, size_t len) {
|
void noise_rand_bytes(void *output, size_t len) {
|
||||||
if (!esphome::random_bytes(reinterpret_cast<uint8_t *>(output), len)) {
|
if (!esphome::random_bytes(reinterpret_cast<uint8_t *>(output), len)) {
|
||||||
ESP_LOGE(TAG, "Failed to acquire random bytes, rebooting!");
|
ESP_LOGE(TAG, "Acquiring random bytes failed; rebooting");
|
||||||
arch_restart();
|
arch_restart();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -800,18 +821,12 @@ APIError APIPlaintextFrameHelper::init() {
|
|||||||
state_ = State::DATA;
|
state_ = State::DATA;
|
||||||
return APIError::OK;
|
return APIError::OK;
|
||||||
}
|
}
|
||||||
/// Not used for plaintext
|
|
||||||
APIError APIPlaintextFrameHelper::loop() {
|
APIError APIPlaintextFrameHelper::loop() {
|
||||||
if (state_ != State::DATA) {
|
if (state_ != State::DATA) {
|
||||||
return APIError::BAD_STATE;
|
return APIError::BAD_STATE;
|
||||||
}
|
}
|
||||||
if (!this->tx_buf_.empty()) {
|
// Use base class implementation for buffer sending
|
||||||
APIError err = try_send_tx_buf_();
|
return APIFrameHelper::loop();
|
||||||
if (err != APIError::OK && err != APIError::WOULD_BLOCK) {
|
|
||||||
return err;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return APIError::OK; // Convert WOULD_BLOCK to OK to avoid connection termination
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Read a packet into the rx_buf_. If successful, stores frame data in the frame parameter
|
/** Read a packet into the rx_buf_. If successful, stores frame data in the frame parameter
|
||||||
@@ -831,70 +846,60 @@ APIError APIPlaintextFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
|
|
||||||
// read header
|
// read header
|
||||||
while (!rx_header_parsed_) {
|
while (!rx_header_parsed_) {
|
||||||
uint8_t data;
|
// Now that we know when the socket is ready, we can read up to 3 bytes
|
||||||
// Reading one byte at a time is fastest in practice for ESP32 when
|
// into the rx_header_buf_ before we have to switch back to reading
|
||||||
// there is no data on the wire (which is the common case).
|
// one byte at a time to ensure we don't read past the message and
|
||||||
// This results in faster failure detection compared to
|
// into the next one.
|
||||||
// attempting to read multiple bytes at once.
|
|
||||||
ssize_t received = this->socket_->read(&data, 1);
|
// Read directly into rx_header_buf_ at the current position
|
||||||
if (received == -1) {
|
// Try to get to at least 3 bytes total (indicator + 2 varint bytes), then read one byte at a time
|
||||||
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
ssize_t received =
|
||||||
return APIError::WOULD_BLOCK;
|
this->socket_->read(&rx_header_buf_[rx_header_buf_pos_], rx_header_buf_pos_ < 3 ? 3 - rx_header_buf_pos_ : 1);
|
||||||
}
|
APIError err = handle_socket_read_result_(received);
|
||||||
state_ = State::FAILED;
|
if (err != APIError::OK) {
|
||||||
HELPER_LOG("Socket read failed with errno %d", errno);
|
return err;
|
||||||
return APIError::SOCKET_READ_FAILED;
|
|
||||||
} else if (received == 0) {
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Connection closed");
|
|
||||||
return APIError::CONNECTION_CLOSED;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Successfully read a byte
|
// If this was the first read, validate the indicator byte
|
||||||
|
if (rx_header_buf_pos_ == 0 && received > 0) {
|
||||||
// Process byte according to current buffer position
|
if (rx_header_buf_[0] != 0x00) {
|
||||||
if (rx_header_buf_pos_ == 0) { // Case 1: First byte (indicator byte)
|
|
||||||
if (data != 0x00) {
|
|
||||||
state_ = State::FAILED;
|
state_ = State::FAILED;
|
||||||
HELPER_LOG("Bad indicator byte %u", data);
|
HELPER_LOG("Bad indicator byte %u", rx_header_buf_[0]);
|
||||||
return APIError::BAD_INDICATOR;
|
return APIError::BAD_INDICATOR;
|
||||||
}
|
}
|
||||||
// We don't store the indicator byte, just increment position
|
|
||||||
rx_header_buf_pos_ = 1; // Set to 1 directly
|
|
||||||
continue; // Need more bytes before we can parse
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check buffer overflow before storing
|
rx_header_buf_pos_ += received;
|
||||||
if (rx_header_buf_pos_ == 5) { // Case 2: Buffer would overflow (5 bytes is max allowed)
|
|
||||||
|
// Check for buffer overflow
|
||||||
|
if (rx_header_buf_pos_ >= sizeof(rx_header_buf_)) {
|
||||||
state_ = State::FAILED;
|
state_ = State::FAILED;
|
||||||
HELPER_LOG("Header buffer overflow");
|
HELPER_LOG("Header buffer overflow");
|
||||||
return APIError::BAD_DATA_PACKET;
|
return APIError::BAD_DATA_PACKET;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Store byte in buffer (adjust index to account for skipped indicator byte)
|
// Need at least 3 bytes total (indicator + 2 varint bytes) before trying to parse
|
||||||
rx_header_buf_[rx_header_buf_pos_ - 1] = data;
|
if (rx_header_buf_pos_ < 3) {
|
||||||
|
continue;
|
||||||
// Increment position after storing
|
|
||||||
rx_header_buf_pos_++;
|
|
||||||
|
|
||||||
// Case 3: If we only have one varint byte, we need more
|
|
||||||
if (rx_header_buf_pos_ == 2) { // Have read indicator + 1 byte
|
|
||||||
continue; // Need more bytes before we can parse
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// At this point, we have at least 3 bytes total:
|
// At this point, we have at least 3 bytes total:
|
||||||
// - Validated indicator byte (0x00) but not stored
|
// - Validated indicator byte (0x00) stored at position 0
|
||||||
// - At least 2 bytes in the buffer for the varints
|
// - At least 2 bytes in the buffer for the varints
|
||||||
// Buffer layout:
|
// Buffer layout:
|
||||||
// First 1-3 bytes: Message size varint (variable length)
|
// [0]: indicator byte (0x00)
|
||||||
|
// [1-3]: Message size varint (variable length)
|
||||||
// - 2 bytes would only allow up to 16383, which is less than noise's UINT16_MAX (65535)
|
// - 2 bytes would only allow up to 16383, which is less than noise's UINT16_MAX (65535)
|
||||||
// - 3 bytes allows up to 2097151, ensuring we support at least as much as noise
|
// - 3 bytes allows up to 2097151, ensuring we support at least as much as noise
|
||||||
// Remaining 1-2 bytes: Message type varint (variable length)
|
// [2-5]: Message type varint (variable length)
|
||||||
// We now attempt to parse both varints. If either is incomplete,
|
// We now attempt to parse both varints. If either is incomplete,
|
||||||
// we'll continue reading more bytes.
|
// we'll continue reading more bytes.
|
||||||
|
|
||||||
|
// Skip indicator byte at position 0
|
||||||
|
uint8_t varint_pos = 1;
|
||||||
uint32_t consumed = 0;
|
uint32_t consumed = 0;
|
||||||
auto msg_size_varint = ProtoVarInt::parse(&rx_header_buf_[0], rx_header_buf_pos_ - 1, &consumed);
|
|
||||||
|
auto msg_size_varint = ProtoVarInt::parse(&rx_header_buf_[varint_pos], rx_header_buf_pos_ - varint_pos, &consumed);
|
||||||
if (!msg_size_varint.has_value()) {
|
if (!msg_size_varint.has_value()) {
|
||||||
// not enough data there yet
|
// not enough data there yet
|
||||||
continue;
|
continue;
|
||||||
@@ -908,7 +913,10 @@ APIError APIPlaintextFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
}
|
}
|
||||||
rx_header_parsed_len_ = msg_size_varint->as_uint16();
|
rx_header_parsed_len_ = msg_size_varint->as_uint16();
|
||||||
|
|
||||||
auto msg_type_varint = ProtoVarInt::parse(&rx_header_buf_[consumed], rx_header_buf_pos_ - 1 - consumed, &consumed);
|
// Move to next varint position
|
||||||
|
varint_pos += consumed;
|
||||||
|
|
||||||
|
auto msg_type_varint = ProtoVarInt::parse(&rx_header_buf_[varint_pos], rx_header_buf_pos_ - varint_pos, &consumed);
|
||||||
if (!msg_type_varint.has_value()) {
|
if (!msg_type_varint.has_value()) {
|
||||||
// not enough data there yet
|
// not enough data there yet
|
||||||
continue;
|
continue;
|
||||||
@@ -933,17 +941,9 @@ APIError APIPlaintextFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
// more data to read
|
// more data to read
|
||||||
uint16_t to_read = rx_header_parsed_len_ - rx_buf_len_;
|
uint16_t to_read = rx_header_parsed_len_ - rx_buf_len_;
|
||||||
ssize_t received = this->socket_->read(&rx_buf_[rx_buf_len_], to_read);
|
ssize_t received = this->socket_->read(&rx_buf_[rx_buf_len_], to_read);
|
||||||
if (received == -1) {
|
APIError err = handle_socket_read_result_(received);
|
||||||
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
if (err != APIError::OK) {
|
||||||
return APIError::WOULD_BLOCK;
|
return err;
|
||||||
}
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Socket read failed with errno %d", errno);
|
|
||||||
return APIError::SOCKET_READ_FAILED;
|
|
||||||
} else if (received == 0) {
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Connection closed");
|
|
||||||
return APIError::CONNECTION_CLOSED;
|
|
||||||
}
|
}
|
||||||
rx_buf_len_ += static_cast<uint16_t>(received);
|
rx_buf_len_ += static_cast<uint16_t>(received);
|
||||||
if (static_cast<uint16_t>(received) != to_read) {
|
if (static_cast<uint16_t>(received) != to_read) {
|
||||||
@@ -1003,65 +1003,74 @@ APIError APIPlaintextFrameHelper::read_packet(ReadPacketBuffer *buffer) {
|
|||||||
return APIError::OK;
|
return APIError::OK;
|
||||||
}
|
}
|
||||||
APIError APIPlaintextFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) {
|
APIError APIPlaintextFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) {
|
||||||
|
PacketInfo packet{type, 0, static_cast<uint16_t>(buffer.get_buffer()->size() - frame_header_padding_)};
|
||||||
|
return write_protobuf_packets(buffer, std::span<const PacketInfo>(&packet, 1));
|
||||||
|
}
|
||||||
|
|
||||||
|
APIError APIPlaintextFrameHelper::write_protobuf_packets(ProtoWriteBuffer buffer, std::span<const PacketInfo> packets) {
|
||||||
if (state_ != State::DATA) {
|
if (state_ != State::DATA) {
|
||||||
return APIError::BAD_STATE;
|
return APIError::BAD_STATE;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<uint8_t> *raw_buffer = buffer.get_buffer();
|
if (packets.empty()) {
|
||||||
// Message data starts after padding (frame_header_padding_ = 6)
|
return APIError::OK;
|
||||||
uint16_t payload_len = static_cast<uint16_t>(raw_buffer->size() - frame_header_padding_);
|
|
||||||
|
|
||||||
// Calculate varint sizes for header components
|
|
||||||
uint8_t size_varint_len = api::ProtoSize::varint(static_cast<uint32_t>(payload_len));
|
|
||||||
uint8_t type_varint_len = api::ProtoSize::varint(static_cast<uint32_t>(type));
|
|
||||||
uint8_t total_header_len = 1 + size_varint_len + type_varint_len;
|
|
||||||
|
|
||||||
if (total_header_len > frame_header_padding_) {
|
|
||||||
// Header is too large to fit in the padding
|
|
||||||
return APIError::BAD_ARG;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calculate where to start writing the header
|
std::vector<uint8_t> *raw_buffer = buffer.get_buffer();
|
||||||
// The header starts at the latest possible position to minimize unused padding
|
uint8_t *buffer_data = raw_buffer->data(); // Cache buffer pointer
|
||||||
//
|
|
||||||
// Example 1 (small values): total_header_len = 3, header_offset = 6 - 3 = 3
|
|
||||||
// [0-2] - Unused padding
|
|
||||||
// [3] - 0x00 indicator byte
|
|
||||||
// [4] - Payload size varint (1 byte, for sizes 0-127)
|
|
||||||
// [5] - Message type varint (1 byte, for types 0-127)
|
|
||||||
// [6...] - Actual payload data
|
|
||||||
//
|
|
||||||
// Example 2 (medium values): total_header_len = 4, header_offset = 6 - 4 = 2
|
|
||||||
// [0-1] - Unused padding
|
|
||||||
// [2] - 0x00 indicator byte
|
|
||||||
// [3-4] - Payload size varint (2 bytes, for sizes 128-16383)
|
|
||||||
// [5] - Message type varint (1 byte, for types 0-127)
|
|
||||||
// [6...] - Actual payload data
|
|
||||||
//
|
|
||||||
// Example 3 (large values): total_header_len = 6, header_offset = 6 - 6 = 0
|
|
||||||
// [0] - 0x00 indicator byte
|
|
||||||
// [1-3] - Payload size varint (3 bytes, for sizes 16384-2097151)
|
|
||||||
// [4-5] - Message type varint (2 bytes, for types 128-32767)
|
|
||||||
// [6...] - Actual payload data
|
|
||||||
uint8_t *buf_start = raw_buffer->data();
|
|
||||||
uint8_t header_offset = frame_header_padding_ - total_header_len;
|
|
||||||
|
|
||||||
// Write the plaintext header
|
this->reusable_iovs_.clear();
|
||||||
buf_start[header_offset] = 0x00; // indicator
|
this->reusable_iovs_.reserve(packets.size());
|
||||||
|
|
||||||
// Encode size varint directly into buffer
|
for (const auto &packet : packets) {
|
||||||
ProtoVarInt(payload_len).encode_to_buffer_unchecked(buf_start + header_offset + 1, size_varint_len);
|
// Calculate varint sizes for header layout
|
||||||
|
uint8_t size_varint_len = api::ProtoSize::varint(static_cast<uint32_t>(packet.payload_size));
|
||||||
|
uint8_t type_varint_len = api::ProtoSize::varint(static_cast<uint32_t>(packet.message_type));
|
||||||
|
uint8_t total_header_len = 1 + size_varint_len + type_varint_len;
|
||||||
|
|
||||||
// Encode type varint directly into buffer
|
// Calculate where to start writing the header
|
||||||
ProtoVarInt(type).encode_to_buffer_unchecked(buf_start + header_offset + 1 + size_varint_len, type_varint_len);
|
// The header starts at the latest possible position to minimize unused padding
|
||||||
|
//
|
||||||
|
// Example 1 (small values): total_header_len = 3, header_offset = 6 - 3 = 3
|
||||||
|
// [0-2] - Unused padding
|
||||||
|
// [3] - 0x00 indicator byte
|
||||||
|
// [4] - Payload size varint (1 byte, for sizes 0-127)
|
||||||
|
// [5] - Message type varint (1 byte, for types 0-127)
|
||||||
|
// [6...] - Actual payload data
|
||||||
|
//
|
||||||
|
// Example 2 (medium values): total_header_len = 4, header_offset = 6 - 4 = 2
|
||||||
|
// [0-1] - Unused padding
|
||||||
|
// [2] - 0x00 indicator byte
|
||||||
|
// [3-4] - Payload size varint (2 bytes, for sizes 128-16383)
|
||||||
|
// [5] - Message type varint (1 byte, for types 0-127)
|
||||||
|
// [6...] - Actual payload data
|
||||||
|
//
|
||||||
|
// Example 3 (large values): total_header_len = 6, header_offset = 6 - 6 = 0
|
||||||
|
// [0] - 0x00 indicator byte
|
||||||
|
// [1-3] - Payload size varint (3 bytes, for sizes 16384-2097151)
|
||||||
|
// [4-5] - Message type varint (2 bytes, for types 128-32767)
|
||||||
|
// [6...] - Actual payload data
|
||||||
|
//
|
||||||
|
// The message starts at offset + frame_header_padding_
|
||||||
|
// So we write the header starting at offset + frame_header_padding_ - total_header_len
|
||||||
|
uint8_t *buf_start = buffer_data + packet.offset;
|
||||||
|
uint32_t header_offset = frame_header_padding_ - total_header_len;
|
||||||
|
|
||||||
struct iovec iov;
|
// Write the plaintext header
|
||||||
// Point iov_base to the beginning of our header (skip unused padding)
|
buf_start[header_offset] = 0x00; // indicator
|
||||||
// This ensures we only send the actual header and payload, not the empty padding bytes
|
|
||||||
iov.iov_base = buf_start + header_offset;
|
|
||||||
iov.iov_len = total_header_len + payload_len;
|
|
||||||
|
|
||||||
return write_raw_(&iov, 1);
|
// Encode varints directly into buffer
|
||||||
|
ProtoVarInt(packet.payload_size).encode_to_buffer_unchecked(buf_start + header_offset + 1, size_varint_len);
|
||||||
|
ProtoVarInt(packet.message_type)
|
||||||
|
.encode_to_buffer_unchecked(buf_start + header_offset + 1 + size_varint_len, type_varint_len);
|
||||||
|
|
||||||
|
// Add iovec for this packet (header + payload)
|
||||||
|
this->reusable_iovs_.push_back(
|
||||||
|
{buf_start + header_offset, static_cast<size_t>(total_header_len + packet.payload_size)});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send all packets in one writev call
|
||||||
|
return write_raw_(this->reusable_iovs_.data(), this->reusable_iovs_.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif // USE_API_PLAINTEXT
|
#endif // USE_API_PLAINTEXT
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <deque>
|
#include <deque>
|
||||||
#include <limits>
|
#include <limits>
|
||||||
|
#include <span>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
@@ -13,6 +14,7 @@
|
|||||||
|
|
||||||
#include "api_noise_context.h"
|
#include "api_noise_context.h"
|
||||||
#include "esphome/components/socket/socket.h"
|
#include "esphome/components/socket/socket.h"
|
||||||
|
#include "esphome/core/application.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace api {
|
namespace api {
|
||||||
@@ -26,7 +28,18 @@ struct ReadPacketBuffer {
|
|||||||
uint16_t data_len;
|
uint16_t data_len;
|
||||||
};
|
};
|
||||||
|
|
||||||
enum class APIError : int {
|
// Packed packet info structure to minimize memory usage
|
||||||
|
struct PacketInfo {
|
||||||
|
uint16_t message_type; // 2 bytes
|
||||||
|
uint16_t offset; // 2 bytes (sufficient for packet size ~1460 bytes)
|
||||||
|
uint16_t payload_size; // 2 bytes (up to 65535 bytes)
|
||||||
|
uint16_t padding; // 2 byte (for alignment)
|
||||||
|
|
||||||
|
PacketInfo(uint16_t type, uint16_t off, uint16_t size)
|
||||||
|
: message_type(type), offset(off), payload_size(size), padding(0) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class APIError : uint16_t {
|
||||||
OK = 0,
|
OK = 0,
|
||||||
WOULD_BLOCK = 1001,
|
WOULD_BLOCK = 1001,
|
||||||
BAD_HANDSHAKE_PACKET_LEN = 1002,
|
BAD_HANDSHAKE_PACKET_LEN = 1002,
|
||||||
@@ -62,7 +75,7 @@ class APIFrameHelper {
|
|||||||
}
|
}
|
||||||
virtual ~APIFrameHelper() = default;
|
virtual ~APIFrameHelper() = default;
|
||||||
virtual APIError init() = 0;
|
virtual APIError init() = 0;
|
||||||
virtual APIError loop() = 0;
|
virtual APIError loop();
|
||||||
virtual APIError read_packet(ReadPacketBuffer *buffer) = 0;
|
virtual APIError read_packet(ReadPacketBuffer *buffer) = 0;
|
||||||
bool can_write_without_blocking() { return state_ == State::DATA && tx_buf_.empty(); }
|
bool can_write_without_blocking() { return state_ == State::DATA && tx_buf_.empty(); }
|
||||||
std::string getpeername() { return socket_->getpeername(); }
|
std::string getpeername() { return socket_->getpeername(); }
|
||||||
@@ -86,10 +99,16 @@ class APIFrameHelper {
|
|||||||
// Give this helper a name for logging
|
// Give this helper a name for logging
|
||||||
void set_log_info(std::string info) { info_ = std::move(info); }
|
void set_log_info(std::string info) { info_ = std::move(info); }
|
||||||
virtual APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) = 0;
|
virtual APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) = 0;
|
||||||
|
// Write multiple protobuf packets in a single operation
|
||||||
|
// packets contains (message_type, offset, length) for each message in the buffer
|
||||||
|
// The buffer contains all messages with appropriate padding before each
|
||||||
|
virtual APIError write_protobuf_packets(ProtoWriteBuffer buffer, std::span<const PacketInfo> packets) = 0;
|
||||||
// Get the frame header padding required by this protocol
|
// Get the frame header padding required by this protocol
|
||||||
virtual uint8_t frame_header_padding() = 0;
|
virtual uint8_t frame_header_padding() = 0;
|
||||||
// Get the frame footer size required by this protocol
|
// Get the frame footer size required by this protocol
|
||||||
virtual uint8_t frame_footer_size() = 0;
|
virtual uint8_t frame_footer_size() = 0;
|
||||||
|
// Check if socket has data ready to read
|
||||||
|
bool is_socket_ready() const { return socket_ != nullptr && socket_->ready(); }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
// Struct for holding parsed frame data
|
// Struct for holding parsed frame data
|
||||||
@@ -107,38 +126,6 @@ class APIFrameHelper {
|
|||||||
const uint8_t *current_data() const { return data.data() + offset; }
|
const uint8_t *current_data() const { return data.data() + offset; }
|
||||||
};
|
};
|
||||||
|
|
||||||
// Queue of data buffers to be sent
|
|
||||||
std::deque<SendBuffer> tx_buf_;
|
|
||||||
|
|
||||||
// Common state enum for all frame helpers
|
|
||||||
// Note: Not all states are used by all implementations
|
|
||||||
// - INITIALIZE: Used by both Noise and Plaintext
|
|
||||||
// - CLIENT_HELLO, SERVER_HELLO, HANDSHAKE: Only used by Noise protocol
|
|
||||||
// - DATA: Used by both Noise and Plaintext
|
|
||||||
// - CLOSED: Used by both Noise and Plaintext
|
|
||||||
// - FAILED: Used by both Noise and Plaintext
|
|
||||||
// - EXPLICIT_REJECT: Only used by Noise protocol
|
|
||||||
enum class State {
|
|
||||||
INITIALIZE = 1,
|
|
||||||
CLIENT_HELLO = 2, // Noise only
|
|
||||||
SERVER_HELLO = 3, // Noise only
|
|
||||||
HANDSHAKE = 4, // Noise only
|
|
||||||
DATA = 5,
|
|
||||||
CLOSED = 6,
|
|
||||||
FAILED = 7,
|
|
||||||
EXPLICIT_REJECT = 8, // Noise only
|
|
||||||
};
|
|
||||||
|
|
||||||
// Current state of the frame helper
|
|
||||||
State state_{State::INITIALIZE};
|
|
||||||
|
|
||||||
// Helper name for logging
|
|
||||||
std::string info_;
|
|
||||||
|
|
||||||
// Socket for communication
|
|
||||||
socket::Socket *socket_{nullptr};
|
|
||||||
std::unique_ptr<socket::Socket> socket_owned_;
|
|
||||||
|
|
||||||
// Common implementation for writing raw data to socket
|
// Common implementation for writing raw data to socket
|
||||||
APIError write_raw_(const struct iovec *iov, int iovcnt);
|
APIError write_raw_(const struct iovec *iov, int iovcnt);
|
||||||
|
|
||||||
@@ -151,15 +138,47 @@ class APIFrameHelper {
|
|||||||
APIError write_raw_(const struct iovec *iov, int iovcnt, socket::Socket *socket, std::vector<uint8_t> &tx_buf,
|
APIError write_raw_(const struct iovec *iov, int iovcnt, socket::Socket *socket, std::vector<uint8_t> &tx_buf,
|
||||||
const std::string &info, StateEnum &state, StateEnum failed_state);
|
const std::string &info, StateEnum &state, StateEnum failed_state);
|
||||||
|
|
||||||
|
// Pointers first (4 bytes each)
|
||||||
|
socket::Socket *socket_{nullptr};
|
||||||
|
std::unique_ptr<socket::Socket> socket_owned_;
|
||||||
|
|
||||||
|
// Common state enum for all frame helpers
|
||||||
|
// Note: Not all states are used by all implementations
|
||||||
|
// - INITIALIZE: Used by both Noise and Plaintext
|
||||||
|
// - CLIENT_HELLO, SERVER_HELLO, HANDSHAKE: Only used by Noise protocol
|
||||||
|
// - DATA: Used by both Noise and Plaintext
|
||||||
|
// - CLOSED: Used by both Noise and Plaintext
|
||||||
|
// - FAILED: Used by both Noise and Plaintext
|
||||||
|
// - EXPLICIT_REJECT: Only used by Noise protocol
|
||||||
|
enum class State : uint8_t {
|
||||||
|
INITIALIZE = 1,
|
||||||
|
CLIENT_HELLO = 2, // Noise only
|
||||||
|
SERVER_HELLO = 3, // Noise only
|
||||||
|
HANDSHAKE = 4, // Noise only
|
||||||
|
DATA = 5,
|
||||||
|
CLOSED = 6,
|
||||||
|
FAILED = 7,
|
||||||
|
EXPLICIT_REJECT = 8, // Noise only
|
||||||
|
};
|
||||||
|
|
||||||
|
// Containers (size varies, but typically 12+ bytes on 32-bit)
|
||||||
|
std::deque<SendBuffer> tx_buf_;
|
||||||
|
std::string info_;
|
||||||
|
std::vector<struct iovec> reusable_iovs_;
|
||||||
|
std::vector<uint8_t> rx_buf_;
|
||||||
|
|
||||||
|
// Group smaller types together
|
||||||
|
uint16_t rx_buf_len_ = 0;
|
||||||
|
State state_{State::INITIALIZE};
|
||||||
uint8_t frame_header_padding_{0};
|
uint8_t frame_header_padding_{0};
|
||||||
uint8_t frame_footer_size_{0};
|
uint8_t frame_footer_size_{0};
|
||||||
|
// 5 bytes total, 3 bytes padding
|
||||||
// Receive buffer for reading frame data
|
|
||||||
std::vector<uint8_t> rx_buf_;
|
|
||||||
uint16_t rx_buf_len_ = 0;
|
|
||||||
|
|
||||||
// Common initialization for both plaintext and noise protocols
|
// Common initialization for both plaintext and noise protocols
|
||||||
APIError init_common_();
|
APIError init_common_();
|
||||||
|
|
||||||
|
// Helper method to handle socket read results
|
||||||
|
APIError handle_socket_read_result_(ssize_t received);
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifdef USE_API_NOISE
|
#ifdef USE_API_NOISE
|
||||||
@@ -179,6 +198,7 @@ class APINoiseFrameHelper : public APIFrameHelper {
|
|||||||
APIError loop() override;
|
APIError loop() override;
|
||||||
APIError read_packet(ReadPacketBuffer *buffer) override;
|
APIError read_packet(ReadPacketBuffer *buffer) override;
|
||||||
APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) override;
|
APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) override;
|
||||||
|
APIError write_protobuf_packets(ProtoWriteBuffer buffer, std::span<const PacketInfo> packets) override;
|
||||||
// Get the frame header padding required by this protocol
|
// Get the frame header padding required by this protocol
|
||||||
uint8_t frame_header_padding() override { return frame_header_padding_; }
|
uint8_t frame_header_padding() override { return frame_header_padding_; }
|
||||||
// Get the frame footer size required by this protocol
|
// Get the frame footer size required by this protocol
|
||||||
@@ -191,19 +211,28 @@ class APINoiseFrameHelper : public APIFrameHelper {
|
|||||||
APIError init_handshake_();
|
APIError init_handshake_();
|
||||||
APIError check_handshake_finished_();
|
APIError check_handshake_finished_();
|
||||||
void send_explicit_handshake_reject_(const std::string &reason);
|
void send_explicit_handshake_reject_(const std::string &reason);
|
||||||
|
|
||||||
|
// Pointers first (4 bytes each)
|
||||||
|
NoiseHandshakeState *handshake_{nullptr};
|
||||||
|
NoiseCipherState *send_cipher_{nullptr};
|
||||||
|
NoiseCipherState *recv_cipher_{nullptr};
|
||||||
|
|
||||||
|
// Shared pointer (8 bytes on 32-bit = 4 bytes control block pointer + 4 bytes object pointer)
|
||||||
|
std::shared_ptr<APINoiseContext> ctx_;
|
||||||
|
|
||||||
|
// Vector (12 bytes on 32-bit)
|
||||||
|
std::vector<uint8_t> prologue_;
|
||||||
|
|
||||||
|
// NoiseProtocolId (size depends on implementation)
|
||||||
|
NoiseProtocolId nid_;
|
||||||
|
|
||||||
|
// Group small types together
|
||||||
// Fixed-size header buffer for noise protocol:
|
// Fixed-size header buffer for noise protocol:
|
||||||
// 1 byte for indicator + 2 bytes for message size (16-bit value, not varint)
|
// 1 byte for indicator + 2 bytes for message size (16-bit value, not varint)
|
||||||
// Note: Maximum message size is UINT16_MAX (65535), with a limit of 128 bytes during handshake phase
|
// Note: Maximum message size is UINT16_MAX (65535), with a limit of 128 bytes during handshake phase
|
||||||
uint8_t rx_header_buf_[3];
|
uint8_t rx_header_buf_[3];
|
||||||
uint8_t rx_header_buf_len_ = 0;
|
uint8_t rx_header_buf_len_ = 0;
|
||||||
|
// 4 bytes total, no padding
|
||||||
std::vector<uint8_t> prologue_;
|
|
||||||
|
|
||||||
std::shared_ptr<APINoiseContext> ctx_;
|
|
||||||
NoiseHandshakeState *handshake_{nullptr};
|
|
||||||
NoiseCipherState *send_cipher_{nullptr};
|
|
||||||
NoiseCipherState *recv_cipher_{nullptr};
|
|
||||||
NoiseProtocolId nid_;
|
|
||||||
};
|
};
|
||||||
#endif // USE_API_NOISE
|
#endif // USE_API_NOISE
|
||||||
|
|
||||||
@@ -223,25 +252,31 @@ class APIPlaintextFrameHelper : public APIFrameHelper {
|
|||||||
APIError loop() override;
|
APIError loop() override;
|
||||||
APIError read_packet(ReadPacketBuffer *buffer) override;
|
APIError read_packet(ReadPacketBuffer *buffer) override;
|
||||||
APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) override;
|
APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) override;
|
||||||
|
APIError write_protobuf_packets(ProtoWriteBuffer buffer, std::span<const PacketInfo> packets) override;
|
||||||
uint8_t frame_header_padding() override { return frame_header_padding_; }
|
uint8_t frame_header_padding() override { return frame_header_padding_; }
|
||||||
// Get the frame footer size required by this protocol
|
// Get the frame footer size required by this protocol
|
||||||
uint8_t frame_footer_size() override { return frame_footer_size_; }
|
uint8_t frame_footer_size() override { return frame_footer_size_; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
APIError try_read_frame_(ParsedFrame *frame);
|
APIError try_read_frame_(ParsedFrame *frame);
|
||||||
|
|
||||||
|
// Group 2-byte aligned types
|
||||||
|
uint16_t rx_header_parsed_type_ = 0;
|
||||||
|
uint16_t rx_header_parsed_len_ = 0;
|
||||||
|
|
||||||
|
// Group 1-byte types together
|
||||||
// Fixed-size header buffer for plaintext protocol:
|
// Fixed-size header buffer for plaintext protocol:
|
||||||
// We only need space for the two varints since we validate the indicator byte separately.
|
// We now store the indicator byte + the two varints.
|
||||||
// To match noise protocol's maximum message size (UINT16_MAX = 65535), we need:
|
// To match noise protocol's maximum message size (UINT16_MAX = 65535), we need:
|
||||||
// 3 bytes for message size varint (supports up to 2097151) + 2 bytes for message type varint
|
// 1 byte for indicator + 3 bytes for message size varint (supports up to 2097151) + 2 bytes for message type varint
|
||||||
//
|
//
|
||||||
// While varints could theoretically be up to 10 bytes each for 64-bit values,
|
// While varints could theoretically be up to 10 bytes each for 64-bit values,
|
||||||
// attempting to process messages with headers that large would likely crash the
|
// attempting to process messages with headers that large would likely crash the
|
||||||
// ESP32 due to memory constraints.
|
// ESP32 due to memory constraints.
|
||||||
uint8_t rx_header_buf_[5]; // 5 bytes for varints (3 for size + 2 for type)
|
uint8_t rx_header_buf_[6]; // 1 byte indicator + 5 bytes for varints (3 for size + 2 for type)
|
||||||
uint8_t rx_header_buf_pos_ = 0;
|
uint8_t rx_header_buf_pos_ = 0;
|
||||||
bool rx_header_parsed_ = false;
|
bool rx_header_parsed_ = false;
|
||||||
uint16_t rx_header_parsed_type_ = 0;
|
// 8 bytes total, no padding needed
|
||||||
uint16_t rx_header_parsed_len_ = 0;
|
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
@@ -21,4 +21,5 @@ extend google.protobuf.MessageOptions {
|
|||||||
optional string ifdef = 1038;
|
optional string ifdef = 1038;
|
||||||
optional bool log = 1039 [default=true];
|
optional bool log = 1039 [default=true];
|
||||||
optional bool no_delay = 1040 [default=false];
|
optional bool no_delay = 1040 [default=false];
|
||||||
|
optional string base_class = 1041;
|
||||||
}
|
}
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
4333
esphome/components/api/api_pb2_dump.cpp
Normal file
4333
esphome/components/api/api_pb2_dump.cpp
Normal file
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -2,170 +2,103 @@
|
|||||||
// See script/api_protobuf/api_protobuf.py
|
// See script/api_protobuf/api_protobuf.py
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "api_pb2.h"
|
|
||||||
#include "esphome/core/defines.h"
|
#include "esphome/core/defines.h"
|
||||||
|
|
||||||
|
#include "api_pb2.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace api {
|
namespace api {
|
||||||
|
|
||||||
class APIServerConnectionBase : public ProtoService {
|
class APIServerConnectionBase : public ProtoService {
|
||||||
public:
|
public:
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
protected:
|
||||||
|
void log_send_message_(const char *name, const std::string &dump);
|
||||||
|
|
||||||
|
public:
|
||||||
|
#endif
|
||||||
|
|
||||||
|
template<typename T> bool send_message(const T &msg) {
|
||||||
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
|
this->log_send_message_(msg.message_name(), msg.dump());
|
||||||
|
#endif
|
||||||
|
return this->send_message_(msg, T::MESSAGE_TYPE);
|
||||||
|
}
|
||||||
|
|
||||||
virtual void on_hello_request(const HelloRequest &value){};
|
virtual void on_hello_request(const HelloRequest &value){};
|
||||||
bool send_hello_response(const HelloResponse &msg);
|
|
||||||
virtual void on_connect_request(const ConnectRequest &value){};
|
virtual void on_connect_request(const ConnectRequest &value){};
|
||||||
bool send_connect_response(const ConnectResponse &msg);
|
|
||||||
bool send_disconnect_request(const DisconnectRequest &msg);
|
|
||||||
virtual void on_disconnect_request(const DisconnectRequest &value){};
|
virtual void on_disconnect_request(const DisconnectRequest &value){};
|
||||||
bool send_disconnect_response(const DisconnectResponse &msg);
|
|
||||||
virtual void on_disconnect_response(const DisconnectResponse &value){};
|
virtual void on_disconnect_response(const DisconnectResponse &value){};
|
||||||
bool send_ping_request(const PingRequest &msg);
|
|
||||||
virtual void on_ping_request(const PingRequest &value){};
|
virtual void on_ping_request(const PingRequest &value){};
|
||||||
bool send_ping_response(const PingResponse &msg);
|
|
||||||
virtual void on_ping_response(const PingResponse &value){};
|
virtual void on_ping_response(const PingResponse &value){};
|
||||||
virtual void on_device_info_request(const DeviceInfoRequest &value){};
|
virtual void on_device_info_request(const DeviceInfoRequest &value){};
|
||||||
bool send_device_info_response(const DeviceInfoResponse &msg);
|
|
||||||
virtual void on_list_entities_request(const ListEntitiesRequest &value){};
|
virtual void on_list_entities_request(const ListEntitiesRequest &value){};
|
||||||
bool send_list_entities_done_response(const ListEntitiesDoneResponse &msg);
|
|
||||||
virtual void on_subscribe_states_request(const SubscribeStatesRequest &value){};
|
virtual void on_subscribe_states_request(const SubscribeStatesRequest &value){};
|
||||||
#ifdef USE_BINARY_SENSOR
|
|
||||||
bool send_list_entities_binary_sensor_response(const ListEntitiesBinarySensorResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BINARY_SENSOR
|
|
||||||
bool send_binary_sensor_state_response(const BinarySensorStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_COVER
|
|
||||||
bool send_list_entities_cover_response(const ListEntitiesCoverResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_COVER
|
|
||||||
bool send_cover_state_response(const CoverStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_COVER
|
#ifdef USE_COVER
|
||||||
virtual void on_cover_command_request(const CoverCommandRequest &value){};
|
virtual void on_cover_command_request(const CoverCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_FAN
|
|
||||||
bool send_list_entities_fan_response(const ListEntitiesFanResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_FAN
|
|
||||||
bool send_fan_state_response(const FanStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_FAN
|
#ifdef USE_FAN
|
||||||
virtual void on_fan_command_request(const FanCommandRequest &value){};
|
virtual void on_fan_command_request(const FanCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LIGHT
|
|
||||||
bool send_list_entities_light_response(const ListEntitiesLightResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_LIGHT
|
|
||||||
bool send_light_state_response(const LightStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_LIGHT
|
#ifdef USE_LIGHT
|
||||||
virtual void on_light_command_request(const LightCommandRequest &value){};
|
virtual void on_light_command_request(const LightCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SENSOR
|
|
||||||
bool send_list_entities_sensor_response(const ListEntitiesSensorResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_SENSOR
|
|
||||||
bool send_sensor_state_response(const SensorStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_SWITCH
|
|
||||||
bool send_list_entities_switch_response(const ListEntitiesSwitchResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_SWITCH
|
|
||||||
bool send_switch_state_response(const SwitchStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_SWITCH
|
#ifdef USE_SWITCH
|
||||||
virtual void on_switch_command_request(const SwitchCommandRequest &value){};
|
virtual void on_switch_command_request(const SwitchCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT_SENSOR
|
|
||||||
bool send_list_entities_text_sensor_response(const ListEntitiesTextSensorResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_TEXT_SENSOR
|
|
||||||
bool send_text_sensor_state_response(const TextSensorStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
virtual void on_subscribe_logs_request(const SubscribeLogsRequest &value){};
|
virtual void on_subscribe_logs_request(const SubscribeLogsRequest &value){};
|
||||||
bool send_subscribe_logs_response(const SubscribeLogsResponse &msg);
|
|
||||||
#ifdef USE_API_NOISE
|
#ifdef USE_API_NOISE
|
||||||
virtual void on_noise_encryption_set_key_request(const NoiseEncryptionSetKeyRequest &value){};
|
virtual void on_noise_encryption_set_key_request(const NoiseEncryptionSetKeyRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_API_NOISE
|
|
||||||
bool send_noise_encryption_set_key_response(const NoiseEncryptionSetKeyResponse &msg);
|
|
||||||
#endif
|
|
||||||
virtual void on_subscribe_homeassistant_services_request(const SubscribeHomeassistantServicesRequest &value){};
|
virtual void on_subscribe_homeassistant_services_request(const SubscribeHomeassistantServicesRequest &value){};
|
||||||
bool send_homeassistant_service_response(const HomeassistantServiceResponse &msg);
|
|
||||||
virtual void on_subscribe_home_assistant_states_request(const SubscribeHomeAssistantStatesRequest &value){};
|
virtual void on_subscribe_home_assistant_states_request(const SubscribeHomeAssistantStatesRequest &value){};
|
||||||
bool send_subscribe_home_assistant_state_response(const SubscribeHomeAssistantStateResponse &msg);
|
|
||||||
virtual void on_home_assistant_state_response(const HomeAssistantStateResponse &value){};
|
virtual void on_home_assistant_state_response(const HomeAssistantStateResponse &value){};
|
||||||
bool send_get_time_request(const GetTimeRequest &msg);
|
|
||||||
virtual void on_get_time_request(const GetTimeRequest &value){};
|
virtual void on_get_time_request(const GetTimeRequest &value){};
|
||||||
bool send_get_time_response(const GetTimeResponse &msg);
|
|
||||||
virtual void on_get_time_response(const GetTimeResponse &value){};
|
virtual void on_get_time_response(const GetTimeResponse &value){};
|
||||||
bool send_list_entities_services_response(const ListEntitiesServicesResponse &msg);
|
|
||||||
virtual void on_execute_service_request(const ExecuteServiceRequest &value){};
|
virtual void on_execute_service_request(const ExecuteServiceRequest &value){};
|
||||||
#ifdef USE_ESP32_CAMERA
|
|
||||||
bool send_list_entities_camera_response(const ListEntitiesCameraResponse &msg);
|
#ifdef USE_CAMERA
|
||||||
#endif
|
|
||||||
#ifdef USE_ESP32_CAMERA
|
|
||||||
bool send_camera_image_response(const CameraImageResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_ESP32_CAMERA
|
|
||||||
virtual void on_camera_image_request(const CameraImageRequest &value){};
|
virtual void on_camera_image_request(const CameraImageRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_CLIMATE
|
|
||||||
bool send_list_entities_climate_response(const ListEntitiesClimateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_CLIMATE
|
|
||||||
bool send_climate_state_response(const ClimateStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_CLIMATE
|
#ifdef USE_CLIMATE
|
||||||
virtual void on_climate_command_request(const ClimateCommandRequest &value){};
|
virtual void on_climate_command_request(const ClimateCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_NUMBER
|
|
||||||
bool send_list_entities_number_response(const ListEntitiesNumberResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_NUMBER
|
|
||||||
bool send_number_state_response(const NumberStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_NUMBER
|
#ifdef USE_NUMBER
|
||||||
virtual void on_number_command_request(const NumberCommandRequest &value){};
|
virtual void on_number_command_request(const NumberCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SELECT
|
|
||||||
bool send_list_entities_select_response(const ListEntitiesSelectResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_SELECT
|
|
||||||
bool send_select_state_response(const SelectStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
virtual void on_select_command_request(const SelectCommandRequest &value){};
|
virtual void on_select_command_request(const SelectCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SIREN
|
|
||||||
bool send_list_entities_siren_response(const ListEntitiesSirenResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_SIREN
|
|
||||||
bool send_siren_state_response(const SirenStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_SIREN
|
#ifdef USE_SIREN
|
||||||
virtual void on_siren_command_request(const SirenCommandRequest &value){};
|
virtual void on_siren_command_request(const SirenCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LOCK
|
|
||||||
bool send_list_entities_lock_response(const ListEntitiesLockResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_LOCK
|
|
||||||
bool send_lock_state_response(const LockStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_LOCK
|
#ifdef USE_LOCK
|
||||||
virtual void on_lock_command_request(const LockCommandRequest &value){};
|
virtual void on_lock_command_request(const LockCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BUTTON
|
|
||||||
bool send_list_entities_button_response(const ListEntitiesButtonResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BUTTON
|
#ifdef USE_BUTTON
|
||||||
virtual void on_button_command_request(const ButtonCommandRequest &value){};
|
virtual void on_button_command_request(const ButtonCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_MEDIA_PLAYER
|
|
||||||
bool send_list_entities_media_player_response(const ListEntitiesMediaPlayerResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_MEDIA_PLAYER
|
|
||||||
bool send_media_player_state_response(const MediaPlayerStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
virtual void on_media_player_command_request(const MediaPlayerCommandRequest &value){};
|
virtual void on_media_player_command_request(const MediaPlayerCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
@@ -173,33 +106,19 @@ class APIServerConnectionBase : public ProtoService {
|
|||||||
virtual void on_subscribe_bluetooth_le_advertisements_request(
|
virtual void on_subscribe_bluetooth_le_advertisements_request(
|
||||||
const SubscribeBluetoothLEAdvertisementsRequest &value){};
|
const SubscribeBluetoothLEAdvertisementsRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_le_advertisement_response(const BluetoothLEAdvertisementResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_le_raw_advertisements_response(const BluetoothLERawAdvertisementsResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual void on_bluetooth_device_request(const BluetoothDeviceRequest &value){};
|
virtual void on_bluetooth_device_request(const BluetoothDeviceRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_device_connection_response(const BluetoothDeviceConnectionResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual void on_bluetooth_gatt_get_services_request(const BluetoothGATTGetServicesRequest &value){};
|
virtual void on_bluetooth_gatt_get_services_request(const BluetoothGATTGetServicesRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_gatt_get_services_response(const BluetoothGATTGetServicesResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_gatt_get_services_done_response(const BluetoothGATTGetServicesDoneResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual void on_bluetooth_gatt_read_request(const BluetoothGATTReadRequest &value){};
|
virtual void on_bluetooth_gatt_read_request(const BluetoothGATTReadRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_gatt_read_response(const BluetoothGATTReadResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual void on_bluetooth_gatt_write_request(const BluetoothGATTWriteRequest &value){};
|
virtual void on_bluetooth_gatt_write_request(const BluetoothGATTWriteRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
@@ -212,49 +131,23 @@ class APIServerConnectionBase : public ProtoService {
|
|||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual void on_bluetooth_gatt_notify_request(const BluetoothGATTNotifyRequest &value){};
|
virtual void on_bluetooth_gatt_notify_request(const BluetoothGATTNotifyRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_gatt_notify_data_response(const BluetoothGATTNotifyDataResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual void on_subscribe_bluetooth_connections_free_request(const SubscribeBluetoothConnectionsFreeRequest &value){};
|
virtual void on_subscribe_bluetooth_connections_free_request(const SubscribeBluetoothConnectionsFreeRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_connections_free_response(const BluetoothConnectionsFreeResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_gatt_error_response(const BluetoothGATTErrorResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_gatt_write_response(const BluetoothGATTWriteResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_gatt_notify_response(const BluetoothGATTNotifyResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_device_pairing_response(const BluetoothDevicePairingResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_device_unpairing_response(const BluetoothDeviceUnpairingResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual void on_unsubscribe_bluetooth_le_advertisements_request(
|
virtual void on_unsubscribe_bluetooth_le_advertisements_request(
|
||||||
const UnsubscribeBluetoothLEAdvertisementsRequest &value){};
|
const UnsubscribeBluetoothLEAdvertisementsRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_device_clear_cache_response(const BluetoothDeviceClearCacheResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_scanner_state_response(const BluetoothScannerStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual void on_bluetooth_scanner_set_mode_request(const BluetoothScannerSetModeRequest &value){};
|
virtual void on_bluetooth_scanner_set_mode_request(const BluetoothScannerSetModeRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
virtual void on_subscribe_voice_assistant_request(const SubscribeVoiceAssistantRequest &value){};
|
virtual void on_subscribe_voice_assistant_request(const SubscribeVoiceAssistantRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
bool send_voice_assistant_request(const VoiceAssistantRequest &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
virtual void on_voice_assistant_response(const VoiceAssistantResponse &value){};
|
virtual void on_voice_assistant_response(const VoiceAssistantResponse &value){};
|
||||||
#endif
|
#endif
|
||||||
@@ -262,7 +155,6 @@ class APIServerConnectionBase : public ProtoService {
|
|||||||
virtual void on_voice_assistant_event_response(const VoiceAssistantEventResponse &value){};
|
virtual void on_voice_assistant_event_response(const VoiceAssistantEventResponse &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
bool send_voice_assistant_audio(const VoiceAssistantAudio &msg);
|
|
||||||
virtual void on_voice_assistant_audio(const VoiceAssistantAudio &value){};
|
virtual void on_voice_assistant_audio(const VoiceAssistantAudio &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
@@ -271,89 +163,44 @@ class APIServerConnectionBase : public ProtoService {
|
|||||||
#ifdef USE_VOICE_ASSISTANT
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
virtual void on_voice_assistant_announce_request(const VoiceAssistantAnnounceRequest &value){};
|
virtual void on_voice_assistant_announce_request(const VoiceAssistantAnnounceRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
bool send_voice_assistant_announce_finished(const VoiceAssistantAnnounceFinished &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
virtual void on_voice_assistant_configuration_request(const VoiceAssistantConfigurationRequest &value){};
|
virtual void on_voice_assistant_configuration_request(const VoiceAssistantConfigurationRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
bool send_voice_assistant_configuration_response(const VoiceAssistantConfigurationResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
#ifdef USE_VOICE_ASSISTANT
|
||||||
virtual void on_voice_assistant_set_configuration(const VoiceAssistantSetConfiguration &value){};
|
virtual void on_voice_assistant_set_configuration(const VoiceAssistantSetConfiguration &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
bool send_list_entities_alarm_control_panel_response(const ListEntitiesAlarmControlPanelResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
bool send_alarm_control_panel_state_response(const AlarmControlPanelStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
virtual void on_alarm_control_panel_command_request(const AlarmControlPanelCommandRequest &value){};
|
virtual void on_alarm_control_panel_command_request(const AlarmControlPanelCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT
|
|
||||||
bool send_list_entities_text_response(const ListEntitiesTextResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_TEXT
|
|
||||||
bool send_text_state_response(const TextStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_TEXT
|
#ifdef USE_TEXT
|
||||||
virtual void on_text_command_request(const TextCommandRequest &value){};
|
virtual void on_text_command_request(const TextCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATE
|
|
||||||
bool send_list_entities_date_response(const ListEntitiesDateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_DATETIME_DATE
|
|
||||||
bool send_date_state_response(const DateStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_DATETIME_DATE
|
#ifdef USE_DATETIME_DATE
|
||||||
virtual void on_date_command_request(const DateCommandRequest &value){};
|
virtual void on_date_command_request(const DateCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_TIME
|
|
||||||
bool send_list_entities_time_response(const ListEntitiesTimeResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_DATETIME_TIME
|
|
||||||
bool send_time_state_response(const TimeStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_DATETIME_TIME
|
#ifdef USE_DATETIME_TIME
|
||||||
virtual void on_time_command_request(const TimeCommandRequest &value){};
|
virtual void on_time_command_request(const TimeCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_EVENT
|
|
||||||
bool send_list_entities_event_response(const ListEntitiesEventResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_EVENT
|
|
||||||
bool send_event_response(const EventResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VALVE
|
|
||||||
bool send_list_entities_valve_response(const ListEntitiesValveResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VALVE
|
|
||||||
bool send_valve_state_response(const ValveStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VALVE
|
#ifdef USE_VALVE
|
||||||
virtual void on_valve_command_request(const ValveCommandRequest &value){};
|
virtual void on_valve_command_request(const ValveCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATETIME
|
|
||||||
bool send_list_entities_date_time_response(const ListEntitiesDateTimeResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_DATETIME_DATETIME
|
|
||||||
bool send_date_time_state_response(const DateTimeStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_DATETIME_DATETIME
|
#ifdef USE_DATETIME_DATETIME
|
||||||
virtual void on_date_time_command_request(const DateTimeCommandRequest &value){};
|
virtual void on_date_time_command_request(const DateTimeCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_UPDATE
|
|
||||||
bool send_list_entities_update_response(const ListEntitiesUpdateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_UPDATE
|
|
||||||
bool send_update_state_response(const UpdateStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_UPDATE
|
#ifdef USE_UPDATE
|
||||||
virtual void on_update_command_request(const UpdateCommandRequest &value){};
|
virtual void on_update_command_request(const UpdateCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
protected:
|
protected:
|
||||||
bool read_message(uint32_t msg_size, uint32_t msg_type, uint8_t *msg_data) override;
|
void read_message(uint32_t msg_size, uint32_t msg_type, uint8_t *msg_data) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
class APIServerConnection : public APIServerConnectionBase {
|
class APIServerConnection : public APIServerConnectionBase {
|
||||||
@@ -376,7 +223,7 @@ class APIServerConnection : public APIServerConnectionBase {
|
|||||||
#ifdef USE_BUTTON
|
#ifdef USE_BUTTON
|
||||||
virtual void button_command(const ButtonCommandRequest &msg) = 0;
|
virtual void button_command(const ButtonCommandRequest &msg) = 0;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ESP32_CAMERA
|
#ifdef USE_CAMERA
|
||||||
virtual void camera_image(const CameraImageRequest &msg) = 0;
|
virtual void camera_image(const CameraImageRequest &msg) = 0;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_CLIMATE
|
#ifdef USE_CLIMATE
|
||||||
@@ -493,7 +340,7 @@ class APIServerConnection : public APIServerConnectionBase {
|
|||||||
#ifdef USE_BUTTON
|
#ifdef USE_BUTTON
|
||||||
void on_button_command_request(const ButtonCommandRequest &msg) override;
|
void on_button_command_request(const ButtonCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ESP32_CAMERA
|
#ifdef USE_CAMERA
|
||||||
void on_camera_image_request(const CameraImageRequest &msg) override;
|
void on_camera_image_request(const CameraImageRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_CLIMATE
|
#ifdef USE_CLIMATE
|
||||||
|
|||||||
@@ -316,15 +316,13 @@ class ProtoSize {
|
|||||||
/**
|
/**
|
||||||
* @brief Calculates and adds the size of a nested message field to the total message size
|
* @brief Calculates and adds the size of a nested message field to the total message size
|
||||||
*
|
*
|
||||||
* This templated version directly takes a message object, calculates its size internally,
|
* This version takes a ProtoMessage object, calculates its size internally,
|
||||||
* and updates the total_size reference. This eliminates the need for a temporary variable
|
* and updates the total_size reference. This eliminates the need for a temporary variable
|
||||||
* at the call site.
|
* at the call site.
|
||||||
*
|
*
|
||||||
* @tparam MessageType The type of the nested message (inferred from parameter)
|
|
||||||
* @param message The nested message object
|
* @param message The nested message object
|
||||||
*/
|
*/
|
||||||
template<typename MessageType>
|
static inline void add_message_object(uint32_t &total_size, uint32_t field_id_size, const ProtoMessage &message,
|
||||||
static inline void add_message_object(uint32_t &total_size, uint32_t field_id_size, const MessageType &message,
|
|
||||||
bool force = false) {
|
bool force = false) {
|
||||||
uint32_t nested_size = 0;
|
uint32_t nested_size = 0;
|
||||||
message.calculate_size(nested_size);
|
message.calculate_size(nested_size);
|
||||||
|
|||||||
@@ -24,10 +24,22 @@ static const char *const TAG = "api";
|
|||||||
// APIServer
|
// APIServer
|
||||||
APIServer *global_api_server = nullptr; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
|
APIServer *global_api_server = nullptr; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
|
||||||
|
|
||||||
APIServer::APIServer() { global_api_server = this; }
|
#ifndef USE_API_YAML_SERVICES
|
||||||
|
// Global empty vector to avoid guard variables (saves 8 bytes)
|
||||||
|
// This is initialized at program startup before any threads
|
||||||
|
static const std::vector<UserServiceDescriptor *> empty_user_services{};
|
||||||
|
|
||||||
|
const std::vector<UserServiceDescriptor *> &get_empty_user_services_instance() { return empty_user_services; }
|
||||||
|
#endif
|
||||||
|
|
||||||
|
APIServer::APIServer() {
|
||||||
|
global_api_server = this;
|
||||||
|
// Pre-allocate shared write buffer
|
||||||
|
shared_write_buffer_.reserve(64);
|
||||||
|
}
|
||||||
|
|
||||||
void APIServer::setup() {
|
void APIServer::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up Home Assistant API server...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->setup_controller();
|
this->setup_controller();
|
||||||
|
|
||||||
#ifdef USE_API_NOISE
|
#ifdef USE_API_NOISE
|
||||||
@@ -43,7 +55,12 @@ void APIServer::setup() {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
this->socket_ = socket::socket_ip(SOCK_STREAM, 0);
|
// Schedule reboot if no clients connect within timeout
|
||||||
|
if (this->reboot_timeout_ != 0) {
|
||||||
|
this->schedule_reboot_timeout_();
|
||||||
|
}
|
||||||
|
|
||||||
|
this->socket_ = socket::socket_ip_loop_monitored(SOCK_STREAM, 0); // monitored for incoming connections
|
||||||
if (this->socket_ == nullptr) {
|
if (this->socket_ == nullptr) {
|
||||||
ESP_LOGW(TAG, "Could not create socket");
|
ESP_LOGW(TAG, "Could not create socket");
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
@@ -87,88 +104,118 @@ void APIServer::setup() {
|
|||||||
|
|
||||||
#ifdef USE_LOGGER
|
#ifdef USE_LOGGER
|
||||||
if (logger::global_logger != nullptr) {
|
if (logger::global_logger != nullptr) {
|
||||||
logger::global_logger->add_on_log_callback([this](int level, const char *tag, const char *message) {
|
logger::global_logger->add_on_log_callback(
|
||||||
for (auto &c : this->clients_) {
|
[this](int level, const char *tag, const char *message, size_t message_len) {
|
||||||
if (!c->remove_)
|
if (this->shutting_down_) {
|
||||||
c->try_send_log_message(level, tag, message);
|
// Don't try to send logs during shutdown
|
||||||
}
|
// as it could result in a recursion and
|
||||||
});
|
// we would be filling a buffer we are trying to clear
|
||||||
}
|
return;
|
||||||
#endif
|
}
|
||||||
|
|
||||||
this->last_connected_ = millis();
|
|
||||||
|
|
||||||
#ifdef USE_ESP32_CAMERA
|
|
||||||
if (esp32_camera::global_esp32_camera != nullptr && !esp32_camera::global_esp32_camera->is_internal()) {
|
|
||||||
esp32_camera::global_esp32_camera->add_image_callback(
|
|
||||||
[this](const std::shared_ptr<esp32_camera::CameraImage> &image) {
|
|
||||||
for (auto &c : this->clients_) {
|
for (auto &c : this->clients_) {
|
||||||
if (!c->remove_)
|
if (!c->flags_.remove)
|
||||||
c->set_camera_state(image);
|
c->try_send_log_message(level, tag, message, message_len);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef USE_CAMERA
|
||||||
|
if (camera::Camera::instance() != nullptr && !camera::Camera::instance()->is_internal()) {
|
||||||
|
camera::Camera::instance()->add_image_callback([this](const std::shared_ptr<camera::CameraImage> &image) {
|
||||||
|
for (auto &c : this->clients_) {
|
||||||
|
if (!c->flags_.remove)
|
||||||
|
c->set_camera_state(image);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void APIServer::schedule_reboot_timeout_() {
|
||||||
|
this->status_set_warning();
|
||||||
|
this->set_timeout("api_reboot", this->reboot_timeout_, []() {
|
||||||
|
if (!global_api_server->is_connected()) {
|
||||||
|
ESP_LOGE(TAG, "No clients; rebooting");
|
||||||
|
App.reboot();
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void APIServer::loop() {
|
void APIServer::loop() {
|
||||||
// Accept new clients
|
// Accept new clients only if the socket exists and has incoming connections
|
||||||
while (true) {
|
if (this->socket_ && this->socket_->ready()) {
|
||||||
struct sockaddr_storage source_addr;
|
while (true) {
|
||||||
socklen_t addr_len = sizeof(source_addr);
|
struct sockaddr_storage source_addr;
|
||||||
auto sock = this->socket_->accept((struct sockaddr *) &source_addr, &addr_len);
|
socklen_t addr_len = sizeof(source_addr);
|
||||||
if (!sock)
|
auto sock = this->socket_->accept_loop_monitored((struct sockaddr *) &source_addr, &addr_len);
|
||||||
break;
|
if (!sock)
|
||||||
ESP_LOGD(TAG, "Accepted %s", sock->getpeername().c_str());
|
break;
|
||||||
|
ESP_LOGD(TAG, "Accept %s", sock->getpeername().c_str());
|
||||||
|
|
||||||
auto *conn = new APIConnection(std::move(sock), this);
|
auto *conn = new APIConnection(std::move(sock), this);
|
||||||
this->clients_.emplace_back(conn);
|
this->clients_.emplace_back(conn);
|
||||||
conn->start();
|
conn->start();
|
||||||
|
|
||||||
|
// Clear warning status and cancel reboot when first client connects
|
||||||
|
if (this->clients_.size() == 1 && this->reboot_timeout_ != 0) {
|
||||||
|
this->status_clear_warning();
|
||||||
|
this->cancel_timeout("api_reboot");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this->clients_.empty()) {
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Process clients and remove disconnected ones in a single pass
|
// Process clients and remove disconnected ones in a single pass
|
||||||
if (!this->clients_.empty()) {
|
// Check network connectivity once for all clients
|
||||||
size_t client_index = 0;
|
if (!network::is_connected()) {
|
||||||
while (client_index < this->clients_.size()) {
|
// Network is down - disconnect all clients
|
||||||
auto &client = this->clients_[client_index];
|
for (auto &client : this->clients_) {
|
||||||
|
client->on_fatal_error();
|
||||||
if (client->remove_) {
|
ESP_LOGW(TAG, "%s: Network down; disconnect", client->get_client_combined_info().c_str());
|
||||||
// Handle disconnection
|
|
||||||
this->client_disconnected_trigger_->trigger(client->client_info_, client->client_peername_);
|
|
||||||
ESP_LOGV(TAG, "Removing connection to %s", client->client_info_.c_str());
|
|
||||||
|
|
||||||
// Swap with the last element and pop (avoids expensive vector shifts)
|
|
||||||
if (client_index < this->clients_.size() - 1) {
|
|
||||||
std::swap(this->clients_[client_index], this->clients_.back());
|
|
||||||
}
|
|
||||||
this->clients_.pop_back();
|
|
||||||
// Don't increment client_index since we need to process the swapped element
|
|
||||||
} else {
|
|
||||||
// Process active client
|
|
||||||
client->loop();
|
|
||||||
client_index++; // Move to next client
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
// Continue to process and clean up the clients below
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->reboot_timeout_ != 0) {
|
size_t client_index = 0;
|
||||||
const uint32_t now = millis();
|
while (client_index < this->clients_.size()) {
|
||||||
if (!this->is_connected()) {
|
auto &client = this->clients_[client_index];
|
||||||
if (now - this->last_connected_ > this->reboot_timeout_) {
|
|
||||||
ESP_LOGE(TAG, "No client connected to API. Rebooting...");
|
if (!client->flags_.remove) {
|
||||||
App.reboot();
|
// Common case: process active client
|
||||||
}
|
client->loop();
|
||||||
this->status_set_warning();
|
client_index++;
|
||||||
} else {
|
continue;
|
||||||
this->last_connected_ = now;
|
|
||||||
this->status_clear_warning();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Rare case: handle disconnection
|
||||||
|
#ifdef USE_API_CLIENT_DISCONNECTED_TRIGGER
|
||||||
|
this->client_disconnected_trigger_->trigger(client->client_info_, client->client_peername_);
|
||||||
|
#endif
|
||||||
|
ESP_LOGV(TAG, "Remove connection %s", client->client_info_.c_str());
|
||||||
|
|
||||||
|
// Swap with the last element and pop (avoids expensive vector shifts)
|
||||||
|
if (client_index < this->clients_.size() - 1) {
|
||||||
|
std::swap(this->clients_[client_index], this->clients_.back());
|
||||||
|
}
|
||||||
|
this->clients_.pop_back();
|
||||||
|
|
||||||
|
// Schedule reboot when last client disconnects
|
||||||
|
if (this->clients_.empty() && this->reboot_timeout_ != 0) {
|
||||||
|
this->schedule_reboot_timeout_();
|
||||||
|
}
|
||||||
|
// Don't increment client_index since we need to process the swapped element
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void APIServer::dump_config() {
|
void APIServer::dump_config() {
|
||||||
ESP_LOGCONFIG(TAG, "API Server:");
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Address: %s:%u", network::get_use_address().c_str(), this->port_);
|
"API Server:\n"
|
||||||
|
" Address: %s:%u",
|
||||||
|
network::get_use_address().c_str(), this->port_);
|
||||||
#ifdef USE_API_NOISE
|
#ifdef USE_API_NOISE
|
||||||
ESP_LOGCONFIG(TAG, " Using noise encryption: %s", YESNO(this->noise_ctx_->has_psk()));
|
ESP_LOGCONFIG(TAG, " Using noise encryption: %s", YESNO(this->noise_ctx_->has_psk()));
|
||||||
if (!this->noise_ctx_->has_psk()) {
|
if (!this->noise_ctx_->has_psk()) {
|
||||||
@@ -179,6 +226,7 @@ void APIServer::dump_config() {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef USE_API_PASSWORD
|
||||||
bool APIServer::uses_password() const { return !this->password_.empty(); }
|
bool APIServer::uses_password() const { return !this->password_.empty(); }
|
||||||
|
|
||||||
bool APIServer::check_password(const std::string &password) const {
|
bool APIServer::check_password(const std::string &password) const {
|
||||||
@@ -209,190 +257,129 @@ bool APIServer::check_password(const std::string &password) const {
|
|||||||
|
|
||||||
return result == 0;
|
return result == 0;
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
void APIServer::handle_disconnect(APIConnection *conn) {}
|
void APIServer::handle_disconnect(APIConnection *conn) {}
|
||||||
|
|
||||||
|
// Macro for entities without extra parameters
|
||||||
|
#define API_DISPATCH_UPDATE(entity_type, entity_name) \
|
||||||
|
void APIServer::on_##entity_name##_update(entity_type *obj) { /* NOLINT(bugprone-macro-parentheses) */ \
|
||||||
|
if (obj->is_internal()) \
|
||||||
|
return; \
|
||||||
|
for (auto &c : this->clients_) \
|
||||||
|
c->send_##entity_name##_state(obj); \
|
||||||
|
}
|
||||||
|
|
||||||
|
// Macro for entities with extra parameters (but parameters not used in send)
|
||||||
|
#define API_DISPATCH_UPDATE_IGNORE_PARAMS(entity_type, entity_name, ...) \
|
||||||
|
void APIServer::on_##entity_name##_update(entity_type *obj, __VA_ARGS__) { /* NOLINT(bugprone-macro-parentheses) */ \
|
||||||
|
if (obj->is_internal()) \
|
||||||
|
return; \
|
||||||
|
for (auto &c : this->clients_) \
|
||||||
|
c->send_##entity_name##_state(obj); \
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef USE_BINARY_SENSOR
|
#ifdef USE_BINARY_SENSOR
|
||||||
void APIServer::on_binary_sensor_update(binary_sensor::BinarySensor *obj, bool state) {
|
API_DISPATCH_UPDATE(binary_sensor::BinarySensor, binary_sensor)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_binary_sensor_state(obj, state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_COVER
|
#ifdef USE_COVER
|
||||||
void APIServer::on_cover_update(cover::Cover *obj) {
|
API_DISPATCH_UPDATE(cover::Cover, cover)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_cover_state(obj);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_FAN
|
#ifdef USE_FAN
|
||||||
void APIServer::on_fan_update(fan::Fan *obj) {
|
API_DISPATCH_UPDATE(fan::Fan, fan)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_fan_state(obj);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_LIGHT
|
#ifdef USE_LIGHT
|
||||||
void APIServer::on_light_update(light::LightState *obj) {
|
API_DISPATCH_UPDATE(light::LightState, light)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_light_state(obj);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_SENSOR
|
#ifdef USE_SENSOR
|
||||||
void APIServer::on_sensor_update(sensor::Sensor *obj, float state) {
|
API_DISPATCH_UPDATE_IGNORE_PARAMS(sensor::Sensor, sensor, float state)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_sensor_state(obj, state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_SWITCH
|
#ifdef USE_SWITCH
|
||||||
void APIServer::on_switch_update(switch_::Switch *obj, bool state) {
|
API_DISPATCH_UPDATE_IGNORE_PARAMS(switch_::Switch, switch, bool state)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_switch_state(obj, state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_TEXT_SENSOR
|
#ifdef USE_TEXT_SENSOR
|
||||||
void APIServer::on_text_sensor_update(text_sensor::TextSensor *obj, const std::string &state) {
|
API_DISPATCH_UPDATE_IGNORE_PARAMS(text_sensor::TextSensor, text_sensor, const std::string &state)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_text_sensor_state(obj, state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_CLIMATE
|
#ifdef USE_CLIMATE
|
||||||
void APIServer::on_climate_update(climate::Climate *obj) {
|
API_DISPATCH_UPDATE(climate::Climate, climate)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_climate_state(obj);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_NUMBER
|
#ifdef USE_NUMBER
|
||||||
void APIServer::on_number_update(number::Number *obj, float state) {
|
API_DISPATCH_UPDATE_IGNORE_PARAMS(number::Number, number, float state)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_number_state(obj, state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_DATETIME_DATE
|
#ifdef USE_DATETIME_DATE
|
||||||
void APIServer::on_date_update(datetime::DateEntity *obj) {
|
API_DISPATCH_UPDATE(datetime::DateEntity, date)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_date_state(obj);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_DATETIME_TIME
|
#ifdef USE_DATETIME_TIME
|
||||||
void APIServer::on_time_update(datetime::TimeEntity *obj) {
|
API_DISPATCH_UPDATE(datetime::TimeEntity, time)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_time_state(obj);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_DATETIME_DATETIME
|
#ifdef USE_DATETIME_DATETIME
|
||||||
void APIServer::on_datetime_update(datetime::DateTimeEntity *obj) {
|
API_DISPATCH_UPDATE(datetime::DateTimeEntity, datetime)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_datetime_state(obj);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_TEXT
|
#ifdef USE_TEXT
|
||||||
void APIServer::on_text_update(text::Text *obj, const std::string &state) {
|
API_DISPATCH_UPDATE_IGNORE_PARAMS(text::Text, text, const std::string &state)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_text_state(obj, state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
void APIServer::on_select_update(select::Select *obj, const std::string &state, size_t index) {
|
API_DISPATCH_UPDATE_IGNORE_PARAMS(select::Select, select, const std::string &state, size_t index)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_select_state(obj, state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_LOCK
|
#ifdef USE_LOCK
|
||||||
void APIServer::on_lock_update(lock::Lock *obj) {
|
API_DISPATCH_UPDATE(lock::Lock, lock)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_lock_state(obj, obj->state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_VALVE
|
#ifdef USE_VALVE
|
||||||
void APIServer::on_valve_update(valve::Valve *obj) {
|
API_DISPATCH_UPDATE(valve::Valve, valve)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_valve_state(obj);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
void APIServer::on_media_player_update(media_player::MediaPlayer *obj) {
|
API_DISPATCH_UPDATE(media_player::MediaPlayer, media_player)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_media_player_state(obj);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_EVENT
|
#ifdef USE_EVENT
|
||||||
|
// Event is a special case - it's the only entity that passes extra parameters to the send method
|
||||||
void APIServer::on_event(event::Event *obj, const std::string &event_type) {
|
void APIServer::on_event(event::Event *obj, const std::string &event_type) {
|
||||||
|
if (obj->is_internal())
|
||||||
|
return;
|
||||||
for (auto &c : this->clients_)
|
for (auto &c : this->clients_)
|
||||||
c->send_event(obj, event_type);
|
c->send_event(obj, event_type);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_UPDATE
|
#ifdef USE_UPDATE
|
||||||
|
// Update is a special case - the method is called on_update, not on_update_update
|
||||||
void APIServer::on_update(update::UpdateEntity *obj) {
|
void APIServer::on_update(update::UpdateEntity *obj) {
|
||||||
|
if (obj->is_internal())
|
||||||
|
return;
|
||||||
for (auto &c : this->clients_)
|
for (auto &c : this->clients_)
|
||||||
c->send_update_state(obj);
|
c->send_update_state(obj);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
void APIServer::on_alarm_control_panel_update(alarm_control_panel::AlarmControlPanel *obj) {
|
API_DISPATCH_UPDATE(alarm_control_panel::AlarmControlPanel, alarm_control_panel)
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_alarm_control_panel_state(obj);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
float APIServer::get_setup_priority() const { return setup_priority::AFTER_WIFI; }
|
float APIServer::get_setup_priority() const { return setup_priority::AFTER_WIFI; }
|
||||||
|
|
||||||
void APIServer::set_port(uint16_t port) { this->port_ = port; }
|
void APIServer::set_port(uint16_t port) { this->port_ = port; }
|
||||||
|
|
||||||
|
#ifdef USE_API_PASSWORD
|
||||||
void APIServer::set_password(const std::string &password) { this->password_ = password; }
|
void APIServer::set_password(const std::string &password) { this->password_ = password; }
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void APIServer::set_batch_delay(uint16_t batch_delay) { this->batch_delay_ = batch_delay; }
|
||||||
|
|
||||||
void APIServer::send_homeassistant_service_call(const HomeassistantServiceResponse &call) {
|
void APIServer::send_homeassistant_service_call(const HomeassistantServiceResponse &call) {
|
||||||
for (auto &client : this->clients_) {
|
for (auto &client : this->clients_) {
|
||||||
@@ -452,7 +439,7 @@ bool APIServer::save_noise_psk(psk_t psk, bool make_active) {
|
|||||||
ESP_LOGW(TAG, "Disconnecting all clients to reset connections");
|
ESP_LOGW(TAG, "Disconnecting all clients to reset connections");
|
||||||
this->set_noise_psk(psk);
|
this->set_noise_psk(psk);
|
||||||
for (auto &c : this->clients_) {
|
for (auto &c : this->clients_) {
|
||||||
c->send_disconnect_request(DisconnectRequest());
|
c->send_message(DisconnectRequest());
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -463,7 +450,7 @@ bool APIServer::save_noise_psk(psk_t psk, bool make_active) {
|
|||||||
#ifdef USE_HOMEASSISTANT_TIME
|
#ifdef USE_HOMEASSISTANT_TIME
|
||||||
void APIServer::request_time() {
|
void APIServer::request_time() {
|
||||||
for (auto &client : this->clients_) {
|
for (auto &client : this->clients_) {
|
||||||
if (!client->remove_ && client->is_authenticated())
|
if (!client->flags_.remove && client->is_authenticated())
|
||||||
client->send_time_request();
|
client->send_time_request();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -472,10 +459,36 @@ void APIServer::request_time() {
|
|||||||
bool APIServer::is_connected() const { return !this->clients_.empty(); }
|
bool APIServer::is_connected() const { return !this->clients_.empty(); }
|
||||||
|
|
||||||
void APIServer::on_shutdown() {
|
void APIServer::on_shutdown() {
|
||||||
for (auto &c : this->clients_) {
|
this->shutting_down_ = true;
|
||||||
c->send_disconnect_request(DisconnectRequest());
|
|
||||||
|
// Close the listening socket to prevent new connections
|
||||||
|
if (this->socket_) {
|
||||||
|
this->socket_->close();
|
||||||
|
this->socket_ = nullptr;
|
||||||
}
|
}
|
||||||
delay(10);
|
|
||||||
|
// Change batch delay to 5ms for quick flushing during shutdown
|
||||||
|
this->batch_delay_ = 5;
|
||||||
|
|
||||||
|
// Send disconnect requests to all connected clients
|
||||||
|
for (auto &c : this->clients_) {
|
||||||
|
if (!c->send_message(DisconnectRequest())) {
|
||||||
|
// If we can't send the disconnect request directly (tx_buffer full),
|
||||||
|
// schedule it at the front of the batch so it will be sent with priority
|
||||||
|
c->schedule_message_front_(nullptr, &APIConnection::try_send_disconnect_request, DisconnectRequest::MESSAGE_TYPE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool APIServer::teardown() {
|
||||||
|
// If network is disconnected, no point trying to flush buffers
|
||||||
|
if (!network::is_connected()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
this->loop();
|
||||||
|
|
||||||
|
// Return true only when all clients have been torn down
|
||||||
|
return this->clients_.empty();
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
|
|||||||
@@ -25,6 +25,11 @@ struct SavedNoisePsk {
|
|||||||
} PACKED; // NOLINT
|
} PACKED; // NOLINT
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifndef USE_API_YAML_SERVICES
|
||||||
|
// Forward declaration of helper function
|
||||||
|
const std::vector<UserServiceDescriptor *> &get_empty_user_services_instance();
|
||||||
|
#endif
|
||||||
|
|
||||||
class APIServer : public Component, public Controller {
|
class APIServer : public Component, public Controller {
|
||||||
public:
|
public:
|
||||||
APIServer();
|
APIServer();
|
||||||
@@ -34,11 +39,19 @@ class APIServer : public Component, public Controller {
|
|||||||
void loop() override;
|
void loop() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
void on_shutdown() override;
|
void on_shutdown() override;
|
||||||
|
bool teardown() override;
|
||||||
|
#ifdef USE_API_PASSWORD
|
||||||
bool check_password(const std::string &password) const;
|
bool check_password(const std::string &password) const;
|
||||||
bool uses_password() const;
|
bool uses_password() const;
|
||||||
void set_port(uint16_t port);
|
|
||||||
void set_password(const std::string &password);
|
void set_password(const std::string &password);
|
||||||
|
#endif
|
||||||
|
void set_port(uint16_t port);
|
||||||
void set_reboot_timeout(uint32_t reboot_timeout);
|
void set_reboot_timeout(uint32_t reboot_timeout);
|
||||||
|
void set_batch_delay(uint16_t batch_delay);
|
||||||
|
uint16_t get_batch_delay() const { return batch_delay_; }
|
||||||
|
|
||||||
|
// Get reference to shared buffer for API connections
|
||||||
|
std::vector<uint8_t> &get_shared_buffer_ref() { return shared_write_buffer_; }
|
||||||
|
|
||||||
#ifdef USE_API_NOISE
|
#ifdef USE_API_NOISE
|
||||||
bool save_noise_psk(psk_t psk, bool make_active = true);
|
bool save_noise_psk(psk_t psk, bool make_active = true);
|
||||||
@@ -48,7 +61,7 @@ class APIServer : public Component, public Controller {
|
|||||||
|
|
||||||
void handle_disconnect(APIConnection *conn);
|
void handle_disconnect(APIConnection *conn);
|
||||||
#ifdef USE_BINARY_SENSOR
|
#ifdef USE_BINARY_SENSOR
|
||||||
void on_binary_sensor_update(binary_sensor::BinarySensor *obj, bool state) override;
|
void on_binary_sensor_update(binary_sensor::BinarySensor *obj) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_COVER
|
#ifdef USE_COVER
|
||||||
void on_cover_update(cover::Cover *obj) override;
|
void on_cover_update(cover::Cover *obj) override;
|
||||||
@@ -99,7 +112,18 @@ class APIServer : public Component, public Controller {
|
|||||||
void on_media_player_update(media_player::MediaPlayer *obj) override;
|
void on_media_player_update(media_player::MediaPlayer *obj) override;
|
||||||
#endif
|
#endif
|
||||||
void send_homeassistant_service_call(const HomeassistantServiceResponse &call);
|
void send_homeassistant_service_call(const HomeassistantServiceResponse &call);
|
||||||
void register_user_service(UserServiceDescriptor *descriptor) { this->user_services_.push_back(descriptor); }
|
void register_user_service(UserServiceDescriptor *descriptor) {
|
||||||
|
#ifdef USE_API_YAML_SERVICES
|
||||||
|
// Vector is pre-allocated when services are defined in YAML
|
||||||
|
this->user_services_.push_back(descriptor);
|
||||||
|
#else
|
||||||
|
// Lazy allocate vector on first use for CustomAPIDevice
|
||||||
|
if (!this->user_services_) {
|
||||||
|
this->user_services_ = std::make_unique<std::vector<UserServiceDescriptor *>>();
|
||||||
|
}
|
||||||
|
this->user_services_->push_back(descriptor);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
#ifdef USE_HOMEASSISTANT_TIME
|
#ifdef USE_HOMEASSISTANT_TIME
|
||||||
void request_time();
|
void request_time();
|
||||||
#endif
|
#endif
|
||||||
@@ -128,24 +152,63 @@ class APIServer : public Component, public Controller {
|
|||||||
void get_home_assistant_state(std::string entity_id, optional<std::string> attribute,
|
void get_home_assistant_state(std::string entity_id, optional<std::string> attribute,
|
||||||
std::function<void(std::string)> f);
|
std::function<void(std::string)> f);
|
||||||
const std::vector<HomeAssistantStateSubscription> &get_state_subs() const;
|
const std::vector<HomeAssistantStateSubscription> &get_state_subs() const;
|
||||||
const std::vector<UserServiceDescriptor *> &get_user_services() const { return this->user_services_; }
|
const std::vector<UserServiceDescriptor *> &get_user_services() const {
|
||||||
|
#ifdef USE_API_YAML_SERVICES
|
||||||
|
return this->user_services_;
|
||||||
|
#else
|
||||||
|
if (this->user_services_) {
|
||||||
|
return *this->user_services_;
|
||||||
|
}
|
||||||
|
// Return reference to global empty instance (no guard needed)
|
||||||
|
return get_empty_user_services_instance();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef USE_API_CLIENT_CONNECTED_TRIGGER
|
||||||
Trigger<std::string, std::string> *get_client_connected_trigger() const { return this->client_connected_trigger_; }
|
Trigger<std::string, std::string> *get_client_connected_trigger() const { return this->client_connected_trigger_; }
|
||||||
|
#endif
|
||||||
|
#ifdef USE_API_CLIENT_DISCONNECTED_TRIGGER
|
||||||
Trigger<std::string, std::string> *get_client_disconnected_trigger() const {
|
Trigger<std::string, std::string> *get_client_disconnected_trigger() const {
|
||||||
return this->client_disconnected_trigger_;
|
return this->client_disconnected_trigger_;
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
void schedule_reboot_timeout_();
|
||||||
|
// Pointers and pointer-like types first (4 bytes each)
|
||||||
std::unique_ptr<socket::Socket> socket_ = nullptr;
|
std::unique_ptr<socket::Socket> socket_ = nullptr;
|
||||||
uint16_t port_{6053};
|
#ifdef USE_API_CLIENT_CONNECTED_TRIGGER
|
||||||
uint32_t reboot_timeout_{300000};
|
|
||||||
uint32_t last_connected_{0};
|
|
||||||
std::vector<std::unique_ptr<APIConnection>> clients_;
|
|
||||||
std::string password_;
|
|
||||||
std::vector<HomeAssistantStateSubscription> state_subs_;
|
|
||||||
std::vector<UserServiceDescriptor *> user_services_;
|
|
||||||
Trigger<std::string, std::string> *client_connected_trigger_ = new Trigger<std::string, std::string>();
|
Trigger<std::string, std::string> *client_connected_trigger_ = new Trigger<std::string, std::string>();
|
||||||
|
#endif
|
||||||
|
#ifdef USE_API_CLIENT_DISCONNECTED_TRIGGER
|
||||||
Trigger<std::string, std::string> *client_disconnected_trigger_ = new Trigger<std::string, std::string>();
|
Trigger<std::string, std::string> *client_disconnected_trigger_ = new Trigger<std::string, std::string>();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// 4-byte aligned types
|
||||||
|
uint32_t reboot_timeout_{300000};
|
||||||
|
|
||||||
|
// Vectors and strings (12 bytes each on 32-bit)
|
||||||
|
std::vector<std::unique_ptr<APIConnection>> clients_;
|
||||||
|
#ifdef USE_API_PASSWORD
|
||||||
|
std::string password_;
|
||||||
|
#endif
|
||||||
|
std::vector<uint8_t> shared_write_buffer_; // Shared proto write buffer for all connections
|
||||||
|
std::vector<HomeAssistantStateSubscription> state_subs_;
|
||||||
|
#ifdef USE_API_YAML_SERVICES
|
||||||
|
// When services are defined in YAML, we know at compile time that services will be registered
|
||||||
|
std::vector<UserServiceDescriptor *> user_services_;
|
||||||
|
#else
|
||||||
|
// Services can still be registered at runtime by CustomAPIDevice components even when not
|
||||||
|
// defined in YAML. Using unique_ptr allows lazy allocation, saving 12 bytes in the common
|
||||||
|
// case where no services (YAML or custom) are used.
|
||||||
|
std::unique_ptr<std::vector<UserServiceDescriptor *>> user_services_;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// Group smaller types together
|
||||||
|
uint16_t port_{6053};
|
||||||
|
uint16_t batch_delay_{100};
|
||||||
|
bool shutting_down_ = false;
|
||||||
|
// 5 bytes used, 3 bytes padding
|
||||||
|
|
||||||
#ifdef USE_API_NOISE
|
#ifdef USE_API_NOISE
|
||||||
std::shared_ptr<APINoiseContext> noise_ctx_ = std::make_shared<APINoiseContext>();
|
std::shared_ptr<APINoiseContext> noise_ctx_ = std::make_shared<APINoiseContext>();
|
||||||
|
|||||||
@@ -4,9 +4,15 @@ import asyncio
|
|||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
import logging
|
import logging
|
||||||
from typing import TYPE_CHECKING, Any
|
from typing import TYPE_CHECKING, Any
|
||||||
|
import warnings
|
||||||
|
|
||||||
from aioesphomeapi import APIClient
|
# Suppress protobuf version warnings
|
||||||
from aioesphomeapi.log_runner import async_run
|
with warnings.catch_warnings():
|
||||||
|
warnings.filterwarnings(
|
||||||
|
"ignore", category=UserWarning, message=".*Protobuf gencode version.*"
|
||||||
|
)
|
||||||
|
from aioesphomeapi import APIClient, parse_log_message
|
||||||
|
from aioesphomeapi.log_runner import async_run
|
||||||
|
|
||||||
from esphome.const import CONF_KEY, CONF_PASSWORD, CONF_PORT, __version__
|
from esphome.const import CONF_KEY, CONF_PASSWORD, CONF_PORT, __version__
|
||||||
from esphome.core import CORE
|
from esphome.core import CORE
|
||||||
@@ -29,8 +35,8 @@ async def async_run_logs(config: dict[str, Any], address: str) -> None:
|
|||||||
port: int = int(conf[CONF_PORT])
|
port: int = int(conf[CONF_PORT])
|
||||||
password: str = conf[CONF_PASSWORD]
|
password: str = conf[CONF_PASSWORD]
|
||||||
noise_psk: str | None = None
|
noise_psk: str | None = None
|
||||||
if CONF_ENCRYPTION in conf:
|
if (encryption := conf.get(CONF_ENCRYPTION)) and (key := encryption.get(CONF_KEY)):
|
||||||
noise_psk = conf[CONF_ENCRYPTION][CONF_KEY]
|
noise_psk = key
|
||||||
_LOGGER.info("Starting log output from %s using esphome API", address)
|
_LOGGER.info("Starting log output from %s using esphome API", address)
|
||||||
cli = APIClient(
|
cli = APIClient(
|
||||||
address,
|
address,
|
||||||
@@ -46,9 +52,10 @@ async def async_run_logs(config: dict[str, Any], address: str) -> None:
|
|||||||
time_ = datetime.now()
|
time_ = datetime.now()
|
||||||
message: bytes = msg.message
|
message: bytes = msg.message
|
||||||
text = message.decode("utf8", "backslashreplace")
|
text = message.decode("utf8", "backslashreplace")
|
||||||
if dashboard:
|
for parsed_msg in parse_log_message(
|
||||||
text = text.replace("\033", "\\033")
|
text, f"[{time_.hour:02}:{time_.minute:02}:{time_.second:02}]"
|
||||||
print(f"[{time_.hour:02}:{time_.minute:02}:{time_.second:02}]{text}")
|
):
|
||||||
|
print(parsed_msg.replace("\033", "\\033") if dashboard else parsed_msg)
|
||||||
|
|
||||||
stop = await async_run(cli, on_log, name=name)
|
stop = await async_run(cli, on_log, name=name)
|
||||||
try:
|
try:
|
||||||
|
|||||||
@@ -3,8 +3,8 @@
|
|||||||
#include "api_server.h"
|
#include "api_server.h"
|
||||||
#ifdef USE_API
|
#ifdef USE_API
|
||||||
#include "api_pb2.h"
|
#include "api_pb2.h"
|
||||||
#include "esphome/core/helpers.h"
|
|
||||||
#include "esphome/core/automation.h"
|
#include "esphome/core/automation.h"
|
||||||
|
#include "esphome/core/helpers.h"
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
#include "list_entities.h"
|
#include "list_entities.h"
|
||||||
#ifdef USE_API
|
#ifdef USE_API
|
||||||
#include "api_connection.h"
|
#include "api_connection.h"
|
||||||
|
#include "api_pb2.h"
|
||||||
#include "esphome/core/application.h"
|
#include "esphome/core/application.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
#include "esphome/core/util.h"
|
#include "esphome/core/util.h"
|
||||||
@@ -8,155 +9,85 @@
|
|||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace api {
|
namespace api {
|
||||||
|
|
||||||
|
// Generate entity handler implementations using macros
|
||||||
#ifdef USE_BINARY_SENSOR
|
#ifdef USE_BINARY_SENSOR
|
||||||
bool ListEntitiesIterator::on_binary_sensor(binary_sensor::BinarySensor *binary_sensor) {
|
LIST_ENTITIES_HANDLER(binary_sensor, binary_sensor::BinarySensor, ListEntitiesBinarySensorResponse)
|
||||||
this->client_->send_binary_sensor_info(binary_sensor);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_COVER
|
#ifdef USE_COVER
|
||||||
bool ListEntitiesIterator::on_cover(cover::Cover *cover) {
|
LIST_ENTITIES_HANDLER(cover, cover::Cover, ListEntitiesCoverResponse)
|
||||||
this->client_->send_cover_info(cover);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_FAN
|
#ifdef USE_FAN
|
||||||
bool ListEntitiesIterator::on_fan(fan::Fan *fan) {
|
LIST_ENTITIES_HANDLER(fan, fan::Fan, ListEntitiesFanResponse)
|
||||||
this->client_->send_fan_info(fan);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LIGHT
|
#ifdef USE_LIGHT
|
||||||
bool ListEntitiesIterator::on_light(light::LightState *light) {
|
LIST_ENTITIES_HANDLER(light, light::LightState, ListEntitiesLightResponse)
|
||||||
this->client_->send_light_info(light);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SENSOR
|
#ifdef USE_SENSOR
|
||||||
bool ListEntitiesIterator::on_sensor(sensor::Sensor *sensor) {
|
LIST_ENTITIES_HANDLER(sensor, sensor::Sensor, ListEntitiesSensorResponse)
|
||||||
this->client_->send_sensor_info(sensor);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SWITCH
|
#ifdef USE_SWITCH
|
||||||
bool ListEntitiesIterator::on_switch(switch_::Switch *a_switch) {
|
LIST_ENTITIES_HANDLER(switch, switch_::Switch, ListEntitiesSwitchResponse)
|
||||||
this->client_->send_switch_info(a_switch);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BUTTON
|
#ifdef USE_BUTTON
|
||||||
bool ListEntitiesIterator::on_button(button::Button *button) {
|
LIST_ENTITIES_HANDLER(button, button::Button, ListEntitiesButtonResponse)
|
||||||
this->client_->send_button_info(button);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT_SENSOR
|
#ifdef USE_TEXT_SENSOR
|
||||||
bool ListEntitiesIterator::on_text_sensor(text_sensor::TextSensor *text_sensor) {
|
LIST_ENTITIES_HANDLER(text_sensor, text_sensor::TextSensor, ListEntitiesTextSensorResponse)
|
||||||
this->client_->send_text_sensor_info(text_sensor);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LOCK
|
#ifdef USE_LOCK
|
||||||
bool ListEntitiesIterator::on_lock(lock::Lock *a_lock) {
|
LIST_ENTITIES_HANDLER(lock, lock::Lock, ListEntitiesLockResponse)
|
||||||
this->client_->send_lock_info(a_lock);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VALVE
|
#ifdef USE_VALVE
|
||||||
bool ListEntitiesIterator::on_valve(valve::Valve *valve) {
|
LIST_ENTITIES_HANDLER(valve, valve::Valve, ListEntitiesValveResponse)
|
||||||
this->client_->send_valve_info(valve);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef USE_CAMERA
|
||||||
bool ListEntitiesIterator::on_end() { return this->client_->send_list_info_done(); }
|
LIST_ENTITIES_HANDLER(camera, camera::Camera, ListEntitiesCameraResponse)
|
||||||
ListEntitiesIterator::ListEntitiesIterator(APIConnection *client) : client_(client) {}
|
|
||||||
bool ListEntitiesIterator::on_service(UserServiceDescriptor *service) {
|
|
||||||
auto resp = service->encode_list_service_response();
|
|
||||||
return this->client_->send_list_entities_services_response(resp);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef USE_ESP32_CAMERA
|
|
||||||
bool ListEntitiesIterator::on_camera(esp32_camera::ESP32Camera *camera) {
|
|
||||||
this->client_->send_camera_info(camera);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_CLIMATE
|
#ifdef USE_CLIMATE
|
||||||
bool ListEntitiesIterator::on_climate(climate::Climate *climate) {
|
LIST_ENTITIES_HANDLER(climate, climate::Climate, ListEntitiesClimateResponse)
|
||||||
this->client_->send_climate_info(climate);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_NUMBER
|
#ifdef USE_NUMBER
|
||||||
bool ListEntitiesIterator::on_number(number::Number *number) {
|
LIST_ENTITIES_HANDLER(number, number::Number, ListEntitiesNumberResponse)
|
||||||
this->client_->send_number_info(number);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_DATETIME_DATE
|
#ifdef USE_DATETIME_DATE
|
||||||
bool ListEntitiesIterator::on_date(datetime::DateEntity *date) {
|
LIST_ENTITIES_HANDLER(date, datetime::DateEntity, ListEntitiesDateResponse)
|
||||||
this->client_->send_date_info(date);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_DATETIME_TIME
|
#ifdef USE_DATETIME_TIME
|
||||||
bool ListEntitiesIterator::on_time(datetime::TimeEntity *time) {
|
LIST_ENTITIES_HANDLER(time, datetime::TimeEntity, ListEntitiesTimeResponse)
|
||||||
this->client_->send_time_info(time);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_DATETIME_DATETIME
|
#ifdef USE_DATETIME_DATETIME
|
||||||
bool ListEntitiesIterator::on_datetime(datetime::DateTimeEntity *datetime) {
|
LIST_ENTITIES_HANDLER(datetime, datetime::DateTimeEntity, ListEntitiesDateTimeResponse)
|
||||||
this->client_->send_datetime_info(datetime);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_TEXT
|
#ifdef USE_TEXT
|
||||||
bool ListEntitiesIterator::on_text(text::Text *text) {
|
LIST_ENTITIES_HANDLER(text, text::Text, ListEntitiesTextResponse)
|
||||||
this->client_->send_text_info(text);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
bool ListEntitiesIterator::on_select(select::Select *select) {
|
LIST_ENTITIES_HANDLER(select, select::Select, ListEntitiesSelectResponse)
|
||||||
this->client_->send_select_info(select);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
bool ListEntitiesIterator::on_media_player(media_player::MediaPlayer *media_player) {
|
LIST_ENTITIES_HANDLER(media_player, media_player::MediaPlayer, ListEntitiesMediaPlayerResponse)
|
||||||
this->client_->send_media_player_info(media_player);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
bool ListEntitiesIterator::on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) {
|
LIST_ENTITIES_HANDLER(alarm_control_panel, alarm_control_panel::AlarmControlPanel,
|
||||||
this->client_->send_alarm_control_panel_info(a_alarm_control_panel);
|
ListEntitiesAlarmControlPanelResponse)
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_EVENT
|
#ifdef USE_EVENT
|
||||||
bool ListEntitiesIterator::on_event(event::Event *event) {
|
LIST_ENTITIES_HANDLER(event, event::Event, ListEntitiesEventResponse)
|
||||||
this->client_->send_event_info(event);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_UPDATE
|
#ifdef USE_UPDATE
|
||||||
bool ListEntitiesIterator::on_update(update::UpdateEntity *update) {
|
LIST_ENTITIES_HANDLER(update, update::UpdateEntity, ListEntitiesUpdateResponse)
|
||||||
this->client_->send_update_info(update);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
// Special cases that don't follow the pattern
|
||||||
|
bool ListEntitiesIterator::on_end() { return this->client_->send_list_info_done(); }
|
||||||
|
|
||||||
|
ListEntitiesIterator::ListEntitiesIterator(APIConnection *client) : client_(client) {}
|
||||||
|
|
||||||
|
bool ListEntitiesIterator::on_service(UserServiceDescriptor *service) {
|
||||||
|
auto resp = service->encode_list_service_response();
|
||||||
|
return this->client_->send_message(resp);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -9,75 +9,83 @@ namespace api {
|
|||||||
|
|
||||||
class APIConnection;
|
class APIConnection;
|
||||||
|
|
||||||
|
// Macro for generating ListEntitiesIterator handlers
|
||||||
|
// Calls schedule_message_ with try_send_*_info
|
||||||
|
#define LIST_ENTITIES_HANDLER(entity_type, EntityClass, ResponseType) \
|
||||||
|
bool ListEntitiesIterator::on_##entity_type(EntityClass *entity) { /* NOLINT(bugprone-macro-parentheses) */ \
|
||||||
|
return this->client_->schedule_message_(entity, &APIConnection::try_send_##entity_type##_info, \
|
||||||
|
ResponseType::MESSAGE_TYPE); \
|
||||||
|
}
|
||||||
|
|
||||||
class ListEntitiesIterator : public ComponentIterator {
|
class ListEntitiesIterator : public ComponentIterator {
|
||||||
public:
|
public:
|
||||||
ListEntitiesIterator(APIConnection *client);
|
ListEntitiesIterator(APIConnection *client);
|
||||||
#ifdef USE_BINARY_SENSOR
|
#ifdef USE_BINARY_SENSOR
|
||||||
bool on_binary_sensor(binary_sensor::BinarySensor *binary_sensor) override;
|
bool on_binary_sensor(binary_sensor::BinarySensor *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_COVER
|
#ifdef USE_COVER
|
||||||
bool on_cover(cover::Cover *cover) override;
|
bool on_cover(cover::Cover *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_FAN
|
#ifdef USE_FAN
|
||||||
bool on_fan(fan::Fan *fan) override;
|
bool on_fan(fan::Fan *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LIGHT
|
#ifdef USE_LIGHT
|
||||||
bool on_light(light::LightState *light) override;
|
bool on_light(light::LightState *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SENSOR
|
#ifdef USE_SENSOR
|
||||||
bool on_sensor(sensor::Sensor *sensor) override;
|
bool on_sensor(sensor::Sensor *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SWITCH
|
#ifdef USE_SWITCH
|
||||||
bool on_switch(switch_::Switch *a_switch) override;
|
bool on_switch(switch_::Switch *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BUTTON
|
#ifdef USE_BUTTON
|
||||||
bool on_button(button::Button *button) override;
|
bool on_button(button::Button *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT_SENSOR
|
#ifdef USE_TEXT_SENSOR
|
||||||
bool on_text_sensor(text_sensor::TextSensor *text_sensor) override;
|
bool on_text_sensor(text_sensor::TextSensor *entity) override;
|
||||||
#endif
|
#endif
|
||||||
bool on_service(UserServiceDescriptor *service) override;
|
bool on_service(UserServiceDescriptor *service) override;
|
||||||
#ifdef USE_ESP32_CAMERA
|
#ifdef USE_CAMERA
|
||||||
bool on_camera(esp32_camera::ESP32Camera *camera) override;
|
bool on_camera(camera::Camera *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_CLIMATE
|
#ifdef USE_CLIMATE
|
||||||
bool on_climate(climate::Climate *climate) override;
|
bool on_climate(climate::Climate *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_NUMBER
|
#ifdef USE_NUMBER
|
||||||
bool on_number(number::Number *number) override;
|
bool on_number(number::Number *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATE
|
#ifdef USE_DATETIME_DATE
|
||||||
bool on_date(datetime::DateEntity *date) override;
|
bool on_date(datetime::DateEntity *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_TIME
|
#ifdef USE_DATETIME_TIME
|
||||||
bool on_time(datetime::TimeEntity *time) override;
|
bool on_time(datetime::TimeEntity *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATETIME
|
#ifdef USE_DATETIME_DATETIME
|
||||||
bool on_datetime(datetime::DateTimeEntity *datetime) override;
|
bool on_datetime(datetime::DateTimeEntity *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT
|
#ifdef USE_TEXT
|
||||||
bool on_text(text::Text *text) override;
|
bool on_text(text::Text *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
bool on_select(select::Select *select) override;
|
bool on_select(select::Select *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LOCK
|
#ifdef USE_LOCK
|
||||||
bool on_lock(lock::Lock *a_lock) override;
|
bool on_lock(lock::Lock *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VALVE
|
#ifdef USE_VALVE
|
||||||
bool on_valve(valve::Valve *valve) override;
|
bool on_valve(valve::Valve *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
bool on_media_player(media_player::MediaPlayer *media_player) override;
|
bool on_media_player(media_player::MediaPlayer *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
bool on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) override;
|
bool on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_EVENT
|
#ifdef USE_EVENT
|
||||||
bool on_event(event::Event *event) override;
|
bool on_event(event::Event *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_UPDATE
|
#ifdef USE_UPDATE
|
||||||
bool on_update(update::UpdateEntity *update) override;
|
bool on_update(update::UpdateEntity *entity) override;
|
||||||
#endif
|
#endif
|
||||||
bool on_end() override;
|
bool on_end() override;
|
||||||
bool completed() { return this->state_ == IteratorState::NONE; }
|
bool completed() { return this->state_ == IteratorState::NONE; }
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
#include "proto.h"
|
#include "proto.h"
|
||||||
#include <cinttypes>
|
#include <cinttypes>
|
||||||
|
#include "esphome/core/helpers.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "esphome/core/component.h"
|
#include "esphome/core/component.h"
|
||||||
#include "esphome/core/log.h"
|
|
||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
@@ -216,7 +216,7 @@ class ProtoWriteBuffer {
|
|||||||
this->buffer_->insert(this->buffer_->end(), data, data + len);
|
this->buffer_->insert(this->buffer_->end(), data, data + len);
|
||||||
}
|
}
|
||||||
void encode_string(uint32_t field_id, const std::string &value, bool force = false) {
|
void encode_string(uint32_t field_id, const std::string &value, bool force = false) {
|
||||||
this->encode_string(field_id, value.data(), value.size());
|
this->encode_string(field_id, value.data(), value.size(), force);
|
||||||
}
|
}
|
||||||
void encode_bytes(uint32_t field_id, const uint8_t *data, size_t len, bool force = false) {
|
void encode_bytes(uint32_t field_id, const uint8_t *data, size_t len, bool force = false) {
|
||||||
this->encode_string(field_id, reinterpret_cast<const char *>(data), len, force);
|
this->encode_string(field_id, reinterpret_cast<const char *>(data), len, force);
|
||||||
@@ -327,12 +327,15 @@ class ProtoWriteBuffer {
|
|||||||
class ProtoMessage {
|
class ProtoMessage {
|
||||||
public:
|
public:
|
||||||
virtual ~ProtoMessage() = default;
|
virtual ~ProtoMessage() = default;
|
||||||
virtual void encode(ProtoWriteBuffer buffer) const = 0;
|
// Default implementation for messages with no fields
|
||||||
|
virtual void encode(ProtoWriteBuffer buffer) const {}
|
||||||
void decode(const uint8_t *buffer, size_t length);
|
void decode(const uint8_t *buffer, size_t length);
|
||||||
virtual void calculate_size(uint32_t &total_size) const = 0;
|
// Default implementation for messages with no fields
|
||||||
|
virtual void calculate_size(uint32_t &total_size) const {}
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
std::string dump() const;
|
std::string dump() const;
|
||||||
virtual void dump_to(std::string &out) const = 0;
|
virtual void dump_to(std::string &out) const = 0;
|
||||||
|
virtual const char *message_name() const { return "unknown"; }
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@@ -360,11 +363,11 @@ class ProtoService {
|
|||||||
* @return A ProtoWriteBuffer object with the reserved size.
|
* @return A ProtoWriteBuffer object with the reserved size.
|
||||||
*/
|
*/
|
||||||
virtual ProtoWriteBuffer create_buffer(uint32_t reserve_size) = 0;
|
virtual ProtoWriteBuffer create_buffer(uint32_t reserve_size) = 0;
|
||||||
virtual bool send_buffer(ProtoWriteBuffer buffer, uint32_t message_type) = 0;
|
virtual bool send_buffer(ProtoWriteBuffer buffer, uint16_t message_type) = 0;
|
||||||
virtual bool read_message(uint32_t msg_size, uint32_t msg_type, uint8_t *msg_data) = 0;
|
virtual void read_message(uint32_t msg_size, uint32_t msg_type, uint8_t *msg_data) = 0;
|
||||||
|
|
||||||
// Optimized method that pre-allocates buffer based on message size
|
// Optimized method that pre-allocates buffer based on message size
|
||||||
template<class C> bool send_message_(const C &msg, uint32_t message_type) {
|
bool send_message_(const ProtoMessage &msg, uint16_t message_type) {
|
||||||
uint32_t msg_size = 0;
|
uint32_t msg_size = 0;
|
||||||
msg.calculate_size(msg_size);
|
msg.calculate_size(msg_size);
|
||||||
|
|
||||||
@@ -377,6 +380,26 @@ class ProtoService {
|
|||||||
// Send the buffer
|
// Send the buffer
|
||||||
return this->send_buffer(buffer, message_type);
|
return this->send_buffer(buffer, message_type);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Authentication helper methods
|
||||||
|
bool check_connection_setup_() {
|
||||||
|
if (!this->is_connection_setup()) {
|
||||||
|
this->on_no_setup_connection();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool check_authenticated_() {
|
||||||
|
if (!this->check_connection_setup_()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (!this->is_authenticated()) {
|
||||||
|
this->on_unauthenticated_access();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
|
|||||||
@@ -6,81 +6,67 @@
|
|||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace api {
|
namespace api {
|
||||||
|
|
||||||
|
// Generate entity handler implementations using macros
|
||||||
#ifdef USE_BINARY_SENSOR
|
#ifdef USE_BINARY_SENSOR
|
||||||
bool InitialStateIterator::on_binary_sensor(binary_sensor::BinarySensor *binary_sensor) {
|
INITIAL_STATE_HANDLER(binary_sensor, binary_sensor::BinarySensor)
|
||||||
return this->client_->send_binary_sensor_state(binary_sensor, binary_sensor->state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_COVER
|
#ifdef USE_COVER
|
||||||
bool InitialStateIterator::on_cover(cover::Cover *cover) { return this->client_->send_cover_state(cover); }
|
INITIAL_STATE_HANDLER(cover, cover::Cover)
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_FAN
|
#ifdef USE_FAN
|
||||||
bool InitialStateIterator::on_fan(fan::Fan *fan) { return this->client_->send_fan_state(fan); }
|
INITIAL_STATE_HANDLER(fan, fan::Fan)
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LIGHT
|
#ifdef USE_LIGHT
|
||||||
bool InitialStateIterator::on_light(light::LightState *light) { return this->client_->send_light_state(light); }
|
INITIAL_STATE_HANDLER(light, light::LightState)
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SENSOR
|
#ifdef USE_SENSOR
|
||||||
bool InitialStateIterator::on_sensor(sensor::Sensor *sensor) {
|
INITIAL_STATE_HANDLER(sensor, sensor::Sensor)
|
||||||
return this->client_->send_sensor_state(sensor, sensor->state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SWITCH
|
#ifdef USE_SWITCH
|
||||||
bool InitialStateIterator::on_switch(switch_::Switch *a_switch) {
|
INITIAL_STATE_HANDLER(switch, switch_::Switch)
|
||||||
return this->client_->send_switch_state(a_switch, a_switch->state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT_SENSOR
|
#ifdef USE_TEXT_SENSOR
|
||||||
bool InitialStateIterator::on_text_sensor(text_sensor::TextSensor *text_sensor) {
|
INITIAL_STATE_HANDLER(text_sensor, text_sensor::TextSensor)
|
||||||
return this->client_->send_text_sensor_state(text_sensor, text_sensor->state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_CLIMATE
|
#ifdef USE_CLIMATE
|
||||||
bool InitialStateIterator::on_climate(climate::Climate *climate) { return this->client_->send_climate_state(climate); }
|
INITIAL_STATE_HANDLER(climate, climate::Climate)
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_NUMBER
|
#ifdef USE_NUMBER
|
||||||
bool InitialStateIterator::on_number(number::Number *number) {
|
INITIAL_STATE_HANDLER(number, number::Number)
|
||||||
return this->client_->send_number_state(number, number->state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATE
|
#ifdef USE_DATETIME_DATE
|
||||||
bool InitialStateIterator::on_date(datetime::DateEntity *date) { return this->client_->send_date_state(date); }
|
INITIAL_STATE_HANDLER(date, datetime::DateEntity)
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_TIME
|
#ifdef USE_DATETIME_TIME
|
||||||
bool InitialStateIterator::on_time(datetime::TimeEntity *time) { return this->client_->send_time_state(time); }
|
INITIAL_STATE_HANDLER(time, datetime::TimeEntity)
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATETIME
|
#ifdef USE_DATETIME_DATETIME
|
||||||
bool InitialStateIterator::on_datetime(datetime::DateTimeEntity *datetime) {
|
INITIAL_STATE_HANDLER(datetime, datetime::DateTimeEntity)
|
||||||
return this->client_->send_datetime_state(datetime);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT
|
#ifdef USE_TEXT
|
||||||
bool InitialStateIterator::on_text(text::Text *text) { return this->client_->send_text_state(text, text->state); }
|
INITIAL_STATE_HANDLER(text, text::Text)
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
bool InitialStateIterator::on_select(select::Select *select) {
|
INITIAL_STATE_HANDLER(select, select::Select)
|
||||||
return this->client_->send_select_state(select, select->state);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LOCK
|
#ifdef USE_LOCK
|
||||||
bool InitialStateIterator::on_lock(lock::Lock *a_lock) { return this->client_->send_lock_state(a_lock, a_lock->state); }
|
INITIAL_STATE_HANDLER(lock, lock::Lock)
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VALVE
|
#ifdef USE_VALVE
|
||||||
bool InitialStateIterator::on_valve(valve::Valve *valve) { return this->client_->send_valve_state(valve); }
|
INITIAL_STATE_HANDLER(valve, valve::Valve)
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
bool InitialStateIterator::on_media_player(media_player::MediaPlayer *media_player) {
|
INITIAL_STATE_HANDLER(media_player, media_player::MediaPlayer)
|
||||||
return this->client_->send_media_player_state(media_player);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
bool InitialStateIterator::on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) {
|
INITIAL_STATE_HANDLER(alarm_control_panel, alarm_control_panel::AlarmControlPanel)
|
||||||
return this->client_->send_alarm_control_panel_state(a_alarm_control_panel);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_UPDATE
|
#ifdef USE_UPDATE
|
||||||
bool InitialStateIterator::on_update(update::UpdateEntity *update) { return this->client_->send_update_state(update); }
|
INITIAL_STATE_HANDLER(update, update::UpdateEntity)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
// Special cases (button and event) are already defined inline in subscribe_state.h
|
||||||
|
|
||||||
InitialStateIterator::InitialStateIterator(APIConnection *client) : client_(client) {}
|
InitialStateIterator::InitialStateIterator(APIConnection *client) : client_(client) {}
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
|
|||||||
@@ -10,71 +10,78 @@ namespace api {
|
|||||||
|
|
||||||
class APIConnection;
|
class APIConnection;
|
||||||
|
|
||||||
|
// Macro for generating InitialStateIterator handlers
|
||||||
|
// Calls send_*_state
|
||||||
|
#define INITIAL_STATE_HANDLER(entity_type, EntityClass) \
|
||||||
|
bool InitialStateIterator::on_##entity_type(EntityClass *entity) { /* NOLINT(bugprone-macro-parentheses) */ \
|
||||||
|
return this->client_->send_##entity_type##_state(entity); \
|
||||||
|
}
|
||||||
|
|
||||||
class InitialStateIterator : public ComponentIterator {
|
class InitialStateIterator : public ComponentIterator {
|
||||||
public:
|
public:
|
||||||
InitialStateIterator(APIConnection *client);
|
InitialStateIterator(APIConnection *client);
|
||||||
#ifdef USE_BINARY_SENSOR
|
#ifdef USE_BINARY_SENSOR
|
||||||
bool on_binary_sensor(binary_sensor::BinarySensor *binary_sensor) override;
|
bool on_binary_sensor(binary_sensor::BinarySensor *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_COVER
|
#ifdef USE_COVER
|
||||||
bool on_cover(cover::Cover *cover) override;
|
bool on_cover(cover::Cover *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_FAN
|
#ifdef USE_FAN
|
||||||
bool on_fan(fan::Fan *fan) override;
|
bool on_fan(fan::Fan *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LIGHT
|
#ifdef USE_LIGHT
|
||||||
bool on_light(light::LightState *light) override;
|
bool on_light(light::LightState *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SENSOR
|
#ifdef USE_SENSOR
|
||||||
bool on_sensor(sensor::Sensor *sensor) override;
|
bool on_sensor(sensor::Sensor *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SWITCH
|
#ifdef USE_SWITCH
|
||||||
bool on_switch(switch_::Switch *a_switch) override;
|
bool on_switch(switch_::Switch *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BUTTON
|
#ifdef USE_BUTTON
|
||||||
bool on_button(button::Button *button) override { return true; };
|
bool on_button(button::Button *button) override { return true; };
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT_SENSOR
|
#ifdef USE_TEXT_SENSOR
|
||||||
bool on_text_sensor(text_sensor::TextSensor *text_sensor) override;
|
bool on_text_sensor(text_sensor::TextSensor *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_CLIMATE
|
#ifdef USE_CLIMATE
|
||||||
bool on_climate(climate::Climate *climate) override;
|
bool on_climate(climate::Climate *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_NUMBER
|
#ifdef USE_NUMBER
|
||||||
bool on_number(number::Number *number) override;
|
bool on_number(number::Number *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATE
|
#ifdef USE_DATETIME_DATE
|
||||||
bool on_date(datetime::DateEntity *date) override;
|
bool on_date(datetime::DateEntity *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_TIME
|
#ifdef USE_DATETIME_TIME
|
||||||
bool on_time(datetime::TimeEntity *time) override;
|
bool on_time(datetime::TimeEntity *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_DATETIME_DATETIME
|
#ifdef USE_DATETIME_DATETIME
|
||||||
bool on_datetime(datetime::DateTimeEntity *datetime) override;
|
bool on_datetime(datetime::DateTimeEntity *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_TEXT
|
#ifdef USE_TEXT
|
||||||
bool on_text(text::Text *text) override;
|
bool on_text(text::Text *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_SELECT
|
#ifdef USE_SELECT
|
||||||
bool on_select(select::Select *select) override;
|
bool on_select(select::Select *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_LOCK
|
#ifdef USE_LOCK
|
||||||
bool on_lock(lock::Lock *a_lock) override;
|
bool on_lock(lock::Lock *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VALVE
|
#ifdef USE_VALVE
|
||||||
bool on_valve(valve::Valve *valve) override;
|
bool on_valve(valve::Valve *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
bool on_media_player(media_player::MediaPlayer *media_player) override;
|
bool on_media_player(media_player::MediaPlayer *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
#ifdef USE_ALARM_CONTROL_PANEL
|
||||||
bool on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) override;
|
bool on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *entity) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_EVENT
|
#ifdef USE_EVENT
|
||||||
bool on_event(event::Event *event) override { return true; };
|
bool on_event(event::Event *event) override { return true; };
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_UPDATE
|
#ifdef USE_UPDATE
|
||||||
bool on_update(update::UpdateEntity *update) override;
|
bool on_update(update::UpdateEntity *entity) override;
|
||||||
#endif
|
#endif
|
||||||
bool completed() { return this->state_ == IteratorState::NONE; }
|
bool completed() { return this->state_ == IteratorState::NONE; }
|
||||||
|
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ namespace as3935 {
|
|||||||
static const char *const TAG = "as3935";
|
static const char *const TAG = "as3935";
|
||||||
|
|
||||||
void AS3935Component::setup() {
|
void AS3935Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AS3935...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
this->irq_pin_->setup();
|
this->irq_pin_->setup();
|
||||||
LOG_PIN(" IRQ Pin: ", this->irq_pin_);
|
LOG_PIN(" IRQ Pin: ", this->irq_pin_);
|
||||||
@@ -282,7 +282,7 @@ void AS3935Component::display_oscillator(bool state, uint8_t osc) {
|
|||||||
// based on the resonance frequency of the antenna and so it should be trimmed
|
// based on the resonance frequency of the antenna and so it should be trimmed
|
||||||
// before the calibration is done.
|
// before the calibration is done.
|
||||||
bool AS3935Component::calibrate_oscillator() {
|
bool AS3935Component::calibrate_oscillator() {
|
||||||
ESP_LOGI(TAG, "Starting oscillators calibration...");
|
ESP_LOGI(TAG, "Starting oscillators calibration");
|
||||||
this->write_register(CALIB_RCO, WIPE_ALL, DIRECT_COMMAND, 0); // Send command to calibrate the oscillators
|
this->write_register(CALIB_RCO, WIPE_ALL, DIRECT_COMMAND, 0); // Send command to calibrate the oscillators
|
||||||
|
|
||||||
this->display_oscillator(true, 2);
|
this->display_oscillator(true, 2);
|
||||||
@@ -307,7 +307,7 @@ bool AS3935Component::calibrate_oscillator() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void AS3935Component::tune_antenna() {
|
void AS3935Component::tune_antenna() {
|
||||||
ESP_LOGI(TAG, "Starting antenna tuning...");
|
ESP_LOGI(TAG, "Starting antenna tuning");
|
||||||
uint8_t div_ratio = this->read_div_ratio();
|
uint8_t div_ratio = this->read_div_ratio();
|
||||||
uint8_t tune_val = this->read_capacitance();
|
uint8_t tune_val = this->read_capacitance();
|
||||||
ESP_LOGI(TAG, "Division Ratio is set to: %d", div_ratio);
|
ESP_LOGI(TAG, "Division Ratio is set to: %d", div_ratio);
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ static const uint8_t REGISTER_AGC = 0x1A; // 8 bytes / R
|
|||||||
static const uint8_t REGISTER_MAGNITUDE = 0x1B; // 16 bytes / R
|
static const uint8_t REGISTER_MAGNITUDE = 0x1B; // 16 bytes / R
|
||||||
|
|
||||||
void AS5600Component::setup() {
|
void AS5600Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AS5600...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
if (!this->read_byte(REGISTER_STATUS).has_value()) {
|
if (!this->read_byte(REGISTER_STATUS).has_value()) {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
@@ -91,15 +91,17 @@ void AS5600Component::dump_config() {
|
|||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
|
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with AS5600 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGCONFIG(TAG, " Watchdog: %d", this->watchdog_);
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Fast Filter: %d", this->fast_filter_);
|
" Watchdog: %d\n"
|
||||||
ESP_LOGCONFIG(TAG, " Slow Filter: %d", this->slow_filter_);
|
" Fast Filter: %d\n"
|
||||||
ESP_LOGCONFIG(TAG, " Hysteresis: %d", this->hysteresis_);
|
" Slow Filter: %d\n"
|
||||||
ESP_LOGCONFIG(TAG, " Start Position: %d", this->start_position_);
|
" Hysteresis: %d\n"
|
||||||
|
" Start Position: %d",
|
||||||
|
this->watchdog_, this->fast_filter_, this->slow_filter_, this->hysteresis_, this->start_position_);
|
||||||
if (this->end_mode_ == END_MODE_POSITION) {
|
if (this->end_mode_ == END_MODE_POSITION) {
|
||||||
ESP_LOGCONFIG(TAG, " End Position: %d", this->end_position_);
|
ESP_LOGCONFIG(TAG, " End Position: %d", this->end_position_);
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -50,7 +50,6 @@ class AS5600Component : public Component, public i2c::I2CDevice {
|
|||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
/// HARDWARE_LATE setup priority
|
/// HARDWARE_LATE setup priority
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
|
|
||||||
// configuration setters
|
// configuration setters
|
||||||
void set_dir_pin(InternalGPIOPin *pin) { this->dir_pin_ = pin; }
|
void set_dir_pin(InternalGPIOPin *pin) { this->dir_pin_ = pin; }
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ namespace as7341 {
|
|||||||
static const char *const TAG = "as7341";
|
static const char *const TAG = "as7341";
|
||||||
|
|
||||||
void AS7341Component::setup() {
|
void AS7341Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AS7341...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
|
|
||||||
// Verify device ID
|
// Verify device ID
|
||||||
@@ -38,12 +38,14 @@ void AS7341Component::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "AS7341:");
|
ESP_LOGCONFIG(TAG, "AS7341:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with AS7341 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
ESP_LOGCONFIG(TAG, " Gain: %u", get_gain());
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " ATIME: %u", get_atime());
|
" Gain: %u\n"
|
||||||
ESP_LOGCONFIG(TAG, " ASTEP: %u", get_astep());
|
" ATIME: %u\n"
|
||||||
|
" ASTEP: %u",
|
||||||
|
get_gain(), get_atime(), get_astep());
|
||||||
|
|
||||||
LOG_SENSOR(" ", "F1", this->f1_);
|
LOG_SENSOR(" ", "F1", this->f1_);
|
||||||
LOG_SENSOR(" ", "F2", this->f2_);
|
LOG_SENSOR(" ", "F2", this->f2_);
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ from esphome.const import (
|
|||||||
PLATFORM_BK72XX,
|
PLATFORM_BK72XX,
|
||||||
PLATFORM_ESP32,
|
PLATFORM_ESP32,
|
||||||
PLATFORM_ESP8266,
|
PLATFORM_ESP8266,
|
||||||
|
PLATFORM_LN882X,
|
||||||
PLATFORM_RTL87XX,
|
PLATFORM_RTL87XX,
|
||||||
)
|
)
|
||||||
from esphome.core import CORE, coroutine_with_priority
|
from esphome.core import CORE, coroutine_with_priority
|
||||||
@@ -14,15 +15,23 @@ CODEOWNERS = ["@OttoWinter"]
|
|||||||
CONFIG_SCHEMA = cv.All(
|
CONFIG_SCHEMA = cv.All(
|
||||||
cv.Schema({}),
|
cv.Schema({}),
|
||||||
cv.only_with_arduino,
|
cv.only_with_arduino,
|
||||||
cv.only_on([PLATFORM_ESP32, PLATFORM_ESP8266, PLATFORM_BK72XX, PLATFORM_RTL87XX]),
|
cv.only_on(
|
||||||
|
[
|
||||||
|
PLATFORM_ESP32,
|
||||||
|
PLATFORM_ESP8266,
|
||||||
|
PLATFORM_BK72XX,
|
||||||
|
PLATFORM_LN882X,
|
||||||
|
PLATFORM_RTL87XX,
|
||||||
|
]
|
||||||
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@coroutine_with_priority(200.0)
|
@coroutine_with_priority(200.0)
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
if CORE.is_esp32 or CORE.is_libretiny:
|
if CORE.is_esp32 or CORE.is_libretiny:
|
||||||
# https://github.com/esphome/AsyncTCP/blob/master/library.json
|
# https://github.com/ESP32Async/AsyncTCP
|
||||||
cg.add_library("esphome/AsyncTCP-esphome", "2.1.4")
|
cg.add_library("ESP32Async/AsyncTCP", "3.4.4")
|
||||||
elif CORE.is_esp8266:
|
elif CORE.is_esp8266:
|
||||||
# https://github.com/esphome/ESPAsyncTCP
|
# https://github.com/ESP32Async/ESPAsyncTCP
|
||||||
cg.add_library("esphome/ESPAsyncTCP-esphome", "2.0.0")
|
cg.add_library("ESP32Async/ESPAsyncTCP", "2.0.0")
|
||||||
|
|||||||
@@ -71,19 +71,22 @@ bool AT581XComponent::i2c_read_reg(uint8_t addr, uint8_t &data) {
|
|||||||
return this->read_register(addr, &data, 1) == esphome::i2c::NO_ERROR;
|
return this->read_register(addr, &data, 1) == esphome::i2c::NO_ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
void AT581XComponent::setup() { ESP_LOGCONFIG(TAG, "Setting up AT581X..."); }
|
void AT581XComponent::setup() { ESP_LOGCONFIG(TAG, "Running setup"); }
|
||||||
void AT581XComponent::dump_config() { LOG_I2C_DEVICE(this); }
|
void AT581XComponent::dump_config() { LOG_I2C_DEVICE(this); }
|
||||||
#define ARRAY_SIZE(X) (sizeof(X) / sizeof((X)[0]))
|
#define ARRAY_SIZE(X) (sizeof(X) / sizeof((X)[0]))
|
||||||
bool AT581XComponent::i2c_write_config() {
|
bool AT581XComponent::i2c_write_config() {
|
||||||
ESP_LOGCONFIG(TAG, "Writing new config for AT581X...");
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, "Frequency: %dMHz", this->freq_);
|
"Writing new config for AT581X\n"
|
||||||
ESP_LOGCONFIG(TAG, "Sensing distance: %d", this->delta_);
|
"Frequency: %dMHz\n"
|
||||||
ESP_LOGCONFIG(TAG, "Power: %dµA", this->power_);
|
"Sensing distance: %d\n"
|
||||||
ESP_LOGCONFIG(TAG, "Gain: %d", this->gain_);
|
"Power: %dµA\n"
|
||||||
ESP_LOGCONFIG(TAG, "Trigger base time: %dms", this->trigger_base_time_ms_);
|
"Gain: %d\n"
|
||||||
ESP_LOGCONFIG(TAG, "Trigger keep time: %dms", this->trigger_keep_time_ms_);
|
"Trigger base time: %dms\n"
|
||||||
ESP_LOGCONFIG(TAG, "Protect time: %dms", this->protect_time_ms_);
|
"Trigger keep time: %dms\n"
|
||||||
ESP_LOGCONFIG(TAG, "Self check time: %dms", this->self_check_time_ms_);
|
"Protect time: %dms\n"
|
||||||
|
"Self check time: %dms",
|
||||||
|
this->freq_, this->delta_, this->power_, this->gain_, this->trigger_base_time_ms_,
|
||||||
|
this->trigger_keep_time_ms_, this->protect_time_ms_, this->self_check_time_ms_);
|
||||||
|
|
||||||
// Set frequency point
|
// Set frequency point
|
||||||
if (!this->i2c_write_reg(FREQ_ADDR, GAIN61_VALUE)) {
|
if (!this->i2c_write_reg(FREQ_ADDR, GAIN61_VALUE)) {
|
||||||
|
|||||||
@@ -25,7 +25,6 @@ class ATCMiThermometer : public Component, public esp32_ble_tracker::ESPBTDevice
|
|||||||
|
|
||||||
bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override;
|
bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
void set_temperature(sensor::Sensor *temperature) { temperature_ = temperature; }
|
void set_temperature(sensor::Sensor *temperature) { temperature_ = temperature; }
|
||||||
void set_humidity(sensor::Sensor *humidity) { humidity_ = humidity; }
|
void set_humidity(sensor::Sensor *humidity) { humidity_ = humidity; }
|
||||||
void set_battery_level(sensor::Sensor *battery_level) { battery_level_ = battery_level; }
|
void set_battery_level(sensor::Sensor *battery_level) { battery_level_ = battery_level; }
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ void ATM90E26Component::update() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ATM90E26Component::setup() {
|
void ATM90E26Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ATM90E26 Component...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->spi_setup();
|
this->spi_setup();
|
||||||
|
|
||||||
uint16_t mmode = 0x422; // default values for everything but L/N line current gains
|
uint16_t mmode = 0x422; // default values for everything but L/N line current gains
|
||||||
@@ -135,7 +135,7 @@ void ATM90E26Component::dump_config() {
|
|||||||
ESP_LOGCONFIG("", "ATM90E26:");
|
ESP_LOGCONFIG("", "ATM90E26:");
|
||||||
LOG_PIN(" CS Pin: ", this->cs_);
|
LOG_PIN(" CS Pin: ", this->cs_);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with ATM90E26 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
LOG_SENSOR(" ", "Voltage A", this->voltage_sensor_);
|
LOG_SENSOR(" ", "Voltage A", this->voltage_sensor_);
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
#include "atm90e32.h"
|
#include "atm90e32.h"
|
||||||
#include <cinttypes>
|
#include <cinttypes>
|
||||||
#include <cmath>
|
#include <cmath>
|
||||||
|
#include <numbers>
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
@@ -108,7 +109,7 @@ void ATM90E32Component::update() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ATM90E32Component::setup() {
|
void ATM90E32Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ATM90E32 Component...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->spi_setup();
|
this->spi_setup();
|
||||||
|
|
||||||
uint16_t mmode0 = 0x87; // 3P4W 50Hz
|
uint16_t mmode0 = 0x87; // 3P4W 50Hz
|
||||||
@@ -217,7 +218,7 @@ void ATM90E32Component::dump_config() {
|
|||||||
ESP_LOGCONFIG("", "ATM90E32:");
|
ESP_LOGCONFIG("", "ATM90E32:");
|
||||||
LOG_PIN(" CS Pin: ", this->cs_);
|
LOG_PIN(" CS Pin: ", this->cs_);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with ATM90E32 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
LOG_SENSOR(" ", "Voltage A", this->phase_[PHASEA].voltage_sensor_);
|
LOG_SENSOR(" ", "Voltage A", this->phase_[PHASEA].voltage_sensor_);
|
||||||
@@ -686,7 +687,7 @@ void ATM90E32Component::restore_power_offset_calibrations_() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ATM90E32Component::clear_gain_calibrations() {
|
void ATM90E32Component::clear_gain_calibrations() {
|
||||||
ESP_LOGI(TAG, "[CALIBRATION] Clearing stored gain calibrations and restoring config-defined values...");
|
ESP_LOGI(TAG, "[CALIBRATION] Clearing stored gain calibrations and restoring config-defined values");
|
||||||
|
|
||||||
for (int phase = 0; phase < 3; phase++) {
|
for (int phase = 0; phase < 3; phase++) {
|
||||||
gain_phase_[phase].voltage_gain = this->phase_[phase].voltage_gain_;
|
gain_phase_[phase].voltage_gain = this->phase_[phase].voltage_gain_;
|
||||||
@@ -848,7 +849,7 @@ uint16_t ATM90E32Component::calculate_voltage_threshold(int line_freq, uint16_t
|
|||||||
float nominal_voltage = (line_freq == 60) ? 120.0f : 220.0f;
|
float nominal_voltage = (line_freq == 60) ? 120.0f : 220.0f;
|
||||||
float target_voltage = nominal_voltage * multiplier;
|
float target_voltage = nominal_voltage * multiplier;
|
||||||
|
|
||||||
float peak_01v = target_voltage * 100.0f * std::sqrt(2.0f); // convert RMS → peak, scale to 0.01V
|
float peak_01v = target_voltage * 100.0f * std::numbers::sqrt2_v<float>; // convert RMS → peak, scale to 0.01V
|
||||||
float divider = (2.0f * ugain) / 32768.0f;
|
float divider = (2.0f * ugain) / 32768.0f;
|
||||||
|
|
||||||
float threshold = peak_01v / divider;
|
float threshold = peak_01v / divider;
|
||||||
|
|||||||
@@ -312,7 +312,7 @@ FileDecoderState AudioDecoder::decode_mp3_() {
|
|||||||
if (err) {
|
if (err) {
|
||||||
switch (err) {
|
switch (err) {
|
||||||
case esp_audio_libs::helix_decoder::ERR_MP3_OUT_OF_MEMORY:
|
case esp_audio_libs::helix_decoder::ERR_MP3_OUT_OF_MEMORY:
|
||||||
// Intentional fallthrough
|
[[fallthrough]];
|
||||||
case esp_audio_libs::helix_decoder::ERR_MP3_NULL_POINTER:
|
case esp_audio_libs::helix_decoder::ERR_MP3_NULL_POINTER:
|
||||||
return FileDecoderState::FAILED;
|
return FileDecoderState::FAILED;
|
||||||
break;
|
break;
|
||||||
|
|||||||
@@ -5,6 +5,7 @@
|
|||||||
#include "esphome/core/defines.h"
|
#include "esphome/core/defines.h"
|
||||||
#include "esphome/core/hal.h"
|
#include "esphome/core/hal.h"
|
||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
#if CONFIG_MBEDTLS_CERTIFICATE_BUNDLE
|
#if CONFIG_MBEDTLS_CERTIFICATE_BUNDLE
|
||||||
#include "esp_crt_bundle.h"
|
#include "esp_crt_bundle.h"
|
||||||
@@ -16,13 +17,13 @@ namespace audio {
|
|||||||
static const uint32_t READ_WRITE_TIMEOUT_MS = 20;
|
static const uint32_t READ_WRITE_TIMEOUT_MS = 20;
|
||||||
|
|
||||||
static const uint32_t CONNECTION_TIMEOUT_MS = 5000;
|
static const uint32_t CONNECTION_TIMEOUT_MS = 5000;
|
||||||
|
static const uint8_t MAX_FETCHING_HEADER_ATTEMPTS = 6;
|
||||||
// The number of times the http read times out with no data before throwing an error
|
|
||||||
static const uint32_t ERROR_COUNT_NO_DATA_READ_TIMEOUT = 100;
|
|
||||||
|
|
||||||
static const size_t HTTP_STREAM_BUFFER_SIZE = 2048;
|
static const size_t HTTP_STREAM_BUFFER_SIZE = 2048;
|
||||||
|
|
||||||
static const uint8_t MAX_REDIRECTION = 5;
|
static const uint8_t MAX_REDIRECTIONS = 5;
|
||||||
|
|
||||||
|
static const char *const TAG = "audio_reader";
|
||||||
|
|
||||||
// Some common HTTP status codes - borrowed from http_request component accessed 20241224
|
// Some common HTTP status codes - borrowed from http_request component accessed 20241224
|
||||||
enum HttpStatus {
|
enum HttpStatus {
|
||||||
@@ -94,7 +95,7 @@ esp_err_t AudioReader::start(const std::string &uri, AudioFileType &file_type) {
|
|||||||
client_config.url = uri.c_str();
|
client_config.url = uri.c_str();
|
||||||
client_config.cert_pem = nullptr;
|
client_config.cert_pem = nullptr;
|
||||||
client_config.disable_auto_redirect = false;
|
client_config.disable_auto_redirect = false;
|
||||||
client_config.max_redirection_count = 10;
|
client_config.max_redirection_count = MAX_REDIRECTIONS;
|
||||||
client_config.event_handler = http_event_handler;
|
client_config.event_handler = http_event_handler;
|
||||||
client_config.user_data = this;
|
client_config.user_data = this;
|
||||||
client_config.buffer_size = HTTP_STREAM_BUFFER_SIZE;
|
client_config.buffer_size = HTTP_STREAM_BUFFER_SIZE;
|
||||||
@@ -116,12 +117,29 @@ esp_err_t AudioReader::start(const std::string &uri, AudioFileType &file_type) {
|
|||||||
esp_err_t err = esp_http_client_open(this->client_, 0);
|
esp_err_t err = esp_http_client_open(this->client_, 0);
|
||||||
|
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
|
ESP_LOGE(TAG, "Failed to open URL");
|
||||||
this->cleanup_connection_();
|
this->cleanup_connection_();
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
int64_t header_length = esp_http_client_fetch_headers(this->client_);
|
int64_t header_length = esp_http_client_fetch_headers(this->client_);
|
||||||
|
uint8_t reattempt_count = 0;
|
||||||
|
while ((header_length < 0) && (reattempt_count < MAX_FETCHING_HEADER_ATTEMPTS)) {
|
||||||
|
this->cleanup_connection_();
|
||||||
|
if (header_length != -ESP_ERR_HTTP_EAGAIN) {
|
||||||
|
// Serious error, no recovery
|
||||||
|
return ESP_FAIL;
|
||||||
|
} else {
|
||||||
|
// Reconnect from a fresh state to avoid a bug where it never reads the headers even if made available
|
||||||
|
this->client_ = esp_http_client_init(&client_config);
|
||||||
|
esp_http_client_open(this->client_, 0);
|
||||||
|
header_length = esp_http_client_fetch_headers(this->client_);
|
||||||
|
++reattempt_count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (header_length < 0) {
|
if (header_length < 0) {
|
||||||
|
ESP_LOGE(TAG, "Failed to fetch headers");
|
||||||
this->cleanup_connection_();
|
this->cleanup_connection_();
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
}
|
}
|
||||||
@@ -135,7 +153,7 @@ esp_err_t AudioReader::start(const std::string &uri, AudioFileType &file_type) {
|
|||||||
|
|
||||||
ssize_t redirect_count = 0;
|
ssize_t redirect_count = 0;
|
||||||
|
|
||||||
while ((esp_http_client_set_redirection(this->client_) == ESP_OK) && (redirect_count < MAX_REDIRECTION)) {
|
while ((esp_http_client_set_redirection(this->client_) == ESP_OK) && (redirect_count < MAX_REDIRECTIONS)) {
|
||||||
err = esp_http_client_open(this->client_, 0);
|
err = esp_http_client_open(this->client_, 0);
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
this->cleanup_connection_();
|
this->cleanup_connection_();
|
||||||
@@ -267,27 +285,29 @@ AudioReaderState AudioReader::http_read_() {
|
|||||||
return AudioReaderState::FINISHED;
|
return AudioReaderState::FINISHED;
|
||||||
}
|
}
|
||||||
} else if (this->output_transfer_buffer_->free() > 0) {
|
} else if (this->output_transfer_buffer_->free() > 0) {
|
||||||
size_t bytes_to_read = this->output_transfer_buffer_->free();
|
int received_len = esp_http_client_read(this->client_, (char *) this->output_transfer_buffer_->get_buffer_end(),
|
||||||
int received_len =
|
this->output_transfer_buffer_->free());
|
||||||
esp_http_client_read(this->client_, (char *) this->output_transfer_buffer_->get_buffer_end(), bytes_to_read);
|
|
||||||
|
|
||||||
if (received_len > 0) {
|
if (received_len > 0) {
|
||||||
this->output_transfer_buffer_->increase_buffer_length(received_len);
|
this->output_transfer_buffer_->increase_buffer_length(received_len);
|
||||||
this->last_data_read_ms_ = millis();
|
this->last_data_read_ms_ = millis();
|
||||||
} else if (received_len < 0) {
|
return AudioReaderState::READING;
|
||||||
|
} else if (received_len <= 0) {
|
||||||
// HTTP read error
|
// HTTP read error
|
||||||
this->cleanup_connection_();
|
if (received_len == -1) {
|
||||||
return AudioReaderState::FAILED;
|
// A true connection error occured, no chance at recovery
|
||||||
} else {
|
this->cleanup_connection_();
|
||||||
if (bytes_to_read > 0) {
|
return AudioReaderState::FAILED;
|
||||||
// Read timed out
|
|
||||||
if ((millis() - this->last_data_read_ms_) > CONNECTION_TIMEOUT_MS) {
|
|
||||||
this->cleanup_connection_();
|
|
||||||
return AudioReaderState::FAILED;
|
|
||||||
}
|
|
||||||
|
|
||||||
delay(READ_WRITE_TIMEOUT_MS);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Read timed out, manually verify if it has been too long since the last successful read
|
||||||
|
if ((millis() - this->last_data_read_ms_) > MAX_FETCHING_HEADER_ATTEMPTS * CONNECTION_TIMEOUT_MS) {
|
||||||
|
ESP_LOGE(TAG, "Timed out");
|
||||||
|
this->cleanup_connection_();
|
||||||
|
return AudioReaderState::FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
delay(READ_WRITE_TIMEOUT_MS);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -86,7 +86,7 @@ bool AudioTransferBuffer::reallocate(size_t new_buffer_size) {
|
|||||||
bool AudioTransferBuffer::allocate_buffer_(size_t buffer_size) {
|
bool AudioTransferBuffer::allocate_buffer_(size_t buffer_size) {
|
||||||
this->buffer_size_ = buffer_size;
|
this->buffer_size_ = buffer_size;
|
||||||
|
|
||||||
RAMAllocator<uint8_t> allocator(ExternalRAMAllocator<uint8_t>::ALLOW_FAILURE);
|
RAMAllocator<uint8_t> allocator;
|
||||||
|
|
||||||
this->buffer_ = allocator.allocate(this->buffer_size_);
|
this->buffer_ = allocator.allocate(this->buffer_size_);
|
||||||
if (this->buffer_ == nullptr) {
|
if (this->buffer_ == nullptr) {
|
||||||
@@ -101,7 +101,7 @@ bool AudioTransferBuffer::allocate_buffer_(size_t buffer_size) {
|
|||||||
|
|
||||||
void AudioTransferBuffer::deallocate_buffer_() {
|
void AudioTransferBuffer::deallocate_buffer_() {
|
||||||
if (this->buffer_ != nullptr) {
|
if (this->buffer_ != nullptr) {
|
||||||
RAMAllocator<uint8_t> allocator(ExternalRAMAllocator<uint8_t>::ALLOW_FAILURE);
|
RAMAllocator<uint8_t> allocator;
|
||||||
allocator.deallocate(this->buffer_, this->buffer_size_);
|
allocator.deallocate(this->buffer_, this->buffer_size_);
|
||||||
this->buffer_ = nullptr;
|
this->buffer_ = nullptr;
|
||||||
this->data_start_ = nullptr;
|
this->data_start_ = nullptr;
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ constexpr static const uint8_t AXS_READ_TOUCHPAD[11] = {0xb5, 0xab, 0xa5, 0x5a,
|
|||||||
}
|
}
|
||||||
|
|
||||||
void AXS15231Touchscreen::setup() {
|
void AXS15231Touchscreen::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AXS15231 Touchscreen...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
if (this->reset_pin_ != nullptr) {
|
if (this->reset_pin_ != nullptr) {
|
||||||
this->reset_pin_->setup();
|
this->reset_pin_->setup();
|
||||||
this->reset_pin_->digital_write(false);
|
this->reset_pin_->digital_write(false);
|
||||||
@@ -60,8 +60,10 @@ void AXS15231Touchscreen::dump_config() {
|
|||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
LOG_PIN(" Interrupt Pin: ", this->interrupt_pin_);
|
LOG_PIN(" Interrupt Pin: ", this->interrupt_pin_);
|
||||||
LOG_PIN(" Reset Pin: ", this->reset_pin_);
|
LOG_PIN(" Reset Pin: ", this->reset_pin_);
|
||||||
ESP_LOGCONFIG(TAG, " Width: %d", this->x_raw_max_);
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Height: %d", this->y_raw_max_);
|
" Width: %d\n"
|
||||||
|
" Height: %d",
|
||||||
|
this->x_raw_max_, this->y_raw_max_);
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace axs15231
|
} // namespace axs15231
|
||||||
|
|||||||
@@ -16,7 +16,6 @@ class BParasite : public Component, public esp32_ble_tracker::ESPBTDeviceListene
|
|||||||
|
|
||||||
bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override;
|
bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
float get_setup_priority() const override { return setup_priority::DATA; }
|
|
||||||
|
|
||||||
void set_battery_voltage(sensor::Sensor *battery_voltage) { battery_voltage_ = battery_voltage; }
|
void set_battery_voltage(sensor::Sensor *battery_voltage) { battery_voltage_ = battery_voltage; }
|
||||||
void set_temperature(sensor::Sensor *temperature) { temperature_ = temperature; }
|
void set_temperature(sensor::Sensor *temperature) { temperature_ = temperature; }
|
||||||
|
|||||||
@@ -194,11 +194,14 @@ Trigger<> *BangBangClimate::get_heat_trigger() const { return this->heat_trigger
|
|||||||
void BangBangClimate::set_supports_heat(bool supports_heat) { this->supports_heat_ = supports_heat; }
|
void BangBangClimate::set_supports_heat(bool supports_heat) { this->supports_heat_ = supports_heat; }
|
||||||
void BangBangClimate::dump_config() {
|
void BangBangClimate::dump_config() {
|
||||||
LOG_CLIMATE("", "Bang Bang Climate", this);
|
LOG_CLIMATE("", "Bang Bang Climate", this);
|
||||||
ESP_LOGCONFIG(TAG, " Supports HEAT: %s", YESNO(this->supports_heat_));
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Supports COOL: %s", YESNO(this->supports_cool_));
|
" Supports HEAT: %s\n"
|
||||||
ESP_LOGCONFIG(TAG, " Supports AWAY mode: %s", YESNO(this->supports_away_));
|
" Supports COOL: %s\n"
|
||||||
ESP_LOGCONFIG(TAG, " Default Target Temperature Low: %.2f°C", this->normal_config_.default_temperature_low);
|
" Supports AWAY mode: %s\n"
|
||||||
ESP_LOGCONFIG(TAG, " Default Target Temperature High: %.2f°C", this->normal_config_.default_temperature_high);
|
" Default Target Temperature Low: %.2f°C\n"
|
||||||
|
" Default Target Temperature High: %.2f°C",
|
||||||
|
YESNO(this->supports_heat_), YESNO(this->supports_cool_), YESNO(this->supports_away_),
|
||||||
|
this->normal_config_.default_temperature_low, this->normal_config_.default_temperature_high);
|
||||||
}
|
}
|
||||||
|
|
||||||
BangBangClimateTargetTempConfig::BangBangClimateTargetTempConfig() = default;
|
BangBangClimateTargetTempConfig::BangBangClimateTargetTempConfig() = default;
|
||||||
|
|||||||
@@ -480,13 +480,19 @@ void BedJetHub::set_clock(uint8_t hour, uint8_t minute) {
|
|||||||
|
|
||||||
/* Internal */
|
/* Internal */
|
||||||
|
|
||||||
void BedJetHub::loop() {}
|
void BedJetHub::loop() {
|
||||||
|
// Parent BLEClientNode has a loop() method, but this component uses
|
||||||
|
// polling via update() and BLE callbacks so loop isn't needed
|
||||||
|
this->disable_loop();
|
||||||
|
}
|
||||||
void BedJetHub::update() { this->dispatch_status_(); }
|
void BedJetHub::update() { this->dispatch_status_(); }
|
||||||
|
|
||||||
void BedJetHub::dump_config() {
|
void BedJetHub::dump_config() {
|
||||||
ESP_LOGCONFIG(TAG, "BedJet Hub '%s'", this->get_name().c_str());
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " ble_client.app_id: %d", this->parent()->app_id);
|
"BedJet Hub '%s'\n"
|
||||||
ESP_LOGCONFIG(TAG, " ble_client.conn_id: %d", this->parent()->get_conn_id());
|
" ble_client.app_id: %d\n"
|
||||||
|
" ble_client.conn_id: %d",
|
||||||
|
this->get_name().c_str(), this->parent()->app_id, this->parent()->get_conn_id());
|
||||||
LOG_UPDATE_INTERVAL(this)
|
LOG_UPDATE_INTERVAL(this)
|
||||||
ESP_LOGCONFIG(TAG, " Child components (%d):", this->children_.size());
|
ESP_LOGCONFIG(TAG, " Child components (%d):", this->children_.size());
|
||||||
for (auto *child : this->children_) {
|
for (auto *child : this->children_) {
|
||||||
@@ -527,7 +533,7 @@ void BedJetHub::dispatch_status_() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (this->timeout_ > 0 && diff > this->timeout_ && this->parent()->enabled) {
|
if (this->timeout_ > 0 && diff > this->timeout_ && this->parent()->enabled) {
|
||||||
ESP_LOGW(TAG, "[%s] Timed out after %" PRId32 " sec. Retrying...", this->get_name().c_str(), this->timeout_);
|
ESP_LOGW(TAG, "[%s] Timed out after %" PRId32 " sec. Retrying", this->get_name().c_str(), this->timeout_);
|
||||||
// set_enabled(false) will only close the connection if state != IDLE.
|
// set_enabled(false) will only close the connection if state != IDLE.
|
||||||
this->parent()->set_state(espbt::ClientState::CONNECTING);
|
this->parent()->set_state(espbt::ClientState::CONNECTING);
|
||||||
this->parent()->set_enabled(false);
|
this->parent()->set_enabled(false);
|
||||||
|
|||||||
@@ -83,7 +83,11 @@ void BedJetClimate::reset_state_() {
|
|||||||
this->publish_state();
|
this->publish_state();
|
||||||
}
|
}
|
||||||
|
|
||||||
void BedJetClimate::loop() {}
|
void BedJetClimate::loop() {
|
||||||
|
// This component is controlled via the parent BedJetHub
|
||||||
|
// Empty loop not needed, disable to save CPU cycles
|
||||||
|
this->disable_loop();
|
||||||
|
}
|
||||||
|
|
||||||
void BedJetClimate::control(const ClimateCall &call) {
|
void BedJetClimate::control(const ClimateCall &call) {
|
||||||
ESP_LOGD(TAG, "Received BedJetClimate::control");
|
ESP_LOGD(TAG, "Received BedJetClimate::control");
|
||||||
|
|||||||
@@ -7,11 +7,13 @@
|
|||||||
|
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#include "rtos_pub.h"
|
#include "rtos_pub.h"
|
||||||
#include "spi.h"
|
// rtos_pub.h must be included before the rest of the includes
|
||||||
|
|
||||||
#include "arm_arch.h"
|
#include "arm_arch.h"
|
||||||
#include "general_dma_pub.h"
|
#include "general_dma_pub.h"
|
||||||
#include "gpio_pub.h"
|
#include "gpio_pub.h"
|
||||||
#include "icu_pub.h"
|
#include "icu_pub.h"
|
||||||
|
#include "spi.h"
|
||||||
#undef SPI_DAT
|
#undef SPI_DAT
|
||||||
#undef SPI_BASE
|
#undef SPI_BASE
|
||||||
};
|
};
|
||||||
@@ -119,12 +121,12 @@ void spi_dma_tx_finish_callback(unsigned int param) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void BekenSPILEDStripLightOutput::setup() {
|
void BekenSPILEDStripLightOutput::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up Beken SPI LED Strip...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
size_t buffer_size = this->get_buffer_size_();
|
size_t buffer_size = this->get_buffer_size_();
|
||||||
size_t dma_buffer_size = (buffer_size * 8) + (2 * 64);
|
size_t dma_buffer_size = (buffer_size * 8) + (2 * 64);
|
||||||
|
|
||||||
ExternalRAMAllocator<uint8_t> allocator(ExternalRAMAllocator<uint8_t>::ALLOW_FAILURE);
|
RAMAllocator<uint8_t> allocator;
|
||||||
this->buf_ = allocator.allocate(buffer_size);
|
this->buf_ = allocator.allocate(buffer_size);
|
||||||
if (this->buf_ == nullptr) {
|
if (this->buf_ == nullptr) {
|
||||||
ESP_LOGE(TAG, "Cannot allocate LED buffer!");
|
ESP_LOGE(TAG, "Cannot allocate LED buffer!");
|
||||||
@@ -256,7 +258,7 @@ void BekenSPILEDStripLightOutput::write_state(light::LightState *state) {
|
|||||||
this->last_refresh_ = now;
|
this->last_refresh_ = now;
|
||||||
this->mark_shown_();
|
this->mark_shown_();
|
||||||
|
|
||||||
ESP_LOGVV(TAG, "Writing RGB values to bus...");
|
ESP_LOGVV(TAG, "Writing RGB values to bus");
|
||||||
|
|
||||||
if (spi_data == nullptr) {
|
if (spi_data == nullptr) {
|
||||||
ESP_LOGE(TAG, "SPI not initialized");
|
ESP_LOGE(TAG, "SPI not initialized");
|
||||||
@@ -345,8 +347,10 @@ light::ESPColorView BekenSPILEDStripLightOutput::get_view_internal(int32_t index
|
|||||||
}
|
}
|
||||||
|
|
||||||
void BekenSPILEDStripLightOutput::dump_config() {
|
void BekenSPILEDStripLightOutput::dump_config() {
|
||||||
ESP_LOGCONFIG(TAG, "Beken SPI LED Strip:");
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Pin: %u", this->pin_);
|
"Beken SPI LED Strip:\n"
|
||||||
|
" Pin: %u",
|
||||||
|
this->pin_);
|
||||||
const char *rgb_order;
|
const char *rgb_order;
|
||||||
switch (this->rgb_order_) {
|
switch (this->rgb_order_) {
|
||||||
case ORDER_RGB:
|
case ORDER_RGB:
|
||||||
@@ -371,9 +375,11 @@ void BekenSPILEDStripLightOutput::dump_config() {
|
|||||||
rgb_order = "UNKNOWN";
|
rgb_order = "UNKNOWN";
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
ESP_LOGCONFIG(TAG, " RGB Order: %s", rgb_order);
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Max refresh rate: %" PRIu32, *this->max_refresh_rate_);
|
" RGB Order: %s\n"
|
||||||
ESP_LOGCONFIG(TAG, " Number of LEDs: %u", this->num_leds_);
|
" Max refresh rate: %" PRIu32 "\n"
|
||||||
|
" Number of LEDs: %u",
|
||||||
|
rgb_order, *this->max_refresh_rate_, this->num_leds_);
|
||||||
}
|
}
|
||||||
|
|
||||||
float BekenSPILEDStripLightOutput::get_setup_priority() const { return setup_priority::HARDWARE; }
|
float BekenSPILEDStripLightOutput::get_setup_priority() const { return setup_priority::HARDWARE; }
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ MTreg:
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
void BH1750Sensor::setup() {
|
void BH1750Sensor::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BH1750 '%s'...", this->name_.c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'", this->name_.c_str());
|
||||||
uint8_t turn_on = BH1750_COMMAND_POWER_ON;
|
uint8_t turn_on = BH1750_COMMAND_POWER_ON;
|
||||||
if (this->write(&turn_on, 1) != i2c::ERROR_OK) {
|
if (this->write(&turn_on, 1) != i2c::ERROR_OK) {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
@@ -50,7 +50,7 @@ void BH1750Sensor::read_lx_(BH1750Mode mode, uint8_t mtreg, const std::function<
|
|||||||
// turn on (after one-shot sensor automatically powers down)
|
// turn on (after one-shot sensor automatically powers down)
|
||||||
uint8_t turn_on = BH1750_COMMAND_POWER_ON;
|
uint8_t turn_on = BH1750_COMMAND_POWER_ON;
|
||||||
if (this->write(&turn_on, 1) != i2c::ERROR_OK) {
|
if (this->write(&turn_on, 1) != i2c::ERROR_OK) {
|
||||||
ESP_LOGW(TAG, "Turning on BH1750 failed");
|
ESP_LOGW(TAG, "Power on failed");
|
||||||
f(NAN);
|
f(NAN);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -60,7 +60,7 @@ void BH1750Sensor::read_lx_(BH1750Mode mode, uint8_t mtreg, const std::function<
|
|||||||
uint8_t mtreg_hi = BH1750_COMMAND_MT_REG_HI | ((mtreg >> 5) & 0b111);
|
uint8_t mtreg_hi = BH1750_COMMAND_MT_REG_HI | ((mtreg >> 5) & 0b111);
|
||||||
uint8_t mtreg_lo = BH1750_COMMAND_MT_REG_LO | ((mtreg >> 0) & 0b11111);
|
uint8_t mtreg_lo = BH1750_COMMAND_MT_REG_LO | ((mtreg >> 0) & 0b11111);
|
||||||
if (this->write(&mtreg_hi, 1) != i2c::ERROR_OK || this->write(&mtreg_lo, 1) != i2c::ERROR_OK) {
|
if (this->write(&mtreg_hi, 1) != i2c::ERROR_OK || this->write(&mtreg_lo, 1) != i2c::ERROR_OK) {
|
||||||
ESP_LOGW(TAG, "Setting measurement time for BH1750 failed");
|
ESP_LOGW(TAG, "Set measurement time failed");
|
||||||
active_mtreg_ = 0;
|
active_mtreg_ = 0;
|
||||||
f(NAN);
|
f(NAN);
|
||||||
return;
|
return;
|
||||||
@@ -88,7 +88,7 @@ void BH1750Sensor::read_lx_(BH1750Mode mode, uint8_t mtreg, const std::function<
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (this->write(&cmd, 1) != i2c::ERROR_OK) {
|
if (this->write(&cmd, 1) != i2c::ERROR_OK) {
|
||||||
ESP_LOGW(TAG, "Starting measurement for BH1750 failed");
|
ESP_LOGW(TAG, "Start measurement failed");
|
||||||
f(NAN);
|
f(NAN);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -99,7 +99,7 @@ void BH1750Sensor::read_lx_(BH1750Mode mode, uint8_t mtreg, const std::function<
|
|||||||
this->set_timeout("read", meas_time, [this, mode, mtreg, f]() {
|
this->set_timeout("read", meas_time, [this, mode, mtreg, f]() {
|
||||||
uint16_t raw_value;
|
uint16_t raw_value;
|
||||||
if (this->read(reinterpret_cast<uint8_t *>(&raw_value), 2) != i2c::ERROR_OK) {
|
if (this->read(reinterpret_cast<uint8_t *>(&raw_value), 2) != i2c::ERROR_OK) {
|
||||||
ESP_LOGW(TAG, "Reading BH1750 data failed");
|
ESP_LOGW(TAG, "Read data failed");
|
||||||
f(NAN);
|
f(NAN);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -118,7 +118,7 @@ void BH1750Sensor::dump_config() {
|
|||||||
LOG_SENSOR("", "BH1750", this);
|
LOG_SENSOR("", "BH1750", this);
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with BH1750 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL_FOR, this->get_name().c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
@@ -156,7 +156,7 @@ void BH1750Sensor::update() {
|
|||||||
this->publish_state(NAN);
|
this->publish_state(NAN);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
ESP_LOGD(TAG, "'%s': Got illuminance=%.1flx", this->get_name().c_str(), val);
|
ESP_LOGD(TAG, "'%s': Illuminance=%.1flx", this->get_name().c_str(), val);
|
||||||
this->status_clear_warning();
|
this->status_clear_warning();
|
||||||
this->publish_state(val);
|
this->publish_state(val);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,7 +1,10 @@
|
|||||||
|
from logging import getLogger
|
||||||
|
|
||||||
from esphome import automation, core
|
from esphome import automation, core
|
||||||
from esphome.automation import Condition, maybe_simple_id
|
from esphome.automation import Condition, maybe_simple_id
|
||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
from esphome.components import mqtt, web_server
|
from esphome.components import mqtt, web_server
|
||||||
|
from esphome.components.const import CONF_ON_STATE_CHANGE
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
CONF_DELAY,
|
CONF_DELAY,
|
||||||
@@ -57,8 +60,8 @@ from esphome.const import (
|
|||||||
DEVICE_CLASS_WINDOW,
|
DEVICE_CLASS_WINDOW,
|
||||||
)
|
)
|
||||||
from esphome.core import CORE, coroutine_with_priority
|
from esphome.core import CORE, coroutine_with_priority
|
||||||
|
from esphome.core.entity_helpers import entity_duplicate_validator, setup_entity
|
||||||
from esphome.cpp_generator import MockObjClass
|
from esphome.cpp_generator import MockObjClass
|
||||||
from esphome.cpp_helpers import setup_entity
|
|
||||||
from esphome.util import Registry
|
from esphome.util import Registry
|
||||||
|
|
||||||
CODEOWNERS = ["@esphome/core"]
|
CODEOWNERS = ["@esphome/core"]
|
||||||
@@ -98,6 +101,7 @@ IS_PLATFORM_COMPONENT = True
|
|||||||
|
|
||||||
CONF_TIME_OFF = "time_off"
|
CONF_TIME_OFF = "time_off"
|
||||||
CONF_TIME_ON = "time_on"
|
CONF_TIME_ON = "time_on"
|
||||||
|
CONF_TRIGGER_ON_INITIAL_STATE = "trigger_on_initial_state"
|
||||||
|
|
||||||
DEFAULT_DELAY = "1s"
|
DEFAULT_DELAY = "1s"
|
||||||
DEFAULT_TIME_OFF = "100ms"
|
DEFAULT_TIME_OFF = "100ms"
|
||||||
@@ -127,15 +131,24 @@ MultiClickTriggerEvent = binary_sensor_ns.struct("MultiClickTriggerEvent")
|
|||||||
StateTrigger = binary_sensor_ns.class_(
|
StateTrigger = binary_sensor_ns.class_(
|
||||||
"StateTrigger", automation.Trigger.template(bool)
|
"StateTrigger", automation.Trigger.template(bool)
|
||||||
)
|
)
|
||||||
|
StateChangeTrigger = binary_sensor_ns.class_(
|
||||||
|
"StateChangeTrigger",
|
||||||
|
automation.Trigger.template(cg.optional.template(bool), cg.optional.template(bool)),
|
||||||
|
)
|
||||||
|
|
||||||
BinarySensorPublishAction = binary_sensor_ns.class_(
|
BinarySensorPublishAction = binary_sensor_ns.class_(
|
||||||
"BinarySensorPublishAction", automation.Action
|
"BinarySensorPublishAction", automation.Action
|
||||||
)
|
)
|
||||||
|
BinarySensorInvalidateAction = binary_sensor_ns.class_(
|
||||||
|
"BinarySensorInvalidateAction", automation.Action
|
||||||
|
)
|
||||||
|
|
||||||
# Condition
|
# Condition
|
||||||
BinarySensorCondition = binary_sensor_ns.class_("BinarySensorCondition", Condition)
|
BinarySensorCondition = binary_sensor_ns.class_("BinarySensorCondition", Condition)
|
||||||
|
|
||||||
# Filters
|
# Filters
|
||||||
Filter = binary_sensor_ns.class_("Filter")
|
Filter = binary_sensor_ns.class_("Filter")
|
||||||
|
TimeoutFilter = binary_sensor_ns.class_("TimeoutFilter", Filter, cg.Component)
|
||||||
DelayedOnOffFilter = binary_sensor_ns.class_("DelayedOnOffFilter", Filter, cg.Component)
|
DelayedOnOffFilter = binary_sensor_ns.class_("DelayedOnOffFilter", Filter, cg.Component)
|
||||||
DelayedOnFilter = binary_sensor_ns.class_("DelayedOnFilter", Filter, cg.Component)
|
DelayedOnFilter = binary_sensor_ns.class_("DelayedOnFilter", Filter, cg.Component)
|
||||||
DelayedOffFilter = binary_sensor_ns.class_("DelayedOffFilter", Filter, cg.Component)
|
DelayedOffFilter = binary_sensor_ns.class_("DelayedOffFilter", Filter, cg.Component)
|
||||||
@@ -144,6 +157,8 @@ AutorepeatFilter = binary_sensor_ns.class_("AutorepeatFilter", Filter, cg.Compon
|
|||||||
LambdaFilter = binary_sensor_ns.class_("LambdaFilter", Filter)
|
LambdaFilter = binary_sensor_ns.class_("LambdaFilter", Filter)
|
||||||
SettleFilter = binary_sensor_ns.class_("SettleFilter", Filter, cg.Component)
|
SettleFilter = binary_sensor_ns.class_("SettleFilter", Filter, cg.Component)
|
||||||
|
|
||||||
|
_LOGGER = getLogger(__name__)
|
||||||
|
|
||||||
FILTER_REGISTRY = Registry()
|
FILTER_REGISTRY = Registry()
|
||||||
validate_filters = cv.validate_registry("filter", FILTER_REGISTRY)
|
validate_filters = cv.validate_registry("filter", FILTER_REGISTRY)
|
||||||
|
|
||||||
@@ -157,6 +172,19 @@ async def invert_filter_to_code(config, filter_id):
|
|||||||
return cg.new_Pvariable(filter_id)
|
return cg.new_Pvariable(filter_id)
|
||||||
|
|
||||||
|
|
||||||
|
@register_filter(
|
||||||
|
"timeout",
|
||||||
|
TimeoutFilter,
|
||||||
|
cv.templatable(cv.positive_time_period_milliseconds),
|
||||||
|
)
|
||||||
|
async def timeout_filter_to_code(config, filter_id):
|
||||||
|
var = cg.new_Pvariable(filter_id)
|
||||||
|
await cg.register_component(var, {})
|
||||||
|
template_ = await cg.templatable(config, [], cg.uint32)
|
||||||
|
cg.add(var.set_timeout_value(template_))
|
||||||
|
return var
|
||||||
|
|
||||||
|
|
||||||
@register_filter(
|
@register_filter(
|
||||||
"delayed_on_off",
|
"delayed_on_off",
|
||||||
DelayedOnOffFilter,
|
DelayedOnOffFilter,
|
||||||
@@ -386,6 +414,14 @@ def validate_click_timing(value):
|
|||||||
return value
|
return value
|
||||||
|
|
||||||
|
|
||||||
|
def validate_publish_initial_state(value):
|
||||||
|
value = cv.boolean(value)
|
||||||
|
_LOGGER.warning(
|
||||||
|
"The 'publish_initial_state' option has been replaced by 'trigger_on_initial_state' and will be removed in a future release"
|
||||||
|
)
|
||||||
|
return value
|
||||||
|
|
||||||
|
|
||||||
_BINARY_SENSOR_SCHEMA = (
|
_BINARY_SENSOR_SCHEMA = (
|
||||||
cv.ENTITY_BASE_SCHEMA.extend(web_server.WEBSERVER_SORTING_SCHEMA)
|
cv.ENTITY_BASE_SCHEMA.extend(web_server.WEBSERVER_SORTING_SCHEMA)
|
||||||
.extend(cv.MQTT_COMPONENT_SCHEMA)
|
.extend(cv.MQTT_COMPONENT_SCHEMA)
|
||||||
@@ -395,7 +431,12 @@ _BINARY_SENSOR_SCHEMA = (
|
|||||||
cv.OnlyWith(CONF_MQTT_ID, "mqtt"): cv.declare_id(
|
cv.OnlyWith(CONF_MQTT_ID, "mqtt"): cv.declare_id(
|
||||||
mqtt.MQTTBinarySensorComponent
|
mqtt.MQTTBinarySensorComponent
|
||||||
),
|
),
|
||||||
cv.Optional(CONF_PUBLISH_INITIAL_STATE): cv.boolean,
|
cv.Exclusive(
|
||||||
|
CONF_PUBLISH_INITIAL_STATE, CONF_TRIGGER_ON_INITIAL_STATE
|
||||||
|
): validate_publish_initial_state,
|
||||||
|
cv.Exclusive(
|
||||||
|
CONF_TRIGGER_ON_INITIAL_STATE, CONF_TRIGGER_ON_INITIAL_STATE
|
||||||
|
): cv.boolean,
|
||||||
cv.Optional(CONF_DEVICE_CLASS): validate_device_class,
|
cv.Optional(CONF_DEVICE_CLASS): validate_device_class,
|
||||||
cv.Optional(CONF_FILTERS): validate_filters,
|
cv.Optional(CONF_FILTERS): validate_filters,
|
||||||
cv.Optional(CONF_ON_PRESS): automation.validate_automation(
|
cv.Optional(CONF_ON_PRESS): automation.validate_automation(
|
||||||
@@ -454,11 +495,19 @@ _BINARY_SENSOR_SCHEMA = (
|
|||||||
cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(StateTrigger),
|
cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(StateTrigger),
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
|
cv.Optional(CONF_ON_STATE_CHANGE): automation.validate_automation(
|
||||||
|
{
|
||||||
|
cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(StateChangeTrigger),
|
||||||
|
}
|
||||||
|
),
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
_BINARY_SENSOR_SCHEMA.add_extra(entity_duplicate_validator("binary_sensor"))
|
||||||
|
|
||||||
|
|
||||||
def binary_sensor_schema(
|
def binary_sensor_schema(
|
||||||
class_: MockObjClass = cv.UNDEFINED,
|
class_: MockObjClass = cv.UNDEFINED,
|
||||||
*,
|
*,
|
||||||
@@ -489,12 +538,14 @@ BINARY_SENSOR_SCHEMA.add_extra(cv.deprecated_schema_constant("binary_sensor"))
|
|||||||
|
|
||||||
|
|
||||||
async def setup_binary_sensor_core_(var, config):
|
async def setup_binary_sensor_core_(var, config):
|
||||||
await setup_entity(var, config)
|
await setup_entity(var, config, "binary_sensor")
|
||||||
|
|
||||||
if (device_class := config.get(CONF_DEVICE_CLASS)) is not None:
|
if (device_class := config.get(CONF_DEVICE_CLASS)) is not None:
|
||||||
cg.add(var.set_device_class(device_class))
|
cg.add(var.set_device_class(device_class))
|
||||||
if publish_initial_state := config.get(CONF_PUBLISH_INITIAL_STATE):
|
trigger = config.get(CONF_TRIGGER_ON_INITIAL_STATE, False) or config.get(
|
||||||
cg.add(var.set_publish_initial_state(publish_initial_state))
|
CONF_PUBLISH_INITIAL_STATE, False
|
||||||
|
)
|
||||||
|
cg.add(var.set_trigger_on_initial_state(trigger))
|
||||||
if inverted := config.get(CONF_INVERTED):
|
if inverted := config.get(CONF_INVERTED):
|
||||||
cg.add(var.set_inverted(inverted))
|
cg.add(var.set_inverted(inverted))
|
||||||
if filters_config := config.get(CONF_FILTERS):
|
if filters_config := config.get(CONF_FILTERS):
|
||||||
@@ -542,6 +593,17 @@ async def setup_binary_sensor_core_(var, config):
|
|||||||
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
||||||
await automation.build_automation(trigger, [(bool, "x")], conf)
|
await automation.build_automation(trigger, [(bool, "x")], conf)
|
||||||
|
|
||||||
|
for conf in config.get(CONF_ON_STATE_CHANGE, []):
|
||||||
|
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
||||||
|
await automation.build_automation(
|
||||||
|
trigger,
|
||||||
|
[
|
||||||
|
(cg.optional.template(bool), "x_previous"),
|
||||||
|
(cg.optional.template(bool), "x"),
|
||||||
|
],
|
||||||
|
conf,
|
||||||
|
)
|
||||||
|
|
||||||
if mqtt_id := config.get(CONF_MQTT_ID):
|
if mqtt_id := config.get(CONF_MQTT_ID):
|
||||||
mqtt_ = cg.new_Pvariable(mqtt_id, var)
|
mqtt_ = cg.new_Pvariable(mqtt_id, var)
|
||||||
await mqtt.register_mqtt_component(mqtt_, config)
|
await mqtt.register_mqtt_component(mqtt_, config)
|
||||||
@@ -554,6 +616,7 @@ async def register_binary_sensor(var, config):
|
|||||||
if not CORE.has_id(config[CONF_ID]):
|
if not CORE.has_id(config[CONF_ID]):
|
||||||
var = cg.Pvariable(config[CONF_ID], var)
|
var = cg.Pvariable(config[CONF_ID], var)
|
||||||
cg.add(cg.App.register_binary_sensor(var))
|
cg.add(cg.App.register_binary_sensor(var))
|
||||||
|
CORE.register_platform_component("binary_sensor", var)
|
||||||
await setup_binary_sensor_core_(var, config)
|
await setup_binary_sensor_core_(var, config)
|
||||||
|
|
||||||
|
|
||||||
@@ -590,3 +653,18 @@ async def binary_sensor_is_off_to_code(config, condition_id, template_arg, args)
|
|||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
cg.add_define("USE_BINARY_SENSOR")
|
cg.add_define("USE_BINARY_SENSOR")
|
||||||
cg.add_global(binary_sensor_ns.using)
|
cg.add_global(binary_sensor_ns.using)
|
||||||
|
|
||||||
|
|
||||||
|
@automation.register_action(
|
||||||
|
"binary_sensor.invalidate_state",
|
||||||
|
BinarySensorInvalidateAction,
|
||||||
|
cv.maybe_simple_value(
|
||||||
|
{
|
||||||
|
cv.Required(CONF_ID): cv.use_id(BinarySensor),
|
||||||
|
},
|
||||||
|
key=CONF_ID,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
async def binary_sensor_invalidate_state_to_code(config, action_id, template_arg, args):
|
||||||
|
paren = await cg.get_variable(config[CONF_ID])
|
||||||
|
return cg.new_Pvariable(action_id, template_arg, paren)
|
||||||
|
|||||||
@@ -68,8 +68,7 @@ void binary_sensor::MultiClickTrigger::on_state_(bool state) {
|
|||||||
*this->at_index_ = *this->at_index_ + 1;
|
*this->at_index_ = *this->at_index_ + 1;
|
||||||
}
|
}
|
||||||
void binary_sensor::MultiClickTrigger::schedule_cooldown_() {
|
void binary_sensor::MultiClickTrigger::schedule_cooldown_() {
|
||||||
ESP_LOGV(TAG, "Multi Click: Invalid length of press, starting cooldown of %" PRIu32 " ms...",
|
ESP_LOGV(TAG, "Multi Click: Invalid length of press, starting cooldown of %" PRIu32 " ms", this->invalid_cooldown_);
|
||||||
this->invalid_cooldown_);
|
|
||||||
this->is_in_cooldown_ = true;
|
this->is_in_cooldown_ = true;
|
||||||
this->set_timeout("cooldown", this->invalid_cooldown_, [this]() {
|
this->set_timeout("cooldown", this->invalid_cooldown_, [this]() {
|
||||||
ESP_LOGV(TAG, "Multi Click: Cooldown ended, matching is now enabled again.");
|
ESP_LOGV(TAG, "Multi Click: Cooldown ended, matching is now enabled again.");
|
||||||
|
|||||||
@@ -96,7 +96,7 @@ class MultiClickTrigger : public Trigger<>, public Component {
|
|||||||
: parent_(parent), timing_(std::move(timing)) {}
|
: parent_(parent), timing_(std::move(timing)) {}
|
||||||
|
|
||||||
void setup() override {
|
void setup() override {
|
||||||
this->last_state_ = this->parent_->state;
|
this->last_state_ = this->parent_->get_state_default(false);
|
||||||
auto f = std::bind(&MultiClickTrigger::on_state_, this, std::placeholders::_1);
|
auto f = std::bind(&MultiClickTrigger::on_state_, this, std::placeholders::_1);
|
||||||
this->parent_->add_on_state_callback(f);
|
this->parent_->add_on_state_callback(f);
|
||||||
}
|
}
|
||||||
@@ -130,6 +130,14 @@ class StateTrigger : public Trigger<bool> {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class StateChangeTrigger : public Trigger<optional<bool>, optional<bool> > {
|
||||||
|
public:
|
||||||
|
explicit StateChangeTrigger(BinarySensor *parent) {
|
||||||
|
parent->add_full_state_callback(
|
||||||
|
[this](optional<bool> old_state, optional<bool> state) { this->trigger(old_state, state); });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
template<typename... Ts> class BinarySensorCondition : public Condition<Ts...> {
|
template<typename... Ts> class BinarySensorCondition : public Condition<Ts...> {
|
||||||
public:
|
public:
|
||||||
BinarySensorCondition(BinarySensor *parent, bool state) : parent_(parent), state_(state) {}
|
BinarySensorCondition(BinarySensor *parent, bool state) : parent_(parent), state_(state) {}
|
||||||
@@ -154,5 +162,15 @@ template<typename... Ts> class BinarySensorPublishAction : public Action<Ts...>
|
|||||||
BinarySensor *sensor_;
|
BinarySensor *sensor_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template<typename... Ts> class BinarySensorInvalidateAction : public Action<Ts...> {
|
||||||
|
public:
|
||||||
|
explicit BinarySensorInvalidateAction(BinarySensor *sensor) : sensor_(sensor) {}
|
||||||
|
|
||||||
|
void play(Ts... x) override { this->sensor_->invalidate_state(); }
|
||||||
|
|
||||||
|
protected:
|
||||||
|
BinarySensor *sensor_;
|
||||||
|
};
|
||||||
|
|
||||||
} // namespace binary_sensor
|
} // namespace binary_sensor
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -7,42 +7,25 @@ namespace binary_sensor {
|
|||||||
|
|
||||||
static const char *const TAG = "binary_sensor";
|
static const char *const TAG = "binary_sensor";
|
||||||
|
|
||||||
void BinarySensor::add_on_state_callback(std::function<void(bool)> &&callback) {
|
void BinarySensor::publish_state(bool new_state) {
|
||||||
this->state_callback_.add(std::move(callback));
|
|
||||||
}
|
|
||||||
|
|
||||||
void BinarySensor::publish_state(bool state) {
|
|
||||||
if (!this->publish_dedup_.next(state))
|
|
||||||
return;
|
|
||||||
if (this->filter_list_ == nullptr) {
|
if (this->filter_list_ == nullptr) {
|
||||||
this->send_state_internal(state, false);
|
this->send_state_internal(new_state);
|
||||||
} else {
|
} else {
|
||||||
this->filter_list_->input(state, false);
|
this->filter_list_->input(new_state);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
void BinarySensor::publish_initial_state(bool state) {
|
void BinarySensor::publish_initial_state(bool new_state) {
|
||||||
if (!this->publish_dedup_.next(state))
|
this->invalidate_state();
|
||||||
return;
|
this->publish_state(new_state);
|
||||||
if (this->filter_list_ == nullptr) {
|
}
|
||||||
this->send_state_internal(state, true);
|
void BinarySensor::send_state_internal(bool new_state) {
|
||||||
} else {
|
// copy the new state to the visible property for backwards compatibility, before any callbacks
|
||||||
this->filter_list_->input(state, true);
|
this->state = new_state;
|
||||||
|
// Note that set_state_ de-dups and will only trigger callbacks if the state has actually changed
|
||||||
|
if (this->set_state_(new_state)) {
|
||||||
|
ESP_LOGD(TAG, "'%s': New state is %s", this->get_name().c_str(), ONOFF(new_state));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
void BinarySensor::send_state_internal(bool state, bool is_initial) {
|
|
||||||
if (is_initial) {
|
|
||||||
ESP_LOGD(TAG, "'%s': Sending initial state %s", this->get_name().c_str(), ONOFF(state));
|
|
||||||
} else {
|
|
||||||
ESP_LOGD(TAG, "'%s': Sending state %s", this->get_name().c_str(), ONOFF(state));
|
|
||||||
}
|
|
||||||
this->has_state_ = true;
|
|
||||||
this->state = state;
|
|
||||||
if (!is_initial || this->publish_initial_state_) {
|
|
||||||
this->state_callback_.call(state);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
BinarySensor::BinarySensor() : state(false) {}
|
|
||||||
|
|
||||||
void BinarySensor::add_filter(Filter *filter) {
|
void BinarySensor::add_filter(Filter *filter) {
|
||||||
filter->parent_ = this;
|
filter->parent_ = this;
|
||||||
@@ -60,7 +43,6 @@ void BinarySensor::add_filters(const std::vector<Filter *> &filters) {
|
|||||||
this->add_filter(filter);
|
this->add_filter(filter);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
bool BinarySensor::has_state() const { return this->has_state_; }
|
|
||||||
bool BinarySensor::is_status_binary_sensor() const { return false; }
|
bool BinarySensor::is_status_binary_sensor() const { return false; }
|
||||||
|
|
||||||
} // namespace binary_sensor
|
} // namespace binary_sensor
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "esphome/core/component.h"
|
|
||||||
#include "esphome/core/entity_base.h"
|
#include "esphome/core/entity_base.h"
|
||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
#include "esphome/components/binary_sensor/filter.h"
|
#include "esphome/components/binary_sensor/filter.h"
|
||||||
@@ -34,52 +33,39 @@ namespace binary_sensor {
|
|||||||
* The sub classes should notify the front-end of new states via the publish_state() method which
|
* The sub classes should notify the front-end of new states via the publish_state() method which
|
||||||
* handles inverted inputs for you.
|
* handles inverted inputs for you.
|
||||||
*/
|
*/
|
||||||
class BinarySensor : public EntityBase, public EntityBase_DeviceClass {
|
class BinarySensor : public StatefulEntityBase<bool>, public EntityBase_DeviceClass {
|
||||||
public:
|
public:
|
||||||
explicit BinarySensor();
|
explicit BinarySensor(){};
|
||||||
|
|
||||||
/** Add a callback to be notified of state changes.
|
|
||||||
*
|
|
||||||
* @param callback The void(bool) callback.
|
|
||||||
*/
|
|
||||||
void add_on_state_callback(std::function<void(bool)> &&callback);
|
|
||||||
|
|
||||||
/** Publish a new state to the front-end.
|
/** Publish a new state to the front-end.
|
||||||
*
|
*
|
||||||
* @param state The new state.
|
* @param new_state The new state.
|
||||||
*/
|
*/
|
||||||
void publish_state(bool state);
|
void publish_state(bool new_state);
|
||||||
|
|
||||||
/** Publish the initial state, this will not make the callback manager send callbacks
|
/** Publish the initial state, this will not make the callback manager send callbacks
|
||||||
* and is meant only for the initial state on boot.
|
* and is meant only for the initial state on boot.
|
||||||
*
|
*
|
||||||
* @param state The new state.
|
* @param new_state The new state.
|
||||||
*/
|
*/
|
||||||
void publish_initial_state(bool state);
|
void publish_initial_state(bool new_state);
|
||||||
|
|
||||||
/// The current reported state of the binary sensor.
|
|
||||||
bool state{false};
|
|
||||||
|
|
||||||
void add_filter(Filter *filter);
|
void add_filter(Filter *filter);
|
||||||
void add_filters(const std::vector<Filter *> &filters);
|
void add_filters(const std::vector<Filter *> &filters);
|
||||||
|
|
||||||
void set_publish_initial_state(bool publish_initial_state) { this->publish_initial_state_ = publish_initial_state; }
|
|
||||||
|
|
||||||
// ========== INTERNAL METHODS ==========
|
// ========== INTERNAL METHODS ==========
|
||||||
// (In most use cases you won't need these)
|
// (In most use cases you won't need these)
|
||||||
void send_state_internal(bool state, bool is_initial);
|
void send_state_internal(bool new_state);
|
||||||
|
|
||||||
/// Return whether this binary sensor has outputted a state.
|
/// Return whether this binary sensor has outputted a state.
|
||||||
virtual bool has_state() const;
|
|
||||||
|
|
||||||
virtual bool is_status_binary_sensor() const;
|
virtual bool is_status_binary_sensor() const;
|
||||||
|
|
||||||
|
// For backward compatibility, provide an accessible property
|
||||||
|
|
||||||
|
bool state{};
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
CallbackManager<void(bool)> state_callback_{};
|
|
||||||
Filter *filter_list_{nullptr};
|
Filter *filter_list_{nullptr};
|
||||||
bool has_state_{false};
|
|
||||||
bool publish_initial_state_{false};
|
|
||||||
Deduplicator<bool> publish_dedup_;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class BinarySensorInitiallyOff : public BinarySensor {
|
class BinarySensorInitiallyOff : public BinarySensor {
|
||||||
|
|||||||
@@ -9,37 +9,42 @@ namespace binary_sensor {
|
|||||||
|
|
||||||
static const char *const TAG = "sensor.filter";
|
static const char *const TAG = "sensor.filter";
|
||||||
|
|
||||||
void Filter::output(bool value, bool is_initial) {
|
void Filter::output(bool value) {
|
||||||
|
if (this->next_ == nullptr) {
|
||||||
|
this->parent_->send_state_internal(value);
|
||||||
|
} else {
|
||||||
|
this->next_->input(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void Filter::input(bool value) {
|
||||||
if (!this->dedup_.next(value))
|
if (!this->dedup_.next(value))
|
||||||
return;
|
return;
|
||||||
|
auto b = this->new_value(value);
|
||||||
if (this->next_ == nullptr) {
|
|
||||||
this->parent_->send_state_internal(value, is_initial);
|
|
||||||
} else {
|
|
||||||
this->next_->input(value, is_initial);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void Filter::input(bool value, bool is_initial) {
|
|
||||||
auto b = this->new_value(value, is_initial);
|
|
||||||
if (b.has_value()) {
|
if (b.has_value()) {
|
||||||
this->output(*b, is_initial);
|
this->output(*b);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
optional<bool> DelayedOnOffFilter::new_value(bool value, bool is_initial) {
|
void TimeoutFilter::input(bool value) {
|
||||||
|
this->set_timeout("timeout", this->timeout_delay_.value(), [this]() { this->parent_->invalidate_state(); });
|
||||||
|
// we do not de-dup here otherwise changes from invalid to valid state will not be output
|
||||||
|
this->output(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
optional<bool> DelayedOnOffFilter::new_value(bool value) {
|
||||||
if (value) {
|
if (value) {
|
||||||
this->set_timeout("ON_OFF", this->on_delay_.value(), [this, is_initial]() { this->output(true, is_initial); });
|
this->set_timeout("ON_OFF", this->on_delay_.value(), [this]() { this->output(true); });
|
||||||
} else {
|
} else {
|
||||||
this->set_timeout("ON_OFF", this->off_delay_.value(), [this, is_initial]() { this->output(false, is_initial); });
|
this->set_timeout("ON_OFF", this->off_delay_.value(), [this]() { this->output(false); });
|
||||||
}
|
}
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
float DelayedOnOffFilter::get_setup_priority() const { return setup_priority::HARDWARE; }
|
float DelayedOnOffFilter::get_setup_priority() const { return setup_priority::HARDWARE; }
|
||||||
|
|
||||||
optional<bool> DelayedOnFilter::new_value(bool value, bool is_initial) {
|
optional<bool> DelayedOnFilter::new_value(bool value) {
|
||||||
if (value) {
|
if (value) {
|
||||||
this->set_timeout("ON", this->delay_.value(), [this, is_initial]() { this->output(true, is_initial); });
|
this->set_timeout("ON", this->delay_.value(), [this]() { this->output(true); });
|
||||||
return {};
|
return {};
|
||||||
} else {
|
} else {
|
||||||
this->cancel_timeout("ON");
|
this->cancel_timeout("ON");
|
||||||
@@ -49,9 +54,9 @@ optional<bool> DelayedOnFilter::new_value(bool value, bool is_initial) {
|
|||||||
|
|
||||||
float DelayedOnFilter::get_setup_priority() const { return setup_priority::HARDWARE; }
|
float DelayedOnFilter::get_setup_priority() const { return setup_priority::HARDWARE; }
|
||||||
|
|
||||||
optional<bool> DelayedOffFilter::new_value(bool value, bool is_initial) {
|
optional<bool> DelayedOffFilter::new_value(bool value) {
|
||||||
if (!value) {
|
if (!value) {
|
||||||
this->set_timeout("OFF", this->delay_.value(), [this, is_initial]() { this->output(false, is_initial); });
|
this->set_timeout("OFF", this->delay_.value(), [this]() { this->output(false); });
|
||||||
return {};
|
return {};
|
||||||
} else {
|
} else {
|
||||||
this->cancel_timeout("OFF");
|
this->cancel_timeout("OFF");
|
||||||
@@ -61,11 +66,11 @@ optional<bool> DelayedOffFilter::new_value(bool value, bool is_initial) {
|
|||||||
|
|
||||||
float DelayedOffFilter::get_setup_priority() const { return setup_priority::HARDWARE; }
|
float DelayedOffFilter::get_setup_priority() const { return setup_priority::HARDWARE; }
|
||||||
|
|
||||||
optional<bool> InvertFilter::new_value(bool value, bool is_initial) { return !value; }
|
optional<bool> InvertFilter::new_value(bool value) { return !value; }
|
||||||
|
|
||||||
AutorepeatFilter::AutorepeatFilter(std::vector<AutorepeatFilterTiming> timings) : timings_(std::move(timings)) {}
|
AutorepeatFilter::AutorepeatFilter(std::vector<AutorepeatFilterTiming> timings) : timings_(std::move(timings)) {}
|
||||||
|
|
||||||
optional<bool> AutorepeatFilter::new_value(bool value, bool is_initial) {
|
optional<bool> AutorepeatFilter::new_value(bool value) {
|
||||||
if (value) {
|
if (value) {
|
||||||
// Ignore if already running
|
// Ignore if already running
|
||||||
if (this->active_timing_ != 0)
|
if (this->active_timing_ != 0)
|
||||||
@@ -101,7 +106,7 @@ void AutorepeatFilter::next_timing_() {
|
|||||||
|
|
||||||
void AutorepeatFilter::next_value_(bool val) {
|
void AutorepeatFilter::next_value_(bool val) {
|
||||||
const AutorepeatFilterTiming &timing = this->timings_[this->active_timing_ - 2];
|
const AutorepeatFilterTiming &timing = this->timings_[this->active_timing_ - 2];
|
||||||
this->output(val, false); // This is at least the second one so not initial
|
this->output(val); // This is at least the second one so not initial
|
||||||
this->set_timeout("ON_OFF", val ? timing.time_on : timing.time_off, [this, val]() { this->next_value_(!val); });
|
this->set_timeout("ON_OFF", val ? timing.time_on : timing.time_off, [this, val]() { this->next_value_(!val); });
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -109,18 +114,18 @@ float AutorepeatFilter::get_setup_priority() const { return setup_priority::HARD
|
|||||||
|
|
||||||
LambdaFilter::LambdaFilter(std::function<optional<bool>(bool)> f) : f_(std::move(f)) {}
|
LambdaFilter::LambdaFilter(std::function<optional<bool>(bool)> f) : f_(std::move(f)) {}
|
||||||
|
|
||||||
optional<bool> LambdaFilter::new_value(bool value, bool is_initial) { return this->f_(value); }
|
optional<bool> LambdaFilter::new_value(bool value) { return this->f_(value); }
|
||||||
|
|
||||||
optional<bool> SettleFilter::new_value(bool value, bool is_initial) {
|
optional<bool> SettleFilter::new_value(bool value) {
|
||||||
if (!this->steady_) {
|
if (!this->steady_) {
|
||||||
this->set_timeout("SETTLE", this->delay_.value(), [this, value, is_initial]() {
|
this->set_timeout("SETTLE", this->delay_.value(), [this, value]() {
|
||||||
this->steady_ = true;
|
this->steady_ = true;
|
||||||
this->output(value, is_initial);
|
this->output(value);
|
||||||
});
|
});
|
||||||
return {};
|
return {};
|
||||||
} else {
|
} else {
|
||||||
this->steady_ = false;
|
this->steady_ = false;
|
||||||
this->output(value, is_initial);
|
this->output(value);
|
||||||
this->set_timeout("SETTLE", this->delay_.value(), [this]() { this->steady_ = true; });
|
this->set_timeout("SETTLE", this->delay_.value(), [this]() { this->steady_ = true; });
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,11 +14,11 @@ class BinarySensor;
|
|||||||
|
|
||||||
class Filter {
|
class Filter {
|
||||||
public:
|
public:
|
||||||
virtual optional<bool> new_value(bool value, bool is_initial) = 0;
|
virtual optional<bool> new_value(bool value) = 0;
|
||||||
|
|
||||||
void input(bool value, bool is_initial);
|
virtual void input(bool value);
|
||||||
|
|
||||||
void output(bool value, bool is_initial);
|
void output(bool value);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
friend BinarySensor;
|
friend BinarySensor;
|
||||||
@@ -28,9 +28,19 @@ class Filter {
|
|||||||
Deduplicator<bool> dedup_;
|
Deduplicator<bool> dedup_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class TimeoutFilter : public Filter, public Component {
|
||||||
|
public:
|
||||||
|
optional<bool> new_value(bool value) override { return value; }
|
||||||
|
void input(bool value) override;
|
||||||
|
template<typename T> void set_timeout_value(T timeout) { this->timeout_delay_ = timeout; }
|
||||||
|
|
||||||
|
protected:
|
||||||
|
TemplatableValue<uint32_t> timeout_delay_{};
|
||||||
|
};
|
||||||
|
|
||||||
class DelayedOnOffFilter : public Filter, public Component {
|
class DelayedOnOffFilter : public Filter, public Component {
|
||||||
public:
|
public:
|
||||||
optional<bool> new_value(bool value, bool is_initial) override;
|
optional<bool> new_value(bool value) override;
|
||||||
|
|
||||||
float get_setup_priority() const override;
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
@@ -44,7 +54,7 @@ class DelayedOnOffFilter : public Filter, public Component {
|
|||||||
|
|
||||||
class DelayedOnFilter : public Filter, public Component {
|
class DelayedOnFilter : public Filter, public Component {
|
||||||
public:
|
public:
|
||||||
optional<bool> new_value(bool value, bool is_initial) override;
|
optional<bool> new_value(bool value) override;
|
||||||
|
|
||||||
float get_setup_priority() const override;
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
@@ -56,7 +66,7 @@ class DelayedOnFilter : public Filter, public Component {
|
|||||||
|
|
||||||
class DelayedOffFilter : public Filter, public Component {
|
class DelayedOffFilter : public Filter, public Component {
|
||||||
public:
|
public:
|
||||||
optional<bool> new_value(bool value, bool is_initial) override;
|
optional<bool> new_value(bool value) override;
|
||||||
|
|
||||||
float get_setup_priority() const override;
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
@@ -68,7 +78,7 @@ class DelayedOffFilter : public Filter, public Component {
|
|||||||
|
|
||||||
class InvertFilter : public Filter {
|
class InvertFilter : public Filter {
|
||||||
public:
|
public:
|
||||||
optional<bool> new_value(bool value, bool is_initial) override;
|
optional<bool> new_value(bool value) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct AutorepeatFilterTiming {
|
struct AutorepeatFilterTiming {
|
||||||
@@ -86,7 +96,7 @@ class AutorepeatFilter : public Filter, public Component {
|
|||||||
public:
|
public:
|
||||||
explicit AutorepeatFilter(std::vector<AutorepeatFilterTiming> timings);
|
explicit AutorepeatFilter(std::vector<AutorepeatFilterTiming> timings);
|
||||||
|
|
||||||
optional<bool> new_value(bool value, bool is_initial) override;
|
optional<bool> new_value(bool value) override;
|
||||||
|
|
||||||
float get_setup_priority() const override;
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
@@ -102,7 +112,7 @@ class LambdaFilter : public Filter {
|
|||||||
public:
|
public:
|
||||||
explicit LambdaFilter(std::function<optional<bool>(bool)> f);
|
explicit LambdaFilter(std::function<optional<bool>(bool)> f);
|
||||||
|
|
||||||
optional<bool> new_value(bool value, bool is_initial) override;
|
optional<bool> new_value(bool value) override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
std::function<optional<bool>(bool)> f_;
|
std::function<optional<bool>(bool)> f_;
|
||||||
@@ -110,7 +120,7 @@ class LambdaFilter : public Filter {
|
|||||||
|
|
||||||
class SettleFilter : public Filter, public Component {
|
class SettleFilter : public Filter, public Component {
|
||||||
public:
|
public:
|
||||||
optional<bool> new_value(bool value, bool is_initial) override;
|
optional<bool> new_value(bool value) override;
|
||||||
|
|
||||||
float get_setup_priority() const override;
|
float get_setup_priority() const override;
|
||||||
|
|
||||||
|
|||||||
@@ -100,7 +100,7 @@ void BL0906::handle_actions_() {
|
|||||||
for (int i = 0; i < this->action_queue_.size(); i++) {
|
for (int i = 0; i < this->action_queue_.size(); i++) {
|
||||||
ptr_func = this->action_queue_[i];
|
ptr_func = this->action_queue_[i];
|
||||||
if (ptr_func) {
|
if (ptr_func) {
|
||||||
ESP_LOGI(TAG, "HandleActionCallback[%d]...", i);
|
ESP_LOGI(TAG, "HandleActionCallback[%d]", i);
|
||||||
(this->*ptr_func)();
|
(this->*ptr_func)();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -196,14 +196,17 @@ void BL0942::received_package_(DataPacket *data) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void BL0942::dump_config() { // NOLINT(readability-function-cognitive-complexity)
|
void BL0942::dump_config() { // NOLINT(readability-function-cognitive-complexity)
|
||||||
ESP_LOGCONFIG(TAG, "BL0942:");
|
ESP_LOGCONFIG(TAG,
|
||||||
ESP_LOGCONFIG(TAG, " Reset: %s", TRUEFALSE(this->reset_));
|
"BL0942:\n"
|
||||||
ESP_LOGCONFIG(TAG, " Address: %d", this->address_);
|
" Reset: %s\n"
|
||||||
ESP_LOGCONFIG(TAG, " Nominal line frequency: %d Hz", this->line_freq_);
|
" Address: %d\n"
|
||||||
ESP_LOGCONFIG(TAG, " Current reference: %f", this->current_reference_);
|
" Nominal line frequency: %d Hz\n"
|
||||||
ESP_LOGCONFIG(TAG, " Energy reference: %f", this->energy_reference_);
|
" Current reference: %f\n"
|
||||||
ESP_LOGCONFIG(TAG, " Power reference: %f", this->power_reference_);
|
" Energy reference: %f\n"
|
||||||
ESP_LOGCONFIG(TAG, " Voltage reference: %f", this->voltage_reference_);
|
" Power reference: %f\n"
|
||||||
|
" Voltage reference: %f",
|
||||||
|
TRUEFALSE(this->reset_), this->address_, this->line_freq_, this->current_reference_,
|
||||||
|
this->energy_reference_, this->power_reference_, this->voltage_reference_);
|
||||||
LOG_SENSOR("", "Voltage", this->voltage_sensor_);
|
LOG_SENSOR("", "Voltage", this->voltage_sensor_);
|
||||||
LOG_SENSOR("", "Current", this->current_sensor_);
|
LOG_SENSOR("", "Current", this->current_sensor_);
|
||||||
LOG_SENSOR("", "Power", this->power_sensor_);
|
LOG_SENSOR("", "Power", this->power_sensor_);
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user