mirror of
https://github.com/esphome/esphome.git
synced 2025-08-10 20:29:24 +00:00
Compare commits
3042 Commits
2021.10.0b
...
2023.11.0b
Author | SHA1 | Date | |
---|---|---|---|
![]() |
a9a17ee89d | ||
![]() |
f094702a16 | ||
![]() |
908f56ff46 | ||
![]() |
bd5905c59a | ||
![]() |
91299f05f7 | ||
![]() |
30e5ff9fff | ||
![]() |
163b38e153 | ||
![]() |
3b486084c8 | ||
![]() |
9d453f0ba2 | ||
![]() |
799851a83a | ||
![]() |
7a9866f1b6 | ||
![]() |
3d30f1f733 | ||
![]() |
1e55764d52 | ||
![]() |
020da89b6a | ||
![]() |
6932422104 | ||
![]() |
29aa15b253 | ||
![]() |
c40519ec6f | ||
![]() |
6c62c00963 | ||
![]() |
1bd2e558d6 | ||
![]() |
dbb1263a36 | ||
![]() |
966c6a4531 | ||
![]() |
fff2d01420 | ||
![]() |
bf217ce252 | ||
![]() |
0101ae768c | ||
![]() |
9bd4b229e3 | ||
![]() |
ccffbfd3ae | ||
![]() |
78e3ce7718 | ||
![]() |
31fec2d692 | ||
![]() |
888c298d7e | ||
![]() |
708ed8f38a | ||
![]() |
a8a9c6192d | ||
![]() |
defe8ac97b | ||
![]() |
7ac9caa169 | ||
![]() |
972c18a7ca | ||
![]() |
dd0270207f | ||
![]() |
d141e1cd67 | ||
![]() |
43b36ac3c7 | ||
![]() |
fce59819f5 | ||
![]() |
b978985aa1 | ||
![]() |
84bbf02bde | ||
![]() |
d5aeb32ca6 | ||
![]() |
ff8b904097 | ||
![]() |
e0cee472c3 | ||
![]() |
3ee85d7516 | ||
![]() |
4ca9aefc43 | ||
![]() |
4ef2e28496 | ||
![]() |
13994d9bd1 | ||
![]() |
22cdb8dfc3 | ||
![]() |
a7ad4482f0 | ||
![]() |
aa17661002 | ||
![]() |
4e65aac7ae | ||
![]() |
d74a8abf9a | ||
![]() |
d809c80286 | ||
![]() |
6ca14331e5 | ||
![]() |
d5f010ee52 | ||
![]() |
4837e114e4 | ||
![]() |
77b530cf9e | ||
![]() |
453600f18e | ||
![]() |
4edf3efdf3 | ||
![]() |
f70d651a39 | ||
![]() |
b0a400d82b | ||
![]() |
29b1233609 | ||
![]() |
c3f7cae2b0 | ||
![]() |
9eea52ea85 | ||
![]() |
40c001bdc2 | ||
![]() |
229ba18e6c | ||
![]() |
1fd9d67e2b | ||
![]() |
b99be250a0 | ||
![]() |
b9d4e2e501 | ||
![]() |
ef2531edf3 | ||
![]() |
eae3089201 | ||
![]() |
0ea4de5f4c | ||
![]() |
8eae882d93 | ||
![]() |
9adda8085a | ||
![]() |
b03eb5ea0a | ||
![]() |
193bac94f4 | ||
![]() |
69ec647f7e | ||
![]() |
9dfcf801a3 | ||
![]() |
1e0daefa16 | ||
![]() |
27b593ba85 | ||
![]() |
6d991a1fc8 | ||
![]() |
a1845e1e72 | ||
![]() |
f96a839bcf | ||
![]() |
1282a15b14 | ||
![]() |
35039b45e4 | ||
![]() |
0c46403eaf | ||
![]() |
f36ec7c726 | ||
![]() |
01a16141f1 | ||
![]() |
7cf056e828 | ||
![]() |
f7f63c9da1 | ||
![]() |
d4cb29a380 | ||
![]() |
563e15e8a7 | ||
![]() |
c83cb30935 | ||
![]() |
907d43827c | ||
![]() |
0c5d5cd623 | ||
![]() |
535568e065 | ||
![]() |
2adda3a994 | ||
![]() |
a9f4922824 | ||
![]() |
bcfbcd9578 | ||
![]() |
cfbf3681f3 | ||
![]() |
f6e8d97981 | ||
![]() |
6b8a75d3b8 | ||
![]() |
8b9a7608f0 | ||
![]() |
4774200f6b | ||
![]() |
93056dead9 | ||
![]() |
841b24f744 | ||
![]() |
258b0fbff3 | ||
![]() |
fdef6c6d46 | ||
![]() |
28aedae8d7 | ||
![]() |
2895cc6c57 | ||
![]() |
ec835c0b05 | ||
![]() |
e80bd8ed3d | ||
![]() |
cf7df9e360 | ||
![]() |
e01ba894ed | ||
![]() |
390766eb67 | ||
![]() |
899d280ac7 | ||
![]() |
96dc7f0259 | ||
![]() |
cb2f5eb781 | ||
![]() |
34dce0acbf | ||
![]() |
96822aaa3a | ||
![]() |
b4765fb5fb | ||
![]() |
da9c2f2256 | ||
![]() |
0104bf3fc8 | ||
![]() |
cf8fb7888a | ||
![]() |
9b1e1bf56c | ||
![]() |
33e0f16b3b | ||
![]() |
0807d60c6a | ||
![]() |
f018fde369 | ||
![]() |
c47f8fc02c | ||
![]() |
76ab923780 | ||
![]() |
11dba3147d | ||
![]() |
8c2d9101d5 | ||
![]() |
5347c9aafe | ||
![]() |
e7d51f9c16 | ||
![]() |
7e27e98bff | ||
![]() |
eead33b6f2 | ||
![]() |
b898f75631 | ||
![]() |
84c00a5867 | ||
![]() |
e3fbf54a1a | ||
![]() |
3d7d0d4f73 | ||
![]() |
05a95f8ee9 | ||
![]() |
76417103c7 | ||
![]() |
d82471942f | ||
![]() |
9f033bce3b | ||
![]() |
b632ae49d4 | ||
![]() |
b9d72231b0 | ||
![]() |
61b8004536 | ||
![]() |
e99b8aaf96 | ||
![]() |
db02c4ea21 | ||
![]() |
f077a5962d | ||
![]() |
fa4ba43eb9 | ||
![]() |
9579423b24 | ||
![]() |
02449f24c9 | ||
![]() |
960d6a1431 | ||
![]() |
53572dcb8a | ||
![]() |
9a087c0767 | ||
![]() |
a7d817656e | ||
![]() |
a794836ebe | ||
![]() |
799c3cf439 | ||
![]() |
b7402ee6ff | ||
![]() |
a8667b680e | ||
![]() |
10c89771e3 | ||
![]() |
b735fc475e | ||
![]() |
19736f6e53 | ||
![]() |
cdc4f7f59b | ||
![]() |
8ef743f25e | ||
![]() |
b973238323 | ||
![]() |
582b8383d2 | ||
![]() |
e1c9418aee | ||
![]() |
2aa787f5f0 | ||
![]() |
2189a40a39 | ||
![]() |
51688d4078 | ||
![]() |
cc4c0e3e0b | ||
![]() |
14aa27f5e2 | ||
![]() |
c19dbdb02d | ||
![]() |
b0ac729a8e | ||
![]() |
6839de69c1 | ||
![]() |
b12dc98150 | ||
![]() |
1a44c6487e | ||
![]() |
5e7ce610a0 | ||
![]() |
1f02096edb | ||
![]() |
fd7d3c4332 | ||
![]() |
61cf566560 | ||
![]() |
97d624114d | ||
![]() |
52e8a2e9e4 | ||
![]() |
261c271d60 | ||
![]() |
a960c8008e | ||
![]() |
fb90e19713 | ||
![]() |
4913b3cc35 | ||
![]() |
e42c51a222 | ||
![]() |
26f12cd3bb | ||
![]() |
06eff72065 | ||
![]() |
357ba1ab0f | ||
![]() |
6143099f60 | ||
![]() |
7ddcdab351 | ||
![]() |
cb6e314336 | ||
![]() |
90315b3c40 | ||
![]() |
5d7c3d1622 | ||
![]() |
8c1ad1e9a6 | ||
![]() |
969f6dbe13 | ||
![]() |
6cce6d4c36 | ||
![]() |
d27e5e9c97 | ||
![]() |
da3e3903dd | ||
![]() |
f5c12b50ce | ||
![]() |
33ebfd221e | ||
![]() |
7c890d8ebc | ||
![]() |
04b708c336 | ||
![]() |
2910eb2ef7 | ||
![]() |
af3b22f8b7 | ||
![]() |
cbc1b29f3e | ||
![]() |
f4ce8b8b6c | ||
![]() |
54363f1246 | ||
![]() |
11727391ad | ||
![]() |
d500531c04 | ||
![]() |
689dcd1e24 | ||
![]() |
2fc4e88271 | ||
![]() |
853d81c6dd | ||
![]() |
e35de626a4 | ||
![]() |
41f29c46d0 | ||
![]() |
9d95f5c1da | ||
![]() |
5d5cc96017 | ||
![]() |
c9a8911029 | ||
![]() |
c77a9ad363 | ||
![]() |
511af5845e | ||
![]() |
6b96089f02 | ||
![]() |
46be886ca6 | ||
![]() |
be7e167c63 | ||
![]() |
c65d78f568 | ||
![]() |
412a866de8 | ||
![]() |
e09c217fde | ||
![]() |
af62c2d9cf | ||
![]() |
aba3cd557a | ||
![]() |
7e7c83b3ca | ||
![]() |
ee4ccf2762 | ||
![]() |
fa0dcac2c7 | ||
![]() |
44e5b0c745 | ||
![]() |
b6d5cb4142 | ||
![]() |
4e8cba49f1 | ||
![]() |
506c2ba6c7 | ||
![]() |
050fa0d4c1 | ||
![]() |
88bb051f37 | ||
![]() |
0d800958aa | ||
![]() |
471533d041 | ||
![]() |
7dfc4c74da | ||
![]() |
f709350b04 | ||
![]() |
85c5928baa | ||
![]() |
f5dfbaff4b | ||
![]() |
689c2f11a3 | ||
![]() |
f73fd97525 | ||
![]() |
f38849828d | ||
![]() |
e95ba57a61 | ||
![]() |
4913256597 | ||
![]() |
401a386219 | ||
![]() |
205f41509b | ||
![]() |
e87c8d550b | ||
![]() |
a33b8abce8 | ||
![]() |
40523e6823 | ||
![]() |
5e1472185c | ||
![]() |
af005a6554 | ||
![]() |
efd31be21c | ||
![]() |
e9bda2810f | ||
![]() |
ec4777b8d0 | ||
![]() |
589b9e10b2 | ||
![]() |
2513ede3ec | ||
![]() |
0b5a57ead4 | ||
![]() |
b3dc2d43a5 | ||
![]() |
d3913be7e5 | ||
![]() |
2c94c3d96f | ||
![]() |
4d81153150 | ||
![]() |
507dc5f496 | ||
![]() |
4335543575 | ||
![]() |
9b75121337 | ||
![]() |
d262548d2e | ||
![]() |
b5b654e054 | ||
![]() |
dae8ab563c | ||
![]() |
12365976c4 | ||
![]() |
4ac4492241 | ||
![]() |
57b7dd0fa2 | ||
![]() |
9d4f471855 | ||
![]() |
5751e9ec59 | ||
![]() |
cc1b7a7a56 | ||
![]() |
29249cdc1b | ||
![]() |
e5bae8187f | ||
![]() |
69adebfefa | ||
![]() |
7dabbb65d0 | ||
![]() |
b30bab8c1b | ||
![]() |
5360e14a9c | ||
![]() |
86db559f6e | ||
![]() |
2f7a378c7b | ||
![]() |
607d0b4264 | ||
![]() |
0a1ed58454 | ||
![]() |
5f5ee9c920 | ||
![]() |
0aeebdd289 | ||
![]() |
33e2aa341e | ||
![]() |
a42788812e | ||
![]() |
b07a038bc8 | ||
![]() |
17e1d4c245 | ||
![]() |
a031cc3b84 | ||
![]() |
c34d5111fc | ||
![]() |
727056a28c | ||
![]() |
0ca8dcd08e | ||
![]() |
3c7c4e1dba | ||
![]() |
518ecb4cc4 | ||
![]() |
55e36ab982 | ||
![]() |
90835ab917 | ||
![]() |
5b46088ae4 | ||
![]() |
d7e267eca5 | ||
![]() |
807c47a076 | ||
![]() |
7ebe6a5894 | ||
![]() |
41c829fa32 | ||
![]() |
8f1ce8c7f7 | ||
![]() |
e55636ed52 | ||
![]() |
e886262055 | ||
![]() |
2fa7f8c511 | ||
![]() |
4622ef770d | ||
![]() |
d76f18b4f2 | ||
![]() |
ec20778d83 | ||
![]() |
b3ca71c6fb | ||
![]() |
1100f67b66 | ||
![]() |
056a28906b | ||
![]() |
2c2821cd96 | ||
![]() |
157a3e53dd | ||
![]() |
61edf8c196 | ||
![]() |
397f57ce74 | ||
![]() |
bf253c21fa | ||
![]() |
e3eef1cc6d | ||
![]() |
11f6e555f9 | ||
![]() |
164631fcec | ||
![]() |
a61e3fadf6 | ||
![]() |
b5f2d69ca5 | ||
![]() |
55b5c0fc32 | ||
![]() |
280b090dfc | ||
![]() |
2d53dd05d8 | ||
![]() |
9d97807587 | ||
![]() |
11433c8c17 | ||
![]() |
68a2c45edf | ||
![]() |
d2616cd6c6 | ||
![]() |
736dbfac13 | ||
![]() |
b8fa737bc9 | ||
![]() |
bf5352b44e | ||
![]() |
bff74af882 | ||
![]() |
dadbc1aefa | ||
![]() |
fc354eec0e | ||
![]() |
47b1b45828 | ||
![]() |
e6da2313e6 | ||
![]() |
fe81bcc003 | ||
![]() |
10eee47b6b | ||
![]() |
c930c86cfa | ||
![]() |
d3196e0e34 | ||
![]() |
deb34c9473 | ||
![]() |
892d2ce34f | ||
![]() |
b107948c47 | ||
![]() |
d2648657fb | ||
![]() |
32b103eb1d | ||
![]() |
e66047e072 | ||
![]() |
0c84224ca2 | ||
![]() |
7bb67ae94b | ||
![]() |
ccc30116ba | ||
![]() |
9cf115a752 | ||
![]() |
d9523a0cbf | ||
![]() |
2fd6942de4 | ||
![]() |
b19a7e006e | ||
![]() |
5c26f95a4b | ||
![]() |
ce171f5c00 | ||
![]() |
ab872b075a | ||
![]() |
87395d259e | ||
![]() |
72f29b1283 | ||
![]() |
f2a6f18553 | ||
![]() |
8cac5ca90c | ||
![]() |
01ec414873 | ||
![]() |
6b0fb3dd06 | ||
![]() |
e89c6494a6 | ||
![]() |
eff76d578b | ||
![]() |
76ebbfefd2 | ||
![]() |
150c9b5fa3 | ||
![]() |
55df88d7ae | ||
![]() |
619787e6d2 | ||
![]() |
3f8bad3ed1 | ||
![]() |
c146712b16 | ||
![]() |
2cabe59c22 | ||
![]() |
feba9ffdc4 | ||
![]() |
74ab940aff | ||
![]() |
82c1988a2d | ||
![]() |
ac5c6ec288 | ||
![]() |
47735d1dae | ||
![]() |
35b5dadb99 | ||
![]() |
e2d784a5b5 | ||
![]() |
b7a16d5a59 | ||
![]() |
97dcbe84da | ||
![]() |
32b24726ed | ||
![]() |
343278b291 | ||
![]() |
b11824b058 | ||
![]() |
562f7c8718 | ||
![]() |
d382ca2401 | ||
![]() |
a9630ac847 | ||
![]() |
22c0b0abaa | ||
![]() |
aabe0091cc | ||
![]() |
3d9af2a67c | ||
![]() |
4ae582c305 | ||
![]() |
5fdafc00e6 | ||
![]() |
2165960ba1 | ||
![]() |
2bb5f53b98 | ||
![]() |
211b3eddea | ||
![]() |
bec53f97a2 | ||
![]() |
c3332e4a39 | ||
![]() |
f8a03be2f1 | ||
![]() |
712634b301 | ||
![]() |
19d53c6643 | ||
![]() |
f14419bab5 | ||
![]() |
3003485dc6 | ||
![]() |
01f6791d1c | ||
![]() |
cdb67fc90e | ||
![]() |
78cb098691 | ||
![]() |
45152ad55e | ||
![]() |
45879e3100 | ||
![]() |
c4adb30ab2 | ||
![]() |
b20bae23cc | ||
![]() |
11ed2d5f18 | ||
![]() |
f814b6d47c | ||
![]() |
a67b92a04c | ||
![]() |
2a48b810a4 | ||
![]() |
9fb8e9edef | ||
![]() |
d2bccbe8ac | ||
![]() |
e44a60e814 | ||
![]() |
02a71cb6a7 | ||
![]() |
e600784ebf | ||
![]() |
5e19a3b892 | ||
![]() |
0443310385 | ||
![]() |
03ab23fec8 | ||
![]() |
da8afd36b2 | ||
![]() |
fe7893d1b3 | ||
![]() |
d19bf5d6ee | ||
![]() |
bfdcfa4766 | ||
![]() |
c47c1a7867 | ||
![]() |
f16a24ddf4 | ||
![]() |
8bf112669f | ||
![]() |
4278664208 | ||
![]() |
0789657fd5 | ||
![]() |
b566c78f00 | ||
![]() |
c287e529a8 | ||
![]() |
164d05fdce | ||
![]() |
c11c4dad2f | ||
![]() |
0af8d0b7ea | ||
![]() |
2b4ed0c273 | ||
![]() |
9fc50e8dbc | ||
![]() |
a35122231c | ||
![]() |
7e4ee32b54 | ||
![]() |
7df80eadcf | ||
![]() |
2aaba1d2b8 | ||
![]() |
7c129a4018 | ||
![]() |
cb66ce069e | ||
![]() |
a27e72362a | ||
![]() |
c5be5e6d12 | ||
![]() |
63fc16d872 | ||
![]() |
5f99ed943a | ||
![]() |
f44e5d3142 | ||
![]() |
532163738e | ||
![]() |
63fa922547 | ||
![]() |
48e4cb5ae2 | ||
![]() |
ff8a73c2d1 | ||
![]() |
afd26c6f1a | ||
![]() |
67b06a88b2 | ||
![]() |
5cb5594288 | ||
![]() |
87629191b3 | ||
![]() |
4a518e3e7a | ||
![]() |
6089526975 | ||
![]() |
b9e9223fdd | ||
![]() |
e963eedb64 | ||
![]() |
3b2c61e813 | ||
![]() |
265e019381 | ||
![]() |
560e36a65c | ||
![]() |
b05a3fbb55 | ||
![]() |
3a899e28dc | ||
![]() |
f26238e824 | ||
![]() |
3717e34bba | ||
![]() |
be6f95d43e | ||
![]() |
99a765dc06 | ||
![]() |
351e7ea16b | ||
![]() |
2fa79a2e2f | ||
![]() |
44a917929d | ||
![]() |
2dd4aa7bf6 | ||
![]() |
4c1af007ca | ||
![]() |
08013be6dd | ||
![]() |
0daf4545a9 | ||
![]() |
1269bf9791 | ||
![]() |
5cb21324a1 | ||
![]() |
3eef80506b | ||
![]() |
283d9a0f5f | ||
![]() |
a84365659b | ||
![]() |
21ebc7f95b | ||
![]() |
72e72d7d4b | ||
![]() |
db9dc11022 | ||
![]() |
02ed2c0ebe | ||
![]() |
0f506ea8eb | ||
![]() |
8e7e8da4a3 | ||
![]() |
b56c606523 | ||
![]() |
f457269a68 | ||
![]() |
5b0b9da0b9 | ||
![]() |
0ed0bdc655 | ||
![]() |
a8fa4b56f9 | ||
![]() |
cd514b140e | ||
![]() |
f3329fdc8c | ||
![]() |
689bbf2419 | ||
![]() |
a6b89e4e8a | ||
![]() |
ffd2cb9814 | ||
![]() |
1d5f088740 | ||
![]() |
4e7011c25d | ||
![]() |
f4ac176d77 | ||
![]() |
e4cf7b86fa | ||
![]() |
9876d5276c | ||
![]() |
0b1b25191d | ||
![]() |
9980b9972f | ||
![]() |
93b7ca77ca | ||
![]() |
40697fea96 | ||
![]() |
c541fa1763 | ||
![]() |
fd08f1e23d | ||
![]() |
3a07121784 | ||
![]() |
1495fada90 | ||
![]() |
62fed4c1eb | ||
![]() |
00f9af70a9 | ||
![]() |
0ae3fcb0b7 | ||
![]() |
dfffa67c0f | ||
![]() |
f81c556b63 | ||
![]() |
ce8091c14e | ||
![]() |
581cb642ff | ||
![]() |
e02aaedc42 | ||
![]() |
8c66de2391 | ||
![]() |
cb8ca433d9 | ||
![]() |
b914d6e305 | ||
![]() |
956e19be7d | ||
![]() |
b3d5a4dfdb | ||
![]() |
c63cdae84f | ||
![]() |
dec044ad8b | ||
![]() |
2a12ec09fb | ||
![]() |
91e920c498 | ||
![]() |
9b19c45735 | ||
![]() |
3843d21dbf | ||
![]() |
73db164fb1 | ||
![]() |
17be6b106b | ||
![]() |
869981cfe4 | ||
![]() |
7dd56fb0fa | ||
![]() |
f0f09d3714 | ||
![]() |
9a66199904 | ||
![]() |
bf732f2a2b | ||
![]() |
c418eecf83 | ||
![]() |
98bf427600 | ||
![]() |
9aa5ee3372 | ||
![]() |
ccb3d3d308 | ||
![]() |
9ff0471274 | ||
![]() |
fdb20e4a30 | ||
![]() |
56630bb717 | ||
![]() |
08a41d9bd6 | ||
![]() |
cd46a69f2c | ||
![]() |
794a4bd9a1 | ||
![]() |
a120a455bf | ||
![]() |
cd72a2ed7e | ||
![]() |
3eff7e76aa | ||
![]() |
959d1944fd | ||
![]() |
b0966532bf | ||
![]() |
827b2def1e | ||
![]() |
80154b280e | ||
![]() |
efd0dd4c3d | ||
![]() |
c91b775b73 | ||
![]() |
1c237aef77 | ||
![]() |
89c5298bb9 | ||
![]() |
76c0d0912f | ||
![]() |
5eb12ac5fe | ||
![]() |
d238155640 | ||
![]() |
de626c0f5f | ||
![]() |
973e78355f | ||
![]() |
ab32dd7420 | ||
![]() |
b82c7ad608 | ||
![]() |
2a7aa2fc0d | ||
![]() |
f5e98eb86f | ||
![]() |
362a19c2e1 | ||
![]() |
f4a4956dd4 | ||
![]() |
746488cabf | ||
![]() |
4449248c6f | ||
![]() |
036e14ab7f | ||
![]() |
f840eee1b7 | ||
![]() |
553132443f | ||
![]() |
417d45939f | ||
![]() |
837c749cd7 | ||
![]() |
b0e286972d | ||
![]() |
3cfe1e3083 | ||
![]() |
6738295475 | ||
![]() |
1617eba764 | ||
![]() |
d20242f589 | ||
![]() |
68affce274 | ||
![]() |
c4b9065749 | ||
![]() |
d57a5d1793 | ||
![]() |
74e062fdb3 | ||
![]() |
6bdc0c92fe | ||
![]() |
d7945de001 | ||
![]() |
899aa31df3 | ||
![]() |
ac81fae855 | ||
![]() |
8c6cddf1bb | ||
![]() |
508392db6e | ||
![]() |
3ac0165f00 | ||
![]() |
1691c13b47 | ||
![]() |
f8df694aa3 | ||
![]() |
ac05495781 | ||
![]() |
3ba2a29e54 | ||
![]() |
306ab0c56c | ||
![]() |
76b438f79c | ||
![]() |
bc14f06a07 | ||
![]() |
844cf316e2 | ||
![]() |
9344d85414 | ||
![]() |
a539197bc4 | ||
![]() |
eb859e83f8 | ||
![]() |
e4a640844c | ||
![]() |
119bbba254 | ||
![]() |
8c5978599a | ||
![]() |
bbf3d382e8 | ||
![]() |
c85f70a236 | ||
![]() |
7e52d4f5d6 | ||
![]() |
6d9dbf9e54 | ||
![]() |
ec37dece12 | ||
![]() |
e0fd8cd850 | ||
![]() |
cf65bd8ad7 | ||
![]() |
8a9352939a | ||
![]() |
6ecc1c14d2 | ||
![]() |
5f531ac9b0 | ||
![]() |
7a551081ee | ||
![]() |
74139985c9 | ||
![]() |
f3cdcc008a | ||
![]() |
a391815921 | ||
![]() |
98fd092053 | ||
![]() |
feee075122 | ||
![]() |
ddde1ee31e | ||
![]() |
c5aacdd682 | ||
![]() |
a77cf1beec | ||
![]() |
d7bfdd0efc | ||
![]() |
62aee36f82 | ||
![]() |
8ca9115dc8 | ||
![]() |
8bf8892ab3 | ||
![]() |
8739552c0b | ||
![]() |
e6834f25ed | ||
![]() |
f9fc438de8 | ||
![]() |
677b2c6618 | ||
![]() |
301a78f983 | ||
![]() |
979f014799 | ||
![]() |
a326dcaf0e | ||
![]() |
5bf2fa5c56 | ||
![]() |
fe0404a084 | ||
![]() |
22a1134f0e | ||
![]() |
fc3d558d47 | ||
![]() |
45c72f1f22 | ||
![]() |
fd9cca565b | ||
![]() |
0709367587 | ||
![]() |
98277f6ceb | ||
![]() |
8dd509ba53 | ||
![]() |
8df455f55b | ||
![]() |
36782f13bf | ||
![]() |
e823067a6b | ||
![]() |
c3ef12d580 | ||
![]() |
d64d1650e3 | ||
![]() |
a74abb8ea8 | ||
![]() |
e74ab00b3e | ||
![]() |
2e2ac53071 | ||
![]() |
87c0f48095 | ||
![]() |
25b9bde0a5 | ||
![]() |
63d3a0e8b3 | ||
![]() |
4cc0f3fd53 | ||
![]() |
5b2176562b | ||
![]() |
099dc8d1d2 | ||
![]() |
cf98c497d5 | ||
![]() |
c5eb3941b9 | ||
![]() |
0e93b8ee0d | ||
![]() |
807621402d | ||
![]() |
321155eb40 | ||
![]() |
d34c074b92 | ||
![]() |
abc8e903c1 | ||
![]() |
832ba38f1b | ||
![]() |
70de2f5278 | ||
![]() |
604d4eec79 | ||
![]() |
ac5246e21d | ||
![]() |
951157dc26 | ||
![]() |
68119ddcd4 | ||
![]() |
c82be2cd60 | ||
![]() |
9a149a7aba | ||
![]() |
108fabe18f | ||
![]() |
8ce98dd15a | ||
![]() |
9d21cccac1 | ||
![]() |
8f4abf6a63 | ||
![]() |
bd9a4ff8de | ||
![]() |
d9398a91d1 | ||
![]() |
ef84937fd6 | ||
![]() |
2a2d20a7fc | ||
![]() |
8a1c49a4ae | ||
![]() |
b806eb6a61 | ||
![]() |
39948db59a | ||
![]() |
fbfb4e2a73 | ||
![]() |
595ac84779 | ||
![]() |
eb145757e5 | ||
![]() |
fc0e1a3cb9 | ||
![]() |
746f72a279 | ||
![]() |
ec3d5fc427 | ||
![]() |
dec6f04499 | ||
![]() |
a90d266017 | ||
![]() |
df9fcf9850 | ||
![]() |
85608a8ab7 | ||
![]() |
5ef9cd5f86 | ||
![]() |
52d7d2cae7 | ||
![]() |
ceca91d1e7 | ||
![]() |
bc7c11be96 | ||
![]() |
a2734330e1 | ||
![]() |
ef8180c8a8 | ||
![]() |
cd773a1dec | ||
![]() |
244a212592 | ||
![]() |
f72b07eb0e | ||
![]() |
314c1c8b5c | ||
![]() |
98e3426769 | ||
![]() |
7ef8d67831 | ||
![]() |
c455a5dd6a | ||
![]() |
74daca668e | ||
![]() |
211453df43 | ||
![]() |
1cc7428445 | ||
![]() |
a40fa22055 | ||
![]() |
2047bba4f7 | ||
![]() |
a064ab5c2b | ||
![]() |
e8ce7048d8 | ||
![]() |
0c37d06936 | ||
![]() |
d919373853 | ||
![]() |
6c00be0a63 | ||
![]() |
0671287b80 | ||
![]() |
867b4719d1 | ||
![]() |
de6c527ca4 | ||
![]() |
9e7e3708e3 | ||
![]() |
8bd9f50659 | ||
![]() |
cb5a01da29 | ||
![]() |
bfe85dd710 | ||
![]() |
fc544dc389 | ||
![]() |
24067312f6 | ||
![]() |
501fe83710 | ||
![]() |
5513b0e121 | ||
![]() |
959e7745a6 | ||
![]() |
76e947651d | ||
![]() |
5ba04eb620 | ||
![]() |
ee12c68b8f | ||
![]() |
b2ccd32cd7 | ||
![]() |
7ceb16cc5a | ||
![]() |
5a8b7c17da | ||
![]() |
41a618737b | ||
![]() |
67771abc9d | ||
![]() |
c151df32bc | ||
![]() |
b346ad8080 | ||
![]() |
cd57271386 | ||
![]() |
b9f20b36cb | ||
![]() |
62d2640c37 | ||
![]() |
54eb52c19a | ||
![]() |
77a7d3f24b | ||
![]() |
8c9d63f48f | ||
![]() |
5a8e93ed0a | ||
![]() |
2d32e89b87 | ||
![]() |
88c13768e3 | ||
![]() |
29f4430658 | ||
![]() |
b558a1c9dd | ||
![]() |
a4ef26749b | ||
![]() |
6aa3092be0 | ||
![]() |
abca47f36f | ||
![]() |
407b5e199e | ||
![]() |
2ffd430b0b | ||
![]() |
d4099d68a7 | ||
![]() |
e1b0d86098 | ||
![]() |
1a7f121ac6 | ||
![]() |
ffa669899a | ||
![]() |
17fed954bf | ||
![]() |
467e42d8aa | ||
![]() |
a023f24a08 | ||
![]() |
81736447ee | ||
![]() |
27f69f5439 | ||
![]() |
5ebb468ccf | ||
![]() |
0ead802333 | ||
![]() |
54474e5b33 | ||
![]() |
0411d52420 | ||
![]() |
cef659b0de | ||
![]() |
035c3ef8fe | ||
![]() |
5afdb1e97f | ||
![]() |
f7c0ec6595 | ||
![]() |
0a407c5425 | ||
![]() |
b9e7fdd2b0 | ||
![]() |
c74105aad7 | ||
![]() |
5f0892dec4 | ||
![]() |
302dea4169 | ||
![]() |
ce13979690 | ||
![]() |
1cf210fa25 | ||
![]() |
d1253922c3 | ||
![]() |
6b00622329 | ||
![]() |
aeb94e166b | ||
![]() |
8bb4c65272 | ||
![]() |
09a4d869f6 | ||
![]() |
ae4c130a61 | ||
![]() |
5621d592b4 | ||
![]() |
4858882816 | ||
![]() |
b06bdc2da3 | ||
![]() |
1ea5d90ea3 | ||
![]() |
f9f335e692 | ||
![]() |
3ead48f0db | ||
![]() |
ccba94197d | ||
![]() |
8adb7dc97c | ||
![]() |
796b64541f | ||
![]() |
b8c27bc17d | ||
![]() |
7dcdf80f49 | ||
![]() |
57023457ee | ||
![]() |
1057ac4db7 | ||
![]() |
69f5674d9e | ||
![]() |
ebad407586 | ||
![]() |
71387846dc | ||
![]() |
97c1c34708 | ||
![]() |
79abd773a2 | ||
![]() |
9cd173ef83 | ||
![]() |
b723728177 | ||
![]() |
316171491f | ||
![]() |
91ff502872 | ||
![]() |
6e414180e0 | ||
![]() |
7d2ae4e252 | ||
![]() |
fb4cb07c6f | ||
![]() |
19f91a7deb | ||
![]() |
bb044a789c | ||
![]() |
2153cfc749 | ||
![]() |
2e8b4fbdc8 | ||
![]() |
4141100b1c | ||
![]() |
baa08160bb | ||
![]() |
35ef4aad60 | ||
![]() |
ffa5e29dab | ||
![]() |
28b5c535ec | ||
![]() |
3807350c61 | ||
![]() |
9d2467cf62 | ||
![]() |
d2480d3194 | ||
![]() |
148eb03d13 | ||
![]() |
ed8aec62fc | ||
![]() |
f7b5c6307c | ||
![]() |
40d110fc3f | ||
![]() |
70aa5d0f6c | ||
![]() |
8fcec8e2cb | ||
![]() |
2d3b48f86f | ||
![]() |
a15ac06771 | ||
![]() |
784cc3bc29 | ||
![]() |
8384bd7fc7 | ||
![]() |
8a518f0def | ||
![]() |
99c10bc6de | ||
![]() |
f30f20db86 | ||
![]() |
9bf946e196 | ||
![]() |
bfaad1f28d | ||
![]() |
c61a3bf431 | ||
![]() |
1b77996ccd | ||
![]() |
9e3ecc8372 | ||
![]() |
18f4e41550 | ||
![]() |
26cdbaa3df | ||
![]() |
c76c1337ba | ||
![]() |
fe3b779016 | ||
![]() |
682638d4de | ||
![]() |
c96663daca | ||
![]() |
b3ed988119 | ||
![]() |
77695aa55b | ||
![]() |
c5a45645a6 | ||
![]() |
0de47e2a4e | ||
![]() |
90598d2405 | ||
![]() |
f90e9ba871 | ||
![]() |
726bdd7be2 | ||
![]() |
52db40eb41 | ||
![]() |
3c371a0c59 | ||
![]() |
c941bc4109 | ||
![]() |
cc76e5353c | ||
![]() |
edfd82fd42 | ||
![]() |
492bad645b | ||
![]() |
ab4517f611 | ||
![]() |
11bb46e393 | ||
![]() |
2bfcfa6dae | ||
![]() |
71c4714a6e | ||
![]() |
a4e63c5f86 | ||
![]() |
c71e7d0132 | ||
![]() |
daa966975e | ||
![]() |
2e5757a3f0 | ||
![]() |
f66024b37c | ||
![]() |
c16ca7be13 | ||
![]() |
e13e754bc4 | ||
![]() |
6ccea59f71 | ||
![]() |
d7fd23d8a8 | ||
![]() |
d0ca69bc27 | ||
![]() |
2a7d6addde | ||
![]() |
8e4aeec3bd | ||
![]() |
7f83a6e667 | ||
![]() |
ae838b13a8 | ||
![]() |
ce5dc6f100 | ||
![]() |
b1551d0436 | ||
![]() |
b4c2433bac | ||
![]() |
71b28be3c8 | ||
![]() |
e25d92e1f5 | ||
![]() |
65cda10884 | ||
![]() |
625126df68 | ||
![]() |
e0ee8ca17c | ||
![]() |
af95e781f5 | ||
![]() |
1d8227788b | ||
![]() |
5fdd8440ac | ||
![]() |
5d8daa6990 | ||
![]() |
cf802d0d38 | ||
![]() |
9b230a7d93 | ||
![]() |
e2f3e7c3a6 | ||
![]() |
2fd2e5ceb2 | ||
![]() |
d88358be8e | ||
![]() |
d80885c7a8 | ||
![]() |
9cbf437509 | ||
![]() |
cd7e8e4bdd | ||
![]() |
ed024a0aa5 | ||
![]() |
c7c9c49f4e | ||
![]() |
75d0514d05 | ||
![]() |
5b02715c7b | ||
![]() |
2c160a8a25 | ||
![]() |
3b0eea69ce | ||
![]() |
cc576cf1a9 | ||
![]() |
97a71482a9 | ||
![]() |
8822b6c808 | ||
![]() |
5099595aee | ||
![]() |
39a650ee54 | ||
![]() |
b19c7d462b | ||
![]() |
a8b821c213 | ||
![]() |
535003014b | ||
![]() |
3c05ae4e1a | ||
![]() |
c835b67bac | ||
![]() |
8c32941428 | ||
![]() |
b5dac00dcb | ||
![]() |
ffdc721c79 | ||
![]() |
0828a9fc11 | ||
![]() |
4b664b6f09 | ||
![]() |
679633245d | ||
![]() |
ce8a77c765 | ||
![]() |
2f78c4acfa | ||
![]() |
72f6841aac | ||
![]() |
10bd9b14fc | ||
![]() |
3498aade85 | ||
![]() |
c4539e10fb | ||
![]() |
2b3052e9d7 | ||
![]() |
e725e15f7a | ||
![]() |
4d1113e265 | ||
![]() |
52352ac27a | ||
![]() |
f60b2b754d | ||
![]() |
4a3f9712b2 | ||
![]() |
fb094fca0f | ||
![]() |
de10b356cf | ||
![]() |
bd6d6caa8a | ||
![]() |
1c4af08ed3 | ||
![]() |
c97d361b6c | ||
![]() |
379b1d84dd | ||
![]() |
c13e20643b | ||
![]() |
76b6fcf554 | ||
![]() |
57e909e790 | ||
![]() |
d6f7876e68 | ||
![]() |
56e0923c22 | ||
![]() |
f4b98f5e32 | ||
![]() |
2d56b70a36 | ||
![]() |
980cfaf295 | ||
![]() |
c2a43c733a | ||
![]() |
568e65a6ab | ||
![]() |
59d6b3afa0 | ||
![]() |
b89c04b928 | ||
![]() |
12090657bb | ||
![]() |
4e21cf0bdd | ||
![]() |
ba4ef72d56 | ||
![]() |
f3e6a4314f | ||
![]() |
e14ce3d950 | ||
![]() |
70aa38f5bd | ||
![]() |
55ec082628 | ||
![]() |
6f27126c8d | ||
![]() |
ee21a91313 | ||
![]() |
c5efaa1c00 | ||
![]() |
6476357596 | ||
![]() |
77f71acbc8 | ||
![]() |
4a08a5413d | ||
![]() |
e3d89cc6b6 | ||
![]() |
64afb07e91 | ||
![]() |
f639f7c280 | ||
![]() |
986dd2ddd2 | ||
![]() |
7abdb5d046 | ||
![]() |
0f1e186189 | ||
![]() |
96d208e0d8 | ||
![]() |
38ed38864e | ||
![]() |
a12ba7bd38 | ||
![]() |
4a177e3931 | ||
![]() |
bef5b38d49 | ||
![]() |
0a95f116fc | ||
![]() |
327cd662b4 | ||
![]() |
bb05ba3d00 | ||
![]() |
c0ad5d1d16 | ||
![]() |
4c39631428 | ||
![]() |
afc2b3b74f | ||
![]() |
19fc1417ae | ||
![]() |
e2fefa51f5 | ||
![]() |
f668d5617f | ||
![]() |
47c4ff15d6 | ||
![]() |
ccf1bdc0b4 | ||
![]() |
e993fcf80c | ||
![]() |
1bdc30a09e | ||
![]() |
0f7e34e7ec | ||
![]() |
f56e89597f | ||
![]() |
9460fb28c4 | ||
![]() |
7207b9734f | ||
![]() |
2be703b329 | ||
![]() |
4cea74ef3b | ||
![]() |
3be3267d06 | ||
![]() |
98db604dba | ||
![]() |
ebf6f8c6de | ||
![]() |
2ebacad398 | ||
![]() |
53c59cf675 | ||
![]() |
9da261cb39 | ||
![]() |
3a587ea0d4 | ||
![]() |
8a60919e1f | ||
![]() |
382dcddf12 | ||
![]() |
6b67acbeb5 | ||
![]() |
7b0fca6824 | ||
![]() |
47555d314a | ||
![]() |
0643b71908 | ||
![]() |
afc848bf22 | ||
![]() |
9fbbcd6d8a | ||
![]() |
cc1eb648f9 | ||
![]() |
04a139fe3d | ||
![]() |
1a86167a47 | ||
![]() |
614ed7fd0c | ||
![]() |
4eb69d6af5 | ||
![]() |
0547f2a931 | ||
![]() |
b5fbe0b145 | ||
![]() |
443c3c2a56 | ||
![]() |
d75daa9644 | ||
![]() |
7963abb27a | ||
![]() |
0b9e8fda34 | ||
![]() |
a3cacc0c8b | ||
![]() |
5a4840f641 | ||
![]() |
3d7d689040 | ||
![]() |
b60c08dd28 | ||
![]() |
80bc567c31 | ||
![]() |
888ac2e180 | ||
![]() |
421ebcc8b2 | ||
![]() |
b56fa8c50a | ||
![]() |
42401775e1 | ||
![]() |
9c9bc58c16 | ||
![]() |
fbc129cccc | ||
![]() |
99638190cb | ||
![]() |
d78e9e6aa8 | ||
![]() |
878155a03d | ||
![]() |
9922eb83e2 | ||
![]() |
79f861f012 | ||
![]() |
4faa9d109e | ||
![]() |
28534ecc61 | ||
![]() |
616e0a21d8 | ||
![]() |
a546ffd490 | ||
![]() |
b5d0aede38 | ||
![]() |
a014d853a4 | ||
![]() |
c4ddf7697d | ||
![]() |
a2931b6774 | ||
![]() |
3ac7bf3761 | ||
![]() |
922344811f | ||
![]() |
cb2fcaa9b1 | ||
![]() |
1f50bd0649 | ||
![]() |
e4b2de5c68 | ||
![]() |
f862b479e7 | ||
![]() |
358c59bd8d | ||
![]() |
74fe135c9c | ||
![]() |
8d3896172d | ||
![]() |
9d9725144d | ||
![]() |
06f83bf1c0 | ||
![]() |
c2756d57d8 | ||
![]() |
56504692af | ||
![]() |
e542e75b9e | ||
![]() |
806e43c34c | ||
![]() |
29e7d00894 | ||
![]() |
9ee661c1e4 | ||
![]() |
36c0e2416d | ||
![]() |
e4ba3ff1db | ||
![]() |
be69b49880 | ||
![]() |
cc317d27f5 | ||
![]() |
c16709ed95 | ||
![]() |
e13eaf6706 | ||
![]() |
a1eb3b8475 | ||
![]() |
d52e425ba2 | ||
![]() |
dd8dc1ef1d | ||
![]() |
bc427de16a | ||
![]() |
db5988bbe1 | ||
![]() |
a3875af4b4 | ||
![]() |
d70e7da0ef | ||
![]() |
d42f35de5d | ||
![]() |
cd57469e06 | ||
![]() |
d98d6ff45f | ||
![]() |
14e38f0469 | ||
![]() |
f0f6d3f1cd | ||
![]() |
0b383542da | ||
![]() |
b2cec10601 | ||
![]() |
48658d5a55 | ||
![]() |
5207ca1d52 | ||
![]() |
7196fb8e82 | ||
![]() |
48ada2eebb | ||
![]() |
0de5808ed2 | ||
![]() |
ebc544e4b4 | ||
![]() |
a31fb3c987 | ||
![]() |
dfc7cd7f5d | ||
![]() |
a8bb2a42a1 | ||
![]() |
3d4c0e6667 | ||
![]() |
25fb288016 | ||
![]() |
1b8b8cdd11 | ||
![]() |
e6737479f7 | ||
![]() |
7f75832bf1 | ||
![]() |
33339e3bd8 | ||
![]() |
c298c1166f | ||
![]() |
c3d9eef01f | ||
![]() |
5ffdc66864 | ||
![]() |
2f50e18eb5 | ||
![]() |
9922c1503a | ||
![]() |
fce99d4b17 | ||
![]() |
11567085d8 | ||
![]() |
83f8e84247 | ||
![]() |
215107e8ea | ||
![]() |
d3f2b93c42 | ||
![]() |
11eb5cb0fa | ||
![]() |
9a7af97b2d | ||
![]() |
5e11469f50 | ||
![]() |
0c7a3d1fff | ||
![]() |
8a705bf4b0 | ||
![]() |
ee7102fcd1 | ||
![]() |
a44e38300b | ||
![]() |
b00e20c29f | ||
![]() |
6a89180deb | ||
![]() |
65d2b806cc | ||
![]() |
c149a3033c | ||
![]() |
4b7c233f1a | ||
![]() |
6e8e9c2aa9 | ||
![]() |
b6f628ee40 | ||
![]() |
bf79a700b7 | ||
![]() |
cdeb6e750f | ||
![]() |
d642aeba0f | ||
![]() |
6a6aee510d | ||
![]() |
ea17a92dbc | ||
![]() |
32a0a60480 | ||
![]() |
5a56644702 | ||
![]() |
29113808ee | ||
![]() |
dd226360bb | ||
![]() |
1a9aedf152 | ||
![]() |
d82c6df57e | ||
![]() |
7c91b4474a | ||
![]() |
a4f21db272 | ||
![]() |
58a8e1859e | ||
![]() |
2bed5b18c1 | ||
![]() |
fb5eb57345 | ||
![]() |
6471361715 | ||
![]() |
01687a9d57 | ||
![]() |
801fbf44c5 | ||
![]() |
ba1416cc0e | ||
![]() |
afc1c83af4 | ||
![]() |
da056866ff | ||
![]() |
336c2d34e6 | ||
![]() |
f3a969d35c | ||
![]() |
63db42a1d4 | ||
![]() |
c12dd77c64 | ||
![]() |
f58ffe41f8 | ||
![]() |
4f138c600b | ||
![]() |
445d2e372c | ||
![]() |
1087cb55b4 | ||
![]() |
600f4be2c4 | ||
![]() |
5e6665494d | ||
![]() |
2ef25f3153 | ||
![]() |
bc28ea1fde | ||
![]() |
623e31ddee | ||
![]() |
ceebe14628 | ||
![]() |
b29cc58144 | ||
![]() |
1b328da265 | ||
![]() |
06ca5354b2 | ||
![]() |
356efdb92c | ||
![]() |
bb5ab8b36d | ||
![]() |
6ecf4ecac6 | ||
![]() |
05ab49a615 | ||
![]() |
3773c385c7 | ||
![]() |
3227ef4bca | ||
![]() |
5a07e8d32b | ||
![]() |
29571a1acd | ||
![]() |
b8538c2c12 | ||
![]() |
7466773ac8 | ||
![]() |
b8ca40170e | ||
![]() |
bd86a0ac3b | ||
![]() |
df3f13ded8 | ||
![]() |
a428e2b689 | ||
![]() |
86407b9f6f | ||
![]() |
eceb79ceab | ||
![]() |
43fb68f8a0 | ||
![]() |
14e7b8a1ef | ||
![]() |
62459a8ae1 | ||
![]() |
86c0e6114f | ||
![]() |
1a9141877d | ||
![]() |
6ec18fc630 | ||
![]() |
4d674392e8 | ||
![]() |
6704b2cedf | ||
![]() |
fe4fb5f1ac | ||
![]() |
350d4e5071 | ||
![]() |
23f47d0ad2 | ||
![]() |
c037e95861 | ||
![]() |
2e1b35959f | ||
![]() |
7f46d9e0f9 | ||
![]() |
069b5f81a0 | ||
![]() |
3a36d0b13f | ||
![]() |
f98d93efa8 | ||
![]() |
91e037346b | ||
![]() |
8e1430243e | ||
![]() |
98b3d294aa | ||
![]() |
38a01988a5 | ||
![]() |
d16eff5039 | ||
![]() |
8fb481751f | ||
![]() |
ba6f89a757 | ||
![]() |
48e76e1538 | ||
![]() |
0e1d018ce3 | ||
![]() |
50fbbf2d3b | ||
![]() |
247916fe89 | ||
![]() |
ed801f7a27 | ||
![]() |
f68d577986 | ||
![]() |
5c49730cb9 | ||
![]() |
04c12823b5 | ||
![]() |
add40c7652 | ||
![]() |
f0760e99b7 | ||
![]() |
72391389a3 | ||
![]() |
e68beb8a43 | ||
![]() |
40e2832e67 | ||
![]() |
18fecf8c09 | ||
![]() |
414cf1b333 | ||
![]() |
d10f891f51 | ||
![]() |
36a1f6cfb1 | ||
![]() |
12bef16d54 | ||
![]() |
77db8c8401 | ||
![]() |
66eecd3675 | ||
![]() |
c03b1fae68 | ||
![]() |
b5927322e6 | ||
![]() |
1cf4107e1c | ||
![]() |
c12408326c | ||
![]() |
37d55b55fc | ||
![]() |
4434e59e5a | ||
![]() |
45180d98f6 | ||
![]() |
44494ad18e | ||
![]() |
9aed758d1b | ||
![]() |
dbe5587806 | ||
![]() |
1447536906 | ||
![]() |
27ec517084 | ||
![]() |
ce1f034bac | ||
![]() |
f1f96f16e9 | ||
![]() |
30eec5adee | ||
![]() |
5307dfee21 | ||
![]() |
8b5b9e508b | ||
![]() |
7665e9b076 | ||
![]() |
227d94f38d | ||
![]() |
b724ae9e0e | ||
![]() |
df6cc14201 | ||
![]() |
d981d7859d | ||
![]() |
0f1ec515c1 | ||
![]() |
78e18256f7 | ||
![]() |
a0d04ba091 | ||
![]() |
c02871fdfe | ||
![]() |
0d52f555b2 | ||
![]() |
025cf6320f | ||
![]() |
5997401e9e | ||
![]() |
b4068dac56 | ||
![]() |
458d6e24fc | ||
![]() |
4f4ca61ada | ||
![]() |
dfeeccfcca | ||
![]() |
3a101e8ec5 | ||
![]() |
7a2d7fdd19 | ||
![]() |
4899dfe642 | ||
![]() |
78f5c417a4 | ||
![]() |
b8c0f88440 | ||
![]() |
d6b6e94059 | ||
![]() |
310355a00b | ||
![]() |
8cf26d6f3c | ||
![]() |
b15a10f905 | ||
![]() |
58eeb6b1b8 | ||
![]() |
f8acc45be4 | ||
![]() |
b7ab00b699 | ||
![]() |
045489e6d7 | ||
![]() |
b14e774a27 | ||
![]() |
2a8745d7e0 | ||
![]() |
499cb615f1 | ||
![]() |
5dcf1debd7 | ||
![]() |
9b57e1ac1d | ||
![]() |
68683e3a50 | ||
![]() |
4d192c7387 | ||
![]() |
d83324c4dc | ||
![]() |
ecde4c1d2d | ||
![]() |
bd8e470726 | ||
![]() |
9dd01b30bd | ||
![]() |
d2913fe627 | ||
![]() |
881cd535b9 | ||
![]() |
43acc7dc2c | ||
![]() |
e2a16d758b | ||
![]() |
17ea0efb08 | ||
![]() |
2fbd33267e | ||
![]() |
cf3977f088 | ||
![]() |
d20d4947ac | ||
![]() |
7810ad40d7 | ||
![]() |
7e1e799b3a | ||
![]() |
dfafc41ce6 | ||
![]() |
e460792c43 | ||
![]() |
a9dc491a54 | ||
![]() |
ac6693f177 | ||
![]() |
c6742117d3 | ||
![]() |
b5c47b9669 | ||
![]() |
40df3aa55e | ||
![]() |
393ca64d70 | ||
![]() |
d3627f0972 | ||
![]() |
124ab31f22 | ||
![]() |
1b66fa5004 | ||
![]() |
9494c27ad8 | ||
![]() |
3facfa5c21 | ||
![]() |
93ddce2e79 | ||
![]() |
0bf6e21e1a | ||
![]() |
6b7b076875 | ||
![]() |
8d6ffb9169 | ||
![]() |
e95d6041d8 | ||
![]() |
0554b06b7e | ||
![]() |
e3d9c44bdc | ||
![]() |
e847766514 | ||
![]() |
d4a8df04b8 | ||
![]() |
4af4649e23 | ||
![]() |
8bcddef39d | ||
![]() |
4ac96ccea2 | ||
![]() |
3c5de77ae9 | ||
![]() |
a2925b1d37 | ||
![]() |
73748e9e20 | ||
![]() |
034b47c23a | ||
![]() |
3e017efa30 | ||
![]() |
aca56fcdcc | ||
![]() |
75c9823899 | ||
![]() |
c8c0bd3351 | ||
![]() |
e1cdeb7c8f | ||
![]() |
7f97f42552 | ||
![]() |
aa7f3569ec | ||
![]() |
2d0a08442e | ||
![]() |
d2380756b2 | ||
![]() |
e778a445d9 | ||
![]() |
ded86493c2 | ||
![]() |
4d72eb42a5 | ||
![]() |
267f0587c6 | ||
![]() |
4a374a466a | ||
![]() |
b27a328d1e | ||
![]() |
d94e9d92ca | ||
![]() |
36c2e770bf | ||
![]() |
79040c116d | ||
![]() |
4aac76c549 | ||
![]() |
0ea97df1af | ||
![]() |
92e66a2764 | ||
![]() |
925e3cb6c9 | ||
![]() |
6757acba56 | ||
![]() |
5cc91cdd95 | ||
![]() |
2b41886819 | ||
![]() |
72c6efd6a0 | ||
![]() |
a1f1804112 | ||
![]() |
a8b1ceb4e9 | ||
![]() |
4fb0f7f8c6 | ||
![]() |
958cadeca8 | ||
![]() |
00f2655f1a | ||
![]() |
074f5029eb | ||
![]() |
615d591367 | ||
![]() |
e236c53f05 | ||
![]() |
10c7055b41 | ||
![]() |
a127e60e1b | ||
![]() |
66a3361e9d | ||
![]() |
13cfe11a19 | ||
![]() |
6d65671f92 | ||
![]() |
f2eafa8fbe | ||
![]() |
e4ca3b18cc | ||
![]() |
84698ae888 | ||
![]() |
fd6d6cfb6c | ||
![]() |
8cad9dfc83 | ||
![]() |
5e2f33fde5 | ||
![]() |
029ac75a04 | ||
![]() |
3aa5953cd9 | ||
![]() |
582d90ad72 | ||
![]() |
bbb0105c2f | ||
![]() |
37d17feecf | ||
![]() |
4bf5faf808 | ||
![]() |
ddedc1cd76 | ||
![]() |
1bb90f304c | ||
![]() |
efc6a8df35 | ||
![]() |
e35f90d6e4 | ||
![]() |
11518364a1 | ||
![]() |
05420291ce | ||
![]() |
442faf92c6 | ||
![]() |
62c68f4d60 | ||
![]() |
c301ae3645 | ||
![]() |
3d2d681a7b | ||
![]() |
a45646af1b | ||
![]() |
27185265f6 | ||
![]() |
a9b7d98194 | ||
![]() |
ed4a7210d3 | ||
![]() |
351ea04517 | ||
![]() |
86a8e1f4a6 | ||
![]() |
1cf3424ebe | ||
![]() |
a19f0c0db0 | ||
![]() |
530df91044 | ||
![]() |
c16c0b11cb | ||
![]() |
74556b28a8 | ||
![]() |
48340d41d6 | ||
![]() |
6306348379 | ||
![]() |
b1f1329cee | ||
![]() |
75dff1e102 | ||
![]() |
fe55f3a43d | ||
![]() |
657fd9d0d5 | ||
![]() |
1511a6ebcd | ||
![]() |
ecac26aeba | ||
![]() |
19bf9b1e36 | ||
![]() |
119a6920f2 | ||
![]() |
8237e13c44 | ||
![]() |
53b60ac817 | ||
![]() |
a18ab748fd | ||
![]() |
917488bbc3 | ||
![]() |
7e376ae952 | ||
![]() |
57a1c207c2 | ||
![]() |
50e8e92f0b | ||
![]() |
ff4fd497c4 | ||
![]() |
33b1a853b9 | ||
![]() |
f2df542cb1 | ||
![]() |
ecbbf2d3f4 | ||
![]() |
b76c7a0131 | ||
![]() |
0b0984f9a0 | ||
![]() |
9767856784 | ||
![]() |
c1f09684e6 | ||
![]() |
22b384363b | ||
![]() |
5b23331751 | ||
![]() |
7a2bb32843 | ||
![]() |
c0a4e07e5a | ||
![]() |
322158cccb | ||
![]() |
8db3b59e0f | ||
![]() |
1691976587 | ||
![]() |
60e6b4d21e | ||
![]() |
5750591df2 | ||
![]() |
0d50caa179 | ||
![]() |
8b06135b41 | ||
![]() |
a75da54455 | ||
![]() |
de7f6c3f5f | ||
![]() |
4245480656 | ||
![]() |
1824c8131e | ||
![]() |
573ea55187 | ||
![]() |
b48b5d6cc7 | ||
![]() |
4e9606d2e0 | ||
![]() |
78500fa933 | ||
![]() |
9c69b98a49 | ||
![]() |
e6d8ef98d3 | ||
![]() |
3f1af1690b | ||
![]() |
39af967433 | ||
![]() |
83b5e01a28 | ||
![]() |
1eacbd50fa | ||
![]() |
84374b6b1e | ||
![]() |
391316c9b5 | ||
![]() |
705c62ebd7 | ||
![]() |
cb520c00a5 | ||
![]() |
2f24138345 | ||
![]() |
96512b80cc | ||
![]() |
fcb9b51978 | ||
![]() |
9bf7c97775 | ||
![]() |
24bf3674f3 | ||
![]() |
f408f1a368 | ||
![]() |
7d8d563c62 | ||
![]() |
0a1f705fda | ||
![]() |
1952c1880b | ||
![]() |
c47dc09d34 | ||
![]() |
db3096c6e1 | ||
![]() |
b03967dac1 | ||
![]() |
bcae2596a6 | ||
![]() |
fc0347c86c | ||
![]() |
eef578f4b8 | ||
![]() |
d9563d4de1 | ||
![]() |
cc7e2bf8db | ||
![]() |
5d98e2923b | ||
![]() |
d4e232f267 | ||
![]() |
cc45945fcf | ||
![]() |
07197d12f6 | ||
![]() |
7b0a298497 | ||
![]() |
21679cf2ba | ||
![]() |
0c24d951ff | ||
![]() |
92e44b8238 | ||
![]() |
4be7cd12a1 | ||
![]() |
34387adbcd | ||
![]() |
dee4d0ccb7 | ||
![]() |
c6885c1bf4 | ||
![]() |
8b8efb57af | ||
![]() |
499e120aa4 | ||
![]() |
6f198a4736 | ||
![]() |
f500f448b1 | ||
![]() |
6ad9baa870 | ||
![]() |
f843925301 | ||
![]() |
4a3b628946 | ||
![]() |
4ffdc38cf5 | ||
![]() |
f83f1bff19 | ||
![]() |
9370ff3dfa | ||
![]() |
2053b02c61 | ||
![]() |
f34e797a0d | ||
![]() |
30a2fc1273 | ||
![]() |
48da5ef1c4 | ||
![]() |
7209dd8bae | ||
![]() |
ab736c89bb | ||
![]() |
6911639617 | ||
![]() |
dbbbba3cf8 | ||
![]() |
16e523ca68 | ||
![]() |
3b2bbd306f | ||
![]() |
54caed36f7 | ||
![]() |
dfcccda69e | ||
![]() |
de352c1609 | ||
![]() |
c30068fc97 | ||
![]() |
f28f712827 | ||
![]() |
b9720d0715 | ||
![]() |
47b3267ed4 | ||
![]() |
e16ba2adb5 | ||
![]() |
0a19b1e32c | ||
![]() |
bae9a950c0 | ||
![]() |
b63ade298f | ||
![]() |
f117d4f50a | ||
![]() |
6e4267b797 | ||
![]() |
2dd032475b | ||
![]() |
db0ed055dd | ||
![]() |
106c1bfac2 | ||
![]() |
eb2a0f45db | ||
![]() |
2b4fdd6c39 | ||
![]() |
3669320398 | ||
![]() |
d706f40ce1 | ||
![]() |
027284c29c | ||
![]() |
c55e01ff3f | ||
![]() |
a59ce7bfa2 | ||
![]() |
a6196267c9 | ||
![]() |
eb664b99ba | ||
![]() |
8414bb9a7a | ||
![]() |
ccef7c322f | ||
![]() |
120327866f | ||
![]() |
bc5c2d4eb4 | ||
![]() |
d5ff8f6117 | ||
![]() |
ad0d6f6337 | ||
![]() |
873de13b3d | ||
![]() |
56de8e5cc4 | ||
![]() |
73c82862cf | ||
![]() |
75573a3ed1 | ||
![]() |
1166d93805 | ||
![]() |
ac112a32c9 | ||
![]() |
cee45c1221 | ||
![]() |
fb56b5388e | ||
![]() |
ed42cefeee | ||
![]() |
9052947a71 | ||
![]() |
53e0af18fb | ||
![]() |
c5f59fad62 | ||
![]() |
b089a4ea80 | ||
![]() |
72b2943332 | ||
![]() |
4ec0ef7548 | ||
![]() |
25bc6761f6 | ||
![]() |
81b6562c25 | ||
![]() |
ae74189fc2 | ||
![]() |
555bba7698 | ||
![]() |
294901fbe9 | ||
![]() |
ec576bf9f9 | ||
![]() |
9273e3775b | ||
![]() |
ce5cedb466 | ||
![]() |
b184b01600 | ||
![]() |
81b4078871 | ||
![]() |
d067c8f80b | ||
![]() |
9e516efe10 | ||
![]() |
366e29439e | ||
![]() |
1c9c700d7f | ||
![]() |
b2e6b9d31f | ||
![]() |
7623f63846 | ||
![]() |
2bfaf9dce3 | ||
![]() |
5c2c1560bb | ||
![]() |
8975b4b3f6 | ||
![]() |
20da03f8c6 | ||
![]() |
ef26677b67 | ||
![]() |
91925b1826 | ||
![]() |
1f33ad037d | ||
![]() |
fef60e335e | ||
![]() |
195c78846f | ||
![]() |
0f9c956c04 | ||
![]() |
7258a82875 | ||
![]() |
c1f696c32a | ||
![]() |
f2b63d9c67 | ||
![]() |
7896a7783b | ||
![]() |
621771e1ee | ||
![]() |
2b032e8606 | ||
![]() |
5e1b724697 | ||
![]() |
e6db61c2f0 | ||
![]() |
c2e198311c | ||
![]() |
d874626662 | ||
![]() |
eead72333e | ||
![]() |
f7096ab78e | ||
![]() |
98f8feb625 | ||
![]() |
9944ca414e | ||
![]() |
719c212009 | ||
![]() |
65030e1c37 | ||
![]() |
3f88b63920 | ||
![]() |
70f1c71a9f | ||
![]() |
816df5ad47 | ||
![]() |
f7b1602adf | ||
![]() |
7e88eea532 | ||
![]() |
d1cdfd3b72 | ||
![]() |
d6a03d48f5 | ||
![]() |
d453b42b1a | ||
![]() |
7c19b961e2 | ||
![]() |
147b113b62 | ||
![]() |
d924702825 | ||
![]() |
e8784ba383 | ||
![]() |
e3a454d1a6 | ||
![]() |
392dc8b0db | ||
![]() |
2f62426f09 | ||
![]() |
58fda40389 | ||
![]() |
6a73699a38 | ||
![]() |
3bd6456fbe | ||
![]() |
608be4e050 | ||
![]() |
10f590324b | ||
![]() |
cb2d9e4bec | ||
![]() |
9e3ee28744 | ||
![]() |
472dcebf2c | ||
![]() |
39f0f748bf | ||
![]() |
9efe59a984 | ||
![]() |
fcb02af782 | ||
![]() |
9f30f53c6b | ||
![]() |
2f18ae00c5 | ||
![]() |
27a339fa12 | ||
![]() |
3aeef1afd4 | ||
![]() |
a45ee8f4ac | ||
![]() |
31b62d7dca | ||
![]() |
22f81475db | ||
![]() |
cc7cf73d59 | ||
![]() |
9682e60a25 | ||
![]() |
2c2e68123a | ||
![]() |
fcec7d45cb | ||
![]() |
7c8f502e7e | ||
![]() |
dc17c47634 | ||
![]() |
3d927c2f44 | ||
![]() |
0ae61410d2 | ||
![]() |
2455589f61 | ||
![]() |
c6afae0da5 | ||
![]() |
3155f02be6 | ||
![]() |
4fa0e860ad | ||
![]() |
8c122aa372 | ||
![]() |
5a0bf9fee9 | ||
![]() |
dc794918ed | ||
![]() |
02b15dbc4a | ||
![]() |
ed316b1ce3 | ||
![]() |
d7858f16c1 | ||
![]() |
291deb12ad | ||
![]() |
3e110681c9 | ||
![]() |
65fbfa2097 | ||
![]() |
16ebf9da4c | ||
![]() |
2c76381fcd | ||
![]() |
90683223dd | ||
![]() |
de79171815 | ||
![]() |
1554c5700e | ||
![]() |
5cf257b251 | ||
![]() |
04883e14f6 | ||
![]() |
0a649c184f | ||
![]() |
0e66c899ce | ||
![]() |
e7d236f939 | ||
![]() |
fae4d03473 | ||
![]() |
fd8b9fb028 | ||
![]() |
97bd3e7320 | ||
![]() |
dfca2f88d3 | ||
![]() |
8cad93de37 | ||
![]() |
bdf1813b3a | ||
![]() |
45b6c93f5f | ||
![]() |
e5b8dd7f2d | ||
![]() |
a1c8b8092b | ||
![]() |
109ca2406d | ||
![]() |
3a689112fd | ||
![]() |
40e0cd0f03 | ||
![]() |
bf4d3df906 | ||
![]() |
0e30c49e3f | ||
![]() |
e61a01f7bb | ||
![]() |
f8640cf2cd | ||
![]() |
4bcfeb6e33 | ||
![]() |
a5d4ca0f6d | ||
![]() |
85faecb2fd | ||
![]() |
991fc54994 | ||
![]() |
2de891dc32 | ||
![]() |
9865cb7f55 | ||
![]() |
f97252b93a | ||
![]() |
6124531479 | ||
![]() |
b8549d323c | ||
![]() |
01adece673 | ||
![]() |
0220934e4c | ||
![]() |
ca09693efa | ||
![]() |
e96d7483b3 | ||
![]() |
f2c4f018de | ||
![]() |
237c7dd169 | ||
![]() |
b781b8d77d | ||
![]() |
60b7d1c8a1 | ||
![]() |
8161222b33 | ||
![]() |
1000c4466f | ||
![]() |
60717b074e | ||
![]() |
c3fba97b4c | ||
![]() |
d93f35701f | ||
![]() |
702b60ce66 | ||
![]() |
f8ce597918 | ||
![]() |
22e0a944c8 | ||
![]() |
3a134ef009 | ||
![]() |
96e8cb66b6 | ||
![]() |
615288c151 | ||
![]() |
6153bcc6ad | ||
![]() |
e87edcc77a | ||
![]() |
a21c3e8e2d | ||
![]() |
d7576f67e8 | ||
![]() |
138de643a2 | ||
![]() |
f30e54d177 | ||
![]() |
41b5cb06d3 | ||
![]() |
4ac72d7d08 | ||
![]() |
06ac4980ba | ||
![]() |
ccbfa20bb9 | ||
![]() |
58cd754e07 | ||
![]() |
d1263e583b | ||
![]() |
67c911c37f | ||
![]() |
288e3c3e3e | ||
![]() |
a84378c6c2 | ||
![]() |
b6073408f4 | ||
![]() |
b2d91ac5de | ||
![]() |
8bf34e09f4 | ||
![]() |
be914f2c15 | ||
![]() |
8bb670521d | ||
![]() |
225b3c1494 | ||
![]() |
4bf94e0757 | ||
![]() |
3b21d1d81e | ||
![]() |
5ec1588110 | ||
![]() |
71387be72e | ||
![]() |
98171c9f49 | ||
![]() |
a6c999dea0 | ||
![]() |
bf15b1d302 | ||
![]() |
f422fabab4 | ||
![]() |
01b130ec59 | ||
![]() |
48a1797e72 | ||
![]() |
b34d24735a | ||
![]() |
fe38b36c26 | ||
![]() |
03fca8d91e | ||
![]() |
9f9980e338 | ||
![]() |
19900b004b | ||
![]() |
a8ff0a8913 | ||
![]() |
45861456f1 | ||
![]() |
44b335e7e3 | ||
![]() |
de23bbace2 | ||
![]() |
edff9ae322 | ||
![]() |
3c2766448d | ||
![]() |
786c8b6cfe | ||
![]() |
e8de6a3a67 | ||
![]() |
3c320c4c83 | ||
![]() |
3b83f967e4 | ||
![]() |
5e96b8ef7c | ||
![]() |
5df0e82c37 | ||
![]() |
fd57b21aff | ||
![]() |
6087183a0c | ||
![]() |
01b7c4200e | ||
![]() |
7171286c3c | ||
![]() |
2d58239b74 | ||
![]() |
6b52f62531 | ||
![]() |
1001d9c04e | ||
![]() |
d220d41182 | ||
![]() |
c3a8972550 | ||
![]() |
263b603188 | ||
![]() |
584b722e7e | ||
![]() |
05edfd0e82 | ||
![]() |
16249c02a5 | ||
![]() |
e8ff36d1f3 | ||
![]() |
ed443c6153 | ||
![]() |
f4a84765cd | ||
![]() |
106de3530d | ||
![]() |
119c3f6f46 | ||
![]() |
d2c1c7507c | ||
![]() |
efdb3d1f40 | ||
![]() |
8095db6715 | ||
![]() |
ce2e161b08 | ||
![]() |
9dbc32b85f | ||
![]() |
66226abb48 | ||
![]() |
34bef2f2ca | ||
![]() |
6ef93452f5 | ||
![]() |
fdd4ca6837 | ||
![]() |
9655362f23 | ||
![]() |
130c9fad22 | ||
![]() |
3de0b601bf | ||
![]() |
91560ae4e9 | ||
![]() |
fd6135aebb | ||
![]() |
68ea59f3ae | ||
![]() |
e5fe5d1249 | ||
![]() |
9a69769a7e | ||
![]() |
63b42f3608 | ||
![]() |
d56107e97f | ||
![]() |
33f296e05b | ||
![]() |
3572c62315 | ||
![]() |
97e067a277 | ||
![]() |
1444cddda9 | ||
![]() |
5f56cf3128 | ||
![]() |
5c4e83ebdc | ||
![]() |
91f1c25fcc | ||
![]() |
47a7a239ae | ||
![]() |
fb9984e21f | ||
![]() |
b2db524366 | ||
![]() |
ab8674a5c7 | ||
![]() |
d1c85fc3fa | ||
![]() |
55ad45e3ee | ||
![]() |
f6e5a8cb2a | ||
![]() |
7a91ca9809 | ||
![]() |
71dd04b09e | ||
![]() |
7deabbb512 | ||
![]() |
625a575e49 | ||
![]() |
df4d0da221 | ||
![]() |
6b23b7cad7 | ||
![]() |
cea7deab91 | ||
![]() |
c61abf6aca | ||
![]() |
917bbc669c | ||
![]() |
0ac4c055de | ||
![]() |
78b55d86e9 | ||
![]() |
aaf50fc2e6 | ||
![]() |
6a8f4e92df | ||
![]() |
6d267fda01 | ||
![]() |
88943103a2 | ||
![]() |
e557dc7208 | ||
![]() |
3558806b0e | ||
![]() |
f1e8cc2cf0 | ||
![]() |
6236db1a27 | ||
![]() |
f4b0917239 | ||
![]() |
a5e3cd1a42 | ||
![]() |
b3cca5dcb6 | ||
![]() |
49465223a4 | ||
![]() |
15f0e54cbf | ||
![]() |
ed8f343aad | ||
![]() |
cbd8d70431 | ||
![]() |
9ff187c3f8 | ||
![]() |
be473b97c4 | ||
![]() |
9a5f865eea | ||
![]() |
790280ace9 | ||
![]() |
8ba207fc7f | ||
![]() |
d66b2a1778 | ||
![]() |
e3f2562047 | ||
![]() |
f77118a90c | ||
![]() |
041eb8f6cc | ||
![]() |
733a84df75 | ||
![]() |
acd0b50b40 | ||
![]() |
635851807a | ||
![]() |
89fd367297 | ||
![]() |
60e46d485e | ||
![]() |
5bf0c92318 | ||
![]() |
39d493c278 | ||
![]() |
d2ce62aa13 | ||
![]() |
c8eb30ef27 | ||
![]() |
c317422ed7 | ||
![]() |
614eb81ad7 | ||
![]() |
219c5953f1 | ||
![]() |
5d712c73ea | ||
![]() |
7097b7677e | ||
![]() |
7a4cf13e0c | ||
![]() |
4788a6182e | ||
![]() |
e3dad7c632 | ||
![]() |
1b4156646e | ||
![]() |
31ad75d01b | ||
![]() |
aa2eb29274 | ||
![]() |
22eb4f9cb9 | ||
![]() |
3acc8e7479 | ||
![]() |
2650441013 | ||
![]() |
71697df2b6 | ||
![]() |
acd55b9601 | ||
![]() |
0907de8662 | ||
![]() |
15eb9605a8 | ||
![]() |
6d5cb866db | ||
![]() |
768490089e | ||
![]() |
4d66fab360 | ||
![]() |
bd6bc283b6 | ||
![]() |
3120a0ba83 | ||
![]() |
b2199d5464 | ||
![]() |
84bac8356a | ||
![]() |
2819166539 | ||
![]() |
8fa18ca7c7 | ||
![]() |
63290a265c | ||
![]() |
b854e17995 | ||
![]() |
5dec9d88f6 | ||
![]() |
3d0a85ee78 | ||
![]() |
ac3cdf487f | ||
![]() |
a47e92f2bc | ||
![]() |
fc15ddfa91 | ||
![]() |
d546ef941f | ||
![]() |
b4bbe3d7b5 | ||
![]() |
5561d4eaeb | ||
![]() |
0f6dab394a | ||
![]() |
adc8c1aa38 | ||
![]() |
3a82f500d4 | ||
![]() |
7d1d4831a8 | ||
![]() |
43539f2dbf | ||
![]() |
df6830110d | ||
![]() |
1a98e882dc | ||
![]() |
c943d84036 | ||
![]() |
d07a6704d5 | ||
![]() |
8cfcd5904c | ||
![]() |
fb8846bb45 | ||
![]() |
0d0733dd94 | ||
![]() |
1a524a5a50 | ||
![]() |
1a2288cccf | ||
![]() |
8df27d4c3f | ||
![]() |
0688deca6b | ||
![]() |
01a4443b6c | ||
![]() |
e008b054cb | ||
![]() |
a67d58948d | ||
![]() |
84c051d097 | ||
![]() |
b918abfd54 | ||
![]() |
7f41b7cd93 | ||
![]() |
4759b4fe2e | ||
![]() |
e2c8e69d12 | ||
![]() |
1cf213dad8 | ||
![]() |
aeb81e547b | ||
![]() |
479f7703a2 | ||
![]() |
c7dc396b6d | ||
![]() |
917e8e155c | ||
![]() |
80e3030811 | ||
![]() |
a97e3d827d | ||
![]() |
029014d9d6 | ||
![]() |
e4c2922536 | ||
![]() |
7133ae6aaa | ||
![]() |
2f7f0ff3a1 | ||
![]() |
df853bf61e | ||
![]() |
f0ac753f9b | ||
![]() |
4d56a975e6 | ||
![]() |
d56c53c848 | ||
![]() |
f83b16320d | ||
![]() |
ac10e27f08 | ||
![]() |
34df7a6072 | ||
![]() |
e2cddf1005 | ||
![]() |
ced423748e | ||
![]() |
77fb02729e | ||
![]() |
fef39b9fbe | ||
![]() |
02810105fb | ||
![]() |
baad92515b | ||
![]() |
ab86ddcf02 | ||
![]() |
bf8eddb13b | ||
![]() |
e5eaf7a3fe | ||
![]() |
50f32a3aa5 | ||
![]() |
eb878710c1 | ||
![]() |
311980e0e4 | ||
![]() |
df73170e5a | ||
![]() |
a12c6b5f35 | ||
![]() |
522646c64d | ||
![]() |
4791093e48 | ||
![]() |
599a455150 | ||
![]() |
2deef16ebe | ||
![]() |
989b7be99b | ||
![]() |
cd473e1395 | ||
![]() |
e246ebfb2e | ||
![]() |
8546ae56da | ||
![]() |
9e227b0192 | ||
![]() |
ba7737e9f8 | ||
![]() |
98aa3d51ed | ||
![]() |
e7cfb5492e | ||
![]() |
9ed136dc3a | ||
![]() |
9217216723 | ||
![]() |
936c408a58 | ||
![]() |
54427eac9a | ||
![]() |
e809488cc0 | ||
![]() |
ed26c57d99 | ||
![]() |
2a49811f6e | ||
![]() |
c95acd2568 | ||
![]() |
6a4e0cf667 | ||
![]() |
04f4dd8a22 | ||
![]() |
f33d829ce9 | ||
![]() |
578671ea94 | ||
![]() |
d10300c330 | ||
![]() |
e0555e140f | ||
![]() |
093989406f | ||
![]() |
cdb16f08f6 | ||
![]() |
53139c293b | ||
![]() |
6a8bdcc315 | ||
![]() |
fe535939a3 | ||
![]() |
09e6c11d73 | ||
![]() |
8112bdfaa8 | ||
![]() |
f7db9aaa9f | ||
![]() |
435f972357 | ||
![]() |
f82b46c16b | ||
![]() |
bca96f91b2 | ||
![]() |
6f83a49c63 | ||
![]() |
72cce391ab | ||
![]() |
ccc13cc9e1 | ||
![]() |
020b2c05c8 | ||
![]() |
4c37c17df1 | ||
![]() |
8f67acadd8 | ||
![]() |
ca13c4c1a6 | ||
![]() |
d0c646c721 | ||
![]() |
8a055675af | ||
![]() |
28d2949ebe | ||
![]() |
c4a0015997 | ||
![]() |
f564be6aea | ||
![]() |
988f15e6af | ||
![]() |
37b6d442bd | ||
![]() |
fb2467f6f0 | ||
![]() |
29045b0435 | ||
![]() |
311a48c64e | ||
![]() |
01b3815f27 | ||
![]() |
b0d1c801bd | ||
![]() |
5aaac06f5b | ||
![]() |
34adbf0588 | ||
![]() |
0a4213182e | ||
![]() |
b0c0258e70 | ||
![]() |
8110e591d0 | ||
![]() |
fe05d7aec1 | ||
![]() |
57f5884070 | ||
![]() |
f329c74a15 | ||
![]() |
7c86f3fa9e | ||
![]() |
203b8b01bf | ||
![]() |
8a1034a92f | ||
![]() |
aa0c2dedd9 | ||
![]() |
d045908e05 | ||
![]() |
f002a23d2d | ||
![]() |
29d6d0a906 | ||
![]() |
c8b58b5c23 | ||
![]() |
01bfafc5f1 | ||
![]() |
8c9948bb56 | ||
![]() |
2d1abaa68e | ||
![]() |
664a3df0b4 | ||
![]() |
9ff893881c | ||
![]() |
94f6c6861a | ||
![]() |
b1d614e6c4 | ||
![]() |
7fceb070e5 | ||
![]() |
06440d0202 | ||
![]() |
0ecf9f4f2f | ||
![]() |
5c7c0834c0 | ||
![]() |
f3a25de11d | ||
![]() |
041bef8bcd | ||
![]() |
8998c5f6dd | ||
![]() |
6e83790308 | ||
![]() |
d2d4eb4eae | ||
![]() |
5942a3898c | ||
![]() |
93421f0fa7 | ||
![]() |
3a9ab50dd2 | ||
![]() |
5abd91d6d5 | ||
![]() |
c3da42516b | ||
![]() |
6cb5cd48c2 | ||
![]() |
ec1fae6883 | ||
![]() |
746fd1122f | ||
![]() |
9663760ec5 | ||
![]() |
a3d73d1e23 | ||
![]() |
d63e14a4b6 | ||
![]() |
03944e6cd8 | ||
![]() |
0d1028be2e | ||
![]() |
6a85259e4d | ||
![]() |
ebca936b7e | ||
![]() |
31c4551890 | ||
![]() |
dd470d4197 | ||
![]() |
612822490b | ||
![]() |
f8969605e8 | ||
![]() |
dd24ffa24e | ||
![]() |
d0dda48932 | ||
![]() |
6349b5f654 | ||
![]() |
a6ff02a3cf | ||
![]() |
4f57bf786b | ||
![]() |
6221f6d47d | ||
![]() |
a922efeafa | ||
![]() |
5aa42e5e66 | ||
![]() |
708672ec7e | ||
![]() |
d2cefbf224 | ||
![]() |
adb7aa6950 | ||
![]() |
77f322166e | ||
![]() |
f3f6e54818 | ||
![]() |
fb0fec1f25 | ||
![]() |
b66af9fb4d | ||
![]() |
6617d576a7 | ||
![]() |
cd35ead890 | ||
![]() |
9dc804ee27 | ||
![]() |
a8ceeaa7b0 | ||
![]() |
7092f7663e | ||
![]() |
d9d2edeb08 | ||
![]() |
dda1ddcb26 | ||
![]() |
f0c890f160 | ||
![]() |
4f52d43347 | ||
![]() |
0ed7db979b | ||
![]() |
9c78049359 | ||
![]() |
7882105661 | ||
![]() |
c000e1d6dd | ||
![]() |
420dacb22d | ||
![]() |
ae2f6ad4d1 | ||
![]() |
2c28d79bf8 | ||
![]() |
c5069edc78 | ||
![]() |
282d9e138c | ||
![]() |
72fcf2cbe1 | ||
![]() |
6f49f5465b | ||
![]() |
17b8bd8316 | ||
![]() |
9b6b9c1fa2 | ||
![]() |
609a2ca592 | ||
![]() |
6dabf24bf3 | ||
![]() |
7e88938932 | ||
![]() |
c707e64685 | ||
![]() |
a639690716 | ||
![]() |
01222dbab7 | ||
![]() |
93e2506279 | ||
![]() |
f62d5d3b9d | ||
![]() |
0665acd190 | ||
![]() |
fea05e9d33 | ||
![]() |
7a03c7d56f | ||
![]() |
2dc2aec954 | ||
![]() |
39c6c2417a | ||
![]() |
ff72d6a146 | ||
![]() |
603d0d0c7c | ||
![]() |
28883f711b | ||
![]() |
e914828add | ||
![]() |
c1480029fb | ||
![]() |
40f622949e | ||
![]() |
63096ac2bc | ||
![]() |
03d5a0ec1d | ||
![]() |
1c873e0034 | ||
![]() |
bcb47c306c | ||
![]() |
01c4d3c225 | ||
![]() |
c2aaae4818 | ||
![]() |
3f678e218d | ||
![]() |
c2a59cb476 | ||
![]() |
f8a1bd4e79 | ||
![]() |
d6e039a1d1 | ||
![]() |
0f1a7c2b69 | ||
![]() |
40ad9f4911 | ||
![]() |
4116caff6a | ||
![]() |
0b69f72315 | ||
![]() |
c569f5ddcf | ||
![]() |
62f9e181e0 | ||
![]() |
235a97ea10 | ||
![]() |
e541ae400c | ||
![]() |
4822abde86 | ||
![]() |
b7e52812f8 | ||
![]() |
69118120d9 | ||
![]() |
7cba0c6fb0 | ||
![]() |
5fac67ce15 | ||
![]() |
98c733108e | ||
![]() |
782186e13d | ||
![]() |
4e1f6518e8 | ||
![]() |
53e0fe8e51 | ||
![]() |
0e547390da | ||
![]() |
86b52df839 | ||
![]() |
d685fdf54a | ||
![]() |
d9caab4108 | ||
![]() |
44b68f140e | ||
![]() |
3a3d97dfa7 | ||
![]() |
47898b527c | ||
![]() |
a35f36ad39 | ||
![]() |
d13a397f8e | ||
![]() |
df999723f8 | ||
![]() |
8236e840a7 | ||
![]() |
e5b3625f73 | ||
![]() |
2e4645310b | ||
![]() |
50a32b387e | ||
![]() |
2059283707 | ||
![]() |
8e3af515c9 | ||
![]() |
6f88f0ea3f | ||
![]() |
d2f37cf3f9 | ||
![]() |
7c30d6254e | ||
![]() |
64fb39a653 | ||
![]() |
91895aa70c | ||
![]() |
68dfaf238b | ||
![]() |
ebf13a0ba0 | ||
![]() |
2bff9937b7 | ||
![]() |
256395c28d | ||
![]() |
3346bc8bba | ||
![]() |
6fe22a7e62 | ||
![]() |
757b98748b | ||
![]() |
7a778f3f33 | ||
![]() |
993044c870 | ||
![]() |
a8c1b63edb | ||
![]() |
db7d946e1b | ||
![]() |
41d9059a2f | ||
![]() |
e26e0d7c01 | ||
![]() |
ad41c07a1f | ||
![]() |
9576d246ee | ||
![]() |
988d3ea8ba | ||
![]() |
0767b92b62 | ||
![]() |
5732f3b044 | ||
![]() |
712115b6ce | ||
![]() |
9283559c6b | ||
![]() |
6b393438e9 | ||
![]() |
2064abe16d | ||
![]() |
b605982f94 | ||
![]() |
343b9ab455 | ||
![]() |
dcb226b202 | ||
![]() |
2243021b58 | ||
![]() |
d5134e88b1 | ||
![]() |
c59adf612f | ||
![]() |
93b628d9a8 | ||
![]() |
6bac551d9f | ||
![]() |
70a35656e4 | ||
![]() |
047c18eac0 | ||
![]() |
b4a86ce6cf | ||
![]() |
a82d8ea0c3 | ||
![]() |
b778eed419 | ||
![]() |
ad57faa9a9 | ||
![]() |
a9b5e8d036 | ||
![]() |
8be704e591 | ||
![]() |
b622a8fa58 | ||
![]() |
a519e5c475 | ||
![]() |
d620b6dd5e | ||
![]() |
99335d986e | ||
![]() |
7895cd92cd | ||
![]() |
8b2c032da6 | ||
![]() |
da336247eb | ||
![]() |
dabd27d4be | ||
![]() |
fdda47db6e | ||
![]() |
efa6fd03e5 | ||
![]() |
9e3e34acf5 | ||
![]() |
a2d0c1bf18 | ||
![]() |
7663716ae8 | ||
![]() |
c2cacb3478 | ||
![]() |
84666b54b9 | ||
![]() |
2b91c23bf3 | ||
![]() |
3297267a16 | ||
![]() |
a9e653724c | ||
![]() |
5e79a1f500 | ||
![]() |
d4ff98680a | ||
![]() |
ba8d255cb4 | ||
![]() |
06f4ad922c | ||
![]() |
bff06e448b | ||
![]() |
d48ffa2913 | ||
![]() |
d97c3a7e01 | ||
![]() |
0b1161f7ef | ||
![]() |
061e1a471d | ||
![]() |
a39d874600 | ||
![]() |
de96376565 | ||
![]() |
c54c20ab3c | ||
![]() |
70fafa473b | ||
![]() |
2e436eae6b | ||
![]() |
fd7e861ff5 | ||
![]() |
792108686c | ||
![]() |
fa1b5117fd | ||
![]() |
b0bd9e0a34 | ||
![]() |
05dc97099a | ||
![]() |
9de61fcf58 | ||
![]() |
fc7348d46d | ||
![]() |
8be2456c7e | ||
![]() |
bb5f7249a6 | ||
![]() |
7f7175b184 | ||
![]() |
cf5c640ae4 | ||
![]() |
6b9371d105 | ||
![]() |
9a82057303 | ||
![]() |
48584e94c4 | ||
![]() |
fc94a5d0ee | ||
![]() |
d8024a5928 | ||
![]() |
2034ab4f6c | ||
![]() |
24029cc918 | ||
![]() |
9a9d5964ee | ||
![]() |
4e4a512107 | ||
![]() |
0729ed538e | ||
![]() |
24b75b7ed6 | ||
![]() |
58b70b42dd | ||
![]() |
1496bc1b07 | ||
![]() |
bfbf88b2ea | ||
![]() |
e621b938e3 | ||
![]() |
ec3618ecb8 | ||
![]() |
792a24f38d | ||
![]() |
652e8a015b | ||
![]() |
59e6e798dd | ||
![]() |
e5c2dbc7ec | ||
![]() |
756f71c382 | ||
![]() |
b7535693fa | ||
![]() |
06a3505698 | ||
![]() |
0372d17a11 | ||
![]() |
4525588116 | ||
![]() |
68e957c147 | ||
![]() |
99f5ed1461 | ||
![]() |
59f67796dc | ||
![]() |
aafdfa933e | ||
![]() |
3208c8ed1e | ||
![]() |
6bf733e24e | ||
![]() |
65d3e8fbfc | ||
![]() |
a29d65d47c | ||
![]() |
efa8f0730d | ||
![]() |
0af1edefff | ||
![]() |
023d26f521 | ||
![]() |
5068619f1b | ||
![]() |
5b2457af0b | ||
![]() |
900b4f1af9 | ||
![]() |
4c22a98b0b | ||
![]() |
3b8ca80900 | ||
![]() |
1ef6fd8fb0 | ||
![]() |
942b0de7fd | ||
![]() |
859cca49d1 | ||
![]() |
dc6eff83ea | ||
![]() |
38ff66debd | ||
![]() |
1d2e0f74ea | ||
![]() |
8f7ff25624 | ||
![]() |
97aca8e54c | ||
![]() |
95acf19067 | ||
![]() |
3d0899aa58 | ||
![]() |
bf60e40d0b | ||
![]() |
c9094ca537 | ||
![]() |
68b3fd6b8f | ||
![]() |
9323b3a248 | ||
![]() |
b55e9329d9 | ||
![]() |
a5b4105971 | ||
![]() |
d1feaa935d | ||
![]() |
6919930aaa | ||
![]() |
69633826bb | ||
![]() |
771162bfb1 | ||
![]() |
ba785e29e9 | ||
![]() |
138d6e505b | ||
![]() |
2748e6ba29 | ||
![]() |
dbd4e927d8 | ||
![]() |
e73d47918f | ||
![]() |
b881bc071e | ||
![]() |
1d0395d1c7 | ||
![]() |
616c787e37 | ||
![]() |
0c4de2bc97 | ||
![]() |
2c7b104f4a | ||
![]() |
78951c197a | ||
![]() |
07c1cf7137 | ||
![]() |
d26141151a | ||
![]() |
f59dbe4a88 | ||
![]() |
8dae7f8225 | ||
![]() |
5811389891 | ||
![]() |
debcaf6fb7 | ||
![]() |
b8d10a62c2 | ||
![]() |
d2b209234f | ||
![]() |
34c9d8be50 | ||
![]() |
ae57ad0c81 | ||
![]() |
0c1520dd9c | ||
![]() |
d594f43ebd | ||
![]() |
125c693e3f | ||
![]() |
ad2f857e15 | ||
![]() |
e445d6aada | ||
![]() |
88fbb0ffbb | ||
![]() |
231908fe9f | ||
![]() |
f137cc10f4 | ||
![]() |
c2f5ac9eba | ||
![]() |
5764c988af | ||
![]() |
ccc2fbfd67 | ||
![]() |
1a8f8adc2a | ||
![]() |
7a242bb4ed | ||
![]() |
10b4adb8e6 | ||
![]() |
3b8bb09ae3 | ||
![]() |
83b7181bcb | ||
![]() |
8886b7e141 | ||
![]() |
7dcc4d030b | ||
![]() |
b9398897c1 | ||
![]() |
140db85d21 | ||
![]() |
ccce4b19e8 | ||
![]() |
8cb9be7560 | ||
![]() |
953f0569fb | ||
![]() |
34c229fd33 | ||
![]() |
958ad0d750 | ||
![]() |
36ddd9dd69 | ||
![]() |
38259c96c9 | ||
![]() |
c054fb8a2c | ||
![]() |
5a0b8328d8 | ||
![]() |
ffa19426d7 | ||
![]() |
c123804294 | ||
![]() |
4e24551b90 | ||
![]() |
657b1c60ae | ||
![]() |
dc54b17778 | ||
![]() |
1fb214165b | ||
![]() |
51cb5da7f0 | ||
![]() |
81b2fd78f5 | ||
![]() |
69002fb1e6 | ||
![]() |
75332a752d | ||
![]() |
b528f48417 | ||
![]() |
ec7a79049a | ||
![]() |
6ddad6b299 | ||
![]() |
16dc7762f9 | ||
![]() |
41f84447cc | ||
![]() |
ce073a704b | ||
![]() |
113232ebb6 | ||
![]() |
dc0ed8857f | ||
![]() |
bb6b77bd98 | ||
![]() |
dcc80f9032 | ||
![]() |
dd554bcdf4 | ||
![]() |
f376a39e55 | ||
![]() |
8dcc9d6b66 | ||
![]() |
a13a1225b7 | ||
![]() |
0ec84be5da | ||
![]() |
b1cefb7e3e | ||
![]() |
cc0c1c08b9 | ||
![]() |
3a67884451 | ||
![]() |
72e716cdf1 | ||
![]() |
40e06c9819 | ||
![]() |
ad6c5ff11d | ||
![]() |
335512e232 | ||
![]() |
2622e59b0b | ||
![]() |
35e6a13cd1 | ||
![]() |
a576c9f21f | ||
![]() |
b48490badc | ||
![]() |
71a438e2cb | ||
![]() |
272d6f2a8b | ||
![]() |
5c22065135 | ||
![]() |
e7dd6c52ac | ||
![]() |
3bf042dce9 | ||
![]() |
09ed1aed93 | ||
![]() |
53d3718028 | ||
![]() |
2b5dce5232 | ||
![]() |
9ad84150aa | ||
![]() |
64f798d4b2 | ||
![]() |
f43e04e15a | ||
![]() |
88d72f8c9a | ||
![]() |
9826726a72 | ||
![]() |
c66d0550e8 | ||
![]() |
4aeacfd16e | ||
![]() |
58fa63ad88 | ||
![]() |
94f944dc9c | ||
![]() |
116ddbdd01 | ||
![]() |
1c0697b5d4 | ||
![]() |
434ca47ea0 | ||
![]() |
397ef72b16 | ||
![]() |
7ca9245735 | ||
![]() |
69856286e8 | ||
![]() |
ad43d6a5bc | ||
![]() |
1e5004f495 | ||
![]() |
253161d3d0 | ||
![]() |
ab47e201c7 | ||
![]() |
42984fa72a | ||
![]() |
e7864a28a1 | ||
![]() |
21803607e7 | ||
![]() |
c0523590b4 | ||
![]() |
c7f091ab10 | ||
![]() |
7479e0aada | ||
![]() |
62b366a5ec | ||
![]() |
2b39988707 | ||
![]() |
f9e7291050 | ||
![]() |
4de642ff28 | ||
![]() |
0384efcfc2 | ||
![]() |
bf91443f38 | ||
![]() |
4a5970b4af | ||
![]() |
e3fd68c849 | ||
![]() |
df0de2fc2d | ||
![]() |
0c3568fad5 | ||
![]() |
976f5d91ed | ||
![]() |
0f3d4d9a47 | ||
![]() |
ad1f4429c9 | ||
![]() |
7590d5eacb | ||
![]() |
c5974b8833 | ||
![]() |
511c8de6f3 | ||
![]() |
a718ac7ee0 | ||
![]() |
ef832becf1 | ||
![]() |
3a62455948 | ||
![]() |
297824e2d7 | ||
![]() |
d92f297bc0 | ||
![]() |
7a0827e3d0 | ||
![]() |
ef256a64b8 | ||
![]() |
1de941e837 | ||
![]() |
28b65cb810 | ||
![]() |
6ff3942e8b | ||
![]() |
ef5d959788 | ||
![]() |
5bbee1a1fe | ||
![]() |
6a2c58fcc0 | ||
![]() |
bdb9546ca3 | ||
![]() |
46af4cad6e | ||
![]() |
76a238912b | ||
![]() |
4e6bdb31ac | ||
![]() |
80d03a631e | ||
![]() |
6b27f2d2cf | ||
![]() |
7cb6729fa7 | ||
![]() |
2f46267994 | ||
![]() |
cdda648360 | ||
![]() |
f2d677d51a | ||
![]() |
c2ee0f0864 | ||
![]() |
2a84db7f85 | ||
![]() |
8187a4bce9 | ||
![]() |
97681d142e | ||
![]() |
b2430097f2 | ||
![]() |
7da12a878f | ||
![]() |
a31700e16f | ||
![]() |
7854522792 | ||
![]() |
a6a9ebfde2 | ||
![]() |
c6cbe2748e | ||
![]() |
f9a7f00843 | ||
![]() |
f0b183a552 | ||
![]() |
338ada5c9f | ||
![]() |
ef88f9923f | ||
![]() |
6f8c7d9ec4 | ||
![]() |
ec769ccf72 | ||
![]() |
045952939e | ||
![]() |
1c51cac5ba | ||
![]() |
ea11462e1e | ||
![]() |
62f9736b1d | ||
![]() |
1f8a1f0046 | ||
![]() |
172507acb5 | ||
![]() |
434ab65c16 | ||
![]() |
909a526967 | ||
![]() |
cd6f4fb93f | ||
![]() |
c19458696e | ||
![]() |
cb5f793ede | ||
![]() |
318b930e9f | ||
![]() |
9296a078a7 | ||
![]() |
5dc776e55f | ||
![]() |
72d60f30f7 | ||
![]() |
869743a742 | ||
![]() |
7b03e07908 | ||
![]() |
348f880e15 | ||
![]() |
737188ae50 | ||
![]() |
db21731b14 | ||
![]() |
cdb4fa2487 | ||
![]() |
514204f0d4 | ||
![]() |
ead597d0fb | ||
![]() |
afbf989715 | ||
![]() |
01b62a16c3 | ||
![]() |
c5eba04517 | ||
![]() |
282313ab52 | ||
![]() |
d274545e77 | ||
![]() |
45ac577c4d | ||
![]() |
09402fdb22 | ||
![]() |
89e7448007 | ||
![]() |
1ea6f957bc | ||
![]() |
f44fca0a4b | ||
![]() |
52d2f62a57 | ||
![]() |
d3fda37615 | ||
![]() |
cbe3092404 | ||
![]() |
6dfe3039d0 | ||
![]() |
d6009453df | ||
![]() |
2a8668ea60 | ||
![]() |
cc0d433621 | ||
![]() |
c81323ef91 | ||
![]() |
1fe89fb364 | ||
![]() |
961c27f1c2 | ||
![]() |
fe4a14e6cc | ||
![]() |
ee58ad1ac0 | ||
![]() |
c0ff899812 | ||
![]() |
d9c938de33 | ||
![]() |
56547b3d50 | ||
![]() |
5026bc7a78 | ||
![]() |
27364ee72c | ||
![]() |
ece71a0228 | ||
![]() |
073828235f | ||
![]() |
41bcc8c0f4 | ||
![]() |
a0ea2aae6e | ||
![]() |
f34b46a621 | ||
![]() |
7217a4f7a4 | ||
![]() |
6383eca54a | ||
![]() |
e55bd1e559 | ||
![]() |
9e8b701dea | ||
![]() |
a4431abea8 | ||
![]() |
5844c1767b | ||
![]() |
9a70bfa471 | ||
![]() |
b406c6403c | ||
![]() |
499625f266 | ||
![]() |
6b773553fc | ||
![]() |
15fe049a99 | ||
![]() |
e4555f6997 | ||
![]() |
470071e0b0 | ||
![]() |
ea1be8e7bf | ||
![]() |
84a830195f | ||
![]() |
e62c3e00c1 | ||
![]() |
07e790f900 | ||
![]() |
640142fc0c | ||
![]() |
5c339d4597 | ||
![]() |
a4931f5d78 | ||
![]() |
5e1e543b06 | ||
![]() |
df929f9445 | ||
![]() |
d8e719d1c4 | ||
![]() |
3067e482fc | ||
![]() |
ed5930e934 | ||
![]() |
ffea3597f4 | ||
![]() |
193d3e0206 | ||
![]() |
c8f4fbb7dd | ||
![]() |
c855bc31b4 | ||
![]() |
b924b179ab | ||
![]() |
3df0fee3de | ||
![]() |
b601560e81 | ||
![]() |
e5775cf812 | ||
![]() |
26dd1f8532 | ||
![]() |
5143a5b5c5 | ||
![]() |
15ce27992e | ||
![]() |
dbc2812022 | ||
![]() |
dce3713f12 | ||
![]() |
f849d45bb6 | ||
![]() |
8ad06fb9ea | ||
![]() |
9124d9d6e6 | ||
![]() |
45ebe51e4f | ||
![]() |
407661d56b | ||
![]() |
998d4229af | ||
![]() |
a02d2e2e11 | ||
![]() |
72fa68849f | ||
![]() |
33f17f75a0 | ||
![]() |
23edb18d7e | ||
![]() |
07ff3a853f | ||
![]() |
2cf36bdb46 | ||
![]() |
50848c2f4d | ||
![]() |
d32633b3c7 | ||
![]() |
b37739eec2 | ||
![]() |
28f87dc804 | ||
![]() |
41879e41e6 | ||
![]() |
fc0a6546a2 | ||
![]() |
ffd4280d6c | ||
![]() |
f859b346a6 | ||
![]() |
cb0677cafe | ||
![]() |
c6956527d1 | ||
![]() |
72c6bfaa50 | ||
![]() |
7927b5f624 | ||
![]() |
b7aad39daf | ||
![]() |
f48de6dd43 | ||
![]() |
79d73d8f8b | ||
![]() |
cc5947467f | ||
![]() |
e152f128c8 | ||
![]() |
99bd808ebe | ||
![]() |
beb5f3dc9d | ||
![]() |
f5c3b3446f | ||
![]() |
db3b955b0f | ||
![]() |
f431c7402f | ||
![]() |
5516f65971 | ||
![]() |
9471df0a1b | ||
![]() |
6d39f64be7 | ||
![]() |
4907e6f6d7 | ||
![]() |
1ccee86705 | ||
![]() |
542fb2175b | ||
![]() |
6ec9cfb044 | ||
![]() |
66e0ff8392 | ||
![]() |
1fb0a7109d | ||
![]() |
b89d0a9a73 | ||
![]() |
4bb779d9a5 | ||
![]() |
386a5b6362 | ||
![]() |
e32a999cd0 | ||
![]() |
192eb49589 | ||
![]() |
5d70ff702b | ||
![]() |
a7b05db2a1 | ||
![]() |
45e346cf1b | ||
![]() |
80e2bfada3 | ||
![]() |
16e7bd0388 | ||
![]() |
b3fb35783e | ||
![]() |
a79c6aa9e0 | ||
![]() |
4bb58b2de9 | ||
![]() |
4e10881331 | ||
![]() |
cec4a81e14 | ||
![]() |
da45923d05 | ||
![]() |
31a61b598b | ||
![]() |
9c0506592b | ||
![]() |
beeb0c7c5a | ||
![]() |
b2f05faee0 | ||
![]() |
bfbc6a4bad | ||
![]() |
8c9e0e552d | ||
![]() |
8aaf9fd83f | ||
![]() |
08057720b8 | ||
![]() |
bfaa648837 | ||
![]() |
d504daef91 | ||
![]() |
8375e1d64d | ||
![]() |
cf5193d3e5 | ||
![]() |
b8d3ef2f49 | ||
![]() |
3bf6320030 | ||
![]() |
708b928c73 | ||
![]() |
649366ff44 | ||
![]() |
e5c9e87fad | ||
![]() |
c490388e80 | ||
![]() |
24ec5a6e9d | ||
![]() |
f3d9d707b6 | ||
![]() |
090e10730c | ||
![]() |
fbc84861c7 | ||
![]() |
e763469af8 | ||
![]() |
6df1d5222d | ||
![]() |
58fb7a02f6 | ||
![]() |
3d51ac8df0 | ||
![]() |
6fe4ff7f85 | ||
![]() |
3c0c514e44 | ||
![]() |
2253d4bc16 | ||
![]() |
e5cc19de43 | ||
![]() |
ed5e2dd332 | ||
![]() |
09b7c6f550 | ||
![]() |
df315a1f51 | ||
![]() |
7ee4bb621c | ||
![]() |
24874f4c3c | ||
![]() |
c128880033 | ||
![]() |
a66e94a0b0 | ||
![]() |
56870ed4a8 | ||
![]() |
3ac720df47 | ||
![]() |
1bc757ad06 | ||
![]() |
f72abc6f3d | ||
![]() |
5ac88de985 | ||
![]() |
5404617d43 | ||
![]() |
12467a18e6 | ||
![]() |
1db7043a4d | ||
![]() |
49932747b3 | ||
![]() |
55db190875 | ||
![]() |
71fe2f7ed3 | ||
![]() |
ffc112c9d0 | ||
![]() |
d3e48e296f | ||
![]() |
14f6ae75ea | ||
![]() |
c84efe64d3 | ||
![]() |
10e89a7dbb | ||
![]() |
ef44acbf10 | ||
![]() |
06da540ab0 | ||
![]() |
0826b367d6 | ||
![]() |
329bf861d6 | ||
![]() |
9dcd3d18a0 | ||
![]() |
40c017fd54 | ||
![]() |
db66cd88b6 | ||
![]() |
f0bcf81a98 | ||
![]() |
86c205fe43 | ||
![]() |
6a0b343289 | ||
![]() |
c6414138c7 | ||
![]() |
36b355eb82 | ||
![]() |
9ca4e8f32a | ||
![]() |
1b88b7a166 | ||
![]() |
caf352ff06 | ||
![]() |
54106179a1 | ||
![]() |
607601b3a4 | ||
![]() |
f58828cb82 | ||
![]() |
11330af05f | ||
![]() |
fbe1bca1b9 | ||
![]() |
24a5325db3 | ||
![]() |
1ec3140759 | ||
![]() |
ca8db7696e | ||
![]() |
c9190574a9 | ||
![]() |
bfeb0b3639 | ||
![]() |
cbc1334b8d | ||
![]() |
08cbb97ec9 | ||
![]() |
5719cc1a24 | ||
![]() |
d9513e5ff2 | ||
![]() |
b5a0e8b2c0 | ||
![]() |
b32b918936 | ||
![]() |
0f47ffd908 | ||
![]() |
cd018ad3a5 | ||
![]() |
24dfecb6f0 | ||
![]() |
ab027a6ae2 | ||
![]() |
556d071e7f | ||
![]() |
939fb313df | ||
![]() |
b5639a6472 | ||
![]() |
f50e40e0b8 | ||
![]() |
6f07421911 | ||
![]() |
adf48246a9 | ||
![]() |
7be9291b13 | ||
![]() |
ea9e75039b | ||
![]() |
a5fb036011 | ||
![]() |
e55506f9db | ||
![]() |
50ec1d0445 | ||
![]() |
3d5e1d8d91 | ||
![]() |
db2128a344 | ||
![]() |
cae283dc86 | ||
![]() |
7afcb0fb04 | ||
![]() |
10f830c3ef | ||
![]() |
7a5c3aa7ed | ||
![]() |
2b50406856 | ||
![]() |
10a2a7e0fc | ||
![]() |
7a564b222d | ||
![]() |
21db43db06 | ||
![]() |
5009b3029f | ||
![]() |
57a029189c | ||
![]() |
671d68bc2c | ||
![]() |
5946c37925 | ||
![]() |
17a37b1de9 | ||
![]() |
e7827a6997 | ||
![]() |
2347e043a9 | ||
![]() |
00965fe19e | ||
![]() |
9681dfb458 | ||
![]() |
5e631bc6ba | ||
![]() |
b5f660398c | ||
![]() |
d50bdf619f | ||
![]() |
4e448b21ff | ||
![]() |
2a78c2970d | ||
![]() |
0cb715bb76 | ||
![]() |
7d03823afd | ||
![]() |
8e1c9f5042 | ||
![]() |
980b7cda8f | ||
![]() |
3a72dd5cb6 | ||
![]() |
3178243811 | ||
![]() |
d30e2f2a4f | ||
![]() |
3637be251e | ||
![]() |
2aea27d272 | ||
![]() |
ceb9b1d1ff | ||
![]() |
ccfa1e23f0 | ||
![]() |
290da8df2d | ||
![]() |
4b1d73791d | ||
![]() |
7e8012c1a0 | ||
![]() |
15cd602e8b | ||
![]() |
598f5b241f | ||
![]() |
335e69e6cd | ||
![]() |
05fe5db030 | ||
![]() |
710096b1c6 | ||
![]() |
07b882c801 | ||
![]() |
3e5331a263 | ||
![]() |
897277992b | ||
![]() |
1424091ee5 | ||
![]() |
61ec16cdfc | ||
![]() |
e5cb5756aa | ||
![]() |
9e1c3e8f01 | ||
![]() |
448e1690aa | ||
![]() |
8267f01ccd | ||
![]() |
6f9439e1bc | ||
![]() |
06994c0dfc | ||
![]() |
dee5d639e2 | ||
![]() |
df6730be55 | ||
![]() |
6226dae05c | ||
![]() |
9c6a475a6e | ||
![]() |
8294d10d5b | ||
![]() |
67558bec47 | ||
![]() |
6c1ef398bb | ||
![]() |
84873d4074 | ||
![]() |
58a0b28a39 | ||
![]() |
0469e19f54 | ||
![]() |
dbcfa7b599 | ||
![]() |
b37d3a66cc | ||
![]() |
7e495a5e27 | ||
![]() |
c41547fd4a | ||
![]() |
0d47d41c85 | ||
![]() |
df68403b6d | ||
![]() |
57bdc2b885 | ||
![]() |
f565ff5def | ||
![]() |
8ece639987 | ||
![]() |
b35f509784 | ||
![]() |
41a3a17456 | ||
![]() |
cbbafbcca2 | ||
![]() |
c75566b374 | ||
![]() |
f1954df573 | ||
![]() |
7279f1fcc1 | ||
![]() |
d7432f7c10 | ||
![]() |
b0a0a153f3 | ||
![]() |
9e4fa5dcf1 | ||
![]() |
024632dbd0 | ||
![]() |
0a545a28b9 | ||
![]() |
0f2df59998 | ||
![]() |
0809673ba9 | ||
![]() |
b386284180 | ||
![]() |
515519bc87 | ||
![]() |
5404163be0 | ||
![]() |
29a7d32f77 | ||
![]() |
687a7e9b2f | ||
![]() |
09e8782318 | ||
![]() |
0b193eee43 | ||
![]() |
f2aea02210 | ||
![]() |
194f922312 | ||
![]() |
fea3c48098 | ||
![]() |
c2f57baec2 | ||
![]() |
f4a140e126 | ||
![]() |
ab506b09fe | ||
![]() |
87e1cdeedb | ||
![]() |
81a36146ef | ||
![]() |
7333123ba4 | ||
![]() |
d99c5ed890 | ||
![]() |
04740fbcbb | ||
![]() |
6a7440f7d3 | ||
![]() |
14299bb2cc | ||
![]() |
66cebfc992 | ||
![]() |
108b8e6705 | ||
![]() |
4eaa6afa4d | ||
![]() |
f643a46bbf | ||
![]() |
aae63a7ff3 | ||
![]() |
582567696e | ||
![]() |
2e0c89409d | ||
![]() |
7fa4a68a27 | ||
![]() |
f1c5e2ef81 | ||
![]() |
b526155cce | ||
![]() |
62c3f301e7 | ||
![]() |
38cb988809 | ||
![]() |
7bb7456a8b | ||
![]() |
0372e12b81 | ||
![]() |
a6873c1520 | ||
![]() |
f11220da3a | ||
![]() |
b976ac54c8 | ||
![]() |
78026e766f | ||
![]() |
b4cd8d21a5 | ||
![]() |
7552893311 | ||
![]() |
21c896d8f8 | ||
![]() |
4b7fe202ec | ||
![]() |
bb9793d5b7 | ||
![]() |
e99af991ec | ||
![]() |
abf3708cc2 | ||
![]() |
4395d6156d | ||
![]() |
9f4519210f | ||
![]() |
b0506afa5b | ||
![]() |
8cbb379898 | ||
![]() |
04ba53c870 | ||
![]() |
b226215593 | ||
![]() |
19970729a9 | ||
![]() |
f310cacd41 | ||
![]() |
5ff7c8418c | ||
![]() |
0bdb48bcac | ||
![]() |
99c775d8cb | ||
![]() |
4d43396835 | ||
![]() |
92321e219a | ||
![]() |
c422b2fb0b | ||
![]() |
8aa72f4c1e | ||
![]() |
d8e33c5a69 | ||
![]() |
15f9677d33 | ||
![]() |
219b225ac0 | ||
![]() |
2ac232e634 | ||
![]() |
710866ff4e | ||
![]() |
662773b075 | ||
![]() |
875b803483 | ||
![]() |
6e5cfac927 | ||
![]() |
97eaf3d4a1 | ||
![]() |
366552a969 | ||
![]() |
57b07441a1 | ||
![]() |
fb57ab0add | ||
![]() |
d6717c0032 | ||
![]() |
f72389147d | ||
![]() |
a509f6ccd2 | ||
![]() |
add484a2ea | ||
![]() |
a17a6d5346 | ||
![]() |
be9439f10d | ||
![]() |
96a50f5c6b | ||
![]() |
3c0414c420 | ||
![]() |
b450d4c734 | ||
![]() |
d536509a63 | ||
![]() |
11f1e28139 | ||
![]() |
379c3e98f5 | ||
![]() |
5ea77894b7 | ||
![]() |
d54b4e7c44 | ||
![]() |
d8b3af3815 | ||
![]() |
2b04152482 | ||
![]() |
331a3ac387 | ||
![]() |
7eee3cdc7f | ||
![]() |
b12c7432e0 | ||
![]() |
696643d037 | ||
![]() |
7e54f97003 | ||
![]() |
77dbf84e55 | ||
![]() |
2350c5054c | ||
![]() |
73accf747f | ||
![]() |
0d3e6b2c4c | ||
![]() |
b3d7cc637b | ||
![]() |
2147bcbc29 | ||
![]() |
980c2d4cae | ||
![]() |
d2ebfd2833 | ||
![]() |
bd782fc828 | ||
![]() |
23560e608c | ||
![]() |
f1377b560e | ||
![]() |
72108684ea | ||
![]() |
c6adaaea97 | ||
![]() |
91999a38ca | ||
![]() |
b34eed125d | ||
![]() |
2abe09529a | ||
![]() |
9aaaf4dd4b | ||
![]() |
cbfbcf7f1b | ||
![]() |
68316cbcf9 | ||
![]() |
2f4b9263c3 | ||
![]() |
c2623a08e3 | ||
![]() |
9f625ee7d1 | ||
![]() |
2f85c27a05 | ||
![]() |
c612a3bf60 | ||
![]() |
a01f5f5cf1 | ||
![]() |
87328686a0 | ||
![]() |
81c11ba1f7 | ||
![]() |
49b17c5a2d | ||
![]() |
de06a781ff | ||
![]() |
8e77e3c685 | ||
![]() |
a687b083ae | ||
![]() |
b9e5c7eb35 | ||
![]() |
1a6a063e04 | ||
![]() |
d85b7a6bd0 | ||
![]() |
1c4700f447 | ||
![]() |
c7651dc40d | ||
![]() |
eda1c471ad | ||
![]() |
c7ef18fbc4 | ||
![]() |
901ec918b1 | ||
![]() |
6bdae55ee1 | ||
![]() |
dfb96e4b7f | ||
![]() |
ff2c316b18 | ||
![]() |
5be52f71f9 | ||
![]() |
42873dd37c | ||
![]() |
f93e7d4e3a | ||
![]() |
bbcd523967 | ||
![]() |
68cbe58d00 | ||
![]() |
115bca98f1 | ||
![]() |
ed0b34b2fe | ||
![]() |
ab34401421 | ||
![]() |
eed0c18d65 | ||
![]() |
83400d0417 | ||
![]() |
77a6461c9d | ||
![]() |
6db9d1122f | ||
![]() |
83bef85415 | ||
![]() |
b5b3914bbf | ||
![]() |
0d90ef94ae | ||
![]() |
c08b21b7cd | ||
![]() |
be3cb9ef00 | ||
![]() |
f7b3f52731 | ||
![]() |
9220d9fc52 | ||
![]() |
68c8547067 | ||
![]() |
1468acfced | ||
![]() |
b141aea4c0 | ||
![]() |
a88c022406 | ||
![]() |
5389382798 | ||
![]() |
4765173778 | ||
![]() |
07a9cb910f | ||
![]() |
f408f074c4 | ||
![]() |
f1f2640d0e | ||
![]() |
27d7d7ca69 | ||
![]() |
c0fc5b48ae | ||
![]() |
8735d3b83e | ||
![]() |
ca59dd1302 | ||
![]() |
eccdef8211 | ||
![]() |
f2ebfe7aef | ||
![]() |
cac5b356db | ||
![]() |
e5a38ce748 | ||
![]() |
7d9d9fcf36 | ||
![]() |
156104d5f5 | ||
![]() |
f0aba6ceb2 | ||
![]() |
c248ba4043 | ||
![]() |
ab07ee57c6 | ||
![]() |
c615dc573a | ||
![]() |
eae3d72a4d | ||
![]() |
7b8d826704 | ||
![]() |
e7baa42e63 | ||
![]() |
2f32833a22 | ||
![]() |
f6935a4b4b | ||
![]() |
332c9e891b | ||
![]() |
1caabb6419 | ||
![]() |
f41f7994a3 | ||
![]() |
e39f314e7a | ||
![]() |
7f34561e53 | ||
![]() |
34606b0f1f | ||
![]() |
c51b509501 | ||
![]() |
b91ee4847f | ||
![]() |
625463d871 | ||
![]() |
6433a01e07 | ||
![]() |
56cc31e8e7 | ||
![]() |
3af297aa76 | ||
![]() |
996ec59d28 | ||
![]() |
95593eeeab | ||
![]() |
dad244fb7a | ||
![]() |
15b5968418 | ||
![]() |
64a45dc6a6 | ||
![]() |
7cfede5b83 | ||
![]() |
e4d17e0b15 | ||
![]() |
e79f7ce290 | ||
![]() |
adb5d27d95 | ||
![]() |
8456a8cecb | ||
![]() |
b9f66373c1 | ||
![]() |
9ac365feef | ||
![]() |
bcc77c73e1 | ||
![]() |
8b11e5aeb1 | ||
![]() |
43bbd58a44 | ||
![]() |
7feffa64f3 | ||
![]() |
ea0977abb4 | ||
![]() |
cb48394e8a | ||
![]() |
4c83dc7c28 | ||
![]() |
e10ab1da78 | ||
![]() |
1b0e60374b | ||
![]() |
3a760fbb44 | ||
![]() |
f5441a87e3 | ||
![]() |
e2a812fa4b | ||
![]() |
5b5ead872b | ||
![]() |
03cfd78c59 | ||
![]() |
ced11bc707 | ||
![]() |
6b9c084162 | ||
![]() |
644ce2a26c | ||
![]() |
5425e45851 | ||
![]() |
12fce7a08d | ||
![]() |
0991ab3543 | ||
![]() |
65d2b37496 | ||
![]() |
94d518a418 | ||
![]() |
7cca673902 | ||
![]() |
384f8d97d8 | ||
![]() |
c82d5d63e3 | ||
![]() |
653a3d5d11 | ||
![]() |
884b7201de | ||
![]() |
85d2f24447 | ||
![]() |
935992bcb3 | ||
![]() |
dc15d1c8ec | ||
![]() |
6beb9e568a | ||
![]() |
7178f10bda | ||
![]() |
1308236429 | ||
![]() |
63d6b610b8 | ||
![]() |
8823024509 | ||
![]() |
4896f870f0 | ||
![]() |
7e482901d9 | ||
![]() |
07b309e65d | ||
![]() |
6bbb5e9b56 | ||
![]() |
867fecd157 | ||
![]() |
05388d2dfc | ||
![]() |
e06b6d7140 | ||
![]() |
859e508392 | ||
![]() |
534ce11d54 |
38
.clang-tidy
38
.clang-tidy
@@ -5,12 +5,12 @@ Checks: >-
|
||||
-altera-*,
|
||||
-android-*,
|
||||
-boost-*,
|
||||
-bugprone-branch-clone,
|
||||
-bugprone-easily-swappable-parameters,
|
||||
-bugprone-implicit-widening-of-multiplication-result,
|
||||
-bugprone-narrowing-conversions,
|
||||
-bugprone-signed-char-misuse,
|
||||
-bugprone-too-small-loop-variable,
|
||||
-cert-dcl50-cpp,
|
||||
-cert-err33-c,
|
||||
-cert-err58-cpp,
|
||||
-cert-oop57-cpp,
|
||||
-cert-str34-c,
|
||||
@@ -18,18 +18,18 @@ Checks: >-
|
||||
-clang-analyzer-osx.*,
|
||||
-clang-diagnostic-delete-abstract-non-virtual-dtor,
|
||||
-clang-diagnostic-delete-non-abstract-non-virtual-dtor,
|
||||
-clang-diagnostic-ignored-optimization-argument,
|
||||
-clang-diagnostic-shadow-field,
|
||||
-clang-diagnostic-sign-compare,
|
||||
-clang-diagnostic-unused-variable,
|
||||
-clang-diagnostic-unused-const-variable,
|
||||
-clang-diagnostic-unused-parameter,
|
||||
-concurrency-*,
|
||||
-cppcoreguidelines-avoid-c-arrays,
|
||||
-cppcoreguidelines-avoid-goto,
|
||||
-cppcoreguidelines-avoid-magic-numbers,
|
||||
-cppcoreguidelines-init-variables,
|
||||
-cppcoreguidelines-macro-usage,
|
||||
-cppcoreguidelines-narrowing-conversions,
|
||||
-cppcoreguidelines-non-private-member-variables-in-classes,
|
||||
-cppcoreguidelines-prefer-member-initializer,
|
||||
-cppcoreguidelines-pro-bounds-array-to-pointer-decay,
|
||||
-cppcoreguidelines-pro-bounds-constant-array-index,
|
||||
-cppcoreguidelines-pro-bounds-pointer-arithmetic,
|
||||
@@ -41,7 +41,7 @@ Checks: >-
|
||||
-cppcoreguidelines-pro-type-union-access,
|
||||
-cppcoreguidelines-pro-type-vararg,
|
||||
-cppcoreguidelines-special-member-functions,
|
||||
-fuchsia-default-arguments,
|
||||
-cppcoreguidelines-virtual-class-destructor,
|
||||
-fuchsia-multiple-inheritance,
|
||||
-fuchsia-overloaded-operator,
|
||||
-fuchsia-statically-constructed-objects,
|
||||
@@ -51,6 +51,7 @@ Checks: >-
|
||||
-google-explicit-constructor,
|
||||
-google-readability-braces-around-statements,
|
||||
-google-readability-casting,
|
||||
-google-readability-namespace-comments,
|
||||
-google-readability-todo,
|
||||
-google-runtime-references,
|
||||
-hicpp-*,
|
||||
@@ -73,8 +74,7 @@ Checks: >-
|
||||
-modernize-use-nodiscard,
|
||||
-mpi-*,
|
||||
-objc-*,
|
||||
-readability-braces-around-statements,
|
||||
-readability-const-return-type,
|
||||
-readability-container-data-pointer,
|
||||
-readability-convert-member-functions-to-static,
|
||||
-readability-else-after-return,
|
||||
-readability-function-cognitive-complexity,
|
||||
@@ -82,10 +82,6 @@ Checks: >-
|
||||
-readability-isolate-declaration,
|
||||
-readability-magic-numbers,
|
||||
-readability-make-member-function-const,
|
||||
-readability-named-parameter,
|
||||
-readability-qualified-auto,
|
||||
-readability-redundant-access-specifiers,
|
||||
-readability-redundant-member-init,
|
||||
-readability-redundant-string-init,
|
||||
-readability-uppercase-literal-suffix,
|
||||
-readability-use-anyofallof,
|
||||
@@ -93,13 +89,13 @@ WarningsAsErrors: '*'
|
||||
AnalyzeTemporaryDtors: false
|
||||
FormatStyle: google
|
||||
CheckOptions:
|
||||
- key: google-readability-braces-around-statements.ShortStatementLines
|
||||
value: '1'
|
||||
- key: google-readability-function-size.StatementThreshold
|
||||
value: '800'
|
||||
- key: google-readability-namespace-comments.ShortNamespaceLines
|
||||
- key: google-runtime-int.TypeSuffix
|
||||
value: '_t'
|
||||
- key: llvm-namespace-comment.ShortNamespaceLines
|
||||
value: '10'
|
||||
- key: google-readability-namespace-comments.SpacesBeforeComments
|
||||
- key: llvm-namespace-comment.SpacesBeforeComments
|
||||
value: '2'
|
||||
- key: modernize-loop-convert.MaxCopySize
|
||||
value: '16'
|
||||
@@ -117,6 +113,8 @@ CheckOptions:
|
||||
value: 'make_unique'
|
||||
- key: modernize-make-unique.MakeSmartPtrFunctionHeader
|
||||
value: 'esphome/core/helpers.h'
|
||||
- key: readability-braces-around-statements.ShortStatementLines
|
||||
value: 2
|
||||
- key: readability-identifier-naming.LocalVariableCase
|
||||
value: 'lower_case'
|
||||
- key: readability-identifier-naming.ClassCase
|
||||
@@ -163,3 +161,11 @@ CheckOptions:
|
||||
value: 'lower_case'
|
||||
- key: readability-identifier-naming.VirtualMethodSuffix
|
||||
value: ''
|
||||
- key: readability-qualified-auto.AddConstToQualified
|
||||
value: 0
|
||||
- key: readability-identifier-length.MinimumVariableNameLength
|
||||
value: 0
|
||||
- key: readability-identifier-length.MinimumParameterNameLength
|
||||
value: 0
|
||||
- key: readability-identifier-length.MinimumLoopCounterNameLength
|
||||
value: 0
|
||||
|
@@ -1,56 +1,60 @@
|
||||
{
|
||||
"name": "ESPHome Dev",
|
||||
"image": "esphome/esphome-lint:dev",
|
||||
"postCreateCommand": [
|
||||
"script/devcontainer-post-create"
|
||||
],
|
||||
"runArgs": [
|
||||
"--privileged",
|
||||
"-e",
|
||||
"ESPHOME_DASHBOARD_USE_PING=1"
|
||||
],
|
||||
"image": "ghcr.io/esphome/esphome-lint:dev",
|
||||
"postCreateCommand": ["script/devcontainer-post-create"],
|
||||
"containerEnv": {
|
||||
"DEVCONTAINER": "1",
|
||||
"PIP_BREAK_SYSTEM_PACKAGES": "1",
|
||||
"PIP_ROOT_USER_ACTION": "ignore"
|
||||
},
|
||||
"runArgs": ["--privileged", "-e", "ESPHOME_DASHBOARD_USE_PING=1"],
|
||||
"appPort": 6052,
|
||||
"extensions": [
|
||||
// python
|
||||
"ms-python.python",
|
||||
"visualstudioexptteam.vscodeintellicode",
|
||||
// yaml
|
||||
"redhat.vscode-yaml",
|
||||
// cpp
|
||||
"ms-vscode.cpptools",
|
||||
// editorconfig
|
||||
"editorconfig.editorconfig",
|
||||
],
|
||||
"settings": {
|
||||
"python.languageServer": "Pylance",
|
||||
"python.pythonPath": "/usr/bin/python3",
|
||||
"python.linting.pylintEnabled": true,
|
||||
"python.linting.enabled": true,
|
||||
"python.formatting.provider": "black",
|
||||
"editor.formatOnPaste": false,
|
||||
"editor.formatOnSave": true,
|
||||
"editor.formatOnType": true,
|
||||
"files.trimTrailingWhitespace": true,
|
||||
"terminal.integrated.defaultProfile.linux": "bash",
|
||||
"yaml.customTags": [
|
||||
"!secret scalar",
|
||||
"!lambda scalar",
|
||||
"!include_dir_named scalar",
|
||||
"!include_dir_list scalar",
|
||||
"!include_dir_merge_list scalar",
|
||||
"!include_dir_merge_named scalar"
|
||||
],
|
||||
"files.exclude": {
|
||||
"**/.git": true,
|
||||
"**/.DS_Store": true,
|
||||
"**/*.pyc": {
|
||||
"when": "$(basename).py"
|
||||
},
|
||||
"**/__pycache__": true
|
||||
},
|
||||
"files.associations": {
|
||||
"**/.vscode/*.json": "jsonc"
|
||||
},
|
||||
"C_Cpp.clang_format_path": "/usr/bin/clang-format-11",
|
||||
"customizations": {
|
||||
"vscode": {
|
||||
"extensions": [
|
||||
// python
|
||||
"ms-python.python",
|
||||
"visualstudioexptteam.vscodeintellicode",
|
||||
// yaml
|
||||
"redhat.vscode-yaml",
|
||||
// cpp
|
||||
"ms-vscode.cpptools",
|
||||
// editorconfig
|
||||
"editorconfig.editorconfig"
|
||||
],
|
||||
"settings": {
|
||||
"python.languageServer": "Pylance",
|
||||
"python.pythonPath": "/usr/bin/python3",
|
||||
"python.linting.pylintEnabled": true,
|
||||
"python.linting.enabled": true,
|
||||
"python.formatting.provider": "black",
|
||||
"editor.formatOnPaste": false,
|
||||
"editor.formatOnSave": true,
|
||||
"editor.formatOnType": true,
|
||||
"files.trimTrailingWhitespace": true,
|
||||
"terminal.integrated.defaultProfile.linux": "bash",
|
||||
"yaml.customTags": [
|
||||
"!secret scalar",
|
||||
"!lambda scalar",
|
||||
"!extend scalar",
|
||||
"!include_dir_named scalar",
|
||||
"!include_dir_list scalar",
|
||||
"!include_dir_merge_list scalar",
|
||||
"!include_dir_merge_named scalar"
|
||||
],
|
||||
"files.exclude": {
|
||||
"**/.git": true,
|
||||
"**/.DS_Store": true,
|
||||
"**/*.pyc": {
|
||||
"when": "$(basename).py"
|
||||
},
|
||||
"**/__pycache__": true
|
||||
},
|
||||
"files.associations": {
|
||||
"**/.vscode/*.json": "jsonc"
|
||||
},
|
||||
"C_Cpp.clang_format_path": "/usr/bin/clang-format-13"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@@ -25,10 +25,9 @@ indent_size = 2
|
||||
[*.{yaml,yml}]
|
||||
indent_style = space
|
||||
indent_size = 2
|
||||
quote_type = single
|
||||
quote_type = double
|
||||
|
||||
# JSON
|
||||
[*.json]
|
||||
indent_style = space
|
||||
indent_size = 2
|
||||
|
||||
|
1
.gitattributes
vendored
1
.gitattributes
vendored
@@ -1,2 +1,3 @@
|
||||
# Normalize line endings to LF in the repository
|
||||
* text eol=lf
|
||||
*.png binary
|
||||
|
1
.github/FUNDING.yml
vendored
1
.github/FUNDING.yml
vendored
@@ -1,3 +1,4 @@
|
||||
---
|
||||
# These are supported funding model platforms
|
||||
|
||||
custom: https://www.nabucasa.com
|
||||
|
9
.github/ISSUE_TEMPLATE/config.yml
vendored
9
.github/ISSUE_TEMPLATE/config.yml
vendored
@@ -1,3 +1,4 @@
|
||||
---
|
||||
blank_issues_enabled: false
|
||||
contact_links:
|
||||
- name: Issue Tracker
|
||||
@@ -5,8 +6,10 @@ contact_links:
|
||||
about: Please create bug reports in the dedicated issue tracker.
|
||||
- name: Feature Request Tracker
|
||||
url: https://github.com/esphome/feature-requests
|
||||
about: Please create feature requests in the dedicated feature request tracker.
|
||||
about: |
|
||||
Please create feature requests in the dedicated feature request tracker.
|
||||
- name: Frequently Asked Question
|
||||
url: https://esphome.io/guides/faq.html
|
||||
about: Please view the FAQ for common questions and what to include in a bug report.
|
||||
|
||||
about: |
|
||||
Please view the FAQ for common questions and what
|
||||
to include in a bug report.
|
||||
|
10
.github/PULL_REQUEST_TEMPLATE.md
vendored
10
.github/PULL_REQUEST_TEMPLATE.md
vendored
@@ -1,6 +1,6 @@
|
||||
# What does this implement/fix?
|
||||
# What does this implement/fix?
|
||||
|
||||
Quick description and explanation of changes
|
||||
<!-- Quick description and explanation of changes -->
|
||||
|
||||
## Types of changes
|
||||
|
||||
@@ -16,7 +16,11 @@ Quick description and explanation of changes
|
||||
## Test Environment
|
||||
|
||||
- [ ] ESP32
|
||||
- [ ] ESP32 IDF
|
||||
- [ ] ESP8266
|
||||
- [ ] RP2040
|
||||
- [ ] BK72xx
|
||||
- [ ] RTL87xx
|
||||
|
||||
## Example entry for `config.yaml`:
|
||||
<!--
|
||||
@@ -34,6 +38,6 @@ Quick description and explanation of changes
|
||||
## Checklist:
|
||||
- [ ] The code change is tested and works locally.
|
||||
- [ ] Tests have been added to verify that the new code works (under `tests/` folder).
|
||||
|
||||
|
||||
If user exposed functionality or configuration variables are added/changed:
|
||||
- [ ] Documentation added/updated in [esphome-docs](https://github.com/esphome/esphome-docs).
|
||||
|
38
.github/actions/restore-python/action.yml
vendored
Normal file
38
.github/actions/restore-python/action.yml
vendored
Normal file
@@ -0,0 +1,38 @@
|
||||
name: Restore Python
|
||||
inputs:
|
||||
python-version:
|
||||
description: Python version to restore
|
||||
required: true
|
||||
type: string
|
||||
cache-key:
|
||||
description: Cache key to use
|
||||
required: true
|
||||
type: string
|
||||
outputs:
|
||||
python-version:
|
||||
description: Python version restored
|
||||
value: ${{ steps.python.outputs.python-version }}
|
||||
runs:
|
||||
using: "composite"
|
||||
steps:
|
||||
- name: Set up Python ${{ inputs.python-version }}
|
||||
id: python
|
||||
uses: actions/setup-python@v4.7.0
|
||||
with:
|
||||
python-version: ${{ inputs.python-version }}
|
||||
- name: Restore Python virtual environment
|
||||
id: cache-venv
|
||||
uses: actions/cache/restore@v3.3.2
|
||||
with:
|
||||
path: venv
|
||||
# yamllint disable-line rule:line-length
|
||||
key: ${{ runner.os }}-${{ steps.python.outputs.python-version }}-venv-${{ inputs.cache-key }}
|
||||
- name: Create Python virtual environment
|
||||
if: steps.cache-venv.outputs.cache-hit != 'true'
|
||||
shell: bash
|
||||
run: |
|
||||
python -m venv venv
|
||||
. venv/bin/activate
|
||||
python --version
|
||||
pip install -r requirements.txt -r requirements_optional.txt -r requirements_test.txt
|
||||
pip install -e .
|
10
.github/dependabot.yml
vendored
10
.github/dependabot.yml
vendored
@@ -1,9 +1,15 @@
|
||||
---
|
||||
version: 2
|
||||
updates:
|
||||
- package-ecosystem: "pip"
|
||||
- package-ecosystem: pip
|
||||
directory: "/"
|
||||
schedule:
|
||||
interval: "daily"
|
||||
interval: daily
|
||||
ignore:
|
||||
# Hypotehsis is only used for testing and is updated quite often
|
||||
- dependency-name: hypothesis
|
||||
- package-ecosystem: github-actions
|
||||
directory: "/"
|
||||
schedule:
|
||||
interval: daily
|
||||
open-pull-requests-limit: 10
|
||||
|
7
.github/issue-close-app.yml
vendored
7
.github/issue-close-app.yml
vendored
@@ -1,7 +0,0 @@
|
||||
comment: >-
|
||||
https://github.com/esphome/esphome/issues/430
|
||||
issueConfigs:
|
||||
- content:
|
||||
- "OTHERWISE THE ISSUE WILL BE CLOSED AUTOMATICALLY"
|
||||
|
||||
caseInsensitive: false
|
36
.github/lock.yml
vendored
36
.github/lock.yml
vendored
@@ -1,36 +0,0 @@
|
||||
# Configuration for Lock Threads - https://github.com/dessant/lock-threads
|
||||
|
||||
# Number of days of inactivity before a closed issue or pull request is locked
|
||||
daysUntilLock: 7
|
||||
|
||||
# Skip issues and pull requests created before a given timestamp. Timestamp must
|
||||
# follow ISO 8601 (`YYYY-MM-DD`). Set to `false` to disable
|
||||
skipCreatedBefore: false
|
||||
|
||||
# Issues and pull requests with these labels will be ignored. Set to `[]` to disable
|
||||
exemptLabels:
|
||||
- keep-open
|
||||
|
||||
# Label to add before locking, such as `outdated`. Set to `false` to disable
|
||||
lockLabel: false
|
||||
|
||||
# Comment to post before locking. Set to `false` to disable
|
||||
lockComment: false
|
||||
|
||||
# Assign `resolved` as the reason for locking. Set to `false` to disable
|
||||
setLockReason: false
|
||||
|
||||
# Limit to only `issues` or `pulls`
|
||||
# only: issues
|
||||
|
||||
# Optionally, specify configuration settings just for `issues` or `pulls`
|
||||
# issues:
|
||||
# exemptLabels:
|
||||
# - help-wanted
|
||||
# lockLabel: outdated
|
||||
|
||||
# pulls:
|
||||
# daysUntilLock: 30
|
||||
|
||||
# Repository to extend settings from
|
||||
# _extends: repo
|
68
.github/workflows/ci-docker.yml
vendored
68
.github/workflows/ci-docker.yml
vendored
@@ -1,49 +1,63 @@
|
||||
---
|
||||
name: CI for docker images
|
||||
|
||||
# Only run when docker paths change
|
||||
# yamllint disable-line rule:truthy
|
||||
on:
|
||||
push:
|
||||
branches: [dev, beta, release]
|
||||
paths:
|
||||
- 'docker/**'
|
||||
- '.github/workflows/**'
|
||||
- 'requirements*.txt'
|
||||
- 'platformio.ini'
|
||||
- "docker/**"
|
||||
- ".github/workflows/ci-docker.yml"
|
||||
- "requirements*.txt"
|
||||
- "platformio.ini"
|
||||
- "script/platformio_install_deps.py"
|
||||
|
||||
pull_request:
|
||||
paths:
|
||||
- 'docker/**'
|
||||
- '.github/workflows/**'
|
||||
- 'requirements*.txt'
|
||||
- 'platformio.ini'
|
||||
- "docker/**"
|
||||
- ".github/workflows/ci-docker.yml"
|
||||
- "requirements*.txt"
|
||||
- "platformio.ini"
|
||||
- "script/platformio_install_deps.py"
|
||||
|
||||
permissions:
|
||||
contents: read
|
||||
packages: read
|
||||
|
||||
concurrency:
|
||||
# yamllint disable-line rule:line-length
|
||||
group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
|
||||
cancel-in-progress: true
|
||||
|
||||
jobs:
|
||||
check-docker:
|
||||
name: Build docker containers
|
||||
runs-on: ubuntu-latest
|
||||
strategy:
|
||||
fail-fast: false
|
||||
matrix:
|
||||
arch: [amd64, armv7, aarch64]
|
||||
build_type: ["ha-addon", "docker", "lint"]
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- name: Set up Python
|
||||
uses: actions/setup-python@v2
|
||||
with:
|
||||
python-version: '3.9'
|
||||
- name: Set up Docker Buildx
|
||||
uses: docker/setup-buildx-action@v1
|
||||
- name: Set up QEMU
|
||||
uses: docker/setup-qemu-action@v1
|
||||
- uses: actions/checkout@v4.1.1
|
||||
- name: Set up Python
|
||||
uses: actions/setup-python@v4.7.1
|
||||
with:
|
||||
python-version: "3.9"
|
||||
- name: Set up Docker Buildx
|
||||
uses: docker/setup-buildx-action@v3.0.0
|
||||
- name: Set up QEMU
|
||||
uses: docker/setup-qemu-action@v3.0.0
|
||||
|
||||
- name: Set TAG
|
||||
run: |
|
||||
echo "TAG=check" >> $GITHUB_ENV
|
||||
- name: Set TAG
|
||||
run: |
|
||||
echo "TAG=check" >> $GITHUB_ENV
|
||||
|
||||
- name: Run build
|
||||
run: |
|
||||
docker/build.py \
|
||||
--tag "${TAG}" \
|
||||
--arch "${{ matrix.arch }}" \
|
||||
--build-type "${{ matrix.build_type }}" \
|
||||
build
|
||||
- name: Run build
|
||||
run: |
|
||||
docker/build.py \
|
||||
--tag "${TAG}" \
|
||||
--arch "${{ matrix.arch }}" \
|
||||
--build-type "${{ matrix.build_type }}" \
|
||||
build
|
||||
|
432
.github/workflows/ci.yml
vendored
432
.github/workflows/ci.yml
vendored
@@ -1,156 +1,378 @@
|
||||
# THESE JOBS ARE COPIED IN release.yml and release-dev.yml
|
||||
# PLEASE ALSO UPDATE THOSE FILES WHEN CHANGING LINES HERE
|
||||
---
|
||||
name: CI
|
||||
|
||||
# yamllint disable-line rule:truthy
|
||||
on:
|
||||
push:
|
||||
branches: [dev, beta, release]
|
||||
|
||||
pull_request:
|
||||
paths:
|
||||
- "**"
|
||||
- "!.github/workflows/*.yml"
|
||||
- ".github/workflows/ci.yml"
|
||||
merge_group:
|
||||
|
||||
permissions:
|
||||
contents: read
|
||||
|
||||
env:
|
||||
DEFAULT_PYTHON: "3.9"
|
||||
PYUPGRADE_TARGET: "--py39-plus"
|
||||
CLANG_FORMAT_VERSION: "13.0.1"
|
||||
|
||||
concurrency:
|
||||
# yamllint disable-line rule:line-length
|
||||
group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
|
||||
cancel-in-progress: true
|
||||
|
||||
jobs:
|
||||
ci:
|
||||
name: ${{ matrix.name }}
|
||||
common:
|
||||
name: Create common environment
|
||||
runs-on: ubuntu-latest
|
||||
outputs:
|
||||
cache-key: ${{ steps.cache-key.outputs.key }}
|
||||
steps:
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Generate cache-key
|
||||
id: cache-key
|
||||
run: echo key="${{ hashFiles('requirements.txt', 'requirements_optional.txt', 'requirements_test.txt') }}" >> $GITHUB_OUTPUT
|
||||
- name: Set up Python ${{ env.DEFAULT_PYTHON }}
|
||||
id: python
|
||||
uses: actions/setup-python@v4.7.1
|
||||
with:
|
||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||
- name: Restore Python virtual environment
|
||||
id: cache-venv
|
||||
uses: actions/cache@v3.3.2
|
||||
with:
|
||||
path: venv
|
||||
# yamllint disable-line rule:line-length
|
||||
key: ${{ runner.os }}-${{ steps.python.outputs.python-version }}-venv-${{ steps.cache-key.outputs.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_optional.txt -r requirements_test.txt
|
||||
pip install -e .
|
||||
|
||||
black:
|
||||
name: Check black
|
||||
runs-on: ubuntu-latest
|
||||
needs:
|
||||
- common
|
||||
steps:
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Restore Python
|
||||
uses: ./.github/actions/restore-python
|
||||
with:
|
||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||
cache-key: ${{ needs.common.outputs.cache-key }}
|
||||
- name: Run black
|
||||
run: |
|
||||
. venv/bin/activate
|
||||
black --verbose esphome tests
|
||||
- name: Suggested changes
|
||||
run: script/ci-suggest-changes
|
||||
if: always()
|
||||
|
||||
flake8:
|
||||
name: Check flake8
|
||||
runs-on: ubuntu-latest
|
||||
needs:
|
||||
- common
|
||||
steps:
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Restore Python
|
||||
uses: ./.github/actions/restore-python
|
||||
with:
|
||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||
cache-key: ${{ needs.common.outputs.cache-key }}
|
||||
- name: Run flake8
|
||||
run: |
|
||||
. venv/bin/activate
|
||||
flake8 esphome
|
||||
- name: Suggested changes
|
||||
run: script/ci-suggest-changes
|
||||
if: always()
|
||||
|
||||
pylint:
|
||||
name: Check pylint
|
||||
runs-on: ubuntu-latest
|
||||
needs:
|
||||
- common
|
||||
steps:
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Restore Python
|
||||
uses: ./.github/actions/restore-python
|
||||
with:
|
||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||
cache-key: ${{ needs.common.outputs.cache-key }}
|
||||
- name: Run pylint
|
||||
run: |
|
||||
. venv/bin/activate
|
||||
pylint -f parseable --persistent=n esphome
|
||||
- name: Suggested changes
|
||||
run: script/ci-suggest-changes
|
||||
if: always()
|
||||
|
||||
pyupgrade:
|
||||
name: Check pyupgrade
|
||||
runs-on: ubuntu-latest
|
||||
needs:
|
||||
- common
|
||||
steps:
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Restore Python
|
||||
uses: ./.github/actions/restore-python
|
||||
with:
|
||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||
cache-key: ${{ needs.common.outputs.cache-key }}
|
||||
- name: Run pyupgrade
|
||||
run: |
|
||||
. venv/bin/activate
|
||||
pyupgrade ${{ env.PYUPGRADE_TARGET }} `find esphome -name "*.py" -type f`
|
||||
- name: Suggested changes
|
||||
run: script/ci-suggest-changes
|
||||
if: always()
|
||||
|
||||
ci-custom:
|
||||
name: Run script/ci-custom
|
||||
runs-on: ubuntu-latest
|
||||
needs:
|
||||
- common
|
||||
steps:
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Restore Python
|
||||
uses: ./.github/actions/restore-python
|
||||
with:
|
||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||
cache-key: ${{ needs.common.outputs.cache-key }}
|
||||
- name: Register matcher
|
||||
run: echo "::add-matcher::.github/workflows/matchers/ci-custom.json"
|
||||
- name: Run script/ci-custom
|
||||
run: |
|
||||
. venv/bin/activate
|
||||
script/ci-custom.py
|
||||
script/build_codeowners.py --check
|
||||
|
||||
pytest:
|
||||
name: Run pytest
|
||||
runs-on: ubuntu-latest
|
||||
needs:
|
||||
- common
|
||||
steps:
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Restore Python
|
||||
uses: ./.github/actions/restore-python
|
||||
with:
|
||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||
cache-key: ${{ needs.common.outputs.cache-key }}
|
||||
- name: Register matcher
|
||||
run: echo "::add-matcher::.github/workflows/matchers/pytest.json"
|
||||
- name: Run pytest
|
||||
run: |
|
||||
. venv/bin/activate
|
||||
pytest -vv --tb=native tests
|
||||
|
||||
clang-format:
|
||||
name: Check clang-format
|
||||
runs-on: ubuntu-latest
|
||||
needs:
|
||||
- common
|
||||
steps:
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Restore Python
|
||||
uses: ./.github/actions/restore-python
|
||||
with:
|
||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||
cache-key: ${{ needs.common.outputs.cache-key }}
|
||||
- name: Install clang-format
|
||||
run: |
|
||||
. venv/bin/activate
|
||||
pip install clang-format==${{ env.CLANG_FORMAT_VERSION }}
|
||||
- name: Run clang-format
|
||||
run: |
|
||||
. venv/bin/activate
|
||||
script/clang-format -i
|
||||
git diff-index --quiet HEAD --
|
||||
- name: Suggested changes
|
||||
run: script/ci-suggest-changes
|
||||
if: always()
|
||||
|
||||
compile-tests-list:
|
||||
runs-on: ubuntu-latest
|
||||
outputs:
|
||||
matrix: ${{ steps.set-matrix.outputs.matrix }}
|
||||
steps:
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Find all YAML test files
|
||||
id: set-matrix
|
||||
run: echo "matrix=$(ls tests/test*.yaml | jq -R -s -c 'split("\n")[:-1]')" >> $GITHUB_OUTPUT
|
||||
|
||||
validate-tests:
|
||||
name: Validate YAML test ${{ matrix.file }}
|
||||
runs-on: ubuntu-latest
|
||||
needs:
|
||||
- common
|
||||
- compile-tests-list
|
||||
strategy:
|
||||
fail-fast: false
|
||||
matrix:
|
||||
file: ${{ fromJson(needs.compile-tests-list.outputs.matrix) }}
|
||||
steps:
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Restore Python
|
||||
uses: ./.github/actions/restore-python
|
||||
with:
|
||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||
cache-key: ${{ needs.common.outputs.cache-key }}
|
||||
- name: Run esphome config ${{ matrix.file }}
|
||||
run: |
|
||||
. venv/bin/activate
|
||||
esphome config ${{ matrix.file }}
|
||||
|
||||
compile-tests:
|
||||
name: Run YAML test ${{ matrix.file }}
|
||||
runs-on: ubuntu-latest
|
||||
needs:
|
||||
- common
|
||||
- black
|
||||
- ci-custom
|
||||
- clang-format
|
||||
- flake8
|
||||
- pylint
|
||||
- pytest
|
||||
- pyupgrade
|
||||
- compile-tests-list
|
||||
- validate-tests
|
||||
strategy:
|
||||
fail-fast: false
|
||||
max-parallel: 2
|
||||
matrix:
|
||||
file: ${{ fromJson(needs.compile-tests-list.outputs.matrix) }}
|
||||
steps:
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Restore Python
|
||||
uses: ./.github/actions/restore-python
|
||||
with:
|
||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||
cache-key: ${{ needs.common.outputs.cache-key }}
|
||||
- name: Run esphome compile ${{ matrix.file }}
|
||||
run: |
|
||||
. venv/bin/activate
|
||||
esphome compile ${{ matrix.file }}
|
||||
|
||||
clang-tidy:
|
||||
name: ${{ matrix.name }}
|
||||
runs-on: ubuntu-latest
|
||||
needs:
|
||||
- common
|
||||
- black
|
||||
- ci-custom
|
||||
- clang-format
|
||||
- flake8
|
||||
- pylint
|
||||
- pytest
|
||||
- pyupgrade
|
||||
strategy:
|
||||
fail-fast: false
|
||||
max-parallel: 2
|
||||
matrix:
|
||||
include:
|
||||
- id: ci-custom
|
||||
name: Run script/ci-custom
|
||||
- id: lint-python
|
||||
name: Run script/lint-python
|
||||
- id: test
|
||||
file: tests/test1.yaml
|
||||
name: Test tests/test1.yaml
|
||||
pio_cache_key: test1
|
||||
- id: test
|
||||
file: tests/test2.yaml
|
||||
name: Test tests/test2.yaml
|
||||
pio_cache_key: test2
|
||||
- id: test
|
||||
file: tests/test3.yaml
|
||||
name: Test tests/test3.yaml
|
||||
pio_cache_key: test1
|
||||
- id: test
|
||||
file: tests/test4.yaml
|
||||
name: Test tests/test4.yaml
|
||||
pio_cache_key: test4
|
||||
- id: test
|
||||
file: tests/test5.yaml
|
||||
name: Test tests/test5.yaml
|
||||
pio_cache_key: test5
|
||||
- id: pytest
|
||||
name: Run pytest
|
||||
- id: clang-format
|
||||
name: Run script/clang-format
|
||||
- id: clang-tidy
|
||||
name: Run script/clang-tidy for ESP8266
|
||||
options: --environment esp8266-tidy --grep USE_ESP8266
|
||||
options: --environment esp8266-arduino-tidy --grep USE_ESP8266
|
||||
pio_cache_key: tidyesp8266
|
||||
- id: clang-tidy
|
||||
name: Run script/clang-tidy for ESP32 1/4
|
||||
options: --environment esp32-tidy --split-num 4 --split-at 1
|
||||
name: Run script/clang-tidy for ESP32 Arduino 1/4
|
||||
options: --environment esp32-arduino-tidy --split-num 4 --split-at 1
|
||||
pio_cache_key: tidyesp32
|
||||
- id: clang-tidy
|
||||
name: Run script/clang-tidy for ESP32 2/4
|
||||
options: --environment esp32-tidy --split-num 4 --split-at 2
|
||||
name: Run script/clang-tidy for ESP32 Arduino 2/4
|
||||
options: --environment esp32-arduino-tidy --split-num 4 --split-at 2
|
||||
pio_cache_key: tidyesp32
|
||||
- id: clang-tidy
|
||||
name: Run script/clang-tidy for ESP32 3/4
|
||||
options: --environment esp32-tidy --split-num 4 --split-at 3
|
||||
name: Run script/clang-tidy for ESP32 Arduino 3/4
|
||||
options: --environment esp32-arduino-tidy --split-num 4 --split-at 3
|
||||
pio_cache_key: tidyesp32
|
||||
- id: clang-tidy
|
||||
name: Run script/clang-tidy for ESP32 4/4
|
||||
options: --environment esp32-tidy --split-num 4 --split-at 4
|
||||
name: Run script/clang-tidy for ESP32 Arduino 4/4
|
||||
options: --environment esp32-arduino-tidy --split-num 4 --split-at 4
|
||||
pio_cache_key: tidyesp32
|
||||
- id: clang-tidy
|
||||
name: Run script/clang-tidy for ESP32 esp-idf
|
||||
name: Run script/clang-tidy for ESP32 IDF
|
||||
options: --environment esp32-idf-tidy --grep USE_ESP_IDF
|
||||
pio_cache_key: tidyesp32-idf
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- name: Set up Python
|
||||
uses: actions/setup-python@v2
|
||||
id: python
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Restore Python
|
||||
uses: ./.github/actions/restore-python
|
||||
with:
|
||||
python-version: '3.7'
|
||||
|
||||
- name: Cache pip modules
|
||||
uses: actions/cache@v2
|
||||
with:
|
||||
path: ~/.cache/pip
|
||||
key: pip-${{ steps.python.outputs.python-version }}-${{ hashFiles('requirements*.txt') }}
|
||||
restore-keys: |
|
||||
pip-${{ steps.python.outputs.python-version }}-
|
||||
|
||||
- name: Set up python environment
|
||||
run: |
|
||||
pip3 install -r requirements.txt -r requirements_optional.txt -r requirements_test.txt
|
||||
pip3 install -e .
|
||||
|
||||
# Use per check platformio cache because checks use different parts
|
||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||
cache-key: ${{ needs.common.outputs.cache-key }}
|
||||
- name: Cache platformio
|
||||
uses: actions/cache@v2
|
||||
uses: actions/cache@v3.3.2
|
||||
with:
|
||||
path: ~/.platformio
|
||||
# yamllint disable-line rule:line-length
|
||||
key: platformio-${{ matrix.pio_cache_key }}-${{ hashFiles('platformio.ini') }}
|
||||
if: matrix.id == 'test' || matrix.id == 'clang-tidy'
|
||||
|
||||
- name: Install clang tools
|
||||
run: |
|
||||
sudo apt-get install \
|
||||
clang-format-11 \
|
||||
clang-tidy-11
|
||||
if: matrix.id == 'clang-tidy' || matrix.id == 'clang-format'
|
||||
- name: Install clang-tidy
|
||||
run: sudo apt-get install clang-tidy-14
|
||||
|
||||
- name: Register problem matchers
|
||||
run: |
|
||||
echo "::add-matcher::.github/workflows/matchers/ci-custom.json"
|
||||
echo "::add-matcher::.github/workflows/matchers/lint-python.json"
|
||||
echo "::add-matcher::.github/workflows/matchers/python.json"
|
||||
echo "::add-matcher::.github/workflows/matchers/pytest.json"
|
||||
echo "::add-matcher::.github/workflows/matchers/gcc.json"
|
||||
echo "::add-matcher::.github/workflows/matchers/clang-tidy.json"
|
||||
|
||||
- name: Lint Custom
|
||||
run: |
|
||||
script/ci-custom.py
|
||||
script/build_codeowners.py --check
|
||||
if: matrix.id == 'ci-custom'
|
||||
|
||||
- name: Lint Python
|
||||
run: script/lint-python
|
||||
if: matrix.id == 'lint-python'
|
||||
|
||||
- run: esphome compile ${{ matrix.file }}
|
||||
if: matrix.id == 'test'
|
||||
env:
|
||||
# Also cache libdeps, store them in a ~/.platformio subfolder
|
||||
PLATFORMIO_LIBDEPS_DIR: ~/.platformio/libdeps
|
||||
|
||||
- name: Run pytest
|
||||
run: |
|
||||
pytest -vv --tb=native tests
|
||||
if: matrix.id == 'pytest'
|
||||
|
||||
# Also run git-diff-index so that the step is marked as failed on formatting errors,
|
||||
# since clang-format doesn't do anything but change files if -i is passed.
|
||||
- name: Run clang-format
|
||||
run: |
|
||||
script/clang-format -i
|
||||
git diff-index --quiet HEAD --
|
||||
if: matrix.id == 'clang-format'
|
||||
|
||||
- name: Run clang-tidy
|
||||
run: |
|
||||
. venv/bin/activate
|
||||
script/clang-tidy --all-headers --fix ${{ matrix.options }}
|
||||
if: matrix.id == 'clang-tidy'
|
||||
env:
|
||||
# Also cache libdeps, store them in a ~/.platformio subfolder
|
||||
PLATFORMIO_LIBDEPS_DIR: ~/.platformio/libdeps
|
||||
|
||||
- name: Suggested changes
|
||||
run: script/ci-suggest-changes
|
||||
if: always() && (matrix.id == 'clang-tidy' || matrix.id == 'clang-format')
|
||||
# yamllint disable-line rule:line-length
|
||||
if: always()
|
||||
|
||||
ci-status:
|
||||
name: CI Status
|
||||
runs-on: ubuntu-latest
|
||||
needs:
|
||||
- common
|
||||
- black
|
||||
- ci-custom
|
||||
- clang-format
|
||||
- flake8
|
||||
- pylint
|
||||
- pytest
|
||||
- pyupgrade
|
||||
- compile-tests
|
||||
- clang-tidy
|
||||
if: always()
|
||||
steps:
|
||||
- name: Success
|
||||
if: ${{ !(contains(needs.*.result, 'failure')) }}
|
||||
run: exit 0
|
||||
- name: Failure
|
||||
if: ${{ contains(needs.*.result, 'failure') }}
|
||||
run: exit 1
|
||||
|
18
.github/workflows/lock.yml
vendored
18
.github/workflows/lock.yml
vendored
@@ -1,21 +1,29 @@
|
||||
---
|
||||
name: Lock
|
||||
|
||||
# yamllint disable-line rule:truthy
|
||||
on:
|
||||
schedule:
|
||||
- cron: '30 0 * * *'
|
||||
- cron: "30 0 * * *"
|
||||
workflow_dispatch:
|
||||
|
||||
permissions:
|
||||
issues: write
|
||||
pull-requests: write
|
||||
|
||||
concurrency:
|
||||
group: lock
|
||||
|
||||
jobs:
|
||||
lock:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: dessant/lock-threads@v2
|
||||
- uses: dessant/lock-threads@v4.0.1
|
||||
with:
|
||||
github-token: ${{ github.token }}
|
||||
pr-lock-inactive-days: "1"
|
||||
pr-inactive-days: "1"
|
||||
pr-lock-reason: ""
|
||||
process-only: prs
|
||||
exclude-any-pr-labels: keep-open
|
||||
|
||||
issue-inactive-days: "7"
|
||||
issue-lock-reason: ""
|
||||
exclude-any-issue-labels: keep-open
|
||||
|
2
.github/workflows/matchers/ci-custom.json
vendored
2
.github/workflows/matchers/ci-custom.json
vendored
@@ -4,7 +4,7 @@
|
||||
"owner": "ci-custom",
|
||||
"pattern": [
|
||||
{
|
||||
"regexp": "^ERROR (.*):(\\d+):(\\d+) - (.*)$",
|
||||
"regexp": "^(.*):(\\d+):(\\d+):\\s+lint:\\s+(.*)$",
|
||||
"file": 1,
|
||||
"line": 2,
|
||||
"column": 3,
|
||||
|
2
.github/workflows/matchers/gcc.json
vendored
2
.github/workflows/matchers/gcc.json
vendored
@@ -5,7 +5,7 @@
|
||||
"severity": "error",
|
||||
"pattern": [
|
||||
{
|
||||
"regexp": "^(.*):(\\d+):(\\d+):\\s+(?:fatal\\s+)?(warning|error):\\s+(.*)$",
|
||||
"regexp": "^src/(.*):(\\d+):(\\d+):\\s+(?:fatal\\s+)?(warning|error):\\s+(.*)$",
|
||||
"file": 1,
|
||||
"line": 2,
|
||||
"column": 3,
|
||||
|
15
.github/workflows/matchers/lint-python.json
vendored
15
.github/workflows/matchers/lint-python.json
vendored
@@ -1,11 +1,22 @@
|
||||
{
|
||||
"problemMatcher": [
|
||||
{
|
||||
"owner": "black",
|
||||
"severity": "error",
|
||||
"pattern": [
|
||||
{
|
||||
"regexp": "^(.*): (Please format this file with the black formatter)",
|
||||
"file": 1,
|
||||
"message": 2
|
||||
}
|
||||
]
|
||||
},
|
||||
{
|
||||
"owner": "flake8",
|
||||
"severity": "error",
|
||||
"pattern": [
|
||||
{
|
||||
"regexp": "^(.*):(\\d+) - ([EFCDNW]\\d{3}.*)$",
|
||||
"regexp": "^(.*):(\\d+): ([EFCDNW]\\d{3}.*)$",
|
||||
"file": 1,
|
||||
"line": 2,
|
||||
"message": 3
|
||||
@@ -17,7 +28,7 @@
|
||||
"severity": "error",
|
||||
"pattern": [
|
||||
{
|
||||
"regexp": "^(.*):(\\d+) - (\\[[EFCRW]\\d{4}\\(.*\\),.*\\].*)$",
|
||||
"regexp": "^(.*):(\\d+): (\\[[EFCRW]\\d{4}\\(.*\\),.*\\].*)$",
|
||||
"file": 1,
|
||||
"line": 2,
|
||||
"message": 3
|
||||
|
24
.github/workflows/needs-docs.yml
vendored
Normal file
24
.github/workflows/needs-docs.yml
vendored
Normal file
@@ -0,0 +1,24 @@
|
||||
name: Needs Docs
|
||||
|
||||
on:
|
||||
pull_request:
|
||||
types: [labeled, unlabeled]
|
||||
|
||||
jobs:
|
||||
check:
|
||||
name: Check
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: Check for needs-docs label
|
||||
uses: actions/github-script@v6.4.1
|
||||
with:
|
||||
script: |
|
||||
const { data: labels } = await github.rest.issues.listLabelsOnIssue({
|
||||
owner: context.repo.owner,
|
||||
repo: context.repo.repo,
|
||||
issue_number: context.issue.number
|
||||
});
|
||||
const needsDocs = labels.find(label => label.name === 'needs-docs');
|
||||
if (needsDocs) {
|
||||
core.setFailed('Pull request needs docs');
|
||||
}
|
174
.github/workflows/release.yml
vendored
174
.github/workflows/release.yml
vendored
@@ -1,5 +1,7 @@
|
||||
---
|
||||
name: Publish Release
|
||||
|
||||
# yamllint disable-line rule:truthy
|
||||
on:
|
||||
workflow_dispatch:
|
||||
release:
|
||||
@@ -7,6 +9,9 @@ on:
|
||||
schedule:
|
||||
- cron: "0 2 * * *"
|
||||
|
||||
permissions:
|
||||
contents: read
|
||||
|
||||
jobs:
|
||||
init:
|
||||
name: Initialize build
|
||||
@@ -14,9 +19,10 @@ jobs:
|
||||
outputs:
|
||||
tag: ${{ steps.tag.outputs.tag }}
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- uses: actions/checkout@v4.1.1
|
||||
- name: Get tag
|
||||
id: tag
|
||||
# yamllint disable rule:line-length
|
||||
run: |
|
||||
if [[ "$GITHUB_EVENT_NAME" = "release" ]]; then
|
||||
TAG="${GITHUB_REF#refs/tags/}"
|
||||
@@ -24,23 +30,30 @@ jobs:
|
||||
TAG=$(cat esphome/const.py | sed -n -E "s/^__version__\s+=\s+\"(.+)\"$/\1/p")
|
||||
today="$(date --utc '+%Y%m%d')"
|
||||
TAG="${TAG}${today}"
|
||||
BRANCH=${GITHUB_REF#refs/heads/}
|
||||
if [[ "$BRANCH" != "dev" ]]; then
|
||||
TAG="${TAG}-${BRANCH}"
|
||||
fi
|
||||
fi
|
||||
echo "::set-output name=tag::${TAG}"
|
||||
echo "tag=${TAG}" >> $GITHUB_OUTPUT
|
||||
# yamllint enable rule:line-length
|
||||
|
||||
deploy-pypi:
|
||||
name: Build and publish to PyPi
|
||||
if: github.repository == 'esphome/esphome' && github.event_name == 'release'
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- uses: actions/checkout@v4.1.1
|
||||
- name: Set up Python
|
||||
uses: actions/setup-python@v1
|
||||
uses: actions/setup-python@v4.7.1
|
||||
with:
|
||||
python-version: '3.x'
|
||||
python-version: "3.x"
|
||||
- name: Set up python environment
|
||||
env:
|
||||
ESPHOME_NO_VENV: 1
|
||||
run: |
|
||||
script/setup
|
||||
pip install setuptools wheel twine
|
||||
pip install twine
|
||||
- name: Build
|
||||
run: python setup.py sdist bdist_wheel
|
||||
- name: Upload
|
||||
@@ -50,96 +63,95 @@ jobs:
|
||||
run: twine upload dist/*
|
||||
|
||||
deploy-docker:
|
||||
name: Build and publish docker containers
|
||||
name: Build and publish ESPHome ${{ matrix.image.title}}
|
||||
if: github.repository == 'esphome/esphome'
|
||||
permissions:
|
||||
contents: read
|
||||
packages: write
|
||||
runs-on: ubuntu-latest
|
||||
continue-on-error: ${{ matrix.image.title == 'lint' }}
|
||||
needs: [init]
|
||||
strategy:
|
||||
fail-fast: false
|
||||
matrix:
|
||||
arch: [amd64, armv7, aarch64]
|
||||
build_type: ["ha-addon", "docker", "lint"]
|
||||
image:
|
||||
- title: "ha-addon"
|
||||
suffix: "hassio"
|
||||
target: "hassio"
|
||||
baseimg: "hassio"
|
||||
- title: "docker"
|
||||
suffix: ""
|
||||
target: "docker"
|
||||
baseimg: "docker"
|
||||
- title: "lint"
|
||||
suffix: "lint"
|
||||
target: "lint"
|
||||
baseimg: "docker"
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- name: Set up Python
|
||||
uses: actions/setup-python@v2
|
||||
with:
|
||||
python-version: '3.9'
|
||||
- uses: actions/checkout@v4.1.1
|
||||
- name: Set up Python
|
||||
uses: actions/setup-python@v4.7.1
|
||||
with:
|
||||
python-version: "3.9"
|
||||
|
||||
- name: Set up Docker Buildx
|
||||
uses: docker/setup-buildx-action@v1
|
||||
- name: Set up QEMU
|
||||
uses: docker/setup-qemu-action@v1
|
||||
- name: Set up Docker Buildx
|
||||
uses: docker/setup-buildx-action@v3.0.0
|
||||
- name: Set up QEMU
|
||||
uses: docker/setup-qemu-action@v3.0.0
|
||||
|
||||
- name: Log in to docker hub
|
||||
uses: docker/login-action@v1
|
||||
with:
|
||||
username: ${{ secrets.DOCKER_USER }}
|
||||
password: ${{ secrets.DOCKER_PASSWORD }}
|
||||
- name: Log in to the GitHub container registry
|
||||
uses: docker/login-action@v1
|
||||
with:
|
||||
- name: Log in to docker hub
|
||||
uses: docker/login-action@v3.0.0
|
||||
with:
|
||||
username: ${{ secrets.DOCKER_USER }}
|
||||
password: ${{ secrets.DOCKER_PASSWORD }}
|
||||
- name: Log in to the GitHub container registry
|
||||
uses: docker/login-action@v3.0.0
|
||||
with:
|
||||
registry: ghcr.io
|
||||
username: ${{ github.actor }}
|
||||
password: ${{ secrets.GITHUB_TOKEN }}
|
||||
|
||||
- name: Build and push
|
||||
run: |
|
||||
docker/build.py \
|
||||
--tag "${{ needs.init.outputs.tag }}" \
|
||||
--arch "${{ matrix.arch }}" \
|
||||
--build-type "${{ matrix.build_type }}" \
|
||||
build \
|
||||
--push
|
||||
- name: Generate short tags
|
||||
id: tags
|
||||
run: |
|
||||
docker/generate_tags.py \
|
||||
--tag "${{ needs.init.outputs.tag }}" \
|
||||
--suffix "${{ matrix.image.suffix }}"
|
||||
|
||||
deploy-docker-manifest:
|
||||
if: github.repository == 'esphome/esphome'
|
||||
runs-on: ubuntu-latest
|
||||
needs: [init, deploy-docker]
|
||||
strategy:
|
||||
matrix:
|
||||
build_type: ["ha-addon", "docker", "lint"]
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
- name: Set up Python
|
||||
uses: actions/setup-python@v2
|
||||
with:
|
||||
python-version: '3.9'
|
||||
- name: Enable experimental manifest support
|
||||
run: |
|
||||
mkdir -p ~/.docker
|
||||
echo "{\"experimental\": \"enabled\"}" > ~/.docker/config.json
|
||||
- name: Build and push
|
||||
uses: docker/build-push-action@v5.0.0
|
||||
with:
|
||||
context: .
|
||||
file: ./docker/Dockerfile
|
||||
platforms: linux/amd64,linux/arm/v7,linux/arm64
|
||||
target: ${{ matrix.image.target }}
|
||||
push: true
|
||||
# yamllint disable rule:line-length
|
||||
cache-from: type=registry,ref=ghcr.io/${{ steps.tags.outputs.image }}:cache-${{ steps.tags.outputs.channel }}
|
||||
cache-to: type=registry,ref=ghcr.io/${{ steps.tags.outputs.image }}:cache-${{ steps.tags.outputs.channel }},mode=max
|
||||
# yamllint enable rule:line-length
|
||||
tags: ${{ steps.tags.outputs.tags }}
|
||||
build-args: |
|
||||
BASEIMGTYPE=${{ matrix.image.baseimg }}
|
||||
BUILD_VERSION=${{ needs.init.outputs.tag }}
|
||||
|
||||
- name: Log in to docker hub
|
||||
uses: docker/login-action@v1
|
||||
with:
|
||||
username: ${{ secrets.DOCKER_USER }}
|
||||
password: ${{ secrets.DOCKER_PASSWORD }}
|
||||
- name: Log in to the GitHub container registry
|
||||
uses: docker/login-action@v1
|
||||
with:
|
||||
registry: ghcr.io
|
||||
username: ${{ github.actor }}
|
||||
password: ${{ secrets.GITHUB_TOKEN }}
|
||||
|
||||
- name: Run manifest
|
||||
run: |
|
||||
docker/build.py \
|
||||
--tag "${{ needs.init.outputs.tag }}" \
|
||||
--build-type "${{ matrix.build_type }}" \
|
||||
manifest
|
||||
|
||||
deploy-hassio-repo:
|
||||
deploy-ha-addon-repo:
|
||||
if: github.repository == 'esphome/esphome' && github.event_name == 'release'
|
||||
runs-on: ubuntu-latest
|
||||
needs: [deploy-docker]
|
||||
steps:
|
||||
- env:
|
||||
TOKEN: ${{ secrets.DEPLOY_HASSIO_TOKEN }}
|
||||
run: |
|
||||
TAG="${GITHUB_REF#refs/tags/}"
|
||||
curl \
|
||||
-u ":$TOKEN" \
|
||||
-X POST \
|
||||
-H "Accept: application/vnd.github.v3+json" \
|
||||
https://api.github.com/repos/esphome/hassio/actions/workflows/bump-version.yml/dispatches \
|
||||
-d "{\"ref\":\"main\",\"inputs\":{\"version\":\"$TAG\"}}"
|
||||
- name: Trigger Workflow
|
||||
uses: actions/github-script@v6.4.1
|
||||
with:
|
||||
github-token: ${{ secrets.DEPLOY_HA_ADDON_REPO_TOKEN }}
|
||||
script: |
|
||||
github.rest.actions.createWorkflowDispatch({
|
||||
owner: "esphome",
|
||||
repo: "home-assistant-addon",
|
||||
workflow_id: "bump-version.yml",
|
||||
ref: "main",
|
||||
inputs: {
|
||||
version: "${{ github.event.release.tag_name }}",
|
||||
content: ${{ toJSON(github.event.release.body) }}
|
||||
}
|
||||
})
|
||||
|
29
.github/workflows/stale.yml
vendored
29
.github/workflows/stale.yml
vendored
@@ -1,30 +1,51 @@
|
||||
---
|
||||
name: Stale
|
||||
|
||||
# yamllint disable-line rule:truthy
|
||||
on:
|
||||
schedule:
|
||||
- cron: '30 0 * * *'
|
||||
- cron: "30 0 * * *"
|
||||
workflow_dispatch:
|
||||
|
||||
permissions:
|
||||
issues: write
|
||||
pull-requests: write
|
||||
|
||||
concurrency:
|
||||
group: lock
|
||||
|
||||
jobs:
|
||||
stale:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/stale@v4
|
||||
- uses: actions/stale@v8.0.0
|
||||
with:
|
||||
repo-token: ${{ github.token }}
|
||||
days-before-pr-stale: 90
|
||||
days-before-pr-close: 7
|
||||
days-before-issue-stale: -1
|
||||
days-before-issue-close: -1
|
||||
remove-stale-when-updated: true
|
||||
stale-pr-label: "stale"
|
||||
exempt-pr-labels: "no-stale"
|
||||
exempt-pr-labels: "not-stale"
|
||||
stale-pr-message: >
|
||||
There hasn't been any activity on this pull request recently. This
|
||||
pull request has been automatically marked as stale because of that
|
||||
and will be closed if no further activity occurs within 7 days.
|
||||
Thank you for your contributions.
|
||||
|
||||
# Use stale to automatically close issues with a
|
||||
# reference to the issue tracker
|
||||
close-issues:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/stale@v8.0.0
|
||||
with:
|
||||
days-before-pr-stale: -1
|
||||
days-before-pr-close: -1
|
||||
days-before-issue-stale: 1
|
||||
days-before-issue-close: 1
|
||||
remove-stale-when-updated: true
|
||||
stale-issue-label: "stale"
|
||||
exempt-issue-labels: "not-stale"
|
||||
stale-issue-message: >
|
||||
https://github.com/esphome/esphome/issues/430
|
||||
|
48
.github/workflows/sync-device-classes.yml
vendored
Normal file
48
.github/workflows/sync-device-classes.yml
vendored
Normal file
@@ -0,0 +1,48 @@
|
||||
---
|
||||
name: Synchronise Device Classes from Home Assistant
|
||||
|
||||
on:
|
||||
workflow_dispatch:
|
||||
schedule:
|
||||
- cron: "45 6 * * *"
|
||||
|
||||
jobs:
|
||||
sync:
|
||||
name: Sync Device Classes
|
||||
runs-on: ubuntu-latest
|
||||
if: github.repository == 'esphome/esphome'
|
||||
steps:
|
||||
- name: Checkout
|
||||
uses: actions/checkout@v4.1.1
|
||||
|
||||
- name: Checkout Home Assistant
|
||||
uses: actions/checkout@v4.1.1
|
||||
with:
|
||||
repository: home-assistant/core
|
||||
path: lib/home-assistant
|
||||
|
||||
- name: Setup Python
|
||||
uses: actions/setup-python@v4.7.1
|
||||
with:
|
||||
python-version: 3.11
|
||||
|
||||
- name: Install Home Assistant
|
||||
run: |
|
||||
python -m pip install --upgrade pip
|
||||
pip install -e lib/home-assistant
|
||||
|
||||
- name: Sync
|
||||
run: |
|
||||
python ./script/sync-device_class.py
|
||||
|
||||
- name: Commit changes
|
||||
uses: peter-evans/create-pull-request@v5.0.2
|
||||
with:
|
||||
commit-message: "Synchronise Device Classes from Home Assistant"
|
||||
committer: esphomebot <esphome@nabucasa.com>
|
||||
author: esphomebot <esphome@nabucasa.com>
|
||||
branch: sync/device-classes
|
||||
delete-branch: true
|
||||
title: "Synchronise Device Classes from Home Assistant"
|
||||
body-path: .github/PULL_REQUEST_TEMPLATE.md
|
||||
token: ${{ secrets.DEVICE_CLASS_SYNC_TOKEN }}
|
22
.github/workflows/yaml-lint.yml
vendored
Normal file
22
.github/workflows/yaml-lint.yml
vendored
Normal file
@@ -0,0 +1,22 @@
|
||||
name: YAML lint
|
||||
|
||||
on:
|
||||
push:
|
||||
branches: [dev, beta, release]
|
||||
paths:
|
||||
- "**.yaml"
|
||||
- "**.yml"
|
||||
pull_request:
|
||||
paths:
|
||||
- "**.yaml"
|
||||
- "**.yml"
|
||||
|
||||
jobs:
|
||||
yamllint:
|
||||
name: yamllint
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: Check out code from GitHub
|
||||
uses: actions/checkout@v4.1.1
|
||||
- name: Run yamllint
|
||||
uses: frenck/action-yamllint@v1.4.1
|
11
.gitignore
vendored
11
.gitignore
vendored
@@ -13,6 +13,12 @@ __pycache__/
|
||||
# Intellij Idea
|
||||
.idea
|
||||
|
||||
# Eclipse
|
||||
.project
|
||||
.cproject
|
||||
.pydevproject
|
||||
.settings/
|
||||
|
||||
# Vim
|
||||
*.swp
|
||||
|
||||
@@ -77,6 +83,7 @@ venv/
|
||||
ENV/
|
||||
env.bak/
|
||||
venv.bak/
|
||||
venv-*/
|
||||
|
||||
# mypy
|
||||
.mypy_cache/
|
||||
@@ -127,3 +134,7 @@ tests/.esphome/
|
||||
|
||||
sdkconfig.*
|
||||
!sdkconfig.defaults
|
||||
|
||||
.tests/
|
||||
|
||||
/components
|
||||
|
@@ -1,6 +0,0 @@
|
||||
ports:
|
||||
- port: 6052
|
||||
onOpen: open-preview
|
||||
tasks:
|
||||
- before: pyenv local $(pyenv version | grep '^3\.' | cut -d ' ' -f 1) && script/setup
|
||||
command: python -m esphome config dashboard
|
@@ -1,16 +1,17 @@
|
||||
---
|
||||
# See https://pre-commit.com for more information
|
||||
# See https://pre-commit.com/hooks.html for more hooks
|
||||
repos:
|
||||
- repo: https://github.com/ambv/black
|
||||
rev: 20.8b1
|
||||
- repo: https://github.com/psf/black-pre-commit-mirror
|
||||
rev: 23.10.1
|
||||
hooks:
|
||||
- id: black
|
||||
args:
|
||||
- --safe
|
||||
- --quiet
|
||||
files: ^((esphome|script|tests)/.+)?[^/]+\.py$
|
||||
- repo: https://gitlab.com/pycqa/flake8
|
||||
rev: 3.8.4
|
||||
- id: black
|
||||
args:
|
||||
- --safe
|
||||
- --quiet
|
||||
files: ^((esphome|script|tests)/.+)?[^/]+\.py$
|
||||
- repo: https://github.com/PyCQA/flake8
|
||||
rev: 6.1.0
|
||||
hooks:
|
||||
- id: flake8
|
||||
additional_dependencies:
|
||||
@@ -25,3 +26,8 @@ repos:
|
||||
- --branch=dev
|
||||
- --branch=release
|
||||
- --branch=beta
|
||||
- repo: https://github.com/asottile/pyupgrade
|
||||
rev: v3.15.0
|
||||
hooks:
|
||||
- id: pyupgrade
|
||||
args: [--py39-plus]
|
||||
|
33
.vscode/tasks.json
vendored
33
.vscode/tasks.json
vendored
@@ -2,15 +2,24 @@
|
||||
"version": "2.0.0",
|
||||
"tasks": [
|
||||
{
|
||||
"label": "run",
|
||||
"label": "Run Dashboard",
|
||||
"type": "shell",
|
||||
"command": "python3 -m esphome dashboard config/",
|
||||
"command": "${command:python.interpreterPath}",
|
||||
"args": [
|
||||
"-m",
|
||||
"esphome",
|
||||
"dashboard",
|
||||
"config/"
|
||||
],
|
||||
"problemMatcher": []
|
||||
},
|
||||
{
|
||||
"label": "clang-tidy",
|
||||
"type": "shell",
|
||||
"command": "./script/clang-tidy",
|
||||
"command": "${command:python.interpreterPath}",
|
||||
"args": [
|
||||
"./script/clang-tidy"
|
||||
],
|
||||
"problemMatcher": [
|
||||
{
|
||||
"owner": "clang-tidy",
|
||||
@@ -27,6 +36,24 @@
|
||||
]
|
||||
}
|
||||
]
|
||||
},
|
||||
{
|
||||
"label": "Generate proto files",
|
||||
"type": "shell",
|
||||
"command": "${command:python.interpreterPath}",
|
||||
"args": [
|
||||
"./script/api_protobuf/api_protobuf.py"
|
||||
],
|
||||
"group": {
|
||||
"kind": "build",
|
||||
"isDefault": true
|
||||
},
|
||||
"presentation": {
|
||||
"reveal": "never",
|
||||
"close": true,
|
||||
"panel": "new"
|
||||
},
|
||||
"problemMatcher": []
|
||||
}
|
||||
]
|
||||
}
|
||||
|
197
CODEOWNERS
197
CODEOWNERS
@@ -11,75 +11,166 @@ esphome/*.py @esphome/core
|
||||
esphome/core/* @esphome/core
|
||||
|
||||
# Integrations
|
||||
esphome/components/a01nyub/* @MrSuicideParrot
|
||||
esphome/components/absolute_humidity/* @DAVe3283
|
||||
esphome/components/ac_dimmer/* @glmnet
|
||||
esphome/components/adc/* @esphome/core
|
||||
esphome/components/adc128s102/* @DeerMaximum
|
||||
esphome/components/addressable_light/* @justfalter
|
||||
esphome/components/ade7953/* @angelnu
|
||||
esphome/components/ade7953_i2c/* @angelnu
|
||||
esphome/components/ade7953_spi/* @angelnu
|
||||
esphome/components/airthings_ble/* @jeromelaban
|
||||
esphome/components/airthings_wave_base/* @jeromelaban @kpfleming @ncareau
|
||||
esphome/components/airthings_wave_mini/* @ncareau
|
||||
esphome/components/airthings_wave_plus/* @jeromelaban
|
||||
esphome/components/alarm_control_panel/* @grahambrown11
|
||||
esphome/components/alpha3/* @jan-hofmeier
|
||||
esphome/components/am43/* @buxtronix
|
||||
esphome/components/am43/cover/* @buxtronix
|
||||
esphome/components/am43/sensor/* @buxtronix
|
||||
esphome/components/analog_threshold/* @ianchi
|
||||
esphome/components/animation/* @syndlex
|
||||
esphome/components/anova/* @buxtronix
|
||||
esphome/components/api/* @OttoWinter
|
||||
esphome/components/as7341/* @mrgnr
|
||||
esphome/components/async_tcp/* @OttoWinter
|
||||
esphome/components/atc_mithermometer/* @ahpohl
|
||||
esphome/components/atm90e26/* @danieltwagner
|
||||
esphome/components/b_parasite/* @rbaron
|
||||
esphome/components/ballu/* @bazuchan
|
||||
esphome/components/bang_bang/* @OttoWinter
|
||||
esphome/components/bedjet/* @jhansche
|
||||
esphome/components/bedjet/climate/* @jhansche
|
||||
esphome/components/bedjet/fan/* @jhansche
|
||||
esphome/components/bh1750/* @OttoWinter
|
||||
esphome/components/binary_sensor/* @esphome/core
|
||||
esphome/components/bk72xx/* @kuba2k2
|
||||
esphome/components/bl0939/* @ziceva
|
||||
esphome/components/bl0940/* @tobias-
|
||||
esphome/components/bl0942/* @dbuezas
|
||||
esphome/components/ble_client/* @buxtronix
|
||||
esphome/components/bluetooth_proxy/* @jesserockz
|
||||
esphome/components/bme680_bsec/* @trvrnrth
|
||||
esphome/components/bmi160/* @flaviut
|
||||
esphome/components/bmp3xx/* @martgras
|
||||
esphome/components/bmp581/* @kahrendt
|
||||
esphome/components/bp1658cj/* @Cossid
|
||||
esphome/components/bp5758d/* @Cossid
|
||||
esphome/components/button/* @esphome/core
|
||||
esphome/components/canbus/* @danielschramm @mvturnho
|
||||
esphome/components/cap1188/* @mreditor97
|
||||
esphome/components/captive_portal/* @OttoWinter
|
||||
esphome/components/ccs811/* @habbie
|
||||
esphome/components/cd74hc4067/* @asoehlke
|
||||
esphome/components/climate/* @esphome/core
|
||||
esphome/components/climate_ir/* @glmnet
|
||||
esphome/components/color_temperature/* @jesserockz
|
||||
esphome/components/coolix/* @glmnet
|
||||
esphome/components/copy/* @OttoWinter
|
||||
esphome/components/cover/* @esphome/core
|
||||
esphome/components/cs5460a/* @balrog-kun
|
||||
esphome/components/cse7761/* @berfenger
|
||||
esphome/components/ct_clamp/* @jesserockz
|
||||
esphome/components/current_based/* @djwmarcx
|
||||
esphome/components/dac7678/* @NickB1
|
||||
esphome/components/daikin_brc/* @hagak
|
||||
esphome/components/daly_bms/* @s1lvi0
|
||||
esphome/components/dashboard_import/* @esphome/core
|
||||
esphome/components/debug/* @OttoWinter
|
||||
esphome/components/delonghi/* @grob6000
|
||||
esphome/components/dfplayer/* @glmnet
|
||||
esphome/components/dfrobot_sen0395/* @niklasweber
|
||||
esphome/components/dht/* @OttoWinter
|
||||
esphome/components/display_menu_base/* @numo68
|
||||
esphome/components/dps310/* @kbx81
|
||||
esphome/components/ds1307/* @badbadc0ffee
|
||||
esphome/components/dsmr/* @glmnet @zuidwijk
|
||||
esphome/components/duty_time/* @dudanov
|
||||
esphome/components/ee895/* @Stock-M
|
||||
esphome/components/ektf2232/* @jesserockz
|
||||
esphome/components/emc2101/* @ellull
|
||||
esphome/components/ens210/* @itn3rd77
|
||||
esphome/components/esp32/* @esphome/core
|
||||
esphome/components/esp32_ble/* @jesserockz
|
||||
esphome/components/esp32_ble_server/* @jesserockz
|
||||
esphome/components/esp32_ble/* @Rapsssito @jesserockz
|
||||
esphome/components/esp32_ble_client/* @jesserockz
|
||||
esphome/components/esp32_ble_server/* @Rapsssito @clydebarrow @jesserockz
|
||||
esphome/components/esp32_camera_web_server/* @ayufan
|
||||
esphome/components/esp32_can/* @Sympatron
|
||||
esphome/components/esp32_improv/* @jesserockz
|
||||
esphome/components/esp32_rmt_led_strip/* @jesserockz
|
||||
esphome/components/esp8266/* @esphome/core
|
||||
esphome/components/ethernet_info/* @gtjadsonsantos
|
||||
esphome/components/exposure_notifications/* @OttoWinter
|
||||
esphome/components/ezo/* @ssieb
|
||||
esphome/components/ezo_pmp/* @carlos-sarmiento
|
||||
esphome/components/factory_reset/* @anatoly-savchenkov
|
||||
esphome/components/fastled_base/* @OttoWinter
|
||||
esphome/components/feedback/* @ianchi
|
||||
esphome/components/fingerprint_grow/* @OnFreund @loongyh
|
||||
esphome/components/fs3000/* @kahrendt
|
||||
esphome/components/gcja5/* @gcormier
|
||||
esphome/components/globals/* @esphome/core
|
||||
esphome/components/gp8403/* @jesserockz
|
||||
esphome/components/gpio/* @esphome/core
|
||||
esphome/components/gps/* @coogle
|
||||
esphome/components/graph/* @synco
|
||||
esphome/components/gree/* @orestismers
|
||||
esphome/components/grove_tb6612fng/* @max246
|
||||
esphome/components/growatt_solar/* @leeuwte
|
||||
esphome/components/haier/* @paveldn
|
||||
esphome/components/havells_solar/* @sourabhjaiswal
|
||||
esphome/components/hbridge/fan/* @WeekendWarrior
|
||||
esphome/components/hbridge/light/* @DotNetDann
|
||||
esphome/components/heatpumpir/* @rob-deutsch
|
||||
esphome/components/hitachi_ac424/* @sourabhjaiswal
|
||||
esphome/components/hm3301/* @freekode
|
||||
esphome/components/homeassistant/* @OttoWinter
|
||||
esphome/components/honeywellabp/* @RubyBailey
|
||||
esphome/components/honeywellabp2_i2c/* @jpfaff
|
||||
esphome/components/host/* @esphome/core
|
||||
esphome/components/hrxl_maxsonar_wr/* @netmikey
|
||||
esphome/components/hte501/* @Stock-M
|
||||
esphome/components/hydreon_rgxx/* @functionpointer
|
||||
esphome/components/hyt271/* @Philippe12
|
||||
esphome/components/i2c/* @esphome/core
|
||||
esphome/components/improv/* @jesserockz
|
||||
esphome/components/i2s_audio/* @jesserockz
|
||||
esphome/components/i2s_audio/media_player/* @jesserockz
|
||||
esphome/components/i2s_audio/microphone/* @jesserockz
|
||||
esphome/components/i2s_audio/speaker/* @jesserockz
|
||||
esphome/components/iaqcore/* @yozik04
|
||||
esphome/components/ili9xxx/* @clydebarrow @nielsnl68
|
||||
esphome/components/improv_base/* @esphome/core
|
||||
esphome/components/improv_serial/* @esphome/core
|
||||
esphome/components/ina260/* @mreditor97
|
||||
esphome/components/inkbird_ibsth1_mini/* @fkirill
|
||||
esphome/components/inkplate6/* @jesserockz
|
||||
esphome/components/integration/* @OttoWinter
|
||||
esphome/components/internal_temperature/* @Mat931
|
||||
esphome/components/interval/* @esphome/core
|
||||
esphome/components/json/* @OttoWinter
|
||||
esphome/components/kalman_combinator/* @Cat-Ion
|
||||
esphome/components/key_collector/* @ssieb
|
||||
esphome/components/key_provider/* @ssieb
|
||||
esphome/components/kuntze/* @ssieb
|
||||
esphome/components/lcd_menu/* @numo68
|
||||
esphome/components/ld2410/* @regevbr @sebcaps
|
||||
esphome/components/ld2420/* @descipher
|
||||
esphome/components/ledc/* @OttoWinter
|
||||
esphome/components/libretiny/* @kuba2k2
|
||||
esphome/components/libretiny_pwm/* @kuba2k2
|
||||
esphome/components/light/* @esphome/core
|
||||
esphome/components/lightwaverf/* @max246
|
||||
esphome/components/lilygo_t5_47/touchscreen/* @jesserockz
|
||||
esphome/components/lock/* @esphome/core
|
||||
esphome/components/logger/* @esphome/core
|
||||
esphome/components/ltr390/* @sjtrny
|
||||
esphome/components/matrix_keypad/* @ssieb
|
||||
esphome/components/max31865/* @DAVe3283
|
||||
esphome/components/max44009/* @berfenger
|
||||
esphome/components/max6956/* @looping40
|
||||
esphome/components/max7219digit/* @rspaargaren
|
||||
esphome/components/max9611/* @mckaymatthew
|
||||
esphome/components/mcp23008/* @jesserockz
|
||||
esphome/components/mcp23017/* @jesserockz
|
||||
esphome/components/mcp23s08/* @SenexCrenshaw @jesserockz
|
||||
@@ -88,18 +179,37 @@ esphome/components/mcp23x08_base/* @jesserockz
|
||||
esphome/components/mcp23x17_base/* @jesserockz
|
||||
esphome/components/mcp23xxx_base/* @jesserockz
|
||||
esphome/components/mcp2515/* @danielschramm @mvturnho
|
||||
esphome/components/mcp3204/* @rsumner
|
||||
esphome/components/mcp4728/* @berfenger
|
||||
esphome/components/mcp47a1/* @jesserockz
|
||||
esphome/components/mcp9600/* @mreditor97
|
||||
esphome/components/mcp9808/* @k7hpn
|
||||
esphome/components/md5/* @esphome/core
|
||||
esphome/components/mdns/* @esphome/core
|
||||
esphome/components/media_player/* @jesserockz
|
||||
esphome/components/micronova/* @jorre05
|
||||
esphome/components/microphone/* @jesserockz
|
||||
esphome/components/mics_4514/* @jesserockz
|
||||
esphome/components/midea/* @dudanov
|
||||
esphome/components/midea_ir/* @dudanov
|
||||
esphome/components/mitsubishi/* @RubyBailey
|
||||
esphome/components/mlx90393/* @functionpointer
|
||||
esphome/components/mlx90614/* @jesserockz
|
||||
esphome/components/mmc5603/* @benhoff
|
||||
esphome/components/mmc5983/* @agoode
|
||||
esphome/components/modbus_controller/* @martgras
|
||||
esphome/components/modbus_controller/binary_sensor/* @martgras
|
||||
esphome/components/modbus_controller/number/* @martgras
|
||||
esphome/components/modbus_controller/output/* @martgras
|
||||
esphome/components/modbus_controller/select/* @martgras @stegm
|
||||
esphome/components/modbus_controller/sensor/* @martgras
|
||||
esphome/components/modbus_controller/switch/* @martgras
|
||||
esphome/components/modbus_controller/text_sensor/* @martgras
|
||||
esphome/components/mopeka_ble/* @Fabian-Schmidt @spbrogan
|
||||
esphome/components/mopeka_pro_check/* @spbrogan
|
||||
esphome/components/mopeka_std_check/* @Fabian-Schmidt
|
||||
esphome/components/mpl3115a2/* @kbickar
|
||||
esphome/components/mpu6886/* @fabaff
|
||||
esphome/components/network/* @esphome/core
|
||||
esphome/components/nextion/* @senexcrenshaw
|
||||
esphome/components/nextion/binary_sensor/* @senexcrenshaw
|
||||
@@ -107,42 +217,79 @@ esphome/components/nextion/sensor/* @senexcrenshaw
|
||||
esphome/components/nextion/switch/* @senexcrenshaw
|
||||
esphome/components/nextion/text_sensor/* @senexcrenshaw
|
||||
esphome/components/nfc/* @jesserockz
|
||||
esphome/components/noblex/* @AGalfra
|
||||
esphome/components/number/* @esphome/core
|
||||
esphome/components/ota/* @esphome/core
|
||||
esphome/components/output/* @esphome/core
|
||||
esphome/components/pca6416a/* @Mat931
|
||||
esphome/components/pca9554/* @clydebarrow @hwstar
|
||||
esphome/components/pcf85063/* @brogon
|
||||
esphome/components/pcf8563/* @KoenBreeman
|
||||
esphome/components/pid/* @OttoWinter
|
||||
esphome/components/pipsolar/* @andreashergert1984
|
||||
esphome/components/pm1006/* @habbie
|
||||
esphome/components/pmsa003i/* @sjtrny
|
||||
esphome/components/pmwcs3/* @SeByDocKy
|
||||
esphome/components/pn532/* @OttoWinter @jesserockz
|
||||
esphome/components/pn532_i2c/* @OttoWinter @jesserockz
|
||||
esphome/components/pn532_spi/* @OttoWinter @jesserockz
|
||||
esphome/components/power_supply/* @esphome/core
|
||||
esphome/components/preferences/* @esphome/core
|
||||
esphome/components/pulse_meter/* @stevebaxter
|
||||
esphome/components/psram/* @esphome/core
|
||||
esphome/components/pulse_meter/* @TrentHouliston @cstaahl @stevebaxter
|
||||
esphome/components/pvvx_mithermometer/* @pasiz
|
||||
esphome/components/qmp6988/* @andrewpc
|
||||
esphome/components/qr_code/* @wjtje
|
||||
esphome/components/qwiic_pir/* @kahrendt
|
||||
esphome/components/radon_eye_ble/* @jeffeb3
|
||||
esphome/components/radon_eye_rd200/* @jeffeb3
|
||||
esphome/components/rc522/* @glmnet
|
||||
esphome/components/rc522_i2c/* @glmnet
|
||||
esphome/components/rc522_spi/* @glmnet
|
||||
esphome/components/resistance_sampler/* @jesserockz
|
||||
esphome/components/restart/* @esphome/core
|
||||
esphome/components/rf_bridge/* @jesserockz
|
||||
esphome/components/rgbct/* @jesserockz
|
||||
esphome/components/rp2040/* @jesserockz
|
||||
esphome/components/rp2040_pio_led_strip/* @Papa-DMan
|
||||
esphome/components/rp2040_pwm/* @jesserockz
|
||||
esphome/components/rtl87xx/* @kuba2k2
|
||||
esphome/components/rtttl/* @glmnet
|
||||
esphome/components/safe_mode/* @paulmonigatti
|
||||
esphome/components/scd4x/* @sjtrny
|
||||
esphome/components/safe_mode/* @jsuanet @paulmonigatti
|
||||
esphome/components/scd4x/* @martgras @sjtrny
|
||||
esphome/components/script/* @esphome/core
|
||||
esphome/components/sdm_meter/* @jesserockz @polyfaces
|
||||
esphome/components/sdp3x/* @Azimath
|
||||
esphome/components/selec_meter/* @sourabhjaiswal
|
||||
esphome/components/select/* @esphome/core
|
||||
esphome/components/sen0321/* @notjj
|
||||
esphome/components/sen21231/* @shreyaskarnik
|
||||
esphome/components/sen5x/* @martgras
|
||||
esphome/components/sensirion_common/* @martgras
|
||||
esphome/components/sensor/* @esphome/core
|
||||
esphome/components/sfa30/* @ghsensdev
|
||||
esphome/components/sgp40/* @SenexCrenshaw
|
||||
esphome/components/sgp4x/* @SenexCrenshaw @martgras
|
||||
esphome/components/shelly_dimmer/* @edge90 @rnauber
|
||||
esphome/components/sht4x/* @sjtrny
|
||||
esphome/components/shutdown/* @esphome/core
|
||||
esphome/components/shutdown/* @esphome/core @jsuanet
|
||||
esphome/components/sigma_delta_output/* @Cat-Ion
|
||||
esphome/components/sim800l/* @glmnet
|
||||
esphome/components/sm2135/* @BoukeHaarsma23
|
||||
esphome/components/sm10bit_base/* @Cossid
|
||||
esphome/components/sm2135/* @BoukeHaarsma23 @dd32 @matika77
|
||||
esphome/components/sm2235/* @Cossid
|
||||
esphome/components/sm2335/* @Cossid
|
||||
esphome/components/sml/* @alengwenus
|
||||
esphome/components/smt100/* @piechade
|
||||
esphome/components/sn74hc165/* @jesserockz
|
||||
esphome/components/socket/* @esphome/core
|
||||
esphome/components/spi/* @esphome/core
|
||||
esphome/components/sonoff_d1/* @anatoly-savchenkov
|
||||
esphome/components/speaker/* @jesserockz
|
||||
esphome/components/spi/* @clydebarrow @esphome/core
|
||||
esphome/components/spi_device/* @clydebarrow
|
||||
esphome/components/spi_led_strip/* @clydebarrow
|
||||
esphome/components/sprinkler/* @kbx81
|
||||
esphome/components/sps30/* @martgras
|
||||
esphome/components/ssd1322_base/* @kbx81
|
||||
esphome/components/ssd1322_spi/* @kbx81
|
||||
esphome/components/ssd1325_base/* @kbx81
|
||||
@@ -163,25 +310,55 @@ esphome/components/switch/* @esphome/core
|
||||
esphome/components/t6615/* @tylermenezes
|
||||
esphome/components/tca9548a/* @andreashergert1984
|
||||
esphome/components/tcl112/* @glmnet
|
||||
esphome/components/tee501/* @Stock-M
|
||||
esphome/components/teleinfo/* @0hax
|
||||
esphome/components/template/alarm_control_panel/* @grahambrown11
|
||||
esphome/components/text/* @mauritskorse
|
||||
esphome/components/thermostat/* @kbx81
|
||||
esphome/components/time/* @OttoWinter
|
||||
esphome/components/tlc5947/* @rnauber
|
||||
esphome/components/tm1621/* @Philippe12
|
||||
esphome/components/tm1637/* @glmnet
|
||||
esphome/components/tm1638/* @skykingjwc
|
||||
esphome/components/tm1651/* @freekode
|
||||
esphome/components/tmp102/* @timsavage
|
||||
esphome/components/tmp1075/* @sybrenstuvel
|
||||
esphome/components/tmp117/* @Azimath
|
||||
esphome/components/tof10120/* @wstrzalka
|
||||
esphome/components/toshiba/* @kbx81
|
||||
esphome/components/touchscreen/* @jesserockz
|
||||
esphome/components/tsl2591/* @wjcarpenter
|
||||
esphome/components/tt21100/* @kroimon
|
||||
esphome/components/tuya/binary_sensor/* @jesserockz
|
||||
esphome/components/tuya/climate/* @jesserockz
|
||||
esphome/components/tuya/number/* @frankiboy1
|
||||
esphome/components/tuya/select/* @bearpawmaxim
|
||||
esphome/components/tuya/sensor/* @jesserockz
|
||||
esphome/components/tuya/switch/* @jesserockz
|
||||
esphome/components/tuya/text_sensor/* @dentra
|
||||
esphome/components/uart/* @esphome/core
|
||||
esphome/components/uart/button/* @ssieb
|
||||
esphome/components/ufire_ec/* @pvizeli
|
||||
esphome/components/ufire_ise/* @pvizeli
|
||||
esphome/components/ultrasonic/* @OttoWinter
|
||||
esphome/components/vbus/* @ssieb
|
||||
esphome/components/version/* @esphome/core
|
||||
esphome/components/voice_assistant/* @jesserockz
|
||||
esphome/components/wake_on_lan/* @willwill2will54
|
||||
esphome/components/web_server_base/* @OttoWinter
|
||||
esphome/components/web_server_idf/* @dentra
|
||||
esphome/components/whirlpool/* @glmnet
|
||||
esphome/components/whynter/* @aeonsablaze
|
||||
esphome/components/wiegand/* @ssieb
|
||||
esphome/components/wireguard/* @droscy @lhoracek @thomas0bernard
|
||||
esphome/components/wl_134/* @hobbypunk90
|
||||
esphome/components/x9c/* @EtienneMD
|
||||
esphome/components/xgzp68xx/* @gcormier
|
||||
esphome/components/xiaomi_lywsd03mmc/* @ahpohl
|
||||
esphome/components/xiaomi_mhoc303/* @drug123
|
||||
esphome/components/xiaomi_mhoc401/* @vevsvevs
|
||||
esphome/components/xpt2046/* @numo68
|
||||
esphome/components/xiaomi_rtcgq02lm/* @jesserockz
|
||||
esphome/components/xl9535/* @mreditor97
|
||||
esphome/components/xpt2046/* @nielsnl68 @numo68
|
||||
esphome/components/zhlt01/* @cfeenstra1024
|
||||
esphome/components/zio_ultrasonic/* @kahrendt
|
||||
|
@@ -5,7 +5,7 @@ For a detailed guide, please see https://esphome.io/guides/contributing.html#con
|
||||
Things to note when contributing:
|
||||
|
||||
- Please test your changes :)
|
||||
- If a new feature is added or an existing user-facing feature is changed, you should also
|
||||
- If a new feature is added or an existing user-facing feature is changed, you should also
|
||||
update the [docs](https://github.com/esphome/esphome-docs). See [contributing to esphome-docs](https://esphome.io/guides/contributing.html#contributing-to-esphomedocs)
|
||||
for more information.
|
||||
- Please also update the tests in the `tests/` folder. You can do so by just adding a line in one of the YAML files
|
||||
|
@@ -1,7 +1,6 @@
|
||||
include LICENSE
|
||||
include README.md
|
||||
include requirements.txt
|
||||
include esphome/dashboard/templates/*.html
|
||||
recursive-include esphome/dashboard/static *.ico *.js *.css *.woff* LICENSE
|
||||
recursive-include esphome *.cpp *.h *.tcc
|
||||
recursive-include esphome *.cpp *.h *.tcc *.c
|
||||
recursive-include esphome *.py.script
|
||||
recursive-include esphome LICENSE.txt
|
||||
|
@@ -5,30 +5,54 @@
|
||||
# One of "docker", "hassio"
|
||||
ARG BASEIMGTYPE=docker
|
||||
|
||||
FROM ghcr.io/hassio-addons/debian-base/amd64:5.1.0 AS base-hassio-amd64
|
||||
FROM ghcr.io/hassio-addons/debian-base/aarch64:5.1.0 AS base-hassio-arm64
|
||||
FROM ghcr.io/hassio-addons/debian-base/armv7:5.1.0 AS base-hassio-armv7
|
||||
FROM debian:bullseye-20210902-slim AS base-docker-amd64
|
||||
FROM debian:bullseye-20210902-slim AS base-docker-arm64
|
||||
FROM debian:bullseye-20210902-slim AS base-docker-armv7
|
||||
|
||||
# Use TARGETARCH/TARGETVARIANT defined by docker
|
||||
# https://docs.docker.com/engine/reference/builder/#automatic-platform-args-in-the-global-scope
|
||||
FROM base-${BASEIMGTYPE}-${TARGETARCH}${TARGETVARIANT} AS base
|
||||
# https://github.com/hassio-addons/addon-debian-base/releases
|
||||
FROM ghcr.io/hassio-addons/debian-base:7.2.0 AS base-hassio
|
||||
# https://hub.docker.com/_/debian?tab=tags&page=1&name=bookworm
|
||||
FROM debian:12.2-slim AS base-docker
|
||||
|
||||
FROM base-${BASEIMGTYPE} AS base
|
||||
|
||||
|
||||
ARG TARGETARCH
|
||||
ARG TARGETVARIANT
|
||||
|
||||
|
||||
# Note that --break-system-packages is used below because
|
||||
# https://peps.python.org/pep-0668/ added a safety check that prevents
|
||||
# installing packages with the same name as a system package. This is
|
||||
# not a problem for us because we are not concerned about overwriting
|
||||
# system packages because we are running in an isolated container.
|
||||
|
||||
RUN \
|
||||
apt-get update \
|
||||
# Use pinned versions so that we get updates with build caching
|
||||
&& apt-get install -y --no-install-recommends \
|
||||
python3=3.9.2-3 \
|
||||
python3-pip=20.3.4-4 \
|
||||
python3-setuptools=52.0.0-4 \
|
||||
python3-pil=8.1.2+dfsg-0.3 \
|
||||
python3-cryptography=3.3.2-1 \
|
||||
iputils-ping=3:20210202-1 \
|
||||
git=1:2.30.2-1 \
|
||||
curl=7.74.0-1.3+b1 \
|
||||
&& rm -rf \
|
||||
python3-pip=23.0.1+dfsg-1 \
|
||||
python3-setuptools=66.1.1-1 \
|
||||
python3-venv=3.11.2-1+b1 \
|
||||
python3-wheel=0.38.4-2 \
|
||||
iputils-ping=3:20221126-1 \
|
||||
git=1:2.39.2-1.1 \
|
||||
curl=7.88.1-10+deb12u4 \
|
||||
openssh-client=1:9.2p1-2+deb12u1 \
|
||||
python3-cffi=1.15.1-5 \
|
||||
libcairo2=1.16.0-7 \
|
||||
patch=2.7.6-7; \
|
||||
if [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]; then \
|
||||
apt-get install -y --no-install-recommends \
|
||||
build-essential=12.9 \
|
||||
python3-dev=3.11.2-1+b1 \
|
||||
zlib1g-dev=1:1.2.13.dfsg-1 \
|
||||
libjpeg-dev=1:2.1.5-2 \
|
||||
libfreetype-dev=2.12.1+dfsg-5 \
|
||||
libssl-dev=3.0.11-1~deb12u2 \
|
||||
libffi-dev=3.4.4-1 \
|
||||
cargo=0.66.0+ds1-1 \
|
||||
pkg-config=1.8.1-1 \
|
||||
gcc-arm-linux-gnueabihf=4:12.2.0-3; \
|
||||
fi; \
|
||||
rm -rf \
|
||||
/tmp/* \
|
||||
/var/{cache,log}/* \
|
||||
/var/lib/apt/lists/*
|
||||
@@ -39,32 +63,51 @@ ENV \
|
||||
# Store globally installed pio libs in /piolibs
|
||||
PLATFORMIO_GLOBALLIB_DIR=/piolibs
|
||||
|
||||
# Support legacy binaries on Debian multiarch system. There is no "correct" way
|
||||
# to do this, other than using properly built toolchains...
|
||||
# See: https://unix.stackexchange.com/questions/553743/correct-way-to-add-lib-ld-linux-so-3-in-debian
|
||||
RUN \
|
||||
if [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]; then \
|
||||
ln -s /lib/arm-linux-gnueabihf/ld-linux.so.3 /lib/ld-linux.so.3; \
|
||||
fi
|
||||
|
||||
RUN \
|
||||
# Ubuntu python3-pip is missing wheel
|
||||
pip3 install --no-cache-dir \
|
||||
wheel==0.36.2 \
|
||||
platformio==5.2.0 \
|
||||
if [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]; then \
|
||||
export PIP_EXTRA_INDEX_URL="https://www.piwheels.org/simple"; \
|
||||
fi; \
|
||||
pip3 install \
|
||||
--break-system-packages --no-cache-dir \
|
||||
platformio==6.1.11 \
|
||||
# Change some platformio settings
|
||||
&& platformio settings set enable_telemetry No \
|
||||
&& platformio settings set check_libraries_interval 1000000 \
|
||||
&& platformio settings set check_platformio_interval 1000000 \
|
||||
&& platformio settings set check_platforms_interval 1000000 \
|
||||
&& mkdir -p /piolibs
|
||||
|
||||
|
||||
# First install requirements to leverage caching when requirements don't change
|
||||
# tmpfs is for https://github.com/rust-lang/cargo/issues/8719
|
||||
|
||||
COPY requirements.txt requirements_optional.txt script/platformio_install_deps.py platformio.ini /
|
||||
RUN --mount=type=tmpfs,target=/root/.cargo if [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]; then \
|
||||
export PIP_EXTRA_INDEX_URL="https://www.piwheels.org/simple"; \
|
||||
fi; \
|
||||
CARGO_REGISTRIES_CRATES_IO_PROTOCOL=sparse CARGO_HOME=/root/.cargo \
|
||||
pip3 install \
|
||||
--break-system-packages --no-cache-dir -r /requirements.txt -r /requirements_optional.txt \
|
||||
&& /platformio_install_deps.py /platformio.ini --libraries
|
||||
|
||||
|
||||
# ======================= docker-type image =======================
|
||||
FROM base AS docker
|
||||
|
||||
# First install requirements to leverage caching when requirements don't change
|
||||
COPY requirements.txt requirements_optional.txt docker/platformio_install_deps.py platformio.ini /
|
||||
RUN \
|
||||
pip3 install --no-cache-dir -r /requirements.txt -r /requirements_optional.txt \
|
||||
&& /platformio_install_deps.py /platformio.ini
|
||||
|
||||
# Copy esphome and install
|
||||
COPY . /esphome
|
||||
RUN pip3 install --no-cache-dir -e /esphome
|
||||
RUN if [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]; then \
|
||||
export PIP_EXTRA_INDEX_URL="https://www.piwheels.org/simple"; \
|
||||
fi; \
|
||||
pip3 install \
|
||||
--break-system-packages --no-cache-dir --no-use-pep517 -e /esphome
|
||||
|
||||
# Settings for dashboard
|
||||
ENV USERNAME="" PASSWORD=""
|
||||
@@ -72,6 +115,10 @@ ENV USERNAME="" PASSWORD=""
|
||||
# Expose the dashboard to Docker
|
||||
EXPOSE 6052
|
||||
|
||||
# Run healthcheck (heartbeat)
|
||||
HEALTHCHECK --interval=30s --timeout=30s \
|
||||
CMD curl --fail http://localhost:6052/version -A "HealthCheck" || exit 1
|
||||
|
||||
COPY docker/docker_entrypoint.sh /entrypoint.sh
|
||||
|
||||
# The directory the user should mount their configuration files to
|
||||
@@ -93,7 +140,7 @@ RUN \
|
||||
apt-get update \
|
||||
# Use pinned versions so that we get updates with build caching
|
||||
&& apt-get install -y --no-install-recommends \
|
||||
nginx=1.18.0-6.1 \
|
||||
nginx-light=1.22.1-9 \
|
||||
&& rm -rf \
|
||||
/tmp/* \
|
||||
/var/{cache,log}/* \
|
||||
@@ -102,17 +149,15 @@ RUN \
|
||||
ARG BUILD_VERSION=dev
|
||||
|
||||
# Copy root filesystem
|
||||
COPY docker/hassio-rootfs/ /
|
||||
|
||||
# First install requirements to leverage caching when requirements don't change
|
||||
COPY requirements.txt requirements_optional.txt docker/platformio_install_deps.py platformio.ini /
|
||||
RUN \
|
||||
pip3 install --no-cache-dir -r /requirements.txt -r /requirements_optional.txt \
|
||||
&& /platformio_install_deps.py /platformio.ini
|
||||
COPY docker/ha-addon-rootfs/ /
|
||||
|
||||
# Copy esphome and install
|
||||
COPY . /esphome
|
||||
RUN pip3 install --no-cache-dir -e /esphome
|
||||
RUN if [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]; then \
|
||||
export PIP_EXTRA_INDEX_URL="https://www.piwheels.org/simple"; \
|
||||
fi; \
|
||||
pip3 install \
|
||||
--break-system-packages --no-cache-dir --no-use-pep517 -e /esphome
|
||||
|
||||
# Labels
|
||||
LABEL \
|
||||
@@ -135,22 +180,24 @@ RUN \
|
||||
apt-get update \
|
||||
# Use pinned versions so that we get updates with build caching
|
||||
&& apt-get install -y --no-install-recommends \
|
||||
clang-format-11=1:11.0.1-2 \
|
||||
clang-tidy-11=1:11.0.1-2 \
|
||||
clang-format-13=1:13.0.1-11+b2 \
|
||||
clang-tidy-14=1:14.0.6-12 \
|
||||
patch=2.7.6-7 \
|
||||
software-properties-common=0.96.20.2-2.1 \
|
||||
nano=5.4-2 \
|
||||
software-properties-common=0.99.30-4 \
|
||||
nano=7.2-1 \
|
||||
build-essential=12.9 \
|
||||
python3-dev=3.9.2-3 \
|
||||
python3-dev=3.11.2-1+b1 \
|
||||
&& rm -rf \
|
||||
/tmp/* \
|
||||
/var/{cache,log}/* \
|
||||
/var/lib/apt/lists/*
|
||||
|
||||
COPY requirements.txt requirements_optional.txt docker/platformio_install_deps.py platformio.ini /
|
||||
RUN \
|
||||
pip3 install --no-cache-dir -r /requirements.txt -r /requirements_optional.txt \
|
||||
&& /platformio_install_deps.py /platformio.ini
|
||||
COPY requirements_test.txt /
|
||||
RUN if [ "$TARGETARCH$TARGETVARIANT" = "armv7" ]; then \
|
||||
export PIP_EXTRA_INDEX_URL="https://www.piwheels.org/simple"; \
|
||||
fi; \
|
||||
pip3 install \
|
||||
--break-system-packages --no-cache-dir -r /requirements_test.txt
|
||||
|
||||
VOLUME ["/esphome"]
|
||||
WORKDIR /esphome
|
||||
|
@@ -8,31 +8,49 @@ import re
|
||||
import sys
|
||||
|
||||
|
||||
CHANNEL_DEV = 'dev'
|
||||
CHANNEL_BETA = 'beta'
|
||||
CHANNEL_RELEASE = 'release'
|
||||
CHANNEL_DEV = "dev"
|
||||
CHANNEL_BETA = "beta"
|
||||
CHANNEL_RELEASE = "release"
|
||||
CHANNELS = [CHANNEL_DEV, CHANNEL_BETA, CHANNEL_RELEASE]
|
||||
|
||||
ARCH_AMD64 = 'amd64'
|
||||
ARCH_ARMV7 = 'armv7'
|
||||
ARCH_AARCH64 = 'aarch64'
|
||||
ARCH_AMD64 = "amd64"
|
||||
ARCH_ARMV7 = "armv7"
|
||||
ARCH_AARCH64 = "aarch64"
|
||||
ARCHS = [ARCH_AMD64, ARCH_ARMV7, ARCH_AARCH64]
|
||||
|
||||
TYPE_DOCKER = 'docker'
|
||||
TYPE_HA_ADDON = 'ha-addon'
|
||||
TYPE_LINT = 'lint'
|
||||
TYPE_DOCKER = "docker"
|
||||
TYPE_HA_ADDON = "ha-addon"
|
||||
TYPE_LINT = "lint"
|
||||
TYPES = [TYPE_DOCKER, TYPE_HA_ADDON, TYPE_LINT]
|
||||
|
||||
|
||||
parser = argparse.ArgumentParser()
|
||||
parser.add_argument("--tag", type=str, required=True, help="The main docker tag to push to. If a version number also adds latest and/or beta tag")
|
||||
parser.add_argument("--arch", choices=ARCHS, required=False, help="The architecture to build for")
|
||||
parser.add_argument("--build-type", choices=TYPES, required=True, help="The type of build to run")
|
||||
parser.add_argument("--dry-run", action="store_true", help="Don't run any commands, just print them")
|
||||
subparsers = parser.add_subparsers(help="Action to perform", dest="command", required=True)
|
||||
parser.add_argument(
|
||||
"--tag",
|
||||
type=str,
|
||||
required=True,
|
||||
help="The main docker tag to push to. If a version number also adds latest and/or beta tag",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--arch", choices=ARCHS, required=False, help="The architecture to build for"
|
||||
)
|
||||
parser.add_argument(
|
||||
"--build-type", choices=TYPES, required=True, help="The type of build to run"
|
||||
)
|
||||
parser.add_argument(
|
||||
"--dry-run", action="store_true", help="Don't run any commands, just print them"
|
||||
)
|
||||
subparsers = parser.add_subparsers(
|
||||
help="Action to perform", dest="command", required=True
|
||||
)
|
||||
build_parser = subparsers.add_parser("build", help="Build the image")
|
||||
build_parser.add_argument("--push", help="Also push the images", action="store_true")
|
||||
manifest_parser = subparsers.add_parser("manifest", help="Create a manifest from already pushed images")
|
||||
build_parser.add_argument(
|
||||
"--load", help="Load the docker image locally", action="store_true"
|
||||
)
|
||||
manifest_parser = subparsers.add_parser(
|
||||
"manifest", help="Create a manifest from already pushed images"
|
||||
)
|
||||
|
||||
|
||||
@dataclass(frozen=True)
|
||||
@@ -48,7 +66,7 @@ class DockerParams:
|
||||
prefix = {
|
||||
TYPE_DOCKER: "esphome/esphome",
|
||||
TYPE_HA_ADDON: "esphome/esphome-hassio",
|
||||
TYPE_LINT: "esphome/esphome-lint"
|
||||
TYPE_LINT: "esphome/esphome-lint",
|
||||
}[build_type]
|
||||
build_to = f"{prefix}-{arch}"
|
||||
baseimgtype = {
|
||||
@@ -87,10 +105,12 @@ def main():
|
||||
sys.exit(1)
|
||||
|
||||
# detect channel from tag
|
||||
match = re.match(r'^\d+\.\d+(?:\.\d+)?(b\d+)?$', args.tag)
|
||||
match = re.match(r"^(\d+\.\d+)(?:\.\d+)?(b\d+)?$", args.tag)
|
||||
major_minor_version = None
|
||||
if match is None:
|
||||
channel = CHANNEL_DEV
|
||||
elif match.group(1) is None:
|
||||
elif match.group(2) is None:
|
||||
major_minor_version = match.group(1)
|
||||
channel = CHANNEL_RELEASE
|
||||
else:
|
||||
channel = CHANNEL_BETA
|
||||
@@ -105,6 +125,11 @@ def main():
|
||||
tags_to_push.append("beta")
|
||||
tags_to_push.append("latest")
|
||||
|
||||
# Compatibility with HA tags
|
||||
if major_minor_version:
|
||||
tags_to_push.append("stable")
|
||||
tags_to_push.append(major_minor_version)
|
||||
|
||||
if args.command == "build":
|
||||
# 1. pull cache image
|
||||
params = DockerParams.for_type_arch(args.build_type, args.arch)
|
||||
@@ -120,18 +145,28 @@ def main():
|
||||
|
||||
# 3. build
|
||||
cmd = [
|
||||
"docker", "buildx", "build",
|
||||
"--build-arg", f"BASEIMGTYPE={params.baseimgtype}",
|
||||
"--build-arg", f"BUILD_VERSION={args.tag}",
|
||||
"--cache-from", f"type=registry,ref={cache_img}",
|
||||
"--file", "docker/Dockerfile",
|
||||
"--platform", params.platform,
|
||||
"--target", params.target,
|
||||
"docker",
|
||||
"buildx",
|
||||
"build",
|
||||
"--build-arg",
|
||||
f"BASEIMGTYPE={params.baseimgtype}",
|
||||
"--build-arg",
|
||||
f"BUILD_VERSION={args.tag}",
|
||||
"--cache-from",
|
||||
f"type=registry,ref={cache_img}",
|
||||
"--file",
|
||||
"docker/Dockerfile",
|
||||
"--platform",
|
||||
params.platform,
|
||||
"--target",
|
||||
params.target,
|
||||
]
|
||||
for img in imgs:
|
||||
cmd += ["--tag", img]
|
||||
if args.push:
|
||||
cmd += ["--push", "--cache-to", f"type=registry,ref={cache_img},mode=max"]
|
||||
if args.load:
|
||||
cmd += ["--load"]
|
||||
|
||||
run_command(*cmd, ".")
|
||||
elif args.command == "manifest":
|
||||
@@ -150,9 +185,7 @@ def main():
|
||||
run_command(*cmd)
|
||||
# 2. Push manifests
|
||||
for target in targets:
|
||||
run_command(
|
||||
"docker", "manifest", "push", target
|
||||
)
|
||||
run_command("docker", "manifest", "push", target)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
|
68
docker/generate_tags.py
Executable file
68
docker/generate_tags.py
Executable file
@@ -0,0 +1,68 @@
|
||||
#!/usr/bin/env python3
|
||||
import re
|
||||
import os
|
||||
import argparse
|
||||
import json
|
||||
|
||||
CHANNEL_DEV = "dev"
|
||||
CHANNEL_BETA = "beta"
|
||||
CHANNEL_RELEASE = "release"
|
||||
|
||||
parser = argparse.ArgumentParser()
|
||||
parser.add_argument(
|
||||
"--tag",
|
||||
type=str,
|
||||
required=True,
|
||||
help="The main docker tag to push to. If a version number also adds latest and/or beta tag",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--suffix",
|
||||
type=str,
|
||||
required=True,
|
||||
help="The suffix of the tag.",
|
||||
)
|
||||
|
||||
|
||||
def main():
|
||||
args = parser.parse_args()
|
||||
|
||||
# detect channel from tag
|
||||
match = re.match(r"^(\d+\.\d+)(?:\.\d+)?(b\d+)?$", args.tag)
|
||||
major_minor_version = None
|
||||
if match is None:
|
||||
channel = CHANNEL_DEV
|
||||
elif match.group(2) is None:
|
||||
major_minor_version = match.group(1)
|
||||
channel = CHANNEL_RELEASE
|
||||
else:
|
||||
channel = CHANNEL_BETA
|
||||
|
||||
tags_to_push = [args.tag]
|
||||
if channel == CHANNEL_DEV:
|
||||
tags_to_push.append("dev")
|
||||
elif channel == CHANNEL_BETA:
|
||||
tags_to_push.append("beta")
|
||||
elif channel == CHANNEL_RELEASE:
|
||||
# Additionally push to beta
|
||||
tags_to_push.append("beta")
|
||||
tags_to_push.append("latest")
|
||||
|
||||
if major_minor_version:
|
||||
tags_to_push.append("stable")
|
||||
tags_to_push.append(major_minor_version)
|
||||
|
||||
suffix = f"-{args.suffix}" if args.suffix else ""
|
||||
|
||||
with open(os.environ["GITHUB_OUTPUT"], "w") as f:
|
||||
print(f"channel={channel}", file=f)
|
||||
print(f"image=esphome/esphome{suffix}", file=f)
|
||||
full_tags = []
|
||||
|
||||
for tag in tags_to_push:
|
||||
full_tags += [f"ghcr.io/esphome/esphome{suffix}:{tag}"]
|
||||
full_tags += [f"esphome/esphome{suffix}:{tag}"]
|
||||
print(f"tags={','.join(full_tags)}", file=f)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
@@ -1,9 +1,9 @@
|
||||
proxy_http_version 1.1;
|
||||
proxy_ignore_client_abort off;
|
||||
proxy_read_timeout 86400s;
|
||||
proxy_redirect off;
|
||||
proxy_send_timeout 86400s;
|
||||
proxy_max_temp_file_size 0;
|
||||
proxy_http_version 1.1;
|
||||
proxy_ignore_client_abort off;
|
||||
proxy_read_timeout 86400s;
|
||||
proxy_redirect off;
|
||||
proxy_send_timeout 86400s;
|
||||
proxy_max_temp_file_size 0;
|
||||
|
||||
proxy_set_header Accept-Encoding "";
|
||||
proxy_set_header Connection $connection_upgrade;
|
@@ -1,5 +1,7 @@
|
||||
root /dev/null;
|
||||
server_name $hostname;
|
||||
root /dev/null;
|
||||
server_name $hostname;
|
||||
|
||||
client_max_body_size 512m;
|
||||
|
||||
add_header X-Content-Type-Options nosniff;
|
||||
add_header X-XSS-Protection "1; mode=block";
|
@@ -0,0 +1,8 @@
|
||||
ssl_protocols TLSv1.2 TLSv1.3;
|
||||
ssl_prefer_server_ciphers off;
|
||||
ssl_ciphers ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-ECDSA-CHACHA20-POLY1305:ECDHE-RSA-CHACHA20-POLY1305:DHE-RSA-AES128-GCM-SHA256:DHE-RSA-AES256-GCM-SHA384;
|
||||
ssl_session_timeout 10m;
|
||||
ssl_session_cache shared:SSL:10m;
|
||||
ssl_session_tickets off;
|
||||
ssl_stapling on;
|
||||
ssl_stapling_verify on;
|
3
docker/ha-addon-rootfs/etc/nginx/includes/upstream.conf
Normal file
3
docker/ha-addon-rootfs/etc/nginx/includes/upstream.conf
Normal file
@@ -0,0 +1,3 @@
|
||||
upstream esphome {
|
||||
server unix:/var/run/esphome.sock;
|
||||
}
|
@@ -2,7 +2,6 @@ daemon off;
|
||||
user root;
|
||||
pid /var/run/nginx.pid;
|
||||
worker_processes 1;
|
||||
# Hass.io addon log
|
||||
error_log /proc/1/fd/1 error;
|
||||
events {
|
||||
worker_connections 1024;
|
||||
@@ -10,24 +9,22 @@ events {
|
||||
|
||||
http {
|
||||
include /etc/nginx/includes/mime.types;
|
||||
access_log stdout;
|
||||
default_type application/octet-stream;
|
||||
gzip on;
|
||||
keepalive_timeout 65;
|
||||
sendfile on;
|
||||
server_tokens off;
|
||||
|
||||
access_log off;
|
||||
default_type application/octet-stream;
|
||||
gzip on;
|
||||
keepalive_timeout 65;
|
||||
sendfile on;
|
||||
server_tokens off;
|
||||
|
||||
tcp_nodelay on;
|
||||
tcp_nopush on;
|
||||
|
||||
map $http_upgrade $connection_upgrade {
|
||||
default upgrade;
|
||||
'' close;
|
||||
}
|
||||
|
||||
# Use Hass.io supervisor as resolver
|
||||
resolver 172.30.32.2;
|
||||
|
||||
upstream esphome {
|
||||
server unix:/var/run/esphome.sock;
|
||||
}
|
||||
|
||||
include /etc/nginx/includes/upstream.conf;
|
||||
include /etc/nginx/servers/*.conf;
|
||||
}
|
1
docker/ha-addon-rootfs/etc/nginx/servers/.gitkeep
Normal file
1
docker/ha-addon-rootfs/etc/nginx/servers/.gitkeep
Normal file
@@ -0,0 +1 @@
|
||||
Without requirements or design, programming is the art of adding bugs to an empty text file. (Louis Srygley)
|
@@ -1,20 +1,26 @@
|
||||
server {
|
||||
listen %%port%% default_server ssl http2;
|
||||
{{ if not .ssl }}
|
||||
listen 6052 default_server;
|
||||
{{ else }}
|
||||
listen 6052 default_server ssl http2;
|
||||
{{ end }}
|
||||
|
||||
include /etc/nginx/includes/server_params.conf;
|
||||
include /etc/nginx/includes/proxy_params.conf;
|
||||
|
||||
{{ if .ssl }}
|
||||
include /etc/nginx/includes/ssl_params.conf;
|
||||
|
||||
ssl on;
|
||||
ssl_certificate /ssl/%%certfile%%;
|
||||
ssl_certificate_key /ssl/%%keyfile%%;
|
||||
|
||||
# Clear Hass.io Ingress header
|
||||
proxy_set_header X-Hassio-Ingress "";
|
||||
ssl_certificate /ssl/{{ .certfile }};
|
||||
ssl_certificate_key /ssl/{{ .keyfile }};
|
||||
|
||||
# Redirect http requests to https on the same port.
|
||||
# https://rageagainstshell.com/2016/11/redirect-http-to-https-on-the-same-port-in-nginx/
|
||||
error_page 497 https://$http_host$request_uri;
|
||||
{{ end }}
|
||||
|
||||
# Clear Home Assistant Ingress header
|
||||
proxy_set_header X-HA-Ingress "";
|
||||
|
||||
location / {
|
||||
proxy_pass http://esphome;
|
@@ -1,14 +1,16 @@
|
||||
server {
|
||||
listen %%interface%%:%%port%% default_server;
|
||||
listen 127.0.0.1:{{ .port }} default_server;
|
||||
listen {{ .interface }}:{{ .port }} default_server;
|
||||
|
||||
include /etc/nginx/includes/server_params.conf;
|
||||
include /etc/nginx/includes/proxy_params.conf;
|
||||
# Set Hass.io Ingress header
|
||||
proxy_set_header X-Hassio-Ingress "YES";
|
||||
|
||||
# Set Home Assistant Ingress header
|
||||
proxy_set_header X-HA-Ingress "YES";
|
||||
|
||||
location / {
|
||||
# Only allow from Hass.io supervisor
|
||||
allow 172.30.32.2;
|
||||
allow 127.0.0.1;
|
||||
deny all;
|
||||
|
||||
proxy_pass http://esphome;
|
32
docker/ha-addon-rootfs/etc/s6-overlay/s6-rc.d/discovery/run
Executable file
32
docker/ha-addon-rootfs/etc/s6-overlay/s6-rc.d/discovery/run
Executable file
@@ -0,0 +1,32 @@
|
||||
#!/command/with-contenv bashio
|
||||
# shellcheck shell=bash
|
||||
# ==============================================================================
|
||||
# Home Assistant Add-on: ESPHome
|
||||
# Sends discovery information to Home Assistant.
|
||||
# ==============================================================================
|
||||
declare config
|
||||
declare port
|
||||
|
||||
# We only disable it when disabled explicitly
|
||||
if bashio::config.false 'home_assistant_dashboard_integration';
|
||||
then
|
||||
bashio::log.info "Home Assistant discovery is disabled for this add-on."
|
||||
bashio::exit.ok
|
||||
fi
|
||||
|
||||
port=$(bashio::addon.ingress_port)
|
||||
|
||||
# Wait for NGINX to become available
|
||||
bashio::net.wait_for "${port}" "127.0.0.1" 300
|
||||
|
||||
config=$(\
|
||||
bashio::var.json \
|
||||
host "127.0.0.1" \
|
||||
port "^${port}" \
|
||||
)
|
||||
|
||||
if bashio::discovery "esphome" "${config}" > /dev/null; then
|
||||
bashio::log.info "Successfully send discovery information to Home Assistant."
|
||||
else
|
||||
bashio::log.error "Discovery message to Home Assistant failed!"
|
||||
fi
|
@@ -0,0 +1 @@
|
||||
oneshot
|
@@ -0,0 +1 @@
|
||||
/etc/s6-overlay/s6-rc.d/discovery/run
|
26
docker/ha-addon-rootfs/etc/s6-overlay/s6-rc.d/esphome/finish
Executable file
26
docker/ha-addon-rootfs/etc/s6-overlay/s6-rc.d/esphome/finish
Executable file
@@ -0,0 +1,26 @@
|
||||
#!/command/with-contenv bashio
|
||||
# shellcheck shell=bash
|
||||
# ==============================================================================
|
||||
# Home Assistant Community Add-on: ESPHome
|
||||
# Take down the S6 supervision tree when ESPHome dashboard fails
|
||||
# ==============================================================================
|
||||
declare exit_code
|
||||
readonly exit_code_container=$(</run/s6-linux-init-container-results/exitcode)
|
||||
readonly exit_code_service="${1}"
|
||||
readonly exit_code_signal="${2}"
|
||||
|
||||
bashio::log.info \
|
||||
"Service ESPHome dashboard exited with code ${exit_code_service}" \
|
||||
"(by signal ${exit_code_signal})"
|
||||
|
||||
if [[ "${exit_code_service}" -eq 256 ]]; then
|
||||
if [[ "${exit_code_container}" -eq 0 ]]; then
|
||||
echo $((128 + $exit_code_signal)) > /run/s6-linux-init-container-results/exitcode
|
||||
fi
|
||||
[[ "${exit_code_signal}" -eq 15 ]] && exec /run/s6/basedir/bin/halt
|
||||
elif [[ "${exit_code_service}" -ne 0 ]]; then
|
||||
if [[ "${exit_code_container}" -eq 0 ]]; then
|
||||
echo "${exit_code_service}" > /run/s6-linux-init-container-results/exitcode
|
||||
fi
|
||||
exec /run/s6/basedir/bin/halt
|
||||
fi
|
@@ -1,10 +1,19 @@
|
||||
#!/usr/bin/with-contenv bashio
|
||||
#!/command/with-contenv bashio
|
||||
# shellcheck shell=bash
|
||||
# ==============================================================================
|
||||
# Community Hass.io Add-ons: ESPHome
|
||||
# Runs the ESPHome dashboard
|
||||
# ==============================================================================
|
||||
readonly pio_cache_base=/data/cache/platformio
|
||||
|
||||
export ESPHOME_IS_HASSIO=true
|
||||
export ESPHOME_IS_HA_ADDON=true
|
||||
export PLATFORMIO_GLOBALLIB_DIR=/piolibs
|
||||
|
||||
# we can't set core_dir, because the settings file is stored in `core_dir/appstate.json`
|
||||
# setting `core_dir` would therefore prevent pio from accessing
|
||||
export PLATFORMIO_PLATFORMS_DIR="${pio_cache_base}/platforms"
|
||||
export PLATFORMIO_PACKAGES_DIR="${pio_cache_base}/packages"
|
||||
export PLATFORMIO_CACHE_DIR="${pio_cache_base}/cache"
|
||||
|
||||
if bashio::config.true 'leave_front_door_open'; then
|
||||
export DISABLE_HA_AUTHENTICATION=true
|
||||
@@ -22,14 +31,27 @@ if bashio::config.has_value 'relative_url'; then
|
||||
export ESPHOME_DASHBOARD_RELATIVE_URL=$(bashio::config 'relative_url')
|
||||
fi
|
||||
|
||||
pio_cache_base=/data/cache/platformio
|
||||
# we can't set core_dir, because the settings file is stored in `core_dir/appstate.json`
|
||||
# setting `core_dir` would therefore prevent pio from accessing
|
||||
export PLATFORMIO_PLATFORMS_DIR="${pio_cache_base}/platforms"
|
||||
export PLATFORMIO_PACKAGES_DIR="${pio_cache_base}/packages"
|
||||
export PLATFORMIO_CACHE_DIR="${pio_cache_base}/cache"
|
||||
if bashio::config.has_value 'default_compile_process_limit'; then
|
||||
export ESPHOME_DEFAULT_COMPILE_PROCESS_LIMIT=$(bashio::config 'default_compile_process_limit')
|
||||
else
|
||||
if grep -q 'Raspberry Pi 3' /proc/cpuinfo; then
|
||||
export ESPHOME_DEFAULT_COMPILE_PROCESS_LIMIT=1
|
||||
fi
|
||||
fi
|
||||
|
||||
export PLATFORMIO_GLOBALLIB_DIR=/piolibs
|
||||
mkdir -p "${pio_cache_base}"
|
||||
|
||||
mkdir -p /config/esphome
|
||||
|
||||
if bashio::fs.directory_exists '/config/esphome/.esphome'; then
|
||||
bashio::log.info "Migrating old .esphome directory..."
|
||||
if bashio::fs.file_exists '/config/esphome/.esphome/esphome.json'; then
|
||||
mv /config/esphome/.esphome/esphome.json /data/esphome.json
|
||||
fi
|
||||
mkdir -p "/data/storage"
|
||||
mv /config/esphome/.esphome/*.json /data/storage/ || true
|
||||
rm -rf /config/esphome/.esphome
|
||||
fi
|
||||
|
||||
bashio::log.info "Starting ESPHome dashboard..."
|
||||
exec esphome dashboard /config/esphome --socket /var/run/esphome.sock --hassio
|
||||
exec esphome dashboard /config/esphome --socket /var/run/esphome.sock --ha-addon
|
@@ -0,0 +1 @@
|
||||
longrun
|
27
docker/ha-addon-rootfs/etc/s6-overlay/s6-rc.d/init-nginx/run
Executable file
27
docker/ha-addon-rootfs/etc/s6-overlay/s6-rc.d/init-nginx/run
Executable file
@@ -0,0 +1,27 @@
|
||||
#!/command/with-contenv bashio
|
||||
# shellcheck shell=bash
|
||||
# ==============================================================================
|
||||
# Community Hass.io Add-ons: ESPHome
|
||||
# Configures NGINX for use with ESPHome
|
||||
# ==============================================================================
|
||||
mkdir -p /var/log/nginx
|
||||
|
||||
# Generate Ingress configuration
|
||||
bashio::var.json \
|
||||
interface "$(bashio::addon.ip_address)" \
|
||||
port "^$(bashio::addon.ingress_port)" \
|
||||
| tempio \
|
||||
-template /etc/nginx/templates/ingress.gtpl \
|
||||
-out /etc/nginx/servers/ingress.conf
|
||||
|
||||
# Generate direct access configuration, if enabled.
|
||||
if bashio::var.has_value "$(bashio::addon.port 6052)"; then
|
||||
bashio::config.require.ssl
|
||||
bashio::var.json \
|
||||
certfile "$(bashio::config 'certfile')" \
|
||||
keyfile "$(bashio::config 'keyfile')" \
|
||||
ssl "^$(bashio::config 'ssl')" \
|
||||
| tempio \
|
||||
-template /etc/nginx/templates/direct.gtpl \
|
||||
-out /etc/nginx/servers/direct.conf
|
||||
fi
|
@@ -0,0 +1 @@
|
||||
oneshot
|
@@ -0,0 +1 @@
|
||||
/etc/s6-overlay/s6-rc.d/init-nginx/run
|
25
docker/ha-addon-rootfs/etc/s6-overlay/s6-rc.d/nginx/finish
Executable file
25
docker/ha-addon-rootfs/etc/s6-overlay/s6-rc.d/nginx/finish
Executable file
@@ -0,0 +1,25 @@
|
||||
#!/command/with-contenv bashio
|
||||
# ==============================================================================
|
||||
# Community Hass.io Add-ons: ESPHome
|
||||
# Take down the S6 supervision tree when NGINX fails
|
||||
# ==============================================================================
|
||||
declare exit_code
|
||||
readonly exit_code_container=$(</run/s6-linux-init-container-results/exitcode)
|
||||
readonly exit_code_service="${1}"
|
||||
readonly exit_code_signal="${2}"
|
||||
|
||||
bashio::log.info \
|
||||
"Service NGINX exited with code ${exit_code_service}" \
|
||||
"(by signal ${exit_code_signal})"
|
||||
|
||||
if [[ "${exit_code_service}" -eq 256 ]]; then
|
||||
if [[ "${exit_code_container}" -eq 0 ]]; then
|
||||
echo $((128 + $exit_code_signal)) > /run/s6-linux-init-container-results/exitcode
|
||||
fi
|
||||
[[ "${exit_code_signal}" -eq 15 ]] && exec /run/s6/basedir/bin/halt
|
||||
elif [[ "${exit_code_service}" -ne 0 ]]; then
|
||||
if [[ "${exit_code_container}" -eq 0 ]]; then
|
||||
echo "${exit_code_service}" > /run/s6-linux-init-container-results/exitcode
|
||||
fi
|
||||
exec /run/s6/basedir/bin/halt
|
||||
fi
|
@@ -1,10 +1,11 @@
|
||||
#!/usr/bin/with-contenv bashio
|
||||
#!/command/with-contenv bashio
|
||||
# shellcheck shell=bash
|
||||
# ==============================================================================
|
||||
# Community Hass.io Add-ons: ESPHome
|
||||
# Runs the NGINX proxy
|
||||
# ==============================================================================
|
||||
|
||||
bashio::log.info "Waiting for dashboard to come up..."
|
||||
bashio::log.info "Waiting for ESPHome dashboard to come up..."
|
||||
|
||||
while [[ ! -S /var/run/esphome.sock ]]; do
|
||||
sleep 0.5
|
1
docker/ha-addon-rootfs/etc/s6-overlay/s6-rc.d/nginx/type
Normal file
1
docker/ha-addon-rootfs/etc/s6-overlay/s6-rc.d/nginx/type
Normal file
@@ -0,0 +1 @@
|
||||
longrun
|
@@ -1,41 +0,0 @@
|
||||
#!/usr/bin/with-contenv bashio
|
||||
# ==============================================================================
|
||||
# Community Hass.io Add-ons: ESPHome
|
||||
# This files check if all user configuration requirements are met
|
||||
# ==============================================================================
|
||||
|
||||
# Check SSL requirements, if enabled
|
||||
if bashio::config.true 'ssl'; then
|
||||
if ! bashio::config.has_value 'certfile'; then
|
||||
bashio::fatal 'SSL is enabled, but no certfile was specified.'
|
||||
bashio::exit.nok
|
||||
fi
|
||||
|
||||
if ! bashio::config.has_value 'keyfile'; then
|
||||
bashio::fatal 'SSL is enabled, but no keyfile was specified'
|
||||
bashio::exit.nok
|
||||
fi
|
||||
|
||||
|
||||
certfile="/ssl/$(bashio::config 'certfile')"
|
||||
keyfile="/ssl/$(bashio::config 'keyfile')"
|
||||
|
||||
if ! bashio::fs.file_exists "${certfile}"; then
|
||||
if ! bashio::fs.file_exists "${keyfile}"; then
|
||||
# Both files are missing, let's print a friendlier error message
|
||||
bashio::log.fatal 'You enabled encrypted connections using the "ssl": true option.'
|
||||
bashio::log.fatal "However, the SSL files '${certfile}' and '${keyfile}'"
|
||||
bashio::log.fatal "were not found. If you're using Hass.io on your local network and don't want"
|
||||
bashio::log.fatal 'to encrypt connections to the ESPHome dashboard, you can manually disable'
|
||||
bashio::log.fatal 'SSL by setting "ssl" to false."'
|
||||
bashio::exit.nok
|
||||
fi
|
||||
bashio::log.fatal "The configured certfile '${certfile}' was not found."
|
||||
bashio::exit.nok
|
||||
fi
|
||||
|
||||
if ! bashio::fs.file_exists "/ssl/$(bashio::config 'keyfile')"; then
|
||||
bashio::log.fatal "The configured keyfile '${keyfile}' was not found."
|
||||
bashio::exit.nok
|
||||
fi
|
||||
fi
|
@@ -1,34 +0,0 @@
|
||||
#!/usr/bin/with-contenv bashio
|
||||
# ==============================================================================
|
||||
# Community Hass.io Add-ons: ESPHome
|
||||
# Configures NGINX for use with ESPHome
|
||||
# ==============================================================================
|
||||
|
||||
declare certfile
|
||||
declare keyfile
|
||||
declare direct_port
|
||||
declare ingress_interface
|
||||
declare ingress_port
|
||||
|
||||
mkdir -p /var/log/nginx
|
||||
|
||||
direct_port=$(bashio::addon.port 6052)
|
||||
if bashio::var.has_value "${direct_port}"; then
|
||||
if bashio::config.true 'ssl'; then
|
||||
certfile=$(bashio::config 'certfile')
|
||||
keyfile=$(bashio::config 'keyfile')
|
||||
|
||||
mv /etc/nginx/servers/direct-ssl.disabled /etc/nginx/servers/direct.conf
|
||||
sed -i "s/%%certfile%%/${certfile}/g" /etc/nginx/servers/direct.conf
|
||||
sed -i "s/%%keyfile%%/${keyfile}/g" /etc/nginx/servers/direct.conf
|
||||
else
|
||||
mv /etc/nginx/servers/direct.disabled /etc/nginx/servers/direct.conf
|
||||
fi
|
||||
|
||||
sed -i "s/%%port%%/${direct_port}/g" /etc/nginx/servers/direct.conf
|
||||
fi
|
||||
|
||||
ingress_port=$(bashio::addon.ingress_port)
|
||||
ingress_interface=$(bashio::addon.ip_address)
|
||||
sed -i "s/%%port%%/${ingress_port}/g" /etc/nginx/servers/ingress.conf
|
||||
sed -i "s/%%interface%%/${ingress_interface}/g" /etc/nginx/servers/ingress.conf
|
@@ -1,9 +0,0 @@
|
||||
#!/usr/bin/with-contenv bashio
|
||||
# ==============================================================================
|
||||
# Community Hass.io Add-ons: ESPHome
|
||||
# This files creates all directories used by esphome
|
||||
# ==============================================================================
|
||||
|
||||
pio_cache_base=/data/cache/platformio
|
||||
|
||||
mkdir -p "${pio_cache_base}"
|
@@ -1,9 +0,0 @@
|
||||
ssl_protocols TLSv1.2;
|
||||
ssl_prefer_server_ciphers on;
|
||||
ssl_ciphers ECDHE-RSA-AES256-GCM-SHA512:DHE-RSA-AES256-GCM-SHA512:ECDHE-RSA-AES256-GCM-SHA384:DHE-RSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-SHA384:ECDHE-RSA-AES256-SHA:DHE-RSA-AES256-SHA;
|
||||
ssl_ecdh_curve secp384r1;
|
||||
ssl_session_timeout 10m;
|
||||
ssl_session_cache shared:SSL:10m;
|
||||
ssl_session_tickets off;
|
||||
ssl_stapling on;
|
||||
ssl_stapling_verify on;
|
@@ -1,12 +0,0 @@
|
||||
server {
|
||||
listen %%port%% default_server;
|
||||
|
||||
include /etc/nginx/includes/server_params.conf;
|
||||
include /etc/nginx/includes/proxy_params.conf;
|
||||
# Clear Hass.io Ingress header
|
||||
proxy_set_header X-Hassio-Ingress "";
|
||||
|
||||
location / {
|
||||
proxy_pass http://esphome;
|
||||
}
|
||||
}
|
@@ -1,9 +0,0 @@
|
||||
#!/usr/bin/execlineb -S0
|
||||
# ==============================================================================
|
||||
# Community Hass.io Add-ons: ESPHome
|
||||
# Take down the S6 supervision tree when ESPHome fails
|
||||
# ==============================================================================
|
||||
if -n { s6-test $# -ne 0 }
|
||||
if -n { s6-test ${1} -eq 256 }
|
||||
|
||||
s6-svscanctl -t /var/run/s6/services
|
@@ -1,9 +0,0 @@
|
||||
#!/usr/bin/execlineb -S0
|
||||
# ==============================================================================
|
||||
# Community Hass.io Add-ons: ESPHome
|
||||
# Take down the S6 supervision tree when NGINX fails
|
||||
# ==============================================================================
|
||||
if -n { s6-test $# -ne 0 }
|
||||
if -n { s6-test ${1} -eq 256 }
|
||||
|
||||
s6-svscanctl -t /var/run/s6/services
|
@@ -1,13 +0,0 @@
|
||||
#!/usr/bin/env python3
|
||||
# This script is used in the docker containers to preinstall
|
||||
# all platformio libraries in the global storage
|
||||
|
||||
import configparser
|
||||
import subprocess
|
||||
import sys
|
||||
|
||||
config = configparser.ConfigParser(inline_comment_prefixes=(';', ))
|
||||
config.read(sys.argv[1])
|
||||
libs = [x for x in config['common']['lib_deps'].splitlines() if len(x) != 0]
|
||||
|
||||
subprocess.check_call(['platformio', 'lib', '-g', 'install', *libs])
|
@@ -1,26 +1,43 @@
|
||||
# PYTHON_ARGCOMPLETE_OK
|
||||
import argparse
|
||||
import functools
|
||||
import logging
|
||||
import os
|
||||
import re
|
||||
import sys
|
||||
import time
|
||||
from datetime import datetime
|
||||
|
||||
import argcomplete
|
||||
|
||||
from esphome import const, writer, yaml_util
|
||||
import esphome.codegen as cg
|
||||
from esphome.config import iter_components, read_config, strip_default_ids
|
||||
from esphome.const import (
|
||||
ALLOWED_NAME_CHARS,
|
||||
CONF_BAUD_RATE,
|
||||
CONF_BROKER,
|
||||
CONF_DEASSERT_RTS_DTR,
|
||||
CONF_LOGGER,
|
||||
CONF_NAME,
|
||||
CONF_OTA,
|
||||
CONF_MQTT,
|
||||
CONF_MDNS,
|
||||
CONF_DISABLED,
|
||||
CONF_PASSWORD,
|
||||
CONF_PORT,
|
||||
CONF_ESPHOME,
|
||||
CONF_PLATFORMIO_OPTIONS,
|
||||
CONF_SUBSTITUTIONS,
|
||||
PLATFORM_BK72XX,
|
||||
PLATFORM_RTL87XX,
|
||||
PLATFORM_ESP32,
|
||||
PLATFORM_ESP8266,
|
||||
PLATFORM_RP2040,
|
||||
SECRETS_FILES,
|
||||
)
|
||||
from esphome.core import CORE, EsphomeError, coroutine
|
||||
from esphome.helpers import indent
|
||||
from esphome.helpers import indent, is_ip_address
|
||||
from esphome.util import (
|
||||
run_external_command,
|
||||
run_external_process,
|
||||
@@ -33,7 +50,7 @@ from esphome.log import color, setup_log, Fore
|
||||
_LOGGER = logging.getLogger(__name__)
|
||||
|
||||
|
||||
def choose_prompt(options):
|
||||
def choose_prompt(options, purpose: str = None):
|
||||
if not options:
|
||||
raise EsphomeError(
|
||||
"Found no valid options for upload/logging, please make sure relevant "
|
||||
@@ -44,7 +61,9 @@ def choose_prompt(options):
|
||||
if len(options) == 1:
|
||||
return options[0][1]
|
||||
|
||||
safe_print("Found multiple options, please choose one:")
|
||||
safe_print(
|
||||
f'Found multiple options{f" for {purpose}" if purpose else ""}, please choose one:'
|
||||
)
|
||||
for i, (desc, _) in enumerate(options):
|
||||
safe_print(f" [{i+1}] {desc}")
|
||||
|
||||
@@ -63,15 +82,19 @@ def choose_prompt(options):
|
||||
return options[opt - 1][1]
|
||||
|
||||
|
||||
def choose_upload_log_host(default, check_default, show_ota, show_mqtt, show_api):
|
||||
def choose_upload_log_host(
|
||||
default, check_default, show_ota, show_mqtt, show_api, purpose: str = None
|
||||
):
|
||||
options = []
|
||||
for port in get_serial_ports():
|
||||
options.append((f"{port.path} ({port.description})", port.path))
|
||||
if default == "SERIAL":
|
||||
return choose_prompt(options, purpose=purpose)
|
||||
if (show_ota and "ota" in CORE.config) or (show_api and "api" in CORE.config):
|
||||
options.append((f"Over The Air ({CORE.address})", CORE.address))
|
||||
if default == "OTA":
|
||||
return CORE.address
|
||||
if show_mqtt and "mqtt" in CORE.config:
|
||||
if show_mqtt and CONF_MQTT in CORE.config:
|
||||
options.append((f"MQTT ({CORE.config['mqtt'][CONF_BROKER]})", "MQTT"))
|
||||
if default == "OTA":
|
||||
return "MQTT"
|
||||
@@ -79,7 +102,7 @@ def choose_upload_log_host(default, check_default, show_ota, show_mqtt, show_api
|
||||
return default
|
||||
if check_default is not None and check_default in [opt[1] for opt in options]:
|
||||
return check_default
|
||||
return choose_prompt(options)
|
||||
return choose_prompt(options, purpose=purpose)
|
||||
|
||||
|
||||
def get_port_type(port):
|
||||
@@ -96,11 +119,11 @@ def run_miniterm(config, port):
|
||||
|
||||
if CONF_LOGGER not in config:
|
||||
_LOGGER.info("Logger is not enabled. Not starting UART logs.")
|
||||
return
|
||||
return 1
|
||||
baud_rate = config["logger"][CONF_BAUD_RATE]
|
||||
if baud_rate == 0:
|
||||
_LOGGER.info("UART logging is disabled (baud_rate=0). Not starting UART logs.")
|
||||
return
|
||||
return 1
|
||||
_LOGGER.info("Starting log output from %s with baud rate %s", port, baud_rate)
|
||||
|
||||
backtrace_state = False
|
||||
@@ -114,25 +137,36 @@ def run_miniterm(config, port):
|
||||
ser.dtr = False
|
||||
ser.rts = False
|
||||
|
||||
with ser:
|
||||
while True:
|
||||
try:
|
||||
raw = ser.readline()
|
||||
except serial.SerialException:
|
||||
_LOGGER.error("Serial port closed!")
|
||||
return
|
||||
line = (
|
||||
raw.replace(b"\r", b"")
|
||||
.replace(b"\n", b"")
|
||||
.decode("utf8", "backslashreplace")
|
||||
)
|
||||
time = datetime.now().time().strftime("[%H:%M:%S]")
|
||||
message = time + line
|
||||
safe_print(message)
|
||||
tries = 0
|
||||
while tries < 5:
|
||||
try:
|
||||
with ser:
|
||||
while True:
|
||||
try:
|
||||
raw = ser.readline()
|
||||
except serial.SerialException:
|
||||
_LOGGER.error("Serial port closed!")
|
||||
return 0
|
||||
line = (
|
||||
raw.replace(b"\r", b"")
|
||||
.replace(b"\n", b"")
|
||||
.decode("utf8", "backslashreplace")
|
||||
)
|
||||
time_str = datetime.now().time().strftime("[%H:%M:%S]")
|
||||
message = time_str + line
|
||||
safe_print(message)
|
||||
|
||||
backtrace_state = platformio_api.process_stacktrace(
|
||||
config, line, backtrace_state=backtrace_state
|
||||
)
|
||||
backtrace_state = platformio_api.process_stacktrace(
|
||||
config, line, backtrace_state=backtrace_state
|
||||
)
|
||||
except serial.SerialException:
|
||||
tries += 1
|
||||
time.sleep(1)
|
||||
if tries >= 5:
|
||||
_LOGGER.error("Could not connect to serial port %s", port)
|
||||
return 1
|
||||
|
||||
return 0
|
||||
|
||||
|
||||
def wrap_to_code(name, comp):
|
||||
@@ -144,6 +178,8 @@ def wrap_to_code(name, comp):
|
||||
if comp.config_schema is not None:
|
||||
conf_str = yaml_util.dump(conf)
|
||||
conf_str = conf_str.replace("//", "")
|
||||
# remove tailing \ to avoid multi-line comment warning
|
||||
conf_str = conf_str.replace("\\\n", "\n")
|
||||
cg.add(cg.LineComment(indent(conf_str)))
|
||||
await coro(conf)
|
||||
|
||||
@@ -187,31 +223,33 @@ def compile_program(args, config):
|
||||
return 0 if idedata is not None else 1
|
||||
|
||||
|
||||
def upload_using_esptool(config, port):
|
||||
def upload_using_esptool(config, port, file):
|
||||
from esphome import platformio_api
|
||||
|
||||
first_baudrate = config[CONF_ESPHOME][CONF_PLATFORMIO_OPTIONS].get(
|
||||
"upload_speed", 460800
|
||||
)
|
||||
|
||||
def run_esptool(baud_rate):
|
||||
if file is not None:
|
||||
flash_images = [platformio_api.FlashImage(path=file, offset="0x0")]
|
||||
else:
|
||||
idedata = platformio_api.get_idedata(config)
|
||||
|
||||
firmware_offset = "0x10000" if CORE.is_esp32 else "0x0"
|
||||
flash_images = [
|
||||
platformio_api.FlashImage(
|
||||
path=idedata.firmware_bin_path,
|
||||
offset=firmware_offset,
|
||||
path=idedata.firmware_bin_path, offset=firmware_offset
|
||||
),
|
||||
*idedata.extra_flash_images,
|
||||
]
|
||||
|
||||
mcu = "esp8266"
|
||||
if CORE.is_esp32:
|
||||
from esphome.components.esp32 import get_esp32_variant
|
||||
mcu = "esp8266"
|
||||
if CORE.is_esp32:
|
||||
from esphome.components.esp32 import get_esp32_variant
|
||||
|
||||
mcu = get_esp32_variant().lower()
|
||||
mcu = get_esp32_variant().lower()
|
||||
|
||||
def run_esptool(baud_rate):
|
||||
cmd = [
|
||||
"esptool.py",
|
||||
"--before",
|
||||
@@ -226,6 +264,8 @@ def upload_using_esptool(config, port):
|
||||
mcu,
|
||||
"write_flash",
|
||||
"-z",
|
||||
"--flash_size",
|
||||
"detect",
|
||||
]
|
||||
for img in flash_images:
|
||||
cmd += [img.offset, img.path]
|
||||
@@ -233,8 +273,7 @@ def upload_using_esptool(config, port):
|
||||
if os.environ.get("ESPHOME_USE_SUBPROCESS") is None:
|
||||
import esptool
|
||||
|
||||
# pylint: disable=protected-access
|
||||
return run_external_command(esptool._main, *cmd)
|
||||
return run_external_command(esptool.main, *cmd) # pylint: disable=no-member
|
||||
|
||||
return run_external_process(*cmd)
|
||||
|
||||
@@ -249,12 +288,28 @@ def upload_using_esptool(config, port):
|
||||
return run_esptool(115200)
|
||||
|
||||
|
||||
def upload_program(config, args, host):
|
||||
# if upload is to a serial port use platformio, otherwise assume ota
|
||||
if get_port_type(host) == "SERIAL":
|
||||
return upload_using_esptool(config, host)
|
||||
def upload_using_platformio(config, port):
|
||||
from esphome import platformio_api
|
||||
|
||||
from esphome import espota2
|
||||
upload_args = ["-t", "upload", "-t", "nobuild"]
|
||||
if port is not None:
|
||||
upload_args += ["--upload-port", port]
|
||||
return platformio_api.run_platformio_cli_run(config, CORE.verbose, *upload_args)
|
||||
|
||||
|
||||
def upload_program(config, args, host):
|
||||
if get_port_type(host) == "SERIAL":
|
||||
if CORE.target_platform in (PLATFORM_ESP32, PLATFORM_ESP8266):
|
||||
file = getattr(args, "file", None)
|
||||
return upload_using_esptool(config, host, file)
|
||||
|
||||
if CORE.target_platform in (PLATFORM_RP2040):
|
||||
return upload_using_platformio(config, args.device)
|
||||
|
||||
if CORE.target_platform in (PLATFORM_BK72XX, PLATFORM_RTL87XX):
|
||||
return upload_using_platformio(config, host)
|
||||
|
||||
return 1 # Unknown target platform
|
||||
|
||||
if CONF_OTA not in config:
|
||||
raise EsphomeError(
|
||||
@@ -262,9 +317,26 @@ def upload_program(config, args, host):
|
||||
"component"
|
||||
)
|
||||
|
||||
from esphome import espota2
|
||||
|
||||
ota_conf = config[CONF_OTA]
|
||||
remote_port = ota_conf[CONF_PORT]
|
||||
password = ota_conf.get(CONF_PASSWORD, "")
|
||||
|
||||
if (
|
||||
not is_ip_address(CORE.address)
|
||||
and (get_port_type(host) == "MQTT" or config[CONF_MDNS][CONF_DISABLED])
|
||||
and CONF_MQTT in config
|
||||
):
|
||||
from esphome import mqtt
|
||||
|
||||
host = mqtt.get_esphome_device_ip(
|
||||
config, args.username, args.password, args.client_id
|
||||
)
|
||||
|
||||
if getattr(args, "file", None) is not None:
|
||||
return espota2.run_ota(host, remote_port, password, args.file)
|
||||
|
||||
return espota2.run_ota(host, remote_port, password, CORE.firmware_bin)
|
||||
|
||||
|
||||
@@ -272,9 +344,15 @@ def show_logs(config, args, port):
|
||||
if "logger" not in config:
|
||||
raise EsphomeError("Logger is not configured!")
|
||||
if get_port_type(port) == "SERIAL":
|
||||
run_miniterm(config, port)
|
||||
return 0
|
||||
return run_miniterm(config, port)
|
||||
if get_port_type(port) == "NETWORK" and "api" in config:
|
||||
if config[CONF_MDNS][CONF_DISABLED] and CONF_MQTT in config:
|
||||
from esphome import mqtt
|
||||
|
||||
port = mqtt.get_esphome_device_ip(
|
||||
config, args.username, args.password, args.client_id
|
||||
)
|
||||
|
||||
from esphome.components.api.client import run_logs
|
||||
|
||||
return run_logs(config, port)
|
||||
@@ -303,10 +381,16 @@ def command_wizard(args):
|
||||
|
||||
|
||||
def command_config(args, config):
|
||||
_LOGGER.info("Configuration is valid!")
|
||||
if not CORE.verbose:
|
||||
config = strip_default_ids(config)
|
||||
safe_print(yaml_util.dump(config))
|
||||
output = yaml_util.dump(config, args.show_secrets)
|
||||
# add the console decoration so the front-end can hide the secrets
|
||||
if not args.show_secrets:
|
||||
output = re.sub(
|
||||
r"(password|key|psk|ssid)\: (.+)", r"\1: \\033[5m\2\\033[6m", output
|
||||
)
|
||||
safe_print(output)
|
||||
_LOGGER.info("Configuration is valid!")
|
||||
return 0
|
||||
|
||||
|
||||
@@ -339,6 +423,7 @@ def command_upload(args, config):
|
||||
show_ota=True,
|
||||
show_mqtt=False,
|
||||
show_api=False,
|
||||
purpose="uploading",
|
||||
)
|
||||
exit_code = upload_program(config, args, port)
|
||||
if exit_code != 0:
|
||||
@@ -347,6 +432,15 @@ def command_upload(args, config):
|
||||
return 0
|
||||
|
||||
|
||||
def command_discover(args, config):
|
||||
if "mqtt" in config:
|
||||
from esphome import mqtt
|
||||
|
||||
return mqtt.show_discover(config, args.username, args.password, args.client_id)
|
||||
|
||||
raise EsphomeError("No discover method configured (mqtt)")
|
||||
|
||||
|
||||
def command_logs(args, config):
|
||||
port = choose_upload_log_host(
|
||||
default=args.device,
|
||||
@@ -354,6 +448,7 @@ def command_logs(args, config):
|
||||
show_ota=False,
|
||||
show_mqtt=True,
|
||||
show_api=True,
|
||||
purpose="logging",
|
||||
)
|
||||
return show_logs(config, args, port)
|
||||
|
||||
@@ -372,6 +467,7 @@ def command_run(args, config):
|
||||
show_ota=True,
|
||||
show_mqtt=False,
|
||||
show_api=True,
|
||||
purpose="uploading",
|
||||
)
|
||||
exit_code = upload_program(config, args, port)
|
||||
if exit_code != 0:
|
||||
@@ -385,6 +481,7 @@ def command_run(args, config):
|
||||
show_ota=False,
|
||||
show_mqtt=True,
|
||||
show_api=True,
|
||||
purpose="logging",
|
||||
)
|
||||
return show_logs(config, args, port)
|
||||
|
||||
@@ -477,6 +574,98 @@ def command_idedata(args, config):
|
||||
return 0
|
||||
|
||||
|
||||
def command_rename(args, config):
|
||||
for c in args.name:
|
||||
if c not in ALLOWED_NAME_CHARS:
|
||||
print(
|
||||
color(
|
||||
Fore.BOLD_RED,
|
||||
f"'{c}' is an invalid character for names. Valid characters are: "
|
||||
f"{ALLOWED_NAME_CHARS} (lowercase, no spaces)",
|
||||
)
|
||||
)
|
||||
return 1
|
||||
# Load existing yaml file
|
||||
with open(CORE.config_path, mode="r+", encoding="utf-8") as raw_file:
|
||||
raw_contents = raw_file.read()
|
||||
|
||||
yaml = yaml_util.load_yaml(CORE.config_path)
|
||||
if CONF_ESPHOME not in yaml or CONF_NAME not in yaml[CONF_ESPHOME]:
|
||||
print(
|
||||
color(Fore.BOLD_RED, "Complex YAML files cannot be automatically renamed.")
|
||||
)
|
||||
return 1
|
||||
old_name = yaml[CONF_ESPHOME][CONF_NAME]
|
||||
match = re.match(r"^\$\{?([a-zA-Z0-9_]+)\}?$", old_name)
|
||||
if match is None:
|
||||
new_raw = re.sub(
|
||||
rf"name:\s+[\"']?{old_name}[\"']?",
|
||||
f'name: "{args.name}"',
|
||||
raw_contents,
|
||||
)
|
||||
else:
|
||||
old_name = yaml[CONF_SUBSTITUTIONS][match.group(1)]
|
||||
if (
|
||||
len(
|
||||
re.findall(
|
||||
rf"^\s+{match.group(1)}:\s+[\"']?{old_name}[\"']?",
|
||||
raw_contents,
|
||||
flags=re.MULTILINE,
|
||||
)
|
||||
)
|
||||
> 1
|
||||
):
|
||||
print(color(Fore.BOLD_RED, "Too many matches in YAML to safely rename"))
|
||||
return 1
|
||||
|
||||
new_raw = re.sub(
|
||||
rf"^(\s+{match.group(1)}):\s+[\"']?{old_name}[\"']?",
|
||||
f'\\1: "{args.name}"',
|
||||
raw_contents,
|
||||
flags=re.MULTILINE,
|
||||
)
|
||||
|
||||
new_path = os.path.join(CORE.config_dir, args.name + ".yaml")
|
||||
print(
|
||||
f"Updating {color(Fore.CYAN, CORE.config_path)} to {color(Fore.CYAN, new_path)}"
|
||||
)
|
||||
print()
|
||||
|
||||
with open(new_path, mode="w", encoding="utf-8") as new_file:
|
||||
new_file.write(new_raw)
|
||||
|
||||
rc = run_external_process("esphome", "config", new_path)
|
||||
if rc != 0:
|
||||
print(color(Fore.BOLD_RED, "Rename failed. Reverting changes."))
|
||||
os.remove(new_path)
|
||||
return 1
|
||||
|
||||
cli_args = [
|
||||
"run",
|
||||
new_path,
|
||||
"--no-logs",
|
||||
"--device",
|
||||
CORE.address,
|
||||
]
|
||||
|
||||
if args.dashboard:
|
||||
cli_args.insert(0, "--dashboard")
|
||||
|
||||
try:
|
||||
rc = run_external_process("esphome", *cli_args)
|
||||
except KeyboardInterrupt:
|
||||
rc = 1
|
||||
if rc != 0:
|
||||
os.remove(new_path)
|
||||
return 1
|
||||
|
||||
os.remove(CORE.config_path)
|
||||
|
||||
print(color(Fore.BOLD_GREEN, "SUCCESS"))
|
||||
print()
|
||||
return 0
|
||||
|
||||
|
||||
PRE_CONFIG_ACTIONS = {
|
||||
"wizard": command_wizard,
|
||||
"version": command_version,
|
||||
@@ -495,6 +684,8 @@ POST_CONFIG_ACTIONS = {
|
||||
"mqtt-fingerprint": command_mqtt_fingerprint,
|
||||
"clean": command_clean,
|
||||
"idedata": command_idedata,
|
||||
"rename": command_rename,
|
||||
"discover": command_discover,
|
||||
}
|
||||
|
||||
|
||||
@@ -539,6 +730,9 @@ def parse_args(argv):
|
||||
parser_config.add_argument(
|
||||
"configuration", help="Your YAML configuration file(s).", nargs="+"
|
||||
)
|
||||
parser_config.add_argument(
|
||||
"--show-secrets", help="Show secrets in output.", action="store_true"
|
||||
)
|
||||
|
||||
parser_compile = subparsers.add_parser(
|
||||
"compile", help="Read the configuration and compile a program."
|
||||
@@ -562,6 +756,10 @@ def parse_args(argv):
|
||||
"--device",
|
||||
help="Manually specify the serial port/address to use, for example /dev/ttyUSB0.",
|
||||
)
|
||||
parser_upload.add_argument(
|
||||
"--file",
|
||||
help="Manually specify the binary file to upload.",
|
||||
)
|
||||
|
||||
parser_logs = subparsers.add_parser(
|
||||
"logs",
|
||||
@@ -576,6 +774,15 @@ def parse_args(argv):
|
||||
help="Manually specify the serial port/address to use, for example /dev/ttyUSB0.",
|
||||
)
|
||||
|
||||
parser_discover = subparsers.add_parser(
|
||||
"discover",
|
||||
help="Validate the configuration and show all discovered devices.",
|
||||
parents=[mqtt_options],
|
||||
)
|
||||
parser_discover.add_argument(
|
||||
"configuration", help="Your YAML configuration file.", nargs=1
|
||||
)
|
||||
|
||||
parser_run = subparsers.add_parser(
|
||||
"run",
|
||||
help="Validate the configuration, create a binary, upload it, and start logs.",
|
||||
@@ -605,10 +812,7 @@ def parse_args(argv):
|
||||
"wizard",
|
||||
help="A helpful setup wizard that will guide you through setting up ESPHome.",
|
||||
)
|
||||
parser_wizard.add_argument(
|
||||
"configuration",
|
||||
help="Your YAML configuration file.",
|
||||
)
|
||||
parser_wizard.add_argument("configuration", help="Your YAML configuration file.")
|
||||
|
||||
parser_fingerprint = subparsers.add_parser(
|
||||
"mqtt-fingerprint", help="Get the SSL fingerprint from a MQTT broker."
|
||||
@@ -630,8 +834,7 @@ def parse_args(argv):
|
||||
"dashboard", help="Create a simple web server for a dashboard."
|
||||
)
|
||||
parser_dashboard.add_argument(
|
||||
"configuration",
|
||||
help="Your YAML configuration file directory.",
|
||||
"configuration", help="Your YAML configuration file directory."
|
||||
)
|
||||
parser_dashboard.add_argument(
|
||||
"--port",
|
||||
@@ -639,6 +842,12 @@ def parse_args(argv):
|
||||
type=int,
|
||||
default=6052,
|
||||
)
|
||||
parser_dashboard.add_argument(
|
||||
"--address",
|
||||
help="The address to bind to.",
|
||||
type=str,
|
||||
default="0.0.0.0",
|
||||
)
|
||||
parser_dashboard.add_argument(
|
||||
"--username",
|
||||
help="The optional username to require for authentication.",
|
||||
@@ -655,7 +864,7 @@ def parse_args(argv):
|
||||
"--open-ui", help="Open the dashboard UI in a browser.", action="store_true"
|
||||
)
|
||||
parser_dashboard.add_argument(
|
||||
"--hassio", help=argparse.SUPPRESS, action="store_true"
|
||||
"--ha-addon", help=argparse.SUPPRESS, action="store_true"
|
||||
)
|
||||
parser_dashboard.add_argument(
|
||||
"--socket", help="Make the dashboard serve under a unix socket", type=str
|
||||
@@ -675,6 +884,15 @@ def parse_args(argv):
|
||||
"configuration", help="Your YAML configuration file(s).", nargs=1
|
||||
)
|
||||
|
||||
parser_rename = subparsers.add_parser(
|
||||
"rename",
|
||||
help="Rename a device in YAML, compile the binary and upload it.",
|
||||
)
|
||||
parser_rename.add_argument(
|
||||
"configuration", help="Your YAML configuration file.", nargs=1
|
||||
)
|
||||
parser_rename.add_argument("name", help="The new name for the device.", type=str)
|
||||
|
||||
# Keep backward compatibility with the old command line format of
|
||||
# esphome <config> <command>.
|
||||
#
|
||||
@@ -751,6 +969,7 @@ def parse_args(argv):
|
||||
# Finally, run the new-style parser again with the possibly swapped arguments,
|
||||
# and let it error out if the command is unparsable.
|
||||
parser.set_defaults(deprecated_argv_suggestion=deprecated_argv_suggestion)
|
||||
argcomplete.autocomplete(parser)
|
||||
return parser.parse_args(arguments)
|
||||
|
||||
|
||||
@@ -772,10 +991,10 @@ def run_esphome(argv):
|
||||
_LOGGER.warning("Please instead use:")
|
||||
_LOGGER.warning(" esphome %s", " ".join(args.deprecated_argv_suggestion))
|
||||
|
||||
if sys.version_info < (3, 7, 0):
|
||||
if sys.version_info < (3, 8, 0):
|
||||
_LOGGER.error(
|
||||
"You're running ESPHome with Python <3.7. ESPHome is no longer compatible "
|
||||
"with this Python version. Please reinstall ESPHome with Python 3.7+"
|
||||
"You're running ESPHome with Python <3.8. ESPHome is no longer compatible "
|
||||
"with this Python version. Please reinstall ESPHome with Python 3.8+"
|
||||
)
|
||||
return 1
|
||||
|
||||
@@ -786,7 +1005,13 @@ def run_esphome(argv):
|
||||
_LOGGER.error(e, exc_info=args.verbose)
|
||||
return 1
|
||||
|
||||
_LOGGER.info("ESPHome %s", const.__version__)
|
||||
|
||||
for conf_path in args.configuration:
|
||||
if any(os.path.basename(conf_path) == x for x in SECRETS_FILES):
|
||||
_LOGGER.warning("Skipping secrets file %s", conf_path)
|
||||
continue
|
||||
|
||||
CORE.config_path = conf_path
|
||||
CORE.dashboard = args.dashboard
|
||||
|
||||
|
@@ -3,6 +3,7 @@ import esphome.config_validation as cv
|
||||
from esphome.const import (
|
||||
CONF_AUTOMATION_ID,
|
||||
CONF_CONDITION,
|
||||
CONF_COUNT,
|
||||
CONF_ELSE,
|
||||
CONF_ID,
|
||||
CONF_THEN,
|
||||
@@ -10,8 +11,9 @@ from esphome.const import (
|
||||
CONF_TRIGGER_ID,
|
||||
CONF_TYPE_ID,
|
||||
CONF_TIME,
|
||||
CONF_UPDATE_INTERVAL,
|
||||
)
|
||||
from esphome.jsonschema import jschema_extractor
|
||||
from esphome.schema_extractors import SCHEMA_EXTRACT, schema_extractor
|
||||
from esphome.util import Registry
|
||||
|
||||
|
||||
@@ -22,11 +24,10 @@ def maybe_simple_id(*validators):
|
||||
def maybe_conf(conf, *validators):
|
||||
validator = cv.All(*validators)
|
||||
|
||||
@jschema_extractor("maybe")
|
||||
@schema_extractor("maybe")
|
||||
def validate(value):
|
||||
# pylint: disable=comparison-with-callable
|
||||
if value == jschema_extractor:
|
||||
return validator
|
||||
if value == SCHEMA_EXTRACT:
|
||||
return (validator, conf)
|
||||
|
||||
if isinstance(value, dict):
|
||||
return validator(value)
|
||||
@@ -66,8 +67,11 @@ DelayAction = cg.esphome_ns.class_("DelayAction", Action, cg.Component)
|
||||
LambdaAction = cg.esphome_ns.class_("LambdaAction", Action)
|
||||
IfAction = cg.esphome_ns.class_("IfAction", Action)
|
||||
WhileAction = cg.esphome_ns.class_("WhileAction", Action)
|
||||
RepeatAction = cg.esphome_ns.class_("RepeatAction", Action)
|
||||
WaitUntilAction = cg.esphome_ns.class_("WaitUntilAction", Action, cg.Component)
|
||||
UpdateComponentAction = cg.esphome_ns.class_("UpdateComponentAction", Action)
|
||||
SuspendComponentAction = cg.esphome_ns.class_("SuspendComponentAction", Action)
|
||||
ResumeComponentAction = cg.esphome_ns.class_("ResumeComponentAction", Action)
|
||||
Automation = cg.esphome_ns.class_("Automation")
|
||||
|
||||
LambdaCondition = cg.esphome_ns.class_("LambdaCondition", Condition)
|
||||
@@ -109,11 +113,9 @@ def validate_automation(extra_schema=None, extra_validators=None, single=False):
|
||||
# This should only happen with invalid configs, but let's have a nice error message.
|
||||
return [schema(value)]
|
||||
|
||||
@jschema_extractor("automation")
|
||||
@schema_extractor("automation")
|
||||
def validator(value):
|
||||
# hack to get the schema
|
||||
# pylint: disable=comparison-with-callable
|
||||
if value == jschema_extractor:
|
||||
if value == SCHEMA_EXTRACT:
|
||||
return schema
|
||||
|
||||
value = validator_(value)
|
||||
@@ -139,6 +141,7 @@ AUTOMATION_SCHEMA = cv.Schema(
|
||||
AndCondition = cg.esphome_ns.class_("AndCondition", Condition)
|
||||
OrCondition = cg.esphome_ns.class_("OrCondition", Condition)
|
||||
NotCondition = cg.esphome_ns.class_("NotCondition", Condition)
|
||||
XorCondition = cg.esphome_ns.class_("XorCondition", Condition)
|
||||
|
||||
|
||||
@register_condition("and", AndCondition, validate_condition_list)
|
||||
@@ -159,6 +162,12 @@ async def not_condition_to_code(config, condition_id, template_arg, args):
|
||||
return cg.new_Pvariable(condition_id, template_arg, condition)
|
||||
|
||||
|
||||
@register_condition("xor", XorCondition, validate_condition_list)
|
||||
async def xor_condition_to_code(config, condition_id, template_arg, args):
|
||||
conditions = await build_condition_list(config, template_arg, args)
|
||||
return cg.new_Pvariable(condition_id, template_arg, conditions)
|
||||
|
||||
|
||||
@register_condition("lambda", LambdaCondition, cv.returning_lambda)
|
||||
async def lambda_condition_to_code(config, condition_id, template_arg, args):
|
||||
lambda_ = await cg.process_lambda(config, args, return_type=bool)
|
||||
@@ -241,21 +250,39 @@ async def while_action_to_code(config, action_id, template_arg, args):
|
||||
return var
|
||||
|
||||
|
||||
def validate_wait_until(value):
|
||||
schema = cv.Schema(
|
||||
@register_action(
|
||||
"repeat",
|
||||
RepeatAction,
|
||||
cv.Schema(
|
||||
{
|
||||
cv.Required(CONF_CONDITION): validate_potentially_and_condition,
|
||||
cv.Optional(CONF_TIMEOUT): cv.templatable(
|
||||
cv.positive_time_period_milliseconds
|
||||
),
|
||||
cv.Required(CONF_COUNT): cv.templatable(cv.positive_not_null_int),
|
||||
cv.Required(CONF_THEN): validate_action_list,
|
||||
}
|
||||
),
|
||||
)
|
||||
async def repeat_action_to_code(config, action_id, template_arg, args):
|
||||
var = cg.new_Pvariable(action_id, template_arg)
|
||||
count_template = await cg.templatable(config[CONF_COUNT], args, cg.uint32)
|
||||
cg.add(var.set_count(count_template))
|
||||
actions = await build_action_list(
|
||||
config[CONF_THEN],
|
||||
cg.TemplateArguments(cg.uint32, *template_arg.args),
|
||||
[(cg.uint32, "iteration"), *args],
|
||||
)
|
||||
if isinstance(value, dict) and CONF_CONDITION in value:
|
||||
return schema(value)
|
||||
return validate_wait_until({CONF_CONDITION: value})
|
||||
cg.add(var.add_then(actions))
|
||||
return var
|
||||
|
||||
|
||||
@register_action("wait_until", WaitUntilAction, validate_wait_until)
|
||||
_validate_wait_until = cv.maybe_simple_value(
|
||||
{
|
||||
cv.Required(CONF_CONDITION): validate_potentially_and_condition,
|
||||
cv.Optional(CONF_TIMEOUT): cv.templatable(cv.positive_time_period_milliseconds),
|
||||
},
|
||||
key=CONF_CONDITION,
|
||||
)
|
||||
|
||||
|
||||
@register_action("wait_until", WaitUntilAction, _validate_wait_until)
|
||||
async def wait_until_action_to_code(config, action_id, template_arg, args):
|
||||
conditions = await build_condition(config[CONF_CONDITION], template_arg, args)
|
||||
var = cg.new_Pvariable(action_id, template_arg, conditions)
|
||||
@@ -286,6 +313,41 @@ async def component_update_action_to_code(config, action_id, template_arg, args)
|
||||
return cg.new_Pvariable(action_id, template_arg, comp)
|
||||
|
||||
|
||||
@register_action(
|
||||
"component.suspend",
|
||||
SuspendComponentAction,
|
||||
maybe_simple_id(
|
||||
{
|
||||
cv.Required(CONF_ID): cv.use_id(cg.PollingComponent),
|
||||
}
|
||||
),
|
||||
)
|
||||
async def component_suspend_action_to_code(config, action_id, template_arg, args):
|
||||
comp = await cg.get_variable(config[CONF_ID])
|
||||
return cg.new_Pvariable(action_id, template_arg, comp)
|
||||
|
||||
|
||||
@register_action(
|
||||
"component.resume",
|
||||
ResumeComponentAction,
|
||||
maybe_simple_id(
|
||||
{
|
||||
cv.Required(CONF_ID): cv.use_id(cg.PollingComponent),
|
||||
cv.Optional(CONF_UPDATE_INTERVAL): cv.templatable(
|
||||
cv.positive_time_period_milliseconds
|
||||
),
|
||||
}
|
||||
),
|
||||
)
|
||||
async def component_resume_action_to_code(config, action_id, template_arg, args):
|
||||
comp = await cg.get_variable(config[CONF_ID])
|
||||
var = cg.new_Pvariable(action_id, template_arg, comp)
|
||||
if CONF_UPDATE_INTERVAL in config:
|
||||
template_ = await cg.templatable(config[CONF_UPDATE_INTERVAL], args, int)
|
||||
cg.add(var.set_update_interval(template_))
|
||||
return var
|
||||
|
||||
|
||||
async def build_action(full_config, template_arg, args):
|
||||
registry_entry, config = cg.extract_registry_entry_config(
|
||||
ACTION_REGISTRY, full_config
|
||||
|
@@ -22,6 +22,7 @@ from esphome.cpp_generator import ( # noqa
|
||||
static_const_array,
|
||||
statement,
|
||||
variable,
|
||||
with_local_variable,
|
||||
new_variable,
|
||||
Pvariable,
|
||||
new_Pvariable,
|
||||
@@ -46,6 +47,7 @@ from esphome.cpp_helpers import ( # noqa
|
||||
build_registry_list,
|
||||
extract_registry_entry_config,
|
||||
register_parented,
|
||||
past_safe_mode,
|
||||
)
|
||||
from esphome.cpp_types import ( # noqa
|
||||
global_ns,
|
||||
@@ -62,7 +64,10 @@ from esphome.cpp_types import ( # noqa
|
||||
uint16,
|
||||
uint32,
|
||||
uint64,
|
||||
int16,
|
||||
int32,
|
||||
int64,
|
||||
size_t,
|
||||
const_char_ptr,
|
||||
NAN,
|
||||
esphome_ns,
|
||||
@@ -75,10 +80,11 @@ from esphome.cpp_types import ( # noqa
|
||||
optional,
|
||||
arduino_json_ns,
|
||||
JsonObject,
|
||||
JsonObjectRef,
|
||||
JsonObjectConstRef,
|
||||
JsonObjectConst,
|
||||
Controller,
|
||||
GPIOPin,
|
||||
InternalGPIOPin,
|
||||
gpio_Flags,
|
||||
EntityCategory,
|
||||
Parented,
|
||||
)
|
||||
|
1
esphome/components/a01nyub/__init__.py
Normal file
1
esphome/components/a01nyub/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
CODEOWNERS = ["@MrSuicideParrot"]
|
57
esphome/components/a01nyub/a01nyub.cpp
Normal file
57
esphome/components/a01nyub/a01nyub.cpp
Normal file
@@ -0,0 +1,57 @@
|
||||
// Datasheet https://wiki.dfrobot.com/A01NYUB%20Waterproof%20Ultrasonic%20Sensor%20SKU:%20SEN0313
|
||||
|
||||
#include "a01nyub.h"
|
||||
#include "esphome/core/helpers.h"
|
||||
#include "esphome/core/log.h"
|
||||
|
||||
namespace esphome {
|
||||
namespace a01nyub {
|
||||
|
||||
static const char *const TAG = "a01nyub.sensor";
|
||||
static const uint8_t MAX_DATA_LENGTH_BYTES = 4;
|
||||
|
||||
void A01nyubComponent::loop() {
|
||||
uint8_t data;
|
||||
while (this->available() > 0) {
|
||||
if (this->read_byte(&data)) {
|
||||
buffer_.push_back(data);
|
||||
this->check_buffer_();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void A01nyubComponent::check_buffer_() {
|
||||
if (this->buffer_.size() >= MAX_DATA_LENGTH_BYTES) {
|
||||
size_t i;
|
||||
for (i = 0; i < this->buffer_.size(); i++) {
|
||||
// Look for the first packet
|
||||
if (this->buffer_[i] == 0xFF) {
|
||||
if (i + 1 + 3 < this->buffer_.size()) { // Packet is not complete
|
||||
return; // Wait for completion
|
||||
}
|
||||
|
||||
uint8_t checksum = (this->buffer_[i] + this->buffer_[i + 1] + this->buffer_[i + 2]) & 0xFF;
|
||||
if (this->buffer_[i + 3] == checksum) {
|
||||
float distance = (this->buffer_[i + 1] << 8) + this->buffer_[i + 2];
|
||||
if (distance > 280) {
|
||||
float meters = distance / 1000.0;
|
||||
ESP_LOGV(TAG, "Distance from sensor: %f mm, %f m", distance, meters);
|
||||
this->publish_state(meters);
|
||||
} else {
|
||||
ESP_LOGW(TAG, "Invalid data read from sensor: %s", format_hex_pretty(this->buffer_).c_str());
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
this->buffer_.clear();
|
||||
}
|
||||
}
|
||||
|
||||
void A01nyubComponent::dump_config() {
|
||||
ESP_LOGCONFIG(TAG, "A01nyub Sensor:");
|
||||
LOG_SENSOR(" ", "Distance", this);
|
||||
}
|
||||
|
||||
} // namespace a01nyub
|
||||
} // namespace esphome
|
27
esphome/components/a01nyub/a01nyub.h
Normal file
27
esphome/components/a01nyub/a01nyub.h
Normal file
@@ -0,0 +1,27 @@
|
||||
#pragma once
|
||||
|
||||
#include <vector>
|
||||
|
||||
#include "esphome/core/component.h"
|
||||
#include "esphome/components/sensor/sensor.h"
|
||||
#include "esphome/components/uart/uart.h"
|
||||
|
||||
namespace esphome {
|
||||
namespace a01nyub {
|
||||
|
||||
class A01nyubComponent : public sensor::Sensor, public Component, public uart::UARTDevice {
|
||||
public:
|
||||
// Nothing really public.
|
||||
|
||||
// ========== INTERNAL METHODS ==========
|
||||
void loop() override;
|
||||
void dump_config() override;
|
||||
|
||||
protected:
|
||||
void check_buffer_();
|
||||
|
||||
std::vector<uint8_t> buffer_;
|
||||
};
|
||||
|
||||
} // namespace a01nyub
|
||||
} // namespace esphome
|
41
esphome/components/a01nyub/sensor.py
Normal file
41
esphome/components/a01nyub/sensor.py
Normal file
@@ -0,0 +1,41 @@
|
||||
import esphome.codegen as cg
|
||||
from esphome.components import sensor, uart
|
||||
from esphome.const import (
|
||||
STATE_CLASS_MEASUREMENT,
|
||||
UNIT_METER,
|
||||
ICON_ARROW_EXPAND_VERTICAL,
|
||||
DEVICE_CLASS_DISTANCE,
|
||||
)
|
||||
|
||||
CODEOWNERS = ["@MrSuicideParrot"]
|
||||
DEPENDENCIES = ["uart"]
|
||||
|
||||
a01nyub_ns = cg.esphome_ns.namespace("a01nyub")
|
||||
A01nyubComponent = a01nyub_ns.class_(
|
||||
"A01nyubComponent", sensor.Sensor, cg.Component, uart.UARTDevice
|
||||
)
|
||||
|
||||
CONFIG_SCHEMA = sensor.sensor_schema(
|
||||
A01nyubComponent,
|
||||
unit_of_measurement=UNIT_METER,
|
||||
icon=ICON_ARROW_EXPAND_VERTICAL,
|
||||
accuracy_decimals=3,
|
||||
state_class=STATE_CLASS_MEASUREMENT,
|
||||
device_class=DEVICE_CLASS_DISTANCE,
|
||||
).extend(uart.UART_DEVICE_SCHEMA)
|
||||
|
||||
FINAL_VALIDATE_SCHEMA = uart.final_validate_device_schema(
|
||||
"a01nyub",
|
||||
baud_rate=9600,
|
||||
require_tx=False,
|
||||
require_rx=True,
|
||||
data_bits=8,
|
||||
parity=None,
|
||||
stop_bits=1,
|
||||
)
|
||||
|
||||
|
||||
async def to_code(config):
|
||||
var = await sensor.new_sensor(config)
|
||||
await cg.register_component(var, config)
|
||||
await uart.register_uart_device(var, config)
|
@@ -46,6 +46,7 @@ void A4988::loop() {
|
||||
return;
|
||||
|
||||
this->dir_pin_->digital_write(dir == 1);
|
||||
delayMicroseconds(50);
|
||||
this->step_pin_->digital_write(true);
|
||||
delayMicroseconds(5);
|
||||
this->step_pin_->digital_write(false);
|
||||
|
@@ -28,6 +28,6 @@ async def to_code(config):
|
||||
dir_pin = await cg.gpio_pin_expression(config[CONF_DIR_PIN])
|
||||
cg.add(var.set_dir_pin(dir_pin))
|
||||
|
||||
if CONF_SLEEP_PIN in config:
|
||||
sleep_pin = await cg.gpio_pin_expression(config[CONF_SLEEP_PIN])
|
||||
if sleep_pin_config := config.get(CONF_SLEEP_PIN):
|
||||
sleep_pin = await cg.gpio_pin_expression(sleep_pin_config)
|
||||
cg.add(var.set_sleep_pin(sleep_pin))
|
||||
|
1
esphome/components/absolute_humidity/__init__.py
Normal file
1
esphome/components/absolute_humidity/__init__.py
Normal file
@@ -0,0 +1 @@
|
||||
CODEOWNERS = ["@DAVe3283"]
|
182
esphome/components/absolute_humidity/absolute_humidity.cpp
Normal file
182
esphome/components/absolute_humidity/absolute_humidity.cpp
Normal file
@@ -0,0 +1,182 @@
|
||||
#include "esphome/core/log.h"
|
||||
#include "absolute_humidity.h"
|
||||
|
||||
namespace esphome {
|
||||
namespace absolute_humidity {
|
||||
|
||||
static const char *const TAG = "absolute_humidity.sensor";
|
||||
|
||||
void AbsoluteHumidityComponent::setup() {
|
||||
ESP_LOGCONFIG(TAG, "Setting up absolute humidity '%s'...", this->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); });
|
||||
if (this->temperature_sensor_->has_state()) {
|
||||
this->temperature_callback_(this->temperature_sensor_->get_state());
|
||||
}
|
||||
|
||||
ESP_LOGD(TAG, " Added callback for relative humidity '%s'", this->humidity_sensor_->get_name().c_str());
|
||||
this->humidity_sensor_->add_on_state_callback([this](float state) { this->humidity_callback_(state); });
|
||||
if (this->humidity_sensor_->has_state()) {
|
||||
this->humidity_callback_(this->humidity_sensor_->get_state());
|
||||
}
|
||||
}
|
||||
|
||||
void AbsoluteHumidityComponent::dump_config() {
|
||||
LOG_SENSOR("", "Absolute Humidity", this);
|
||||
|
||||
switch (this->equation_) {
|
||||
case BUCK:
|
||||
ESP_LOGCONFIG(TAG, "Saturation Vapor Pressure Equation: Buck");
|
||||
break;
|
||||
case TETENS:
|
||||
ESP_LOGCONFIG(TAG, "Saturation Vapor Pressure Equation: Tetens");
|
||||
break;
|
||||
case WOBUS:
|
||||
ESP_LOGCONFIG(TAG, "Saturation Vapor Pressure Equation: Wobus");
|
||||
break;
|
||||
default:
|
||||
ESP_LOGE(TAG, "Invalid saturation vapor pressure equation selection!");
|
||||
break;
|
||||
}
|
||||
|
||||
ESP_LOGCONFIG(TAG, "Sources");
|
||||
ESP_LOGCONFIG(TAG, " Temperature: '%s'", this->temperature_sensor_->get_name().c_str());
|
||||
ESP_LOGCONFIG(TAG, " Relative Humidity: '%s'", this->humidity_sensor_->get_name().c_str());
|
||||
}
|
||||
|
||||
float AbsoluteHumidityComponent::get_setup_priority() const { return setup_priority::DATA; }
|
||||
|
||||
void AbsoluteHumidityComponent::loop() {
|
||||
if (!this->next_update_) {
|
||||
return;
|
||||
}
|
||||
this->next_update_ = false;
|
||||
|
||||
// Ensure we have source data
|
||||
const bool no_temperature = std::isnan(this->temperature_);
|
||||
const bool no_humidity = std::isnan(this->humidity_);
|
||||
if (no_temperature || no_humidity) {
|
||||
if (no_temperature) {
|
||||
ESP_LOGW(TAG, "No valid state from temperature sensor!");
|
||||
}
|
||||
if (no_humidity) {
|
||||
ESP_LOGW(TAG, "No valid state from temperature sensor!");
|
||||
}
|
||||
ESP_LOGW(TAG, "Unable to calculate absolute humidity.");
|
||||
this->publish_state(NAN);
|
||||
this->status_set_warning();
|
||||
return;
|
||||
}
|
||||
|
||||
// Convert to desired units
|
||||
const float temperature_c = this->temperature_;
|
||||
const float temperature_k = temperature_c + 273.15;
|
||||
const float hr = this->humidity_ / 100;
|
||||
|
||||
// Calculate saturation vapor pressure
|
||||
float es;
|
||||
switch (this->equation_) {
|
||||
case BUCK:
|
||||
es = es_buck(temperature_c);
|
||||
break;
|
||||
case TETENS:
|
||||
es = es_tetens(temperature_c);
|
||||
break;
|
||||
case WOBUS:
|
||||
es = es_wobus(temperature_c);
|
||||
break;
|
||||
default:
|
||||
ESP_LOGE(TAG, "Invalid saturation vapor pressure equation selection!");
|
||||
this->publish_state(NAN);
|
||||
this->status_set_error();
|
||||
return;
|
||||
}
|
||||
ESP_LOGD(TAG, "Saturation vapor pressure %f kPa", es);
|
||||
|
||||
// Calculate absolute humidity
|
||||
const float absolute_humidity = vapor_density(es, hr, temperature_k);
|
||||
|
||||
// Publish absolute humidity
|
||||
ESP_LOGD(TAG, "Publishing absolute humidity %f g/m³", absolute_humidity);
|
||||
this->status_clear_warning();
|
||||
this->publish_state(absolute_humidity);
|
||||
}
|
||||
|
||||
// Buck equation (https://en.wikipedia.org/wiki/Arden_Buck_equation)
|
||||
// More accurate than Tetens in normal meteorologic conditions
|
||||
float AbsoluteHumidityComponent::es_buck(float temperature_c) {
|
||||
float a, b, c, d;
|
||||
if (temperature_c >= 0) {
|
||||
a = 0.61121;
|
||||
b = 18.678;
|
||||
c = 234.5;
|
||||
d = 257.14;
|
||||
} else {
|
||||
a = 0.61115;
|
||||
b = 18.678;
|
||||
c = 233.7;
|
||||
d = 279.82;
|
||||
}
|
||||
return a * expf((b - (temperature_c / c)) * (temperature_c / (d + temperature_c)));
|
||||
}
|
||||
|
||||
// Tetens equation (https://en.wikipedia.org/wiki/Tetens_equation)
|
||||
float AbsoluteHumidityComponent::es_tetens(float temperature_c) {
|
||||
float a, b;
|
||||
if (temperature_c >= 0) {
|
||||
a = 17.27;
|
||||
b = 237.3;
|
||||
} else {
|
||||
a = 21.875;
|
||||
b = 265.5;
|
||||
}
|
||||
return 0.61078 * expf((a * temperature_c) / (temperature_c + b));
|
||||
}
|
||||
|
||||
// Wobus equation
|
||||
// https://wahiduddin.net/calc/density_altitude.htm
|
||||
// https://wahiduddin.net/calc/density_algorithms.htm
|
||||
// Calculate the saturation vapor pressure (kPa)
|
||||
float AbsoluteHumidityComponent::es_wobus(float t) {
|
||||
// THIS FUNCTION RETURNS THE SATURATION VAPOR PRESSURE ESW (MILLIBARS)
|
||||
// OVER LIQUID WATER GIVEN THE TEMPERATURE T (CELSIUS). THE POLYNOMIAL
|
||||
// APPROXIMATION BELOW IS DUE TO HERMAN WOBUS, A MATHEMATICIAN WHO
|
||||
// WORKED AT THE NAVY WEATHER RESEARCH FACILITY, NORFOLK, VIRGINIA,
|
||||
// BUT WHO IS NOW RETIRED. THE COEFFICIENTS OF THE POLYNOMIAL WERE
|
||||
// CHOSEN TO FIT THE VALUES IN TABLE 94 ON PP. 351-353 OF THE SMITH-
|
||||
// SONIAN METEOROLOGICAL TABLES BY ROLAND LIST (6TH EDITION). THE
|
||||
// APPROXIMATION IS VALID FOR -50 < T < 100C.
|
||||
//
|
||||
// Baker, Schlatter 17-MAY-1982 Original version.
|
||||
|
||||
const float c0 = +0.99999683e00;
|
||||
const float c1 = -0.90826951e-02;
|
||||
const float c2 = +0.78736169e-04;
|
||||
const float c3 = -0.61117958e-06;
|
||||
const float c4 = +0.43884187e-08;
|
||||
const float c5 = -0.29883885e-10;
|
||||
const float c6 = +0.21874425e-12;
|
||||
const float c7 = -0.17892321e-14;
|
||||
const float c8 = +0.11112018e-16;
|
||||
const float c9 = -0.30994571e-19;
|
||||
const float p = c0 + t * (c1 + t * (c2 + t * (c3 + t * (c4 + t * (c5 + t * (c6 + t * (c7 + t * (c8 + t * (c9)))))))));
|
||||
return 0.61078 / pow(p, 8);
|
||||
}
|
||||
|
||||
// From https://www.environmentalbiophysics.org/chalk-talk-how-to-calculate-absolute-humidity/
|
||||
// H/T to https://esphome.io/cookbook/bme280_environment.html
|
||||
// H/T to https://carnotcycle.wordpress.com/2012/08/04/how-to-convert-relative-humidity-to-absolute-humidity/
|
||||
float AbsoluteHumidityComponent::vapor_density(float es, float hr, float ta) {
|
||||
// es = saturated vapor pressure (kPa)
|
||||
// hr = relative humidity [0-1]
|
||||
// ta = absolute temperature (K)
|
||||
|
||||
const float ea = hr * es * 1000; // vapor pressure of the air (Pa)
|
||||
const float mw = 18.01528; // molar mass of water (g⋅mol⁻¹)
|
||||
const float r = 8.31446261815324; // molar gas constant (J⋅K⁻¹)
|
||||
return (ea * mw) / (r * ta);
|
||||
}
|
||||
|
||||
} // namespace absolute_humidity
|
||||
} // namespace esphome
|
76
esphome/components/absolute_humidity/absolute_humidity.h
Normal file
76
esphome/components/absolute_humidity/absolute_humidity.h
Normal file
@@ -0,0 +1,76 @@
|
||||
#pragma once
|
||||
|
||||
#include "esphome/core/component.h"
|
||||
#include "esphome/components/sensor/sensor.h"
|
||||
|
||||
namespace esphome {
|
||||
namespace absolute_humidity {
|
||||
|
||||
/// Enum listing all implemented saturation vapor pressure equations.
|
||||
enum SaturationVaporPressureEquation {
|
||||
BUCK,
|
||||
TETENS,
|
||||
WOBUS,
|
||||
};
|
||||
|
||||
/// This class implements calculation of absolute humidity from temperature and relative humidity.
|
||||
class AbsoluteHumidityComponent : public sensor::Sensor, public Component {
|
||||
public:
|
||||
AbsoluteHumidityComponent() = default;
|
||||
|
||||
void set_temperature_sensor(sensor::Sensor *temperature_sensor) { this->temperature_sensor_ = temperature_sensor; }
|
||||
void set_humidity_sensor(sensor::Sensor *humidity_sensor) { this->humidity_sensor_ = humidity_sensor; }
|
||||
void set_equation(SaturationVaporPressureEquation equation) { this->equation_ = equation; }
|
||||
|
||||
void setup() override;
|
||||
void dump_config() override;
|
||||
float get_setup_priority() const override;
|
||||
void loop() override;
|
||||
|
||||
protected:
|
||||
void temperature_callback_(float state) {
|
||||
this->next_update_ = true;
|
||||
this->temperature_ = state;
|
||||
}
|
||||
void humidity_callback_(float state) {
|
||||
this->next_update_ = true;
|
||||
this->humidity_ = state;
|
||||
}
|
||||
|
||||
/** Buck equation for saturation vapor pressure in kPa.
|
||||
*
|
||||
* @param temperature_c Air temperature in °C.
|
||||
*/
|
||||
static float es_buck(float temperature_c);
|
||||
/** Tetens equation for saturation vapor pressure in kPa.
|
||||
*
|
||||
* @param temperature_c Air temperature in °C.
|
||||
*/
|
||||
static float es_tetens(float temperature_c);
|
||||
/** Wobus equation for saturation vapor pressure in kPa.
|
||||
*
|
||||
* @param temperature_c Air temperature in °C.
|
||||
*/
|
||||
static float es_wobus(float temperature_c);
|
||||
|
||||
/** Calculate vapor density (absolute humidity) in g/m³.
|
||||
*
|
||||
* @param es Saturation vapor pressure in kPa.
|
||||
* @param hr Relative humidity 0 to 1.
|
||||
* @param ta Absolute temperature in K.
|
||||
* @param heater_duration The duration in ms that the heater should turn on for when measuring.
|
||||
*/
|
||||
static float vapor_density(float es, float hr, float ta);
|
||||
|
||||
sensor::Sensor *temperature_sensor_{nullptr};
|
||||
sensor::Sensor *humidity_sensor_{nullptr};
|
||||
|
||||
bool next_update_{false};
|
||||
|
||||
float temperature_{NAN};
|
||||
float humidity_{NAN};
|
||||
SaturationVaporPressureEquation equation_;
|
||||
};
|
||||
|
||||
} // namespace absolute_humidity
|
||||
} // namespace esphome
|
56
esphome/components/absolute_humidity/sensor.py
Normal file
56
esphome/components/absolute_humidity/sensor.py
Normal file
@@ -0,0 +1,56 @@
|
||||
import esphome.codegen as cg
|
||||
import esphome.config_validation as cv
|
||||
from esphome.components import sensor
|
||||
from esphome.const import (
|
||||
CONF_HUMIDITY,
|
||||
CONF_TEMPERATURE,
|
||||
STATE_CLASS_MEASUREMENT,
|
||||
CONF_EQUATION,
|
||||
ICON_WATER,
|
||||
UNIT_GRAMS_PER_CUBIC_METER,
|
||||
)
|
||||
|
||||
absolute_humidity_ns = cg.esphome_ns.namespace("absolute_humidity")
|
||||
AbsoluteHumidityComponent = absolute_humidity_ns.class_(
|
||||
"AbsoluteHumidityComponent", sensor.Sensor, cg.Component
|
||||
)
|
||||
|
||||
SaturationVaporPressureEquation = absolute_humidity_ns.enum(
|
||||
"SaturationVaporPressureEquation"
|
||||
)
|
||||
EQUATION = {
|
||||
"BUCK": SaturationVaporPressureEquation.BUCK,
|
||||
"TETENS": SaturationVaporPressureEquation.TETENS,
|
||||
"WOBUS": SaturationVaporPressureEquation.WOBUS,
|
||||
}
|
||||
|
||||
CONFIG_SCHEMA = (
|
||||
sensor.sensor_schema(
|
||||
unit_of_measurement=UNIT_GRAMS_PER_CUBIC_METER,
|
||||
icon=ICON_WATER,
|
||||
accuracy_decimals=2,
|
||||
state_class=STATE_CLASS_MEASUREMENT,
|
||||
)
|
||||
.extend(
|
||||
{
|
||||
cv.GenerateID(): cv.declare_id(AbsoluteHumidityComponent),
|
||||
cv.Required(CONF_TEMPERATURE): cv.use_id(sensor.Sensor),
|
||||
cv.Required(CONF_HUMIDITY): cv.use_id(sensor.Sensor),
|
||||
cv.Optional(CONF_EQUATION, default="WOBUS"): cv.enum(EQUATION, upper=True),
|
||||
}
|
||||
)
|
||||
.extend(cv.COMPONENT_SCHEMA)
|
||||
)
|
||||
|
||||
|
||||
async def to_code(config):
|
||||
var = await sensor.new_sensor(config)
|
||||
await cg.register_component(var, config)
|
||||
|
||||
temperature_sensor = await cg.get_variable(config[CONF_TEMPERATURE])
|
||||
cg.add(var.set_temperature_sensor(temperature_sensor))
|
||||
|
||||
humidity_sensor = await cg.get_variable(config[CONF_HUMIDITY])
|
||||
cg.add(var.set_humidity_sensor(humidity_sensor))
|
||||
|
||||
cg.add(var.set_equation(config[CONF_EQUATION]))
|
@@ -52,10 +52,10 @@ uint32_t IRAM_ATTR HOT AcDimmerDataStore::timer_intr(uint32_t now) {
|
||||
this->gate_pin.digital_write(false);
|
||||
}
|
||||
|
||||
if (time_since_zc < this->enable_time_us)
|
||||
if (time_since_zc < this->enable_time_us) {
|
||||
// Next event is enable, return time until that event
|
||||
return this->enable_time_us - time_since_zc;
|
||||
else if (time_since_zc < disable_time_us) {
|
||||
} else if (time_since_zc < disable_time_us) {
|
||||
// Next event is disable, return time until that event
|
||||
return this->disable_time_us - time_since_zc;
|
||||
}
|
||||
@@ -74,9 +74,10 @@ uint32_t IRAM_ATTR HOT timer_interrupt() {
|
||||
uint32_t min_dt_us = 1000;
|
||||
uint32_t now = micros();
|
||||
for (auto *dimmer : all_dimmers) {
|
||||
if (dimmer == nullptr)
|
||||
if (dimmer == nullptr) {
|
||||
// no more dimmers
|
||||
break;
|
||||
}
|
||||
uint32_t res = dimmer->timer_intr(now);
|
||||
if (res != 0 && res < min_dt_us)
|
||||
min_dt_us = res;
|
||||
@@ -121,6 +122,7 @@ void IRAM_ATTR HOT AcDimmerDataStore::gpio_intr() {
|
||||
// also take into account min_power
|
||||
auto min_us = this->cycle_time_us * this->min_power / 1000;
|
||||
this->enable_time_us = std::max((uint32_t) 1, ((65535 - this->value) * (this->cycle_time_us - min_us)) / 65535);
|
||||
|
||||
if (this->method == DIM_METHOD_LEADING_PULSE) {
|
||||
// Minimum pulse time should be enough for the triac to trigger when it is close to the ZC zone
|
||||
// this is for brightness near 99%
|
||||
@@ -201,6 +203,7 @@ void AcDimmer::setup() {
|
||||
#endif
|
||||
}
|
||||
void AcDimmer::write_state(float state) {
|
||||
state = std::acos(1 - (2 * state)) / 3.14159; // RMS power compensation
|
||||
auto new_value = static_cast<uint16_t>(roundf(state * 65535));
|
||||
if (new_value != 0 && this->store_.value == 0)
|
||||
this->store_.init_cycle = this->init_with_half_cycle_;
|
||||
@@ -212,12 +215,13 @@ void AcDimmer::dump_config() {
|
||||
LOG_PIN(" Zero-Cross Pin: ", this->zero_cross_pin_);
|
||||
ESP_LOGCONFIG(TAG, " Min Power: %.1f%%", this->store_.min_power / 10.0f);
|
||||
ESP_LOGCONFIG(TAG, " Init with half cycle: %s", YESNO(this->init_with_half_cycle_));
|
||||
if (method_ == DIM_METHOD_LEADING_PULSE)
|
||||
if (method_ == DIM_METHOD_LEADING_PULSE) {
|
||||
ESP_LOGCONFIG(TAG, " Method: leading pulse");
|
||||
else if (method_ == DIM_METHOD_LEADING)
|
||||
} else if (method_ == DIM_METHOD_LEADING) {
|
||||
ESP_LOGCONFIG(TAG, " Method: leading");
|
||||
else
|
||||
} else {
|
||||
ESP_LOGCONFIG(TAG, " Method: trailing");
|
||||
}
|
||||
|
||||
LOG_FLOAT_OUTPUT(this);
|
||||
ESP_LOGV(TAG, " Estimated Frequency: %.3fHz", 1e6f / this->store_.cycle_time_us / 2);
|
||||
|
@@ -25,7 +25,7 @@ void AdalightLightEffect::stop() {
|
||||
AddressableLightEffect::stop();
|
||||
}
|
||||
|
||||
int AdalightLightEffect::get_frame_size_(int led_count) const {
|
||||
unsigned int AdalightLightEffect::get_frame_size_(int led_count) const {
|
||||
// 3 bytes: Ada
|
||||
// 2 bytes: LED count
|
||||
// 1 byte: checksum
|
||||
|
@@ -13,7 +13,6 @@ class AdalightLightEffect : public light::AddressableLightEffect, public uart::U
|
||||
public:
|
||||
AdalightLightEffect(const std::string &name);
|
||||
|
||||
public:
|
||||
void start() override;
|
||||
void stop() override;
|
||||
void apply(light::AddressableLight &it, const Color ¤t_color) override;
|
||||
@@ -25,12 +24,11 @@ class AdalightLightEffect : public light::AddressableLightEffect, public uart::U
|
||||
CONSUMED,
|
||||
};
|
||||
|
||||
int get_frame_size_(int led_count) const;
|
||||
unsigned int get_frame_size_(int led_count) const;
|
||||
void reset_frame_(light::AddressableLight &it);
|
||||
void blank_all_leds_(light::AddressableLight &it);
|
||||
Frame parse_frame_(light::AddressableLight &it);
|
||||
|
||||
protected:
|
||||
uint32_t last_ack_{0};
|
||||
uint32_t last_byte_{0};
|
||||
uint32_t last_reset_{0};
|
||||
|
@@ -1 +1,193 @@
|
||||
import esphome.codegen as cg
|
||||
import esphome.config_validation as cv
|
||||
from esphome import pins
|
||||
from esphome.const import CONF_ANALOG, CONF_INPUT
|
||||
|
||||
from esphome.core import CORE
|
||||
from esphome.components.esp32 import get_esp32_variant
|
||||
from esphome.const import PLATFORM_ESP8266
|
||||
from esphome.components.esp32.const import (
|
||||
VARIANT_ESP32,
|
||||
VARIANT_ESP32C2,
|
||||
VARIANT_ESP32C3,
|
||||
VARIANT_ESP32C6,
|
||||
VARIANT_ESP32H2,
|
||||
VARIANT_ESP32S2,
|
||||
VARIANT_ESP32S3,
|
||||
)
|
||||
|
||||
CODEOWNERS = ["@esphome/core"]
|
||||
|
||||
ATTENUATION_MODES = {
|
||||
"0db": cg.global_ns.ADC_ATTEN_DB_0,
|
||||
"2.5db": cg.global_ns.ADC_ATTEN_DB_2_5,
|
||||
"6db": cg.global_ns.ADC_ATTEN_DB_6,
|
||||
"11db": cg.global_ns.ADC_ATTEN_DB_11,
|
||||
"auto": "auto",
|
||||
}
|
||||
|
||||
adc1_channel_t = cg.global_ns.enum("adc1_channel_t")
|
||||
adc2_channel_t = cg.global_ns.enum("adc2_channel_t")
|
||||
|
||||
# From https://github.com/espressif/esp-idf/blob/master/components/driver/include/driver/adc_common.h
|
||||
# pin to adc1 channel mapping
|
||||
ESP32_VARIANT_ADC1_PIN_TO_CHANNEL = {
|
||||
VARIANT_ESP32: {
|
||||
36: adc1_channel_t.ADC1_CHANNEL_0,
|
||||
37: adc1_channel_t.ADC1_CHANNEL_1,
|
||||
38: adc1_channel_t.ADC1_CHANNEL_2,
|
||||
39: adc1_channel_t.ADC1_CHANNEL_3,
|
||||
32: adc1_channel_t.ADC1_CHANNEL_4,
|
||||
33: adc1_channel_t.ADC1_CHANNEL_5,
|
||||
34: adc1_channel_t.ADC1_CHANNEL_6,
|
||||
35: adc1_channel_t.ADC1_CHANNEL_7,
|
||||
},
|
||||
VARIANT_ESP32S2: {
|
||||
1: adc1_channel_t.ADC1_CHANNEL_0,
|
||||
2: adc1_channel_t.ADC1_CHANNEL_1,
|
||||
3: adc1_channel_t.ADC1_CHANNEL_2,
|
||||
4: adc1_channel_t.ADC1_CHANNEL_3,
|
||||
5: adc1_channel_t.ADC1_CHANNEL_4,
|
||||
6: adc1_channel_t.ADC1_CHANNEL_5,
|
||||
7: adc1_channel_t.ADC1_CHANNEL_6,
|
||||
8: adc1_channel_t.ADC1_CHANNEL_7,
|
||||
9: adc1_channel_t.ADC1_CHANNEL_8,
|
||||
10: adc1_channel_t.ADC1_CHANNEL_9,
|
||||
},
|
||||
VARIANT_ESP32S3: {
|
||||
1: adc1_channel_t.ADC1_CHANNEL_0,
|
||||
2: adc1_channel_t.ADC1_CHANNEL_1,
|
||||
3: adc1_channel_t.ADC1_CHANNEL_2,
|
||||
4: adc1_channel_t.ADC1_CHANNEL_3,
|
||||
5: adc1_channel_t.ADC1_CHANNEL_4,
|
||||
6: adc1_channel_t.ADC1_CHANNEL_5,
|
||||
7: adc1_channel_t.ADC1_CHANNEL_6,
|
||||
8: adc1_channel_t.ADC1_CHANNEL_7,
|
||||
9: adc1_channel_t.ADC1_CHANNEL_8,
|
||||
10: adc1_channel_t.ADC1_CHANNEL_9,
|
||||
},
|
||||
VARIANT_ESP32C3: {
|
||||
0: adc1_channel_t.ADC1_CHANNEL_0,
|
||||
1: adc1_channel_t.ADC1_CHANNEL_1,
|
||||
2: adc1_channel_t.ADC1_CHANNEL_2,
|
||||
3: adc1_channel_t.ADC1_CHANNEL_3,
|
||||
4: adc1_channel_t.ADC1_CHANNEL_4,
|
||||
},
|
||||
VARIANT_ESP32C2: {
|
||||
0: adc1_channel_t.ADC1_CHANNEL_0,
|
||||
1: adc1_channel_t.ADC1_CHANNEL_1,
|
||||
2: adc1_channel_t.ADC1_CHANNEL_2,
|
||||
3: adc1_channel_t.ADC1_CHANNEL_3,
|
||||
4: adc1_channel_t.ADC1_CHANNEL_4,
|
||||
},
|
||||
VARIANT_ESP32C6: {
|
||||
0: adc1_channel_t.ADC1_CHANNEL_0,
|
||||
1: adc1_channel_t.ADC1_CHANNEL_1,
|
||||
2: adc1_channel_t.ADC1_CHANNEL_2,
|
||||
3: adc1_channel_t.ADC1_CHANNEL_3,
|
||||
4: adc1_channel_t.ADC1_CHANNEL_4,
|
||||
5: adc1_channel_t.ADC1_CHANNEL_5,
|
||||
6: adc1_channel_t.ADC1_CHANNEL_6,
|
||||
},
|
||||
VARIANT_ESP32H2: {
|
||||
0: adc1_channel_t.ADC1_CHANNEL_0,
|
||||
1: adc1_channel_t.ADC1_CHANNEL_1,
|
||||
2: adc1_channel_t.ADC1_CHANNEL_2,
|
||||
3: adc1_channel_t.ADC1_CHANNEL_3,
|
||||
4: adc1_channel_t.ADC1_CHANNEL_4,
|
||||
},
|
||||
}
|
||||
|
||||
ESP32_VARIANT_ADC2_PIN_TO_CHANNEL = {
|
||||
# TODO: add other variants
|
||||
VARIANT_ESP32: {
|
||||
4: adc2_channel_t.ADC2_CHANNEL_0,
|
||||
0: adc2_channel_t.ADC2_CHANNEL_1,
|
||||
2: adc2_channel_t.ADC2_CHANNEL_2,
|
||||
15: adc2_channel_t.ADC2_CHANNEL_3,
|
||||
13: adc2_channel_t.ADC2_CHANNEL_4,
|
||||
12: adc2_channel_t.ADC2_CHANNEL_5,
|
||||
14: adc2_channel_t.ADC2_CHANNEL_6,
|
||||
27: adc2_channel_t.ADC2_CHANNEL_7,
|
||||
25: adc2_channel_t.ADC2_CHANNEL_8,
|
||||
26: adc2_channel_t.ADC2_CHANNEL_9,
|
||||
},
|
||||
VARIANT_ESP32S2: {
|
||||
11: adc2_channel_t.ADC2_CHANNEL_0,
|
||||
12: adc2_channel_t.ADC2_CHANNEL_1,
|
||||
13: adc2_channel_t.ADC2_CHANNEL_2,
|
||||
14: adc2_channel_t.ADC2_CHANNEL_3,
|
||||
15: adc2_channel_t.ADC2_CHANNEL_4,
|
||||
16: adc2_channel_t.ADC2_CHANNEL_5,
|
||||
17: adc2_channel_t.ADC2_CHANNEL_6,
|
||||
18: adc2_channel_t.ADC2_CHANNEL_7,
|
||||
19: adc2_channel_t.ADC2_CHANNEL_8,
|
||||
20: adc2_channel_t.ADC2_CHANNEL_9,
|
||||
},
|
||||
VARIANT_ESP32S3: {
|
||||
11: adc2_channel_t.ADC2_CHANNEL_0,
|
||||
12: adc2_channel_t.ADC2_CHANNEL_1,
|
||||
13: adc2_channel_t.ADC2_CHANNEL_2,
|
||||
14: adc2_channel_t.ADC2_CHANNEL_3,
|
||||
15: adc2_channel_t.ADC2_CHANNEL_4,
|
||||
16: adc2_channel_t.ADC2_CHANNEL_5,
|
||||
17: adc2_channel_t.ADC2_CHANNEL_6,
|
||||
18: adc2_channel_t.ADC2_CHANNEL_7,
|
||||
19: adc2_channel_t.ADC2_CHANNEL_8,
|
||||
20: adc2_channel_t.ADC2_CHANNEL_9,
|
||||
},
|
||||
VARIANT_ESP32C3: {
|
||||
5: adc2_channel_t.ADC2_CHANNEL_0,
|
||||
},
|
||||
}
|
||||
|
||||
|
||||
def validate_adc_pin(value):
|
||||
if str(value).upper() == "VCC":
|
||||
if CORE.is_rp2040:
|
||||
return pins.internal_gpio_input_pin_schema(29)
|
||||
return cv.only_on([PLATFORM_ESP8266])("VCC")
|
||||
|
||||
if str(value).upper() == "TEMPERATURE":
|
||||
return cv.only_on_rp2040("TEMPERATURE")
|
||||
|
||||
if CORE.is_esp32:
|
||||
value = pins.internal_gpio_input_pin_number(value)
|
||||
variant = get_esp32_variant()
|
||||
if (
|
||||
variant not in ESP32_VARIANT_ADC1_PIN_TO_CHANNEL
|
||||
and variant not in ESP32_VARIANT_ADC2_PIN_TO_CHANNEL
|
||||
):
|
||||
raise cv.Invalid(f"This ESP32 variant ({variant}) is not supported")
|
||||
|
||||
if (
|
||||
value not in ESP32_VARIANT_ADC1_PIN_TO_CHANNEL[variant]
|
||||
and value not in ESP32_VARIANT_ADC2_PIN_TO_CHANNEL[variant]
|
||||
):
|
||||
raise cv.Invalid(f"{variant} doesn't support ADC on this pin")
|
||||
|
||||
return pins.internal_gpio_input_pin_schema(value)
|
||||
|
||||
if CORE.is_esp8266:
|
||||
value = pins.internal_gpio_pin_number({CONF_ANALOG: True, CONF_INPUT: True})(
|
||||
value
|
||||
)
|
||||
|
||||
if value != 17: # A0
|
||||
raise cv.Invalid("ESP8266: Only pin A0 (GPIO17) supports ADC")
|
||||
return pins.gpio_pin_schema(
|
||||
{CONF_ANALOG: True, CONF_INPUT: True}, internal=True
|
||||
)(value)
|
||||
|
||||
if CORE.is_rp2040:
|
||||
value = pins.internal_gpio_input_pin_number(value)
|
||||
if value not in (26, 27, 28, 29):
|
||||
raise cv.Invalid("RP2040: Only pins 26, 27, 28 and 29 support ADC")
|
||||
return pins.internal_gpio_input_pin_schema(value)
|
||||
|
||||
if CORE.is_libretiny:
|
||||
return pins.gpio_pin_schema(
|
||||
{CONF_ANALOG: True, CONF_INPUT: True}, internal=True
|
||||
)(value)
|
||||
|
||||
raise NotImplementedError
|
||||
|
@@ -1,4 +1,5 @@
|
||||
#include "adc_sensor.h"
|
||||
#include "esphome/core/helpers.h"
|
||||
#include "esphome/core/log.h"
|
||||
|
||||
#ifdef USE_ESP8266
|
||||
@@ -10,155 +11,289 @@ ADC_MODE(ADC_VCC)
|
||||
#endif
|
||||
#endif
|
||||
|
||||
#ifdef USE_RP2040
|
||||
#ifdef CYW43_USES_VSYS_PIN
|
||||
#include "pico/cyw43_arch.h"
|
||||
#endif
|
||||
#include <hardware/adc.h>
|
||||
#endif
|
||||
|
||||
namespace esphome {
|
||||
namespace adc {
|
||||
|
||||
static const char *const TAG = "adc";
|
||||
|
||||
// 13-bit for S2, 12-bit for all other ESP32 variants
|
||||
#ifdef USE_ESP32
|
||||
void ADCSensor::set_attenuation(adc_atten_t attenuation) { this->attenuation_ = attenuation; }
|
||||
static const adc_bits_width_t ADC_WIDTH_MAX_SOC_BITS = static_cast<adc_bits_width_t>(ADC_WIDTH_MAX - 1);
|
||||
|
||||
inline adc1_channel_t gpio_to_adc1(uint8_t pin) {
|
||||
#if CONFIG_IDF_TARGET_ESP32
|
||||
switch (pin) {
|
||||
case 36:
|
||||
return ADC1_CHANNEL_0;
|
||||
case 37:
|
||||
return ADC1_CHANNEL_1;
|
||||
case 38:
|
||||
return ADC1_CHANNEL_2;
|
||||
case 39:
|
||||
return ADC1_CHANNEL_3;
|
||||
case 32:
|
||||
return ADC1_CHANNEL_4;
|
||||
case 33:
|
||||
return ADC1_CHANNEL_5;
|
||||
case 34:
|
||||
return ADC1_CHANNEL_6;
|
||||
case 35:
|
||||
return ADC1_CHANNEL_7;
|
||||
default:
|
||||
return ADC1_CHANNEL_MAX;
|
||||
}
|
||||
#elif CONFIG_IDF_TARGET_ESP32C3 || CONFIG_IDF_TARGET_ESP32H2
|
||||
switch (pin) {
|
||||
case 0:
|
||||
return ADC1_CHANNEL_0;
|
||||
case 1:
|
||||
return ADC1_CHANNEL_1;
|
||||
case 2:
|
||||
return ADC1_CHANNEL_2;
|
||||
case 3:
|
||||
return ADC1_CHANNEL_3;
|
||||
case 4:
|
||||
return ADC1_CHANNEL_4;
|
||||
default:
|
||||
return ADC1_CHANNEL_MAX;
|
||||
}
|
||||
#ifndef SOC_ADC_RTC_MAX_BITWIDTH
|
||||
#if USE_ESP32_VARIANT_ESP32S2
|
||||
static const int32_t SOC_ADC_RTC_MAX_BITWIDTH = 13;
|
||||
#else
|
||||
static const int32_t SOC_ADC_RTC_MAX_BITWIDTH = 12;
|
||||
#endif
|
||||
}
|
||||
#endif
|
||||
|
||||
void ADCSensor::setup() {
|
||||
static const int ADC_MAX = (1 << SOC_ADC_RTC_MAX_BITWIDTH) - 1; // 4095 (12 bit) or 8191 (13 bit)
|
||||
static const int ADC_HALF = (1 << SOC_ADC_RTC_MAX_BITWIDTH) >> 1; // 2048 (12 bit) or 4096 (13 bit)
|
||||
#endif
|
||||
|
||||
#ifdef USE_RP2040
|
||||
extern "C"
|
||||
#endif
|
||||
void
|
||||
ADCSensor::setup() {
|
||||
ESP_LOGCONFIG(TAG, "Setting up ADC '%s'...", this->get_name().c_str());
|
||||
#ifndef USE_ADC_SENSOR_VCC
|
||||
#if !defined(USE_ADC_SENSOR_VCC) && !defined(USE_RP2040)
|
||||
pin_->setup();
|
||||
#endif
|
||||
|
||||
#ifdef USE_ESP32
|
||||
adc1_config_channel_atten(gpio_to_adc1(pin_->get_pin()), attenuation_);
|
||||
adc1_config_width(ADC_WIDTH_BIT_12);
|
||||
#if !CONFIG_IDF_TARGET_ESP32C3 && !CONFIG_IDF_TARGET_ESP32H2
|
||||
adc_gpio_init(ADC_UNIT_1, (adc_channel_t) gpio_to_adc1(pin_->get_pin()));
|
||||
#endif
|
||||
if (channel1_ != ADC1_CHANNEL_MAX) {
|
||||
adc1_config_width(ADC_WIDTH_MAX_SOC_BITS);
|
||||
if (!autorange_) {
|
||||
adc1_config_channel_atten(channel1_, attenuation_);
|
||||
}
|
||||
} else if (channel2_ != ADC2_CHANNEL_MAX) {
|
||||
if (!autorange_) {
|
||||
adc2_config_channel_atten(channel2_, attenuation_);
|
||||
}
|
||||
}
|
||||
|
||||
// load characteristics for each attenuation
|
||||
for (int32_t i = 0; i <= ADC_ATTEN_DB_11; i++) {
|
||||
auto adc_unit = channel1_ != ADC1_CHANNEL_MAX ? ADC_UNIT_1 : ADC_UNIT_2;
|
||||
auto cal_value = esp_adc_cal_characterize(adc_unit, (adc_atten_t) i, ADC_WIDTH_MAX_SOC_BITS,
|
||||
1100, // default vref
|
||||
&cal_characteristics_[i]);
|
||||
switch (cal_value) {
|
||||
case ESP_ADC_CAL_VAL_EFUSE_VREF:
|
||||
ESP_LOGV(TAG, "Using eFuse Vref for calibration");
|
||||
break;
|
||||
case ESP_ADC_CAL_VAL_EFUSE_TP:
|
||||
ESP_LOGV(TAG, "Using two-point eFuse Vref for calibration");
|
||||
break;
|
||||
case ESP_ADC_CAL_VAL_DEFAULT_VREF:
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
#endif // USE_ESP32
|
||||
|
||||
#ifdef USE_RP2040
|
||||
static bool initialized = false;
|
||||
if (!initialized) {
|
||||
adc_init();
|
||||
initialized = true;
|
||||
}
|
||||
#endif
|
||||
|
||||
ESP_LOGCONFIG(TAG, "ADC '%s' setup finished!", this->get_name().c_str());
|
||||
}
|
||||
|
||||
void ADCSensor::dump_config() {
|
||||
LOG_SENSOR("", "ADC Sensor", this);
|
||||
#ifdef USE_ESP8266
|
||||
#if defined(USE_ESP8266) || defined(USE_LIBRETINY)
|
||||
#ifdef USE_ADC_SENSOR_VCC
|
||||
ESP_LOGCONFIG(TAG, " Pin: VCC");
|
||||
#else
|
||||
LOG_PIN(" Pin: ", pin_);
|
||||
#endif
|
||||
#endif
|
||||
#endif // USE_ESP8266 || USE_LIBRETINY
|
||||
|
||||
#ifdef USE_ESP32
|
||||
LOG_PIN(" Pin: ", pin_);
|
||||
switch (this->attenuation_) {
|
||||
case ADC_ATTEN_DB_0:
|
||||
ESP_LOGCONFIG(TAG, " Attenuation: 0db (max 1.1V)");
|
||||
break;
|
||||
case ADC_ATTEN_DB_2_5:
|
||||
ESP_LOGCONFIG(TAG, " Attenuation: 2.5db (max 1.5V)");
|
||||
break;
|
||||
case ADC_ATTEN_DB_6:
|
||||
ESP_LOGCONFIG(TAG, " Attenuation: 6db (max 2.2V)");
|
||||
break;
|
||||
case ADC_ATTEN_DB_11:
|
||||
ESP_LOGCONFIG(TAG, " Attenuation: 11db (max 3.9V)");
|
||||
break;
|
||||
default: // This is to satisfy the unused ADC_ATTEN_MAX
|
||||
break;
|
||||
if (autorange_) {
|
||||
ESP_LOGCONFIG(TAG, " Attenuation: auto");
|
||||
} else {
|
||||
switch (this->attenuation_) {
|
||||
case ADC_ATTEN_DB_0:
|
||||
ESP_LOGCONFIG(TAG, " Attenuation: 0db");
|
||||
break;
|
||||
case ADC_ATTEN_DB_2_5:
|
||||
ESP_LOGCONFIG(TAG, " Attenuation: 2.5db");
|
||||
break;
|
||||
case ADC_ATTEN_DB_6:
|
||||
ESP_LOGCONFIG(TAG, " Attenuation: 6db");
|
||||
break;
|
||||
case ADC_ATTEN_DB_11:
|
||||
ESP_LOGCONFIG(TAG, " Attenuation: 11db");
|
||||
break;
|
||||
default: // This is to satisfy the unused ADC_ATTEN_MAX
|
||||
break;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
#endif // USE_ESP32
|
||||
|
||||
#ifdef USE_RP2040
|
||||
if (this->is_temperature_) {
|
||||
ESP_LOGCONFIG(TAG, " Pin: Temperature");
|
||||
} else {
|
||||
#ifdef USE_ADC_SENSOR_VCC
|
||||
ESP_LOGCONFIG(TAG, " Pin: VCC");
|
||||
#else
|
||||
LOG_PIN(" Pin: ", pin_);
|
||||
#endif // USE_ADC_SENSOR_VCC
|
||||
}
|
||||
#endif // USE_RP2040
|
||||
|
||||
LOG_UPDATE_INTERVAL(this);
|
||||
}
|
||||
|
||||
float ADCSensor::get_setup_priority() const { return setup_priority::DATA; }
|
||||
void ADCSensor::update() {
|
||||
float value_v = this->sample();
|
||||
ESP_LOGD(TAG, "'%s': Got voltage=%.2fV", this->get_name().c_str(), value_v);
|
||||
ESP_LOGV(TAG, "'%s': Got voltage=%.4fV", this->get_name().c_str(), value_v);
|
||||
this->publish_state(value_v);
|
||||
}
|
||||
float ADCSensor::sample() {
|
||||
#ifdef USE_ESP32
|
||||
int raw = adc1_get_raw(gpio_to_adc1(pin_->get_pin()));
|
||||
float value_v = raw / 4095.0f;
|
||||
#if CONFIG_IDF_TARGET_ESP32
|
||||
switch (this->attenuation_) {
|
||||
case ADC_ATTEN_DB_0:
|
||||
value_v *= 1.1;
|
||||
break;
|
||||
case ADC_ATTEN_DB_2_5:
|
||||
value_v *= 1.5;
|
||||
break;
|
||||
case ADC_ATTEN_DB_6:
|
||||
value_v *= 2.2;
|
||||
break;
|
||||
case ADC_ATTEN_DB_11:
|
||||
value_v *= 3.9;
|
||||
break;
|
||||
default: // This is to satisfy the unused ADC_ATTEN_MAX
|
||||
break;
|
||||
}
|
||||
#elif CONFIG_IDF_TARGET_ESP32C3 || CONFIG_IDF_TARGET_ESP32H2
|
||||
switch (this->attenuation_) {
|
||||
case ADC_ATTEN_DB_0:
|
||||
value_v *= 0.84;
|
||||
break;
|
||||
case ADC_ATTEN_DB_2_5:
|
||||
value_v *= 1.13;
|
||||
break;
|
||||
case ADC_ATTEN_DB_6:
|
||||
value_v *= 1.56;
|
||||
break;
|
||||
case ADC_ATTEN_DB_11:
|
||||
value_v *= 3.0;
|
||||
break;
|
||||
default: // This is to satisfy the unused ADC_ATTEN_MAX
|
||||
break;
|
||||
}
|
||||
#endif
|
||||
return value_v;
|
||||
#endif
|
||||
|
||||
#ifdef USE_ESP8266
|
||||
float ADCSensor::sample() {
|
||||
#ifdef USE_ADC_SENSOR_VCC
|
||||
return ESP.getVcc() / 1024.0f; // NOLINT(readability-static-accessed-through-instance)
|
||||
int32_t raw = ESP.getVcc(); // NOLINT(readability-static-accessed-through-instance)
|
||||
#else
|
||||
return analogRead(this->pin_->get_pin()) / 1024.0f; // NOLINT
|
||||
#endif
|
||||
int32_t raw = analogRead(this->pin_->get_pin()); // NOLINT
|
||||
#endif
|
||||
if (output_raw_) {
|
||||
return raw;
|
||||
}
|
||||
return raw / 1024.0f;
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef USE_ESP32
|
||||
float ADCSensor::sample() {
|
||||
if (!autorange_) {
|
||||
int raw = -1;
|
||||
if (channel1_ != ADC1_CHANNEL_MAX) {
|
||||
raw = adc1_get_raw(channel1_);
|
||||
} else if (channel2_ != ADC2_CHANNEL_MAX) {
|
||||
adc2_get_raw(channel2_, ADC_WIDTH_MAX_SOC_BITS, &raw);
|
||||
}
|
||||
|
||||
if (raw == -1) {
|
||||
return NAN;
|
||||
}
|
||||
if (output_raw_) {
|
||||
return raw;
|
||||
}
|
||||
uint32_t mv = esp_adc_cal_raw_to_voltage(raw, &cal_characteristics_[(int32_t) attenuation_]);
|
||||
return mv / 1000.0f;
|
||||
}
|
||||
|
||||
int raw11 = ADC_MAX, raw6 = ADC_MAX, raw2 = ADC_MAX, raw0 = ADC_MAX;
|
||||
|
||||
if (channel1_ != ADC1_CHANNEL_MAX) {
|
||||
adc1_config_channel_atten(channel1_, ADC_ATTEN_DB_11);
|
||||
raw11 = adc1_get_raw(channel1_);
|
||||
if (raw11 < ADC_MAX) {
|
||||
adc1_config_channel_atten(channel1_, ADC_ATTEN_DB_6);
|
||||
raw6 = adc1_get_raw(channel1_);
|
||||
if (raw6 < ADC_MAX) {
|
||||
adc1_config_channel_atten(channel1_, ADC_ATTEN_DB_2_5);
|
||||
raw2 = adc1_get_raw(channel1_);
|
||||
if (raw2 < ADC_MAX) {
|
||||
adc1_config_channel_atten(channel1_, ADC_ATTEN_DB_0);
|
||||
raw0 = adc1_get_raw(channel1_);
|
||||
}
|
||||
}
|
||||
}
|
||||
} else if (channel2_ != ADC2_CHANNEL_MAX) {
|
||||
adc2_config_channel_atten(channel2_, ADC_ATTEN_DB_11);
|
||||
adc2_get_raw(channel2_, ADC_WIDTH_MAX_SOC_BITS, &raw11);
|
||||
if (raw11 < ADC_MAX) {
|
||||
adc2_config_channel_atten(channel2_, ADC_ATTEN_DB_6);
|
||||
adc2_get_raw(channel2_, ADC_WIDTH_MAX_SOC_BITS, &raw6);
|
||||
if (raw6 < ADC_MAX) {
|
||||
adc2_config_channel_atten(channel2_, ADC_ATTEN_DB_2_5);
|
||||
adc2_get_raw(channel2_, ADC_WIDTH_MAX_SOC_BITS, &raw2);
|
||||
if (raw2 < ADC_MAX) {
|
||||
adc2_config_channel_atten(channel2_, ADC_ATTEN_DB_0);
|
||||
adc2_get_raw(channel2_, ADC_WIDTH_MAX_SOC_BITS, &raw0);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (raw0 == -1 || raw2 == -1 || raw6 == -1 || raw11 == -1) {
|
||||
return NAN;
|
||||
}
|
||||
|
||||
uint32_t mv11 = esp_adc_cal_raw_to_voltage(raw11, &cal_characteristics_[(int32_t) ADC_ATTEN_DB_11]);
|
||||
uint32_t mv6 = esp_adc_cal_raw_to_voltage(raw6, &cal_characteristics_[(int32_t) ADC_ATTEN_DB_6]);
|
||||
uint32_t mv2 = esp_adc_cal_raw_to_voltage(raw2, &cal_characteristics_[(int32_t) ADC_ATTEN_DB_2_5]);
|
||||
uint32_t mv0 = esp_adc_cal_raw_to_voltage(raw0, &cal_characteristics_[(int32_t) ADC_ATTEN_DB_0]);
|
||||
|
||||
// Contribution of each value, in range 0-2048 (12 bit ADC) or 0-4096 (13 bit ADC)
|
||||
uint32_t c11 = std::min(raw11, ADC_HALF);
|
||||
uint32_t c6 = ADC_HALF - std::abs(raw6 - ADC_HALF);
|
||||
uint32_t c2 = ADC_HALF - std::abs(raw2 - ADC_HALF);
|
||||
uint32_t c0 = std::min(ADC_MAX - raw0, ADC_HALF);
|
||||
// max theoretical csum value is 4096*4 = 16384
|
||||
uint32_t csum = c11 + c6 + c2 + c0;
|
||||
|
||||
// each mv is max 3900; so max value is 3900*4096*4, fits in unsigned32
|
||||
uint32_t mv_scaled = (mv11 * c11) + (mv6 * c6) + (mv2 * c2) + (mv0 * c0);
|
||||
return mv_scaled / (float) (csum * 1000U);
|
||||
}
|
||||
#endif // USE_ESP32
|
||||
|
||||
#ifdef USE_RP2040
|
||||
float ADCSensor::sample() {
|
||||
if (this->is_temperature_) {
|
||||
adc_set_temp_sensor_enabled(true);
|
||||
delay(1);
|
||||
adc_select_input(4);
|
||||
|
||||
int32_t raw = adc_read();
|
||||
adc_set_temp_sensor_enabled(false);
|
||||
if (this->output_raw_) {
|
||||
return raw;
|
||||
}
|
||||
return raw * 3.3f / 4096.0f;
|
||||
} else {
|
||||
uint8_t pin = this->pin_->get_pin();
|
||||
#ifdef CYW43_USES_VSYS_PIN
|
||||
if (pin == PICO_VSYS_PIN) {
|
||||
// Measuring VSYS on Raspberry Pico W needs to be wrapped with
|
||||
// `cyw43_thread_enter()`/`cyw43_thread_exit()` as discussed in
|
||||
// https://github.com/raspberrypi/pico-sdk/issues/1222, since Wifi chip and
|
||||
// VSYS ADC both share GPIO29
|
||||
cyw43_thread_enter();
|
||||
}
|
||||
#endif // CYW43_USES_VSYS_PIN
|
||||
|
||||
adc_gpio_init(pin);
|
||||
adc_select_input(pin - 26);
|
||||
|
||||
int32_t raw = adc_read();
|
||||
|
||||
#ifdef CYW43_USES_VSYS_PIN
|
||||
if (pin == PICO_VSYS_PIN) {
|
||||
cyw43_thread_exit();
|
||||
}
|
||||
#endif // CYW43_USES_VSYS_PIN
|
||||
|
||||
if (output_raw_) {
|
||||
return raw;
|
||||
}
|
||||
float coeff = pin == PICO_VSYS_PIN ? 3.0 : 1.0;
|
||||
return raw * 3.3f / 4096.0f * coeff;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef USE_LIBRETINY
|
||||
float ADCSensor::sample() {
|
||||
if (output_raw_) {
|
||||
return analogRead(this->pin_->get_pin()); // NOLINT
|
||||
}
|
||||
return analogReadVoltage(this->pin_->get_pin()) / 1000.0f; // NOLINT
|
||||
}
|
||||
#endif // USE_LIBRETINY
|
||||
|
||||
#ifdef USE_ESP8266
|
||||
std::string ADCSensor::unique_id() { return get_mac_address() + "-adc"; }
|
||||
#endif
|
||||
|
@@ -8,6 +8,7 @@
|
||||
|
||||
#ifdef USE_ESP32
|
||||
#include "driver/adc.h"
|
||||
#include <esp_adc_cal.h>
|
||||
#endif
|
||||
|
||||
namespace esphome {
|
||||
@@ -17,28 +18,55 @@ class ADCSensor : public sensor::Sensor, public PollingComponent, public voltage
|
||||
public:
|
||||
#ifdef USE_ESP32
|
||||
/// Set the attenuation for this pin. Only available on the ESP32.
|
||||
void set_attenuation(adc_atten_t attenuation);
|
||||
void set_attenuation(adc_atten_t attenuation) { attenuation_ = attenuation; }
|
||||
void set_channel1(adc1_channel_t channel) {
|
||||
channel1_ = channel;
|
||||
channel2_ = ADC2_CHANNEL_MAX;
|
||||
}
|
||||
void set_channel2(adc2_channel_t channel) {
|
||||
channel2_ = channel;
|
||||
channel1_ = ADC1_CHANNEL_MAX;
|
||||
}
|
||||
void set_autorange(bool autorange) { autorange_ = autorange; }
|
||||
#endif
|
||||
|
||||
/// Update adc values.
|
||||
/// Update ADC values
|
||||
void update() override;
|
||||
/// Setup ADc
|
||||
/// Setup ADC
|
||||
void setup() override;
|
||||
void dump_config() override;
|
||||
/// `HARDWARE_LATE` setup priority.
|
||||
/// `HARDWARE_LATE` setup priority
|
||||
float get_setup_priority() const override;
|
||||
void set_pin(InternalGPIOPin *pin) { this->pin_ = pin; }
|
||||
void set_output_raw(bool output_raw) { output_raw_ = output_raw; }
|
||||
float sample() override;
|
||||
|
||||
#ifdef USE_ESP8266
|
||||
std::string unique_id() override;
|
||||
#endif
|
||||
|
||||
#ifdef USE_RP2040
|
||||
void set_is_temperature() { is_temperature_ = true; }
|
||||
#endif
|
||||
|
||||
protected:
|
||||
InternalGPIOPin *pin_;
|
||||
bool output_raw_{false};
|
||||
|
||||
#ifdef USE_RP2040
|
||||
bool is_temperature_{false};
|
||||
#endif
|
||||
|
||||
#ifdef USE_ESP32
|
||||
adc_atten_t attenuation_{ADC_ATTEN_DB_0};
|
||||
adc1_channel_t channel1_{ADC1_CHANNEL_MAX};
|
||||
adc2_channel_t channel2_{ADC2_CHANNEL_MAX};
|
||||
bool autorange_{false};
|
||||
#if ESP_IDF_VERSION_MAJOR >= 5
|
||||
esp_adc_cal_characteristics_t cal_characteristics_[SOC_ADC_ATTEN_NUM] = {};
|
||||
#else
|
||||
esp_adc_cal_characteristics_t cal_characteristics_[ADC_ATTEN_MAX] = {};
|
||||
#endif
|
||||
#endif
|
||||
};
|
||||
|
||||
|
@@ -1,58 +1,50 @@
|
||||
import esphome.codegen as cg
|
||||
import esphome.config_validation as cv
|
||||
from esphome import pins
|
||||
import esphome.final_validate as fv
|
||||
from esphome.core import CORE
|
||||
from esphome.components import sensor, voltage_sampler
|
||||
from esphome.components.esp32 import get_esp32_variant
|
||||
from esphome.const import (
|
||||
CONF_ATTENUATION,
|
||||
CONF_ID,
|
||||
CONF_INPUT,
|
||||
CONF_NUMBER,
|
||||
CONF_PIN,
|
||||
CONF_RAW,
|
||||
CONF_WIFI,
|
||||
DEVICE_CLASS_VOLTAGE,
|
||||
STATE_CLASS_MEASUREMENT,
|
||||
UNIT_VOLT,
|
||||
)
|
||||
from esphome.core import CORE
|
||||
|
||||
from . import (
|
||||
ATTENUATION_MODES,
|
||||
ESP32_VARIANT_ADC1_PIN_TO_CHANNEL,
|
||||
ESP32_VARIANT_ADC2_PIN_TO_CHANNEL,
|
||||
validate_adc_pin,
|
||||
)
|
||||
|
||||
AUTO_LOAD = ["voltage_sampler"]
|
||||
|
||||
ATTENUATION_MODES = {
|
||||
"0db": cg.global_ns.ADC_ATTEN_DB_0,
|
||||
"2.5db": cg.global_ns.ADC_ATTEN_DB_2_5,
|
||||
"6db": cg.global_ns.ADC_ATTEN_DB_6,
|
||||
"11db": cg.global_ns.ADC_ATTEN_DB_11,
|
||||
}
|
||||
|
||||
def validate_config(config):
|
||||
if config[CONF_RAW] and config.get(CONF_ATTENUATION, None) == "auto":
|
||||
raise cv.Invalid("Automatic attenuation cannot be used when raw output is set")
|
||||
|
||||
return config
|
||||
|
||||
|
||||
def validate_adc_pin(value):
|
||||
if str(value).upper() == "VCC":
|
||||
return cv.only_on_esp8266("VCC")
|
||||
|
||||
def final_validate_config(config):
|
||||
if CORE.is_esp32:
|
||||
from esphome.components.esp32 import is_esp32c3
|
||||
variant = get_esp32_variant()
|
||||
if (
|
||||
CONF_WIFI in fv.full_config.get()
|
||||
and config[CONF_PIN][CONF_NUMBER]
|
||||
in ESP32_VARIANT_ADC2_PIN_TO_CHANNEL[variant]
|
||||
):
|
||||
raise cv.Invalid(
|
||||
f"{variant} doesn't support ADC on this pin when Wi-Fi is configured"
|
||||
)
|
||||
|
||||
value = pins.internal_gpio_input_pin_number(value)
|
||||
if is_esp32c3():
|
||||
if not (0 <= value <= 4): # ADC1
|
||||
raise cv.Invalid("ESP32-C3: Only pins 0 though 4 support ADC.")
|
||||
if not (32 <= value <= 39): # ADC1
|
||||
raise cv.Invalid("ESP32: Only pins 32 though 39 support ADC.")
|
||||
elif CORE.is_esp8266:
|
||||
from esphome.components.esp8266.gpio import CONF_ANALOG
|
||||
|
||||
value = pins.internal_gpio_pin_number({CONF_ANALOG: True, CONF_INPUT: True})(
|
||||
value
|
||||
)
|
||||
|
||||
if value != 17: # A0
|
||||
raise cv.Invalid("ESP8266: Only pin A0 (GPIO17) supports ADC.")
|
||||
return pins.gpio_pin_schema(
|
||||
{CONF_ANALOG: True, CONF_INPUT: True}, internal=True
|
||||
)(value)
|
||||
else:
|
||||
raise NotImplementedError
|
||||
|
||||
return pins.internal_gpio_input_pin_schema(value)
|
||||
return config
|
||||
|
||||
|
||||
adc_ns = cg.esphome_ns.namespace("adc")
|
||||
@@ -60,8 +52,9 @@ ADCSensor = adc_ns.class_(
|
||||
"ADCSensor", sensor.Sensor, cg.PollingComponent, voltage_sampler.VoltageSampler
|
||||
)
|
||||
|
||||
CONFIG_SCHEMA = (
|
||||
CONFIG_SCHEMA = cv.All(
|
||||
sensor.sensor_schema(
|
||||
ADCSensor,
|
||||
unit_of_measurement=UNIT_VOLT,
|
||||
accuracy_decimals=2,
|
||||
device_class=DEVICE_CLASS_VOLTAGE,
|
||||
@@ -69,16 +62,19 @@ CONFIG_SCHEMA = (
|
||||
)
|
||||
.extend(
|
||||
{
|
||||
cv.GenerateID(): cv.declare_id(ADCSensor),
|
||||
cv.Required(CONF_PIN): validate_adc_pin,
|
||||
cv.Optional(CONF_RAW, default=False): cv.boolean,
|
||||
cv.SplitDefault(CONF_ATTENUATION, esp32="0db"): cv.All(
|
||||
cv.only_on_esp32, cv.enum(ATTENUATION_MODES, lower=True)
|
||||
),
|
||||
}
|
||||
)
|
||||
.extend(cv.polling_component_schema("60s"))
|
||||
.extend(cv.polling_component_schema("60s")),
|
||||
validate_config,
|
||||
)
|
||||
|
||||
FINAL_VALIDATE_SCHEMA = final_validate_config
|
||||
|
||||
|
||||
async def to_code(config):
|
||||
var = cg.new_Pvariable(config[CONF_ID])
|
||||
@@ -87,9 +83,32 @@ async def to_code(config):
|
||||
|
||||
if config[CONF_PIN] == "VCC":
|
||||
cg.add_define("USE_ADC_SENSOR_VCC")
|
||||
elif config[CONF_PIN] == "TEMPERATURE":
|
||||
cg.add(var.set_is_temperature())
|
||||
else:
|
||||
pin = await cg.gpio_pin_expression(config[CONF_PIN])
|
||||
cg.add(var.set_pin(pin))
|
||||
|
||||
if CONF_ATTENUATION in config:
|
||||
cg.add(var.set_attenuation(config[CONF_ATTENUATION]))
|
||||
cg.add(var.set_output_raw(config[CONF_RAW]))
|
||||
|
||||
if attenuation := config.get(CONF_ATTENUATION):
|
||||
if attenuation == "auto":
|
||||
cg.add(var.set_autorange(cg.global_ns.true))
|
||||
else:
|
||||
cg.add(var.set_attenuation(attenuation))
|
||||
|
||||
if CORE.is_esp32:
|
||||
variant = get_esp32_variant()
|
||||
pin_num = config[CONF_PIN][CONF_NUMBER]
|
||||
if (
|
||||
variant in ESP32_VARIANT_ADC1_PIN_TO_CHANNEL
|
||||
and pin_num in ESP32_VARIANT_ADC1_PIN_TO_CHANNEL[variant]
|
||||
):
|
||||
chan = ESP32_VARIANT_ADC1_PIN_TO_CHANNEL[variant][pin_num]
|
||||
cg.add(var.set_channel1(chan))
|
||||
elif (
|
||||
variant in ESP32_VARIANT_ADC2_PIN_TO_CHANNEL
|
||||
and pin_num in ESP32_VARIANT_ADC2_PIN_TO_CHANNEL[variant]
|
||||
):
|
||||
chan = ESP32_VARIANT_ADC2_PIN_TO_CHANNEL[variant][pin_num]
|
||||
cg.add(var.set_channel2(chan))
|
||||
|
23
esphome/components/adc128s102/__init__.py
Normal file
23
esphome/components/adc128s102/__init__.py
Normal file
@@ -0,0 +1,23 @@
|
||||
import esphome.codegen as cg
|
||||
import esphome.config_validation as cv
|
||||
from esphome.components import spi
|
||||
from esphome.const import CONF_ID
|
||||
|
||||
DEPENDENCIES = ["spi"]
|
||||
MULTI_CONF = True
|
||||
CODEOWNERS = ["@DeerMaximum"]
|
||||
|
||||
adc128s102_ns = cg.esphome_ns.namespace("adc128s102")
|
||||
ADC128S102 = adc128s102_ns.class_("ADC128S102", cg.Component, spi.SPIDevice)
|
||||
|
||||
CONFIG_SCHEMA = cv.Schema(
|
||||
{
|
||||
cv.GenerateID(): cv.declare_id(ADC128S102),
|
||||
}
|
||||
).extend(spi.spi_device_schema(cs_pin_required=True))
|
||||
|
||||
|
||||
async def to_code(config):
|
||||
var = cg.new_Pvariable(config[CONF_ID])
|
||||
await cg.register_component(var, config)
|
||||
await spi.register_spi_device(var, config)
|
35
esphome/components/adc128s102/adc128s102.cpp
Normal file
35
esphome/components/adc128s102/adc128s102.cpp
Normal file
@@ -0,0 +1,35 @@
|
||||
#include "adc128s102.h"
|
||||
#include "esphome/core/log.h"
|
||||
|
||||
namespace esphome {
|
||||
namespace adc128s102 {
|
||||
|
||||
static const char *const TAG = "adc128s102";
|
||||
|
||||
float ADC128S102::get_setup_priority() const { return setup_priority::HARDWARE; }
|
||||
|
||||
void ADC128S102::setup() {
|
||||
ESP_LOGCONFIG(TAG, "Setting up adc128s102");
|
||||
this->spi_setup();
|
||||
}
|
||||
|
||||
void ADC128S102::dump_config() {
|
||||
ESP_LOGCONFIG(TAG, "ADC128S102:");
|
||||
LOG_PIN(" CS Pin:", this->cs_);
|
||||
}
|
||||
|
||||
uint16_t ADC128S102::read_data(uint8_t channel) {
|
||||
uint8_t control = channel << 3;
|
||||
|
||||
this->enable();
|
||||
uint8_t adc_primary_byte = this->transfer_byte(control);
|
||||
uint8_t adc_secondary_byte = this->transfer_byte(0x00);
|
||||
this->disable();
|
||||
|
||||
uint16_t digital_value = adc_primary_byte << 8 | adc_secondary_byte;
|
||||
|
||||
return digital_value;
|
||||
}
|
||||
|
||||
} // namespace adc128s102
|
||||
} // namespace esphome
|
23
esphome/components/adc128s102/adc128s102.h
Normal file
23
esphome/components/adc128s102/adc128s102.h
Normal file
@@ -0,0 +1,23 @@
|
||||
#pragma once
|
||||
|
||||
#include "esphome/core/component.h"
|
||||
#include "esphome/core/hal.h"
|
||||
#include "esphome/components/spi/spi.h"
|
||||
|
||||
namespace esphome {
|
||||
namespace adc128s102 {
|
||||
|
||||
class ADC128S102 : public Component,
|
||||
public spi::SPIDevice<spi::BIT_ORDER_MSB_FIRST, spi::CLOCK_POLARITY_LOW, spi::CLOCK_PHASE_LEADING,
|
||||
spi::DATA_RATE_10MHZ> {
|
||||
public:
|
||||
ADC128S102() = default;
|
||||
|
||||
void setup() override;
|
||||
void dump_config() override;
|
||||
float get_setup_priority() const override;
|
||||
uint16_t read_data(uint8_t channel);
|
||||
};
|
||||
|
||||
} // namespace adc128s102
|
||||
} // namespace esphome
|
38
esphome/components/adc128s102/sensor/__init__.py
Normal file
38
esphome/components/adc128s102/sensor/__init__.py
Normal file
@@ -0,0 +1,38 @@
|
||||
import esphome.codegen as cg
|
||||
import esphome.config_validation as cv
|
||||
from esphome.components import sensor, voltage_sampler
|
||||
from esphome.const import CONF_ID, CONF_CHANNEL
|
||||
|
||||
from .. import adc128s102_ns, ADC128S102
|
||||
|
||||
AUTO_LOAD = ["voltage_sampler"]
|
||||
DEPENDENCIES = ["adc128s102"]
|
||||
|
||||
ADC128S102Sensor = adc128s102_ns.class_(
|
||||
"ADC128S102Sensor",
|
||||
sensor.Sensor,
|
||||
cg.PollingComponent,
|
||||
voltage_sampler.VoltageSampler,
|
||||
)
|
||||
CONF_ADC128S102_ID = "adc128s102_id"
|
||||
|
||||
CONFIG_SCHEMA = (
|
||||
sensor.sensor_schema(ADC128S102Sensor)
|
||||
.extend(
|
||||
{
|
||||
cv.GenerateID(CONF_ADC128S102_ID): cv.use_id(ADC128S102),
|
||||
cv.Required(CONF_CHANNEL): cv.int_range(min=0, max=7),
|
||||
}
|
||||
)
|
||||
.extend(cv.polling_component_schema("60s"))
|
||||
)
|
||||
|
||||
|
||||
async def to_code(config):
|
||||
var = cg.new_Pvariable(
|
||||
config[CONF_ID],
|
||||
config[CONF_CHANNEL],
|
||||
)
|
||||
await cg.register_parented(var, config[CONF_ADC128S102_ID])
|
||||
await cg.register_component(var, config)
|
||||
await sensor.register_sensor(var, config)
|
24
esphome/components/adc128s102/sensor/adc128s102_sensor.cpp
Normal file
24
esphome/components/adc128s102/sensor/adc128s102_sensor.cpp
Normal file
@@ -0,0 +1,24 @@
|
||||
#include "adc128s102_sensor.h"
|
||||
|
||||
#include "esphome/core/log.h"
|
||||
|
||||
namespace esphome {
|
||||
namespace adc128s102 {
|
||||
|
||||
static const char *const TAG = "adc128s102.sensor";
|
||||
|
||||
ADC128S102Sensor::ADC128S102Sensor(uint8_t channel) : channel_(channel) {}
|
||||
|
||||
float ADC128S102Sensor::get_setup_priority() const { return setup_priority::DATA; }
|
||||
|
||||
void ADC128S102Sensor::dump_config() {
|
||||
LOG_SENSOR("", "ADC128S102 Sensor", this);
|
||||
ESP_LOGCONFIG(TAG, " Pin: %u", this->channel_);
|
||||
LOG_UPDATE_INTERVAL(this);
|
||||
}
|
||||
|
||||
float ADC128S102Sensor::sample() { return this->parent_->read_data(this->channel_); }
|
||||
void ADC128S102Sensor::update() { this->publish_state(this->sample()); }
|
||||
|
||||
} // namespace adc128s102
|
||||
} // namespace esphome
|
29
esphome/components/adc128s102/sensor/adc128s102_sensor.h
Normal file
29
esphome/components/adc128s102/sensor/adc128s102_sensor.h
Normal file
@@ -0,0 +1,29 @@
|
||||
#pragma once
|
||||
|
||||
#include "esphome/components/sensor/sensor.h"
|
||||
#include "esphome/components/voltage_sampler/voltage_sampler.h"
|
||||
#include "esphome/core/component.h"
|
||||
#include "esphome/core/hal.h"
|
||||
|
||||
#include "../adc128s102.h"
|
||||
|
||||
namespace esphome {
|
||||
namespace adc128s102 {
|
||||
|
||||
class ADC128S102Sensor : public PollingComponent,
|
||||
public Parented<ADC128S102>,
|
||||
public sensor::Sensor,
|
||||
public voltage_sampler::VoltageSampler {
|
||||
public:
|
||||
ADC128S102Sensor(uint8_t channel);
|
||||
|
||||
void update() override;
|
||||
void dump_config() override;
|
||||
float get_setup_priority() const override;
|
||||
float sample() override;
|
||||
|
||||
protected:
|
||||
uint8_t channel_;
|
||||
};
|
||||
} // namespace adc128s102
|
||||
} // namespace esphome
|
@@ -5,6 +5,8 @@
|
||||
#include "esphome/components/display/display_buffer.h"
|
||||
#include "esphome/components/light/addressable_light.h"
|
||||
|
||||
#include <vector>
|
||||
|
||||
namespace esphome {
|
||||
namespace addressable_light {
|
||||
|
||||
@@ -40,6 +42,8 @@ class AddressableLightDisplay : public display::DisplayBuffer, public PollingCom
|
||||
void setup() override;
|
||||
void display();
|
||||
|
||||
display::DisplayType get_display_type() override { return display::DisplayType::DISPLAY_TYPE_COLOR; }
|
||||
|
||||
protected:
|
||||
int get_width_internal() override;
|
||||
int get_height_internal() override;
|
||||
|
@@ -48,16 +48,16 @@ async def to_code(config):
|
||||
await cg.register_component(var, config)
|
||||
await display.register_display(var, config)
|
||||
|
||||
if CONF_PIXEL_MAPPER in config:
|
||||
if pixel_mapper := config.get(CONF_PIXEL_MAPPER):
|
||||
pixel_mapper_template_ = await cg.process_lambda(
|
||||
config[CONF_PIXEL_MAPPER],
|
||||
pixel_mapper,
|
||||
[(int, "x"), (int, "y")],
|
||||
return_type=cg.int_,
|
||||
)
|
||||
cg.add(var.set_pixel_mapper(pixel_mapper_template_))
|
||||
|
||||
if CONF_LAMBDA in config:
|
||||
if lambda_config := config.get(CONF_LAMBDA):
|
||||
lambda_ = await cg.process_lambda(
|
||||
config[CONF_LAMBDA], [(display.DisplayBufferRef, "it")], return_type=cg.void
|
||||
lambda_config, [(display.DisplayRef, "it")], return_type=cg.void
|
||||
)
|
||||
cg.add(var.set_writer(lambda_))
|
||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user