mirror of
				https://github.com/home-assistant/core.git
				synced 2025-10-25 19:49:37 +00:00 
			
		
		
		
	Compare commits
	
		
			1123 Commits
		
	
	
		
			hassfest-e
			...
			refactor-h
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 2dd40ed15f | ||
|   | b6d8b56b5b | ||
|   | 929d76e236 | ||
|   | fe1ff083de | ||
|   | 90c68f8ad0 | ||
|   | 6b79aa7738 | ||
|   | f6fb4c8d5a | ||
|   | a6e575ecfa | ||
|   | 85392ae167 | ||
|   | 9d124be491 | ||
|   | 8bca3931ab | ||
|   | 0367a01287 | ||
|   | 86e2c2f361 | ||
|   | 335c8e50a2 | ||
|   | 8152a9e5da | ||
|   | 250e562caf | ||
|   | a3b641e53d | ||
|   | 135ea4c02e | ||
|   | bc980c1212 | ||
|   | 59ca88a7e8 | ||
|   | d45114cd11 | ||
|   | 2eba650064 | ||
|   | de4adb8855 | ||
|   | 1d86c03b02 | ||
|   | 77fb1036cc | ||
|   | b15b4e4888 | ||
|   | dddf6d5f1a | ||
|   | 66fb5f4d95 | ||
|   | 42a9d5d4e3 | ||
|   | 93fa162913 | ||
|   | c432b1c8da | ||
|   | 00955b8e6a | ||
|   | 045b9d7f01 | ||
|   | 438c4c7871 | ||
|   | abc360460c | ||
|   | 26437bb253 | ||
|   | 56d953ac1e | ||
|   | fe4eb8766d | ||
|   | 2d9f14c401 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7b6ccb07fd | ||
|   | 2ba5728060 | ||
|   | b5f163cc85 | ||
|   | 65540a3e0b | ||
|   | cbf1b39edb | ||
|   | 142daf5e49 | ||
|   | 8bd0ff7cca | ||
|   | ac676e12f6 | ||
|   | c0ac3292cd | ||
|   | 80fd07c128 | ||
|   | 3701d8859a | ||
|   | 6dd26bae88 | ||
|   | 1a0abe296c | ||
|   | de6c61a4ab | ||
|   | 33c677596e | ||
|   | e9b4b8e99b | ||
|   | 0525c04c42 | ||
|   | d57b502551 | ||
|   | 9fb708baf4 | ||
|   | abdf24b7a0 | ||
|   | 29bfbd27bb | ||
|   | 224553f8d9 | ||
|   | 7c9f6a061f | ||
|   | 8e115d4685 | ||
|   | 00c189844f | ||
|   | 4587c286bb | ||
|   | b46097a7fc | ||
|   | 299cb6a2ff | ||
|   | 1b7b91b328 | ||
|   | 01a1480ebd | ||
|   | 26b8abb118 | ||
|   | 53d1bbb530 | ||
|   | a3ef55274e | ||
|   | 2034915457 | ||
|   | 9e46d7964a | ||
|   | f9828a227b | ||
|   | 3341fa5f33 | ||
|   | e38ae47e76 | ||
|   | 934c0e3c4c | ||
|   | 994a6ae7ed | ||
|   | cdbe93c289 | ||
|   | 56f90e4d96 | ||
|   | 34977abfec | ||
|   | 5622103eb1 | ||
|   | b9a1ab4a44 | ||
|   | 18997833c4 | ||
|   | f99b194afc | ||
|   | 566a347da7 | ||
|   | 881306f6a4 | ||
|   | f63504af01 | ||
|   | d140b82a70 | ||
|   | 681211b1a5 | ||
|   | 6c8b1f3618 | ||
|   | d341065c34 | ||
|   | 81b1346080 | ||
|   | 5613be3980 | ||
|   | fbcf0eb94c | ||
|   | 1c7b9cc354 | ||
|   | 75e900606e | ||
|   | 7c665c53b5 | ||
|   | f72047eb02 | ||
|   | ade424c074 | ||
|   | 5ad805de3c | ||
|   | ece77cf620 | ||
|   | 7eaa559056 | ||
|   | 08a9377373 | ||
|   | a2837e6aee | ||
|   | fa03f6194d | ||
|   | d2851ea1df | ||
|   | 72f8ac7857 | ||
|   | 77a267bc2f | ||
|   | ad238daadc | ||
|   | 42370ba203 | ||
|   | d9691c2a3b | ||
|   | 66cca981a9 | ||
|   | 9640ebb593 | ||
|   | 645f32fd65 | ||
|   | cb6e65f972 | ||
|   | 425bdc0ba6 | ||
|   | c36341e51f | ||
|   | 553d896899 | ||
|   | ac79b3072e | ||
|   | c0aa9bfd4b | ||
|   | e97100028d | ||
|   | da89617432 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e6203dffd3 | ||
|   | c13cfe9c37 | ||
|   | 2447df9341 | ||
|   | 1c1fbe0ec1 | ||
|   | 4a6d2017fd | ||
|   | b4997a52df | ||
|   | 464dec1dcb | ||
|   | 85506ac78a | ||
|   | 6d97355b42 | ||
|   | f9e75c616a | ||
|   | a821d02dfb | ||
|   | e05169c7a4 | ||
|   | 1cc3431529 | ||
|   | 4ba765f265 | ||
|   | 50a7af4179 | ||
|   | e0a2116e88 | ||
|   | d8e1ed5f4a | ||
|   | f1b8e8a963 | ||
|   | 9a9fd44c62 | ||
|   | bc3fe7a18e | ||
|   | 19f3559345 | ||
|   | fad0e23797 | ||
|   | 7f931e4d70 | ||
|   | a04835629b | ||
|   | 78cd80746d | ||
|   | 9ac93920d8 | ||
|   | 1818fce1ae | ||
|   | f524edc4b9 | ||
|   | 19f990ed31 | ||
|   | 5d83c82b81 | ||
|   | d63d154457 | ||
|   | 933b15ce36 | ||
|   | 6ec7b63ebe | ||
|   | 26bfbc55e9 | ||
|   | d75ca0f5f3 | ||
|   | fed8f137e9 | ||
|   | f44d65e023 | ||
|   | a270bd76de | ||
|   | 9209e419ec | ||
|   | 98f8f15e90 | ||
|   | b2a2868afd | ||
|   | 0d4737d360 | ||
|   | 2b370a0eca | ||
|   | 618fe81207 | ||
|   | c0fe4861f9 | ||
|   | dfd33fdab1 | ||
|   | cceee05c15 | ||
|   | f560d2a05e | ||
|   | 3601cff88e | ||
|   | ca5c0a759f | ||
|   | 6f9e6909ce | ||
|   | ccf563437b | ||
|   | 78e97428fd | ||
|   | 8b4c730993 | ||
|   | 0a071a13e2 | ||
|   | ab80991eac | ||
|   | ee7262efb4 | ||
|   | ea5a52cdc8 | ||
|   | 31fe0322ab | ||
|   | e8e0eabb99 | ||
|   | 1629dad1a8 | ||
|   | d9baad530a | ||
|   | 4a1d00e59a | ||
|   | 437e4e027c | ||
|   | 3726f7eca9 | ||
|   | c943cf515c | ||
|   | 3b0c2a7e56 | ||
|   | 6ebaa9cd1d | ||
|   | f81c32f6ea | ||
|   | c0cd7a1a62 | ||
|   | 7a61c818c6 | ||
|   | 2800625bcf | ||
|   | cfec998221 | ||
|   | 7203cffbd7 | ||
|   | 23397ef6a9 | ||
|   | 0e154635ff | ||
|   | 2e6e518722 | ||
|   | e0cded97c7 | ||
|   | 87a6a029bb | ||
|   | 1cc3c22d3f | ||
|   | 2341d1d965 | ||
|   | a0bae9485c | ||
|   | f281b0fc6b | ||
|   | 6f89fe81cc | ||
|   | 34f6ead7a1 | ||
|   | 8985527a87 | ||
|   | bd87a3aa4d | ||
|   | 768a505904 | ||
|   | d97c1f0fc3 | ||
|   | c3fcd34d4c | ||
|   | 44d9eaea95 | ||
|   | 0f34f5139a | ||
|   | 2afb1a673d | ||
|   | c2f7f29630 | ||
|   | b01f5dd24b | ||
|   | 0cda0c449f | ||
|   | 40fdf12bc9 | ||
|   | 3939a80302 | ||
|   | d32a102613 | ||
|   | 20949d39c4 | ||
|   | 310a0c8d13 | ||
|   | c9e80ac7e9 | ||
|   | 5df4e9e1cf | ||
|   | 4022ee74e8 | ||
|   | 80a4115c44 | ||
|   | ce548efd80 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2edf622b41 | ||
|   | 66ac9078aa | ||
|   | ba75f18f5a | ||
|   | 8ee2ece03e | ||
|   | 7060ab8c44 | ||
|   | 85d8244b8a | ||
|   | 3f9421ab08 | ||
|   | 2f3fbf00b7 | ||
|   | d595ec8a07 | ||
|   | 4ff5462cc4 | ||
|   | 404f95b442 | ||
|   | 89cf784022 | ||
|   | 02142f352d | ||
|   | ec3dd7d1e5 | ||
|   | 7355799030 | ||
|   | 982166df3c | ||
|   | c7d3512ad2 | ||
|   | ada6f7b3fb | ||
|   | 78e16495bd | ||
|   | 12085e6152 | ||
|   | 6764463689 | ||
|   | 7055276665 | ||
|   | 71b3ebd15a | ||
|   | b87910e596 | ||
|   | e19bfd670b | ||
|   | 7b3c96e80b | ||
|   | 01ff3cf9d9 | ||
|   | d66da0c10d | ||
|   | 3491bb1b40 | ||
|   | 3bf995eb71 | ||
|   | 2169ce1722 | ||
|   | 275e9485e9 | ||
|   | 95198ae540 | ||
|   | aed2d3899d | ||
|   | 4011d62ac7 | ||
|   | d2aa0573de | ||
|   | 571b2e3ab6 | ||
|   | a7f48360b7 | ||
|   | 22f2f8680a | ||
|   | d92004a9e7 | ||
|   | 64875894d6 | ||
|   | 3f7a288526 | ||
|   | a2a067a81c | ||
|   | f9f61b8da7 | ||
|   | cd69b82fc9 | ||
|   | d20631598e | ||
|   | 229ebe16f3 | ||
|   | a172f67d37 | ||
|   | ee4a1de566 | ||
|   | 7ab99c028c | ||
|   | 0e1d12b1ae | ||
|   | e090ddd761 | ||
|   | 9721ce6877 | ||
|   | 8dde94f421 | ||
|   | f5f6b22af1 | ||
|   | f8a93b6561 | ||
|   | 840a03f048 | ||
|   | 85f3b5ce78 | ||
|   | f4284fec2f | ||
|   | 3a89b3152f | ||
|   | a0356328c3 | ||
|   | 4b6f37b1d7 | ||
|   | 716705fb5a | ||
|   | d246836480 | ||
|   | 6ee2b82d15 | ||
|   | 73ff8d36a5 | ||
|   | 1397def3b8 | ||
|   | d443529041 | ||
|   | 373bb20f1b | ||
|   | 3b44cce6dc | ||
|   | 46056fe45b | ||
|   | 1816c190b2 | ||
|   | 00abaee6b3 | ||
|   | 3a301f54e0 | ||
|   | 762accbd6d | ||
|   | e0422d7d34 | ||
|   | 6ba2057a88 | ||
|   | 752969bce5 | ||
|   | efbdfd2954 | ||
|   | bb7a177a5d | ||
|   | 9b56ca8cde | ||
|   | b0a08782e0 | ||
|   | 6c9955f220 | ||
|   | f56b94c0f9 | ||
|   | 3cf035820b | ||
|   | 99a796d066 | ||
|   | 1cd1b1aba8 | ||
|   | 4131c14629 | ||
|   | c2acda5796 | ||
|   | 4806e7e9d9 | ||
|   | 76606fd44f | ||
|   | 2983f1a3b6 | ||
|   | 8019779b3a | ||
|   | 62cdcbf422 | ||
|   | b12a5a36e1 | ||
|   | e32763e464 | ||
|   | b85cf3f9d2 | ||
|   | 3777bcc2af | ||
|   | 9a29cc53ef | ||
|   | 52cde48ff0 | ||
|   | bf1da35303 | ||
|   | 55d5e769b2 | ||
|   | c1bf11da34 | ||
|   | 3c20325b37 | ||
|   | 6cd1283b00 | ||
|   | dde60cdecb | ||
|   | f03b16bdf8 | ||
|   | fd8ccb8d8f | ||
|   | d76e947021 | ||
|   | c91ed96543 | ||
|   | b164531ba8 | ||
|   | 7c623a8704 | ||
|   | 7ae3340336 | ||
|   | 653b73c601 | ||
|   | f616e5a4e3 | ||
|   | c0317f60cc | ||
|   | 8abfe424e1 | ||
|   | 8de200de0b | ||
|   | f242e294be | ||
|   | 58cc7c8f84 | ||
|   | bd10f6ec08 | ||
|   | ed9cfb4c4b | ||
|   | a6b6e4c4b8 | ||
|   | 36ff5c0d45 | ||
|   | de6d34fec5 | ||
|   | 38f9067970 | ||
|   | 53a8a250d0 | ||
|   | 00f6d26ede | ||
|   | 6d09411c07 | ||
|   | 037e2bfd31 | ||
|   | c893552d4a | ||
|   | 4fd10162c9 | ||
|   | 392ee5ae7e | ||
|   | bf190609a0 | ||
|   | e982ac1e53 | ||
|   | b4747ea87b | ||
|   | df69bcecb7 | ||
|   | 7c93d91bae | ||
|   | 07da0cfb2b | ||
|   | b411a11c2c | ||
|   | 0555b84d05 | ||
|   | 790bddef63 | ||
|   | a3089b8aa7 | ||
|   | 77c8426d63 | ||
|   | faf226f6c2 | ||
|   | 06d143b81a | ||
|   | 08b6a0a702 | ||
|   | a20d1e3656 | ||
|   | 36cc3682ca | ||
|   | 1b495ecafa | ||
|   | 7d1a0be07e | ||
|   | 327f65c991 | ||
|   | 4ac89f6849 | ||
|   | db3b070ed0 | ||
|   | 6d940f476a | ||
|   | 1ca701dda4 | ||
|   | 291c44100c | ||
|   | c8d676e06b | ||
|   | 4c1ae0eddc | ||
|   | 39eadc814f | ||
|   | f7ecad61ba | ||
|   | fa4cb54549 | ||
|   | 2be33c5e0a | ||
|   | 904d7e5d5a | ||
|   | dbc4a65d48 | ||
|   | b93f4aabf1 | ||
|   | 9eaa40c7a4 | ||
|   | b308a882fb | ||
|   | 7f63ba2087 | ||
|   | d7269cfcc6 | ||
|   | 2850a574f6 | ||
|   | dcb8d4f702 | ||
|   | aeadc0c4b0 | ||
|   | 683c6b17be | ||
|   | 69dd5c91b7 | ||
|   | 5cf7dfca8f | ||
|   | 62a49d4244 | ||
|   | 93ee6322f2 | ||
|   | 914990b58a | ||
|   | f78bb5adb6 | ||
|   | 905f5e7289 | ||
|   | ec503618c3 | ||
|   | 7a41cbc314 | ||
|   | c58ba734e7 | ||
|   | 68f63be62f | ||
|   | 2aa4ca1351 | ||
|   | fbabb27787 | ||
|   | 0960d78eb5 | ||
|   | 474b40511f | ||
|   | 18b80aced3 | ||
|   | c75dca743a | ||
|   | 00d667ed51 | ||
|   | 51e098e807 | ||
|   | 5e2b27699e | ||
|   | be942c2888 | ||
|   | 584c1fbd97 | ||
|   | abc5c6e2b4 | ||
|   | d9de964035 | ||
|   | bb02158d1a | ||
|   | be10f097c7 | ||
|   | 7084bca783 | ||
|   | cd6f3a0fe5 | ||
|   | af2888331d | ||
|   | b92e5d7131 | ||
|   | f7265c85d0 | ||
|   | 8466dbf69f | ||
|   | 2dd0d69bcd | ||
|   | 6783c4ad83 | ||
|   | 07d7f4e18d | ||
|   | 54b1749986 | ||
|   | eaf264361f | ||
|   | d8f6f17a4f | ||
|   | 9a969cea63 | ||
|   | ef16327b2b | ||
|   | a6a6261168 | ||
|   | a01eb48db8 | ||
|   | eb103a8d9a | ||
|   | 2b5f989855 | ||
|   | 4e247a6ebe | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b964d362b7 | ||
|   | 3914e41f3c | ||
|   | 82bdfcb99b | ||
|   | 976cea600f | ||
|   | 8c8713c3f7 | ||
|   | 2359ae6ce7 | ||
|   | b570fd35c8 | ||
|   | 9d94e6b3b4 | ||
|   | cfab789823 | ||
|   | 81917425dc | ||
|   | bfb62709d4 | ||
|   | ca3f2ee782 | ||
|   | fc8703a40f | ||
|   | 80517c7ac1 | ||
|   | 2b4b46eaf8 | ||
|   | 40b9dae608 | ||
|   | 5975cd6e09 | ||
|   | 258c9ff52b | ||
|   | 89c5d498a4 | ||
|   | 76cb4d123a | ||
|   | f0c29c7699 | ||
|   | aa4151ced7 | ||
|   | 0a6fa978fa | ||
|   | dc02002b9d | ||
|   | f071a3f38b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b935231e47 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b9f7613567 | ||
|   | 1289a031ab | ||
|   | 289546ef6d | ||
|   | aacff4db5d | ||
|   | f833b56122 | ||
|   | 7eb0f2993f | ||
|   | abb341abfe | ||
|   | 0d90614369 | ||
|   | ec84bebeea | ||
|   | 9176867d6b | ||
|   | 281a137ff5 | ||
|   | d6543480ac | ||
|   | ae6391b866 | ||
|   | 10b56e4258 | ||
|   | 0ff2597957 | ||
|   | 026b28e962 | ||
|   | 9a1e67294a | ||
|   | cdb448a5cc | ||
|   | ab80e726e2 | ||
|   | 2d5d0f67b2 | ||
|   | d4100b6096 | ||
|   | 955e854d77 | ||
|   | 0c37f88c49 | ||
|   | 48167eeb9c | ||
|   | 24177197f7 | ||
|   | 863fc0ba97 | ||
|   | 9f7b229d02 | ||
|   | ffd909f3d9 | ||
|   | 1ebf096a33 | ||
|   | 96d51965e5 | ||
|   | 04b510b020 | ||
|   | c9a301d50e | ||
|   | b304bd1a8b | ||
|   | b99525b231 | ||
|   | 634db13990 | ||
|   | ad51a77989 | ||
|   | 3348a39e8a | ||
|   | 81c2e356ec | ||
|   | de6c3512d2 | ||
|   | 36dc1e938a | ||
|   | 07a78cf6f7 | ||
|   | eaa673e0c3 | ||
|   | f2c4ca081f | ||
|   | e3d707f0b4 | ||
|   | fb93fed2e5 | ||
|   | 95dfc2f23d | ||
|   | 77f897a768 | ||
|   | 4f0a6ef9a1 | ||
|   | 408df2093a | ||
|   | 66c6b0f5fc | ||
|   | dd01243391 | ||
|   | 66c17e250a | ||
|   | 723902e233 | ||
|   | 59fdb9f3b5 | ||
|   | d83502514a | ||
|   | 08e81b2ba6 | ||
|   | 1e808c965d | ||
|   | 563b58c9aa | ||
|   | cf223880e8 | ||
|   | 4058ca59ed | ||
|   | 1386c01733 | ||
|   | 46504947f7 | ||
|   | 0a44682014 | ||
|   | 06a57473a9 | ||
|   | fbed66ef1f | ||
|   | 99a0380ec5 | ||
|   | 68c51dc7aa | ||
|   | 3d945b0fc5 | ||
|   | 7b26a93d38 | ||
|   | 1b2eab00be | ||
|   | 750e849f09 | ||
|   | f32bf0cc3e | ||
|   | dbbe3145b6 | ||
|   | f8bf3ea2ef | ||
|   | 053bd31d43 | ||
|   | 1aefc3f37a | ||
|   | 3de955d9ce | ||
|   | 0ff88fd366 | ||
|   | eb84020773 | ||
|   | 4bbfea3c7c | ||
|   | 63d4fb7558 | ||
|   | 953895cd81 | ||
|   | a6c3f4efc0 | ||
|   | 11e880d034 | ||
|   | e4d6bdb398 | ||
|   | 6ced1783e3 | ||
|   | 8051f78d10 | ||
|   | b724176b23 | ||
|   | fdca16ea92 | ||
|   | f8fd8b432a | ||
|   | 9148ae70ce | ||
|   | 447cb26d28 | ||
|   | 2af36465f6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d5f7265424 | ||
|   | cc16af7f2d | ||
|   | 7a4d75bc44 | ||
|   | ec0380fd3b | ||
|   | b17cc71dfb | ||
|   | 89b327ed7b | ||
|   | 9bf361a1b8 | ||
|   | d11c171c75 | ||
|   | c523c45d17 | ||
|   | c1b9c0e1b6 | ||
|   | 487b9ff03e | ||
|   | ec62b0cdfb | ||
|   | 6d0470064f | ||
|   | 7450b3fd1a | ||
|   | 5b70910d77 | ||
|   | 6aaddad56b | ||
|   | a5af974209 | ||
|   | 09e45f6f54 | ||
|   | d857d8850c | ||
|   | ccc50f2412 | ||
|   | 3905723900 | ||
|   | cee88473a2 | ||
|   | cdf613d3f8 | ||
|   | 52de5ff5ff | ||
|   | c4389a1679 | ||
|   | 35faaa6cae | ||
|   | 3c0b13975a | ||
|   | bc88696339 | ||
|   | 8f99c3f64a | ||
|   | 88016d96d4 | ||
|   | 47df73b18f | ||
|   | 1c12d2b8cd | ||
|   | eb38837a8c | ||
|   | 159c7fbfd1 | ||
|   | 7ee31f0884 | ||
|   | 0c5e12571a | ||
|   | 9db973217f | ||
|   | cf1a745283 | ||
|   | 834e3f1963 | ||
|   | 3f8f7573c9 | ||
|   | 156a0f1a3d | ||
|   | cc2a5b43dd | ||
|   | 0ae272f1f6 | ||
|   | 8774295e2e | ||
|   | 731064f7e9 | ||
|   | 2f75661c20 | ||
|   | be6f056f30 | ||
|   | 79599e1284 | ||
|   | a255585ab6 | ||
|   | e9bde225fe | ||
|   | d9521ac2a0 | ||
|   | d8b24ccccd | ||
|   | b4417a76d5 | ||
|   | 274f6eb54a | ||
|   | 21a5aaf35c | ||
|   | 05820a49d0 | ||
|   | 0c8d2594ef | ||
|   | 205bd2676b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 25849fd9cc | ||
|   | 7d6eac9ff7 | ||
|   | 31017ebc98 | ||
|   | 724a7b0ecc | ||
|   | 91e13d447a | ||
|   | 7c8ad9d535 | ||
|   | 9cd3ab853d | ||
|   | 0b0f8c5829 | ||
|   | ae7bc7fb1b | ||
|   | 09750872b5 | ||
|   | 076e51017b | ||
|   | 95e7b00996 | ||
|   | ddecf1ac21 | ||
|   | 17b12d29af | ||
|   | 9cc78680d6 | ||
|   | 14d42e43bf | ||
|   | ed5f5d4b33 | ||
|   | c3ba086fad | ||
|   | 7b5314605c | ||
|   | 3a806d6603 | ||
|   | 6dd33f900d | ||
|   | 2844bd474a | ||
|   | d865fcf999 | ||
|   | 79a2fc5a01 | ||
|   | 19d87abb8a | ||
|   | c4de46a85b | ||
|   | e79a434d9b | ||
|   | 9a801424c7 | ||
|   | 5cb186980a | ||
|   | 1629ade97f | ||
|   | ccf0011ac2 | ||
|   | 70077511a3 | ||
|   | dfbaf66021 | ||
|   | 62cea48a58 | ||
|   | c493c7dd67 | ||
|   | fdaceaddfd | ||
|   | a2f4073d54 | ||
|   | 2d01a99ec2 | ||
|   | 311d4c4262 | ||
|   | e14f5ba44d | ||
|   | 9babc85517 | ||
|   | 332a3fad3c | ||
|   | 8782aa4f60 | ||
|   | 475b84cc5f | ||
|   | 0f904d418b | ||
|   | 4ea4eec2d8 | ||
|   | afefa16615 | ||
|   | 1dccbee45c | ||
|   | 711a56db2f | ||
|   | 9d1c7dadff | ||
|   | 7d1953e387 | ||
|   | 023ecf2a64 | ||
|   | 934db458a3 | ||
|   | 0a6ae3b52a | ||
|   | bdd0b74d51 | ||
|   | 8837f2aca7 | ||
|   | 403cd2d8ef | ||
|   | ddfc528d63 | ||
|   | ddea2206c3 | ||
|   | 32aacac550 | ||
|   | dadba274aa | ||
|   | 14b5b9742c | ||
|   | a0be737925 | ||
|   | ff47839c61 | ||
|   | 9ba7dda864 | ||
|   | 911f901d9d | ||
|   | 2008a73657 | ||
|   | 60bf298ca6 | ||
|   | 3bc2ea7b5f | ||
|   | 3bac6b86df | ||
|   | 20293e2a11 | ||
|   | 15cc28e6c1 | ||
|   | 874ca1323b | ||
|   | ca186925af | ||
|   | 2ab051b716 | ||
|   | a2a726de34 | ||
|   | 5d543d2185 | ||
|   | a78c909b34 | ||
|   | f00ab80d17 | ||
|   | 014881d985 | ||
|   | 29a42a8e58 | ||
|   | 3f70084d7f | ||
|   | b1ae9c95c9 | ||
|   | 8be79ecdb0 | ||
|   | f6b8aa893b | ||
|   | c867026bdd | ||
|   | da3a164e66 | ||
|   | 32688e1108 | ||
|   | 4305ea9b4c | ||
|   | 61153ec456 | ||
|   | 9e4a2d5fa9 | ||
|   | 72e608918b | ||
|   | 86db60c442 | ||
|   | 25806615a9 | ||
|   | a0f67381e5 | ||
|   | 90bfadda9b | ||
|   | 0f8e700965 | ||
|   | 21d4ed2837 | ||
|   | ce363b3835 | ||
|   | dd3e6b8df5 | ||
|   | abbf8390ac | ||
|   | 689039959c | ||
|   | 52c25cfc88 | ||
|   | 00b2017767 | ||
|   | dd7f7be6ad | ||
|   | 22709506c6 | ||
|   | f0c0492375 | ||
|   | 58459cb80f | ||
|   | a19e378447 | ||
|   | 38a5a3ed4b | ||
|   | e76bed4a83 | ||
|   | d73309ba60 | ||
|   | 19fdea024c | ||
|   | a3cfd7f707 | ||
|   | 3dd941eff7 | ||
|   | d389141aee | ||
|   | 3c542b8d43 | ||
|   | 2367df89d9 | ||
|   | 7bfdfb3fc7 | ||
|   | 485916265a | ||
|   | 1bb3c96fc1 | ||
|   | 4eaf6784af | ||
|   | 7b7265a6b0 | ||
|   | 9059e3dadc | ||
|   | d9d42b3ad5 | ||
|   | d565fb3cb4 | ||
|   | 6e93e480d1 | ||
|   | 5a3570702d | ||
|   | b26b1df143 | ||
|   | fdbff76733 | ||
|   | 018d59a892 | ||
|   | 4b6dd0eb8f | ||
|   | b7db87bd3d | ||
|   | 86dc453c55 | ||
|   | a4f2c88c7f | ||
|   | 3cdb894e61 | ||
|   | cb837aaae5 | ||
|   | 82443ded34 | ||
|   | 71cc3b7fcd | ||
|   | e5658f9747 | ||
|   | 868ded141f | ||
|   | 1151fa698d | ||
|   | 2796d6110a | ||
|   | 844b97bd32 | ||
|   | 286b2500bd | ||
|   | 4b7746ab51 | ||
|   | ca1c366f4f | ||
|   | de42ac14ac | ||
|   | 7f7bd5a97f | ||
|   | 8a70a1badb | ||
|   | 181741cab6 | ||
|   | 1e14fb6dab | ||
|   | 2b6a125927 | ||
|   | e61ad10708 | ||
|   | 5177f9e8c2 | ||
|   | 850aeeb5eb | ||
|   | a1b9061060 | ||
|   | 0ec1f27489 | ||
|   | befc93bc73 | ||
|   | 1526d953bf | ||
|   | d38082a5c8 | ||
|   | 42850421d2 | ||
|   | 21a835c4b4 | ||
|   | e9294dbf72 | ||
|   | 5c4dfbff1b | ||
|   | abe628506d | ||
|   | 12cc0ed18d | ||
|   | 8ca7562390 | ||
|   | 942f7eebb1 | ||
|   | 1a167e6aee | ||
|   | 9531ae10f2 | ||
|   | bfc9616abf | ||
|   | 054a5d751a | ||
|   | a43ba4f966 | ||
|   | 1a5cae125f | ||
|   | f3b9bda876 | ||
|   | 3f3aaa2815 | ||
|   | 6dc7870779 | ||
|   | be83416c72 | ||
|   | c745ee18eb | ||
|   | cf907ae196 | ||
|   | 8eee53036a | ||
|   | b37237d24b | ||
|   | 950e758b62 | ||
|   | 9cd940b7df | ||
|   | 10b186a20d | ||
|   | 757aec1c6b | ||
|   | 0b159bdb9c | ||
|   | 8728312e87 | ||
|   | bbb67db354 | ||
|   | 265f5da21a | ||
|   | 54859e8a83 | ||
|   | c87dba878d | ||
|   | 8d8e008123 | ||
|   | b30667a469 | ||
|   | 8920c548d5 | ||
|   | eac719f9af | ||
|   | 71c274cb91 | ||
|   | d4902361e6 | ||
|   | f63eee3889 | ||
|   | 21bfe610d1 | ||
|   | 21c174e895 | ||
|   | ec148e0459 | ||
|   | 286763b998 | ||
|   | 5f88122a2b | ||
|   | 31968d16ab | ||
|   | c125554817 | ||
|   | 10f2955d34 | ||
|   | 55712b784c | ||
|   | fe3a929556 | ||
|   | 534801e80d | ||
|   | 8aeda5a0c0 | ||
|   | eb1cbbc75c | ||
|   | fa8a4d7098 | ||
|   | 2623ebac4d | ||
|   | 1746c51ce4 | ||
|   | 8b984a2105 | ||
|   | ebee370a56 | ||
|   | dabd096587 | ||
|   | 21399818af | ||
|   | 4354214fbf | ||
|   | 5bd39804f1 | ||
|   | 6d3ad3ab9c | ||
|   | 4c212bdcd4 | ||
|   | b91b39580f | ||
|   | 472d70b6c9 | ||
|   | 017a84a859 | ||
|   | d184540967 | ||
|   | 1740984b3b | ||
|   | 4db8592c61 | ||
|   | 27e630c107 | ||
|   | ea8833342d | ||
|   | 87be2ba823 | ||
|   | 51c35eb631 | ||
|   | 24a86d042f | ||
|   | cd6f653123 | ||
|   | fd05ddca28 | ||
|   | a1f2eb44ae | ||
|   | c4ddc03dbc | ||
|   | 9db5aafb71 | ||
|   | 64cdcfb613 | ||
|   | c761ce699c | ||
|   | 40ebce4ae8 | ||
|   | 29914d6722 | ||
|   | 5eef6edded | ||
|   | db729273a5 | ||
|   | 946d75d651 | ||
|   | 093f779edb | ||
|   | 87658e77a7 | ||
|   | 38f65cda98 | ||
|   | 797c6ddedd | ||
|   | fe8a53407a | ||
|   | ae5f57fd99 | ||
|   | a93c3cc23c | ||
|   | 804b42e1fb | ||
|   | a4f15e4840 | ||
|   | 2471177c84 | ||
|   | a494d3ec69 | ||
|   | b10a9721a7 | ||
|   | 04c0bb20d6 | ||
|   | 1598c4ebe8 | ||
|   | d67ec7593a | ||
|   | 4a4c124181 | ||
|   | c34af4be86 | ||
|   | 823071b722 | ||
|   | 462fa77ba1 | ||
|   | 24fc8b9297 | ||
|   | 2596ab2940 | ||
|   | 23fa84e20e | ||
|   | 7f13141297 | ||
|   | 770f41d079 | ||
|   | df16e85359 | ||
|   | 3c6db923a3 | ||
|   | 450c47f932 | ||
|   | 048f64eccf | ||
|   | c4c523e8b7 | ||
|   | 87e30e0907 | ||
|   | 74660da2d2 | ||
|   | 6b8c180509 | ||
|   | eb4a873c43 | ||
|   | 6aafa666d6 | ||
|   | 9ee9bb368d | ||
|   | 6e4258c8a9 | ||
|   | d65e704823 | ||
|   | aadaf87c16 | ||
|   | e70b147c0c | ||
|   | 031b12752f | ||
|   | df0cfd69a9 | ||
|   | b2c53f2d78 | ||
|   | 3649e949b1 | ||
|   | de7e2303a7 | ||
|   | 892f3f267b | ||
|   | 0254285285 | ||
|   | 44a95242dc | ||
|   | f9b1c52d65 | ||
|   | aa8d78622c | ||
|   | ca6289a576 | ||
|   | 0f372f4b47 | ||
|   | 4bba167ab3 | ||
|   | 962c0c443d | ||
|   | c6b4cac28a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3c7e3a5e30 | ||
|   | fa698956c3 | ||
|   | 32f136b12f | ||
|   | e1f617df25 | ||
|   | 84f1b8a5cc | ||
|   | e9cedf4852 | ||
|   | 9c72b40ab4 | ||
|   | 65f655e5f5 | ||
|   | af28573894 | ||
|   | c5fc1de3df | ||
|   | 1df1144eb9 | ||
|   | d51c0e3752 | ||
|   | f5157878c2 | ||
|   | fb723571b6 | ||
|   | dbf80c3ce3 | ||
|   | e0a774b598 | ||
|   | 168afc5f0e | ||
|   | af23670854 | ||
|   | 935ce421df | ||
|   | c60ad8179d | ||
|   | 14ad3364e3 | ||
|   | e229f36648 | ||
|   | f4f99e015c | ||
|   | 5dc509cba0 | ||
|   | 75597ac98d | ||
|   | b503f792b5 | ||
|   | 410c3df6dd | ||
|   | f1bf28df18 | ||
|   | 99fb64af9b | ||
|   | c0af0159e3 | ||
|   | 71749da3a3 | ||
|   | b01be94034 | ||
|   | 47ec8b7f12 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 93ec9e448e | ||
|   | 90bc41dd02 | ||
|   | 410d869f3d | ||
|   | d75d9f2589 | ||
|   | afbb832a57 | ||
|   | bdc881c87a | ||
|   | 22ea269ed8 | ||
|   | 10fecbaf4d | ||
|   | cbdc1dc5b6 | ||
|   | b203a831c9 | ||
|   | 5ccbee4c9a | ||
|   | 1483c9488f | ||
|   | f5535db24c | ||
|   | e40ecdfb00 | ||
|   | 2f4c69bbd5 | ||
|   | dd0f6a702b | ||
|   | 5ba580bc25 | ||
|   | c13002bdd5 | ||
|   | 75d22191a0 | ||
|   | 58d6549f1c | ||
|   | 1fcc6df1fd | ||
|   | 9bf467e6d1 | ||
|   | d877d6d93f | ||
|   | d2b255ba92 | ||
|   | 1509c429d6 | ||
|   | af9717c1cd | ||
|   | 49e75c9cf8 | ||
|   | c97f16a96d | ||
|   | a3a4433d62 | ||
|   | f832002afd | ||
|   | dbc7f2b43c | ||
|   | 1cd3a1eede | ||
|   | 7d6e0d44b0 | ||
|   | 2bb6d745ca | ||
|   | beb9d7856c | ||
|   | 6a4c8a550a | ||
|   | 7d23752a3f | ||
|   | c2b2a78db5 | ||
|   | 0fb6bbee59 | ||
|   | d93e0a105a | ||
|   | ab1619c0b4 | ||
|   | 70df7b8503 | ||
|   | 0e2c2ad355 | ||
|   | 4c26718739 | ||
|   | 96034e1525 | ||
|   | df1302fc1c | ||
|   | 5a5b639aa4 | ||
|   | e9fbe2227f | ||
|   | 82b57568a0 | ||
|   | be692ab2fd | ||
|   | 24c04cceee | ||
|   | 97077898bb | ||
|   | 08485f4e09 | ||
|   | b64d60fce4 | ||
|   | 3690497e1f | ||
|   | 3499ed7a98 | ||
|   | 2c809d5903 | ||
|   | 40988198f3 | ||
|   | b87e581cde | ||
|   | f1c55ee7e2 | ||
|   | 9f17a82acf | ||
|   | 3955391cda | ||
|   | d9a757c7e6 | ||
|   | aa1ec944c0 | ||
|   | 88c3b6a9f5 | ||
|   | ada73953f6 | ||
|   | 42e9b9a0bc | ||
|   | ec6a052ff5 | ||
|   | c91d64e04d | ||
|   | ab5d1d27f1 | ||
|   | 1c10b85fed | ||
|   | 91a7db08ff | ||
|   | a764d54123 | ||
|   | dc09e33556 | ||
|   | 14173bd9ec | ||
|   | d2e7537629 | ||
|   | 9a165a64fe | ||
|   | 9c749a6abc | ||
|   | 2e33222c71 | ||
|   | ab1c2c4f70 | ||
|   | 529219ae69 | ||
|   | d6ce71fa61 | ||
|   | e5b67d513a | ||
|   | a547179f66 | ||
|   | 8c61788a7d | ||
|   | 6b934d94db | ||
|   | d30ad82774 | ||
|   | 4618b33e93 | ||
|   | d6299094db | ||
|   | 087d9d30c0 | ||
|   | f07890cf5c | ||
|   | e5b78cc481 | ||
|   | 12b409d8e1 | ||
|   | def5408db8 | ||
|   | f105b45ee2 | ||
|   | 9d904c30a7 | ||
|   | 99b047939f | ||
|   | 3a615908ee | ||
|   | baff541f46 | ||
|   | 6d8c35cfe9 | ||
|   | b8d9883e74 | ||
|   | c3c65af450 | ||
|   | 3af8616764 | ||
|   | 64ec4609c5 | ||
|   | c78bc26b83 | ||
|   | 0c093646c9 | ||
|   | 1b27acdde0 | ||
|   | 9dafc0e02f | ||
|   | 0091dafcb0 | ||
|   | b387acffb7 | ||
|   | 36b3133fa2 | ||
|   | fe01e96012 | ||
|   | 0b56ec16ed | ||
|   | ca79f4c963 | ||
|   | 9a43f2776d | ||
|   | 0ac7cb311d | ||
|   | 3472020812 | ||
|   | dcd09523a6 | ||
|   | a5bfdc697b | ||
|   | dbb29a7c7d | ||
|   | 124a63d846 | ||
|   | 3de701a9ab | ||
|   | bfe1dd65b3 | ||
|   | 71bf5e14cc | ||
|   | 6d231c2c99 | ||
|   | b93072865b | ||
|   | 14ebb6cd74 | ||
|   | 2ddbcd560e | ||
|   | c5ff7ed1c9 | ||
|   | c4bea5616c | ||
|   | 17fe147726 | ||
|   | 9fae4e7e1f | ||
|   | 0cebca498c | ||
|   | 521ff62aae | ||
|   | fd1df5ad88 | ||
|   | 91e7a35a07 | ||
|   | 09381abf46 | ||
|   | 3713c03c07 | ||
|   | bd8ddd7cd8 | ||
|   | f0dc1f927b | ||
|   | 984590c6d1 | ||
|   | d324021a3f | ||
|   | 1f4c0b3e9b | ||
|   | 69893aba4b | ||
|   | b9dcf89b37 | ||
|   | 0cda883b56 | ||
|   | ae58e633f0 | ||
|   | 06480bfd9d | ||
|   | 625f586945 | ||
|   | 7dbeaa475d | ||
|   | dff3d5f8af | ||
|   | 89c335919a | ||
|   | 2bb4573357 | ||
|   | 7037ce989c | ||
|   | bfdd2053ba | ||
|   | fcc3f92f8c | ||
|   | 8710267d53 | ||
|   | 85b6adcc9a | ||
|   | beec6e86e0 | ||
|   | 3dacffaaf9 | ||
|   | d90f2a1de1 | ||
|   | b6c9217429 | ||
|   | 7fc8da6769 | 
							
								
								
									
										77
									
								
								.claude/agents/quality-scale-rule-verifier.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								.claude/agents/quality-scale-rule-verifier.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | |||||||
|  | --- | ||||||
|  | name: quality-scale-rule-verifier | ||||||
|  | description: | | ||||||
|  |   Use this agent when you need to verify that a Home Assistant integration follows a specific quality scale rule. This includes checking if the integration implements required patterns, configurations, or code structures defined by the quality scale system. | ||||||
|  |  | ||||||
|  |   <example> | ||||||
|  |   Context: The user wants to verify if an integration follows a specific quality scale rule. | ||||||
|  |   user: "Check if the peblar integration follows the config-flow rule" | ||||||
|  |   assistant: "I'll use the quality scale rule verifier to check if the peblar integration properly implements the config-flow rule." | ||||||
|  |   <commentary> | ||||||
|  |   Since the user is asking to verify a quality scale rule implementation, use the quality-scale-rule-verifier agent. | ||||||
|  |   </commentary> | ||||||
|  |   </example> | ||||||
|  |  | ||||||
|  |   <example> | ||||||
|  |   Context: The user is reviewing if an integration reaches a specific quality scale level. | ||||||
|  |   user: "Verify that this integration reaches the bronze quality scale" | ||||||
|  |   assistant: "Let me use the quality scale rule verifier to check the bronze quality scale implementation." | ||||||
|  |   <commentary> | ||||||
|  |   The user wants to verify the integration has reached a certain quality level, so use multiple quality-scale-rule-verifier agents to verify each bronze rule. | ||||||
|  |   </commentary> | ||||||
|  |   </example> | ||||||
|  | model: inherit | ||||||
|  | color: yellow | ||||||
|  | tools: Read, Bash, Grep, Glob, WebFetch | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | You are an expert Home Assistant integration quality scale auditor specializing in verifying compliance with specific quality scale rules. You have deep knowledge of Home Assistant's architecture, best practices, and the quality scale system that ensures integration consistency and reliability. | ||||||
|  |  | ||||||
|  | You will verify if an integration follows a specific quality scale rule by: | ||||||
|  |  | ||||||
|  | 1. **Fetching Rule Documentation**: Retrieve the official rule documentation from: | ||||||
|  |    `https://raw.githubusercontent.com/home-assistant/developers.home-assistant/refs/heads/master/docs/core/integration-quality-scale/rules/{rule_name}.md` | ||||||
|  |    where `{rule_name}` is the rule identifier (e.g., 'config-flow', 'entity-unique-id', 'parallel-updates') | ||||||
|  |  | ||||||
|  | 2. **Understanding Rule Requirements**: Parse the rule documentation to identify: | ||||||
|  |    - Core requirements and mandatory implementations | ||||||
|  |    - Specific code patterns or configurations required | ||||||
|  |    - Common violations and anti-patterns | ||||||
|  |    - Exemption criteria (when a rule might not apply) | ||||||
|  |    - The quality tier this rule belongs to (Bronze, Silver, Gold, Platinum) | ||||||
|  |  | ||||||
|  | 3. **Analyzing Integration Code**: Examine the integration's codebase at `homeassistant/components/<integration domain>` focusing on: | ||||||
|  |    - `manifest.json` for quality scale declaration and configuration | ||||||
|  |    - `quality_scale.yaml` for rule status (done, todo, exempt) | ||||||
|  |    - Relevant Python modules based on the rule requirements | ||||||
|  |    - Configuration files and service definitions as needed | ||||||
|  |  | ||||||
|  | 4. **Verification Process**: | ||||||
|  |    - Check if the rule is marked as 'done', 'todo', or 'exempt' in quality_scale.yaml | ||||||
|  |    - If marked 'exempt', verify the exemption reason is valid | ||||||
|  |    - If marked 'done', verify the actual implementation matches requirements | ||||||
|  |    - Identify specific files and code sections that demonstrate compliance or violations | ||||||
|  |    - Consider the integration's declared quality tier when applying rules | ||||||
|  |    - To fetch the integration docs, use WebFetch to fetch from `https://raw.githubusercontent.com/home-assistant/home-assistant.io/refs/heads/current/source/_integrations/<integration domain>.markdown` | ||||||
|  |    - To fetch information about a PyPI package, use the URL `https://pypi.org/pypi/<package>/json` | ||||||
|  |  | ||||||
|  | 5. **Reporting Findings**: Provide a comprehensive verification report that includes: | ||||||
|  |    - **Rule Summary**: Brief description of what the rule requires | ||||||
|  |    - **Compliance Status**: Clear pass/fail/exempt determination | ||||||
|  |    - **Evidence**: Specific code examples showing compliance or violations | ||||||
|  |    - **Issues Found**: Detailed list of any non-compliance issues with file locations | ||||||
|  |    - **Recommendations**: Actionable steps to achieve compliance if needed | ||||||
|  |    - **Exemption Analysis**: If applicable, whether the exemption is justified | ||||||
|  |  | ||||||
|  | When examining code, you will: | ||||||
|  | - Look for exact implementation patterns specified in the rule | ||||||
|  | - Verify all required components are present and properly configured | ||||||
|  | - Check for common mistakes and anti-patterns | ||||||
|  | - Consider edge cases and error handling requirements | ||||||
|  | - Validate that implementations follow Home Assistant conventions | ||||||
|  |  | ||||||
|  | You will be thorough but focused, examining only the aspects relevant to the specific rule being verified. You will provide clear, actionable feedback that helps developers understand both what needs to be fixed and why it matters for integration quality. | ||||||
|  |  | ||||||
|  | If you cannot access the rule documentation or find the integration code, clearly state what information is missing and what you would need to complete the verification. | ||||||
|  |  | ||||||
|  | Remember that quality scale rules are cumulative - Bronze rules apply to all integrations with a quality scale, Silver rules apply to Silver+ integrations, and so on. Always consider the integration's target quality level when determining which rules should be enforced. | ||||||
| @@ -58,6 +58,7 @@ base_platforms: &base_platforms | |||||||
| # Extra components that trigger the full suite | # Extra components that trigger the full suite | ||||||
| components: &components | components: &components | ||||||
|   - homeassistant/components/alexa/** |   - homeassistant/components/alexa/** | ||||||
|  |   - homeassistant/components/analytics/** | ||||||
|   - homeassistant/components/application_credentials/** |   - homeassistant/components/application_credentials/** | ||||||
|   - homeassistant/components/assist_pipeline/** |   - homeassistant/components/assist_pipeline/** | ||||||
|   - homeassistant/components/auth/** |   - homeassistant/components/auth/** | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -55,8 +55,12 @@ | |||||||
|   creating the PR. If you're unsure about any of them, don't hesitate to ask. |   creating the PR. If you're unsure about any of them, don't hesitate to ask. | ||||||
|   We're here to help! This is simply a reminder of what we are going to look |   We're here to help! This is simply a reminder of what we are going to look | ||||||
|   for before merging your code. |   for before merging your code. | ||||||
|  |  | ||||||
|  |   AI tools are welcome, but contributors are responsible for *fully* | ||||||
|  |   understanding the code before submitting a PR. | ||||||
| --> | --> | ||||||
|  |  | ||||||
|  | - [ ] I understand the code I am submitting and can explain how it works. | ||||||
| - [ ] The code change is tested and works locally. | - [ ] The code change is tested and works locally. | ||||||
| - [ ] Local tests pass. **Your PR cannot be merged unless tests pass** | - [ ] Local tests pass. **Your PR cannot be merged unless tests pass** | ||||||
| - [ ] There is no commented out code in this PR. | - [ ] There is no commented out code in this PR. | ||||||
| @@ -64,6 +68,7 @@ | |||||||
| - [ ] I have followed the [perfect PR recommendations][perfect-pr] | - [ ] I have followed the [perfect PR recommendations][perfect-pr] | ||||||
| - [ ] The code has been formatted using Ruff (`ruff format homeassistant tests`) | - [ ] The code has been formatted using Ruff (`ruff format homeassistant tests`) | ||||||
| - [ ] Tests have been added to verify that the new code works. | - [ ] Tests have been added to verify that the new code works. | ||||||
|  | - [ ] Any generated code has been carefully reviewed for correctness and compliance with project standards. | ||||||
|  |  | ||||||
| If user exposed functionality or configuration variables are added/changed: | If user exposed functionality or configuration variables are added/changed: | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										48
									
								
								.github/workflows/builder.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										48
									
								
								.github/workflows/builder.yml
									
									
									
									
										vendored
									
									
								
							| @@ -27,12 +27,12 @@ jobs: | |||||||
|       publish: ${{ steps.version.outputs.publish }} |       publish: ${{ steps.version.outputs.publish }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           fetch-depth: 0 |           fetch-depth: 0 | ||||||
|  |  | ||||||
|       - name: Set up Python ${{ env.DEFAULT_PYTHON }} |       - name: Set up Python ${{ env.DEFAULT_PYTHON }} | ||||||
|         uses: actions/setup-python@v6.0.0 |         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON }} |           python-version: ${{ env.DEFAULT_PYTHON }} | ||||||
|  |  | ||||||
| @@ -69,7 +69,7 @@ jobs: | |||||||
|         run: find ./homeassistant/components/*/translations -name "*.json" | tar zcvf translations.tar.gz -T - |         run: find ./homeassistant/components/*/translations -name "*.json" | tar zcvf translations.tar.gz -T - | ||||||
|  |  | ||||||
|       - name: Upload translations |       - name: Upload translations | ||||||
|         uses: actions/upload-artifact@v4.6.2 |         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||||
|         with: |         with: | ||||||
|           name: translations |           name: translations | ||||||
|           path: translations.tar.gz |           path: translations.tar.gz | ||||||
| @@ -90,11 +90,11 @@ jobs: | |||||||
|         arch: ${{ fromJson(needs.init.outputs.architectures) }} |         arch: ${{ fromJson(needs.init.outputs.architectures) }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Download nightly wheels of frontend |       - name: Download nightly wheels of frontend | ||||||
|         if: needs.init.outputs.channel == 'dev' |         if: needs.init.outputs.channel == 'dev' | ||||||
|         uses: dawidd6/action-download-artifact@v11 |         uses: dawidd6/action-download-artifact@ac66b43f0e6a346234dd65d4d0c8fbb31cb316e5 # v11 | ||||||
|         with: |         with: | ||||||
|           github_token: ${{secrets.GITHUB_TOKEN}} |           github_token: ${{secrets.GITHUB_TOKEN}} | ||||||
|           repo: home-assistant/frontend |           repo: home-assistant/frontend | ||||||
| @@ -105,7 +105,7 @@ jobs: | |||||||
|  |  | ||||||
|       - name: Download nightly wheels of intents |       - name: Download nightly wheels of intents | ||||||
|         if: needs.init.outputs.channel == 'dev' |         if: needs.init.outputs.channel == 'dev' | ||||||
|         uses: dawidd6/action-download-artifact@v11 |         uses: dawidd6/action-download-artifact@ac66b43f0e6a346234dd65d4d0c8fbb31cb316e5 # v11 | ||||||
|         with: |         with: | ||||||
|           github_token: ${{secrets.GITHUB_TOKEN}} |           github_token: ${{secrets.GITHUB_TOKEN}} | ||||||
|           repo: OHF-Voice/intents-package |           repo: OHF-Voice/intents-package | ||||||
| @@ -116,7 +116,7 @@ jobs: | |||||||
|  |  | ||||||
|       - name: Set up Python ${{ env.DEFAULT_PYTHON }} |       - name: Set up Python ${{ env.DEFAULT_PYTHON }} | ||||||
|         if: needs.init.outputs.channel == 'dev' |         if: needs.init.outputs.channel == 'dev' | ||||||
|         uses: actions/setup-python@v6.0.0 |         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON }} |           python-version: ${{ env.DEFAULT_PYTHON }} | ||||||
|  |  | ||||||
| @@ -175,7 +175,7 @@ jobs: | |||||||
|           sed -i "s|pykrakenapi|# pykrakenapi|g" requirements_all.txt |           sed -i "s|pykrakenapi|# pykrakenapi|g" requirements_all.txt | ||||||
|  |  | ||||||
|       - name: Download translations |       - name: Download translations | ||||||
|         uses: actions/download-artifact@v5.0.0 |         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           name: translations |           name: translations | ||||||
|  |  | ||||||
| @@ -190,14 +190,15 @@ jobs: | |||||||
|           echo "${{ github.sha }};${{ github.ref }};${{ github.event_name }};${{ github.actor }}" > rootfs/OFFICIAL_IMAGE |           echo "${{ github.sha }};${{ github.ref }};${{ github.event_name }};${{ github.actor }}" > rootfs/OFFICIAL_IMAGE | ||||||
|  |  | ||||||
|       - name: Login to GitHub Container Registry |       - name: Login to GitHub Container Registry | ||||||
|         uses: docker/login-action@v3.5.0 |         uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0 | ||||||
|         with: |         with: | ||||||
|           registry: ghcr.io |           registry: ghcr.io | ||||||
|           username: ${{ github.repository_owner }} |           username: ${{ github.repository_owner }} | ||||||
|           password: ${{ secrets.GITHUB_TOKEN }} |           password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |  | ||||||
|  |       # home-assistant/builder doesn't support sha pinning | ||||||
|       - name: Build base image |       - name: Build base image | ||||||
|         uses: home-assistant/builder@2025.03.0 |         uses: home-assistant/builder@2025.09.0 | ||||||
|         with: |         with: | ||||||
|           args: | |           args: | | ||||||
|             $BUILD_ARGS \ |             $BUILD_ARGS \ | ||||||
| @@ -242,7 +243,7 @@ jobs: | |||||||
|           - green |           - green | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Set build additional args |       - name: Set build additional args | ||||||
|         run: | |         run: | | ||||||
| @@ -256,14 +257,15 @@ jobs: | |||||||
|           fi |           fi | ||||||
|  |  | ||||||
|       - name: Login to GitHub Container Registry |       - name: Login to GitHub Container Registry | ||||||
|         uses: docker/login-action@v3.5.0 |         uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0 | ||||||
|         with: |         with: | ||||||
|           registry: ghcr.io |           registry: ghcr.io | ||||||
|           username: ${{ github.repository_owner }} |           username: ${{ github.repository_owner }} | ||||||
|           password: ${{ secrets.GITHUB_TOKEN }} |           password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |  | ||||||
|  |       # home-assistant/builder doesn't support sha pinning | ||||||
|       - name: Build base image |       - name: Build base image | ||||||
|         uses: home-assistant/builder@2025.03.0 |         uses: home-assistant/builder@2025.09.0 | ||||||
|         with: |         with: | ||||||
|           args: | |           args: | | ||||||
|             $BUILD_ARGS \ |             $BUILD_ARGS \ | ||||||
| @@ -279,7 +281,7 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Initialize git |       - name: Initialize git | ||||||
|         uses: home-assistant/actions/helpers/git-init@master |         uses: home-assistant/actions/helpers/git-init@master | ||||||
| @@ -321,23 +323,23 @@ jobs: | |||||||
|         registry: ["ghcr.io/home-assistant", "docker.io/homeassistant"] |         registry: ["ghcr.io/home-assistant", "docker.io/homeassistant"] | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Install Cosign |       - name: Install Cosign | ||||||
|         uses: sigstore/cosign-installer@v3.9.2 |         uses: sigstore/cosign-installer@d7543c93d881b35a8faa02e8e3605f69b7a1ce62 # v3.10.0 | ||||||
|         with: |         with: | ||||||
|           cosign-release: "v2.2.3" |           cosign-release: "v2.2.3" | ||||||
|  |  | ||||||
|       - name: Login to DockerHub |       - name: Login to DockerHub | ||||||
|         if: matrix.registry == 'docker.io/homeassistant' |         if: matrix.registry == 'docker.io/homeassistant' | ||||||
|         uses: docker/login-action@v3.5.0 |         uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0 | ||||||
|         with: |         with: | ||||||
|           username: ${{ secrets.DOCKERHUB_USERNAME }} |           username: ${{ secrets.DOCKERHUB_USERNAME }} | ||||||
|           password: ${{ secrets.DOCKERHUB_TOKEN }} |           password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||||
|  |  | ||||||
|       - name: Login to GitHub Container Registry |       - name: Login to GitHub Container Registry | ||||||
|         if: matrix.registry == 'ghcr.io/home-assistant' |         if: matrix.registry == 'ghcr.io/home-assistant' | ||||||
|         uses: docker/login-action@v3.5.0 |         uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0 | ||||||
|         with: |         with: | ||||||
|           registry: ghcr.io |           registry: ghcr.io | ||||||
|           username: ${{ github.repository_owner }} |           username: ${{ github.repository_owner }} | ||||||
| @@ -454,15 +456,15 @@ jobs: | |||||||
|     if: github.repository_owner == 'home-assistant' && needs.init.outputs.publish == 'true' |     if: github.repository_owner == 'home-assistant' && needs.init.outputs.publish == 'true' | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Set up Python ${{ env.DEFAULT_PYTHON }} |       - name: Set up Python ${{ env.DEFAULT_PYTHON }} | ||||||
|         uses: actions/setup-python@v6.0.0 |         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON }} |           python-version: ${{ env.DEFAULT_PYTHON }} | ||||||
|  |  | ||||||
|       - name: Download translations |       - name: Download translations | ||||||
|         uses: actions/download-artifact@v5.0.0 |         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           name: translations |           name: translations | ||||||
|  |  | ||||||
| @@ -480,7 +482,7 @@ jobs: | |||||||
|           python -m build |           python -m build | ||||||
|  |  | ||||||
|       - name: Upload package to PyPI |       - name: Upload package to PyPI | ||||||
|         uses: pypa/gh-action-pypi-publish@v1.13.0 |         uses: pypa/gh-action-pypi-publish@ed0c53931b1dc9bd32cbe73a98c7f6766f8a527e # v1.13.0 | ||||||
|         with: |         with: | ||||||
|           skip-existing: true |           skip-existing: true | ||||||
|  |  | ||||||
| @@ -502,7 +504,7 @@ jobs: | |||||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Login to GitHub Container Registry |       - name: Login to GitHub Container Registry | ||||||
|         uses: docker/login-action@184bdaa0721073962dff0199f1fb9940f07167d1 # v3.5.0 |         uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0 | ||||||
|         with: |         with: | ||||||
|           registry: ghcr.io |           registry: ghcr.io | ||||||
|           username: ${{ github.repository_owner }} |           username: ${{ github.repository_owner }} | ||||||
|   | |||||||
							
								
								
									
										745
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										745
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										6
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
								
							| @@ -21,14 +21,14 @@ jobs: | |||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out code from GitHub |       - name: Check out code from GitHub | ||||||
|         uses: actions/checkout@v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Initialize CodeQL |       - name: Initialize CodeQL | ||||||
|         uses: github/codeql-action/init@v3.30.3 |         uses: github/codeql-action/init@e296a935590eb16afc0c0108289f68c87e2a89a5 # v4.30.7 | ||||||
|         with: |         with: | ||||||
|           languages: python |           languages: python | ||||||
|  |  | ||||||
|       - name: Perform CodeQL Analysis |       - name: Perform CodeQL Analysis | ||||||
|         uses: github/codeql-action/analyze@v3.30.3 |         uses: github/codeql-action/analyze@e296a935590eb16afc0c0108289f68c87e2a89a5 # v4.30.7 | ||||||
|         with: |         with: | ||||||
|           category: "/language:python" |           category: "/language:python" | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - name: Check if integration label was added and extract details |       - name: Check if integration label was added and extract details | ||||||
|         id: extract |         id: extract | ||||||
|         uses: actions/github-script@v8 |         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             // Debug: Log the event payload |             // Debug: Log the event payload | ||||||
| @@ -113,7 +113,7 @@ jobs: | |||||||
|       - name: Fetch similar issues |       - name: Fetch similar issues | ||||||
|         id: fetch_similar |         id: fetch_similar | ||||||
|         if: steps.extract.outputs.should_continue == 'true' |         if: steps.extract.outputs.should_continue == 'true' | ||||||
|         uses: actions/github-script@v8 |         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | ||||||
|         env: |         env: | ||||||
|           INTEGRATION_LABELS: ${{ steps.extract.outputs.integration_labels }} |           INTEGRATION_LABELS: ${{ steps.extract.outputs.integration_labels }} | ||||||
|           CURRENT_NUMBER: ${{ steps.extract.outputs.current_number }} |           CURRENT_NUMBER: ${{ steps.extract.outputs.current_number }} | ||||||
| @@ -231,7 +231,7 @@ jobs: | |||||||
|       - name: Detect duplicates using AI |       - name: Detect duplicates using AI | ||||||
|         id: ai_detection |         id: ai_detection | ||||||
|         if: steps.extract.outputs.should_continue == 'true' && steps.fetch_similar.outputs.has_similar == 'true' |         if: steps.extract.outputs.should_continue == 'true' && steps.fetch_similar.outputs.has_similar == 'true' | ||||||
|         uses: actions/ai-inference@v2.0.1 |         uses: actions/ai-inference@a1c11829223a786afe3b5663db904a3aa1eac3a2 # v2.0.1 | ||||||
|         with: |         with: | ||||||
|           model: openai/gpt-4o |           model: openai/gpt-4o | ||||||
|           system-prompt: | |           system-prompt: | | ||||||
| @@ -280,7 +280,7 @@ jobs: | |||||||
|       - name: Post duplicate detection results |       - name: Post duplicate detection results | ||||||
|         id: post_results |         id: post_results | ||||||
|         if: steps.extract.outputs.should_continue == 'true' && steps.fetch_similar.outputs.has_similar == 'true' |         if: steps.extract.outputs.should_continue == 'true' && steps.fetch_similar.outputs.has_similar == 'true' | ||||||
|         uses: actions/github-script@v8 |         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | ||||||
|         env: |         env: | ||||||
|           AI_RESPONSE: ${{ steps.ai_detection.outputs.response }} |           AI_RESPONSE: ${{ steps.ai_detection.outputs.response }} | ||||||
|           SIMILAR_ISSUES: ${{ steps.fetch_similar.outputs.similar_issues }} |           SIMILAR_ISSUES: ${{ steps.fetch_similar.outputs.similar_issues }} | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - name: Check issue language |       - name: Check issue language | ||||||
|         id: detect_language |         id: detect_language | ||||||
|         uses: actions/github-script@v8 |         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | ||||||
|         env: |         env: | ||||||
|           ISSUE_NUMBER: ${{ github.event.issue.number }} |           ISSUE_NUMBER: ${{ github.event.issue.number }} | ||||||
|           ISSUE_TITLE: ${{ github.event.issue.title }} |           ISSUE_TITLE: ${{ github.event.issue.title }} | ||||||
| @@ -57,7 +57,7 @@ jobs: | |||||||
|       - name: Detect language using AI |       - name: Detect language using AI | ||||||
|         id: ai_language_detection |         id: ai_language_detection | ||||||
|         if: steps.detect_language.outputs.should_continue == 'true' |         if: steps.detect_language.outputs.should_continue == 'true' | ||||||
|         uses: actions/ai-inference@v2.0.1 |         uses: actions/ai-inference@a1c11829223a786afe3b5663db904a3aa1eac3a2 # v2.0.1 | ||||||
|         with: |         with: | ||||||
|           model: openai/gpt-4o-mini |           model: openai/gpt-4o-mini | ||||||
|           system-prompt: | |           system-prompt: | | ||||||
| @@ -90,7 +90,7 @@ jobs: | |||||||
|  |  | ||||||
|       - name: Process non-English issues |       - name: Process non-English issues | ||||||
|         if: steps.detect_language.outputs.should_continue == 'true' |         if: steps.detect_language.outputs.should_continue == 'true' | ||||||
|         uses: actions/github-script@v8 |         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | ||||||
|         env: |         env: | ||||||
|           AI_RESPONSE: ${{ steps.ai_language_detection.outputs.response }} |           AI_RESPONSE: ${{ steps.ai_language_detection.outputs.response }} | ||||||
|           ISSUE_NUMBER: ${{ steps.detect_language.outputs.issue_number }} |           ISSUE_NUMBER: ${{ steps.detect_language.outputs.issue_number }} | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,7 +10,7 @@ jobs: | |||||||
|     if: github.repository_owner == 'home-assistant' |     if: github.repository_owner == 'home-assistant' | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: dessant/lock-threads@v5.0.1 |       - uses: dessant/lock-threads@1bf7ec25051fe7c00bdd17e6a7cf3d7bfb7dc771 # v5.0.1 | ||||||
|         with: |         with: | ||||||
|           github-token: ${{ github.token }} |           github-token: ${{ github.token }} | ||||||
|           issue-inactive-days: "30" |           issue-inactive-days: "30" | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/restrict-task-creation.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/restrict-task-creation.yml
									
									
									
									
										vendored
									
									
								
							| @@ -12,7 +12,7 @@ jobs: | |||||||
|     if: github.event.issue.type.name == 'Task' |     if: github.event.issue.type.name == 'Task' | ||||||
|     steps: |     steps: | ||||||
|       - name: Check if user is authorized |       - name: Check if user is authorized | ||||||
|         uses: actions/github-script@v8 |         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             const issueAuthor = context.payload.issue.user.login; |             const issueAuthor = context.payload.issue.user.login; | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							| @@ -17,7 +17,7 @@ jobs: | |||||||
|       # - No PRs marked as no-stale |       # - No PRs marked as no-stale | ||||||
|       # - No issues (-1) |       # - No issues (-1) | ||||||
|       - name: 60 days stale PRs policy |       - name: 60 days stale PRs policy | ||||||
|         uses: actions/stale@v10.0.0 |         uses: actions/stale@5f858e3efba33a5ca4407a664cc011ad407f2008 # v10.1.0 | ||||||
|         with: |         with: | ||||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} |           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|           days-before-stale: 60 |           days-before-stale: 60 | ||||||
| @@ -57,7 +57,7 @@ jobs: | |||||||
|       # - No issues marked as no-stale or help-wanted |       # - No issues marked as no-stale or help-wanted | ||||||
|       # - No PRs (-1) |       # - No PRs (-1) | ||||||
|       - name: 90 days stale issues |       - name: 90 days stale issues | ||||||
|         uses: actions/stale@v10.0.0 |         uses: actions/stale@5f858e3efba33a5ca4407a664cc011ad407f2008 # v10.1.0 | ||||||
|         with: |         with: | ||||||
|           repo-token: ${{ steps.token.outputs.token }} |           repo-token: ${{ steps.token.outputs.token }} | ||||||
|           days-before-stale: 90 |           days-before-stale: 90 | ||||||
| @@ -87,7 +87,7 @@ jobs: | |||||||
|       # - No Issues marked as no-stale or help-wanted |       # - No Issues marked as no-stale or help-wanted | ||||||
|       # - No PRs (-1) |       # - No PRs (-1) | ||||||
|       - name: Needs more information stale issues policy |       - name: Needs more information stale issues policy | ||||||
|         uses: actions/stale@v10.0.0 |         uses: actions/stale@5f858e3efba33a5ca4407a664cc011ad407f2008 # v10.1.0 | ||||||
|         with: |         with: | ||||||
|           repo-token: ${{ steps.token.outputs.token }} |           repo-token: ${{ steps.token.outputs.token }} | ||||||
|           only-labels: "needs-more-information" |           only-labels: "needs-more-information" | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								.github/workflows/translations.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/translations.yml
									
									
									
									
										vendored
									
									
								
							| @@ -19,10 +19,10 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Set up Python ${{ env.DEFAULT_PYTHON }} |       - name: Set up Python ${{ env.DEFAULT_PYTHON }} | ||||||
|         uses: actions/setup-python@v6.0.0 |         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON }} |           python-version: ${{ env.DEFAULT_PYTHON }} | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										36
									
								
								.github/workflows/wheels.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										36
									
								
								.github/workflows/wheels.yml
									
									
									
									
										vendored
									
									
								
							| @@ -32,11 +32,11 @@ jobs: | |||||||
|       architectures: ${{ steps.info.outputs.architectures }} |       architectures: ${{ steps.info.outputs.architectures }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Set up Python ${{ env.DEFAULT_PYTHON }} |       - name: Set up Python ${{ env.DEFAULT_PYTHON }} | ||||||
|         id: python |         id: python | ||||||
|         uses: actions/setup-python@v6.0.0 |         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.DEFAULT_PYTHON }} |           python-version: ${{ env.DEFAULT_PYTHON }} | ||||||
|           check-latest: true |           check-latest: true | ||||||
| @@ -91,7 +91,7 @@ jobs: | |||||||
|           ) > build_constraints.txt |           ) > build_constraints.txt | ||||||
|  |  | ||||||
|       - name: Upload env_file |       - name: Upload env_file | ||||||
|         uses: actions/upload-artifact@v4.6.2 |         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||||
|         with: |         with: | ||||||
|           name: env_file |           name: env_file | ||||||
|           path: ./.env_file |           path: ./.env_file | ||||||
| @@ -99,14 +99,14 @@ jobs: | |||||||
|           overwrite: true |           overwrite: true | ||||||
|  |  | ||||||
|       - name: Upload build_constraints |       - name: Upload build_constraints | ||||||
|         uses: actions/upload-artifact@v4.6.2 |         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||||
|         with: |         with: | ||||||
|           name: build_constraints |           name: build_constraints | ||||||
|           path: ./build_constraints.txt |           path: ./build_constraints.txt | ||||||
|           overwrite: true |           overwrite: true | ||||||
|  |  | ||||||
|       - name: Upload requirements_diff |       - name: Upload requirements_diff | ||||||
|         uses: actions/upload-artifact@v4.6.2 |         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||||
|         with: |         with: | ||||||
|           name: requirements_diff |           name: requirements_diff | ||||||
|           path: ./requirements_diff.txt |           path: ./requirements_diff.txt | ||||||
| @@ -118,7 +118,7 @@ jobs: | |||||||
|           python -m script.gen_requirements_all ci |           python -m script.gen_requirements_all ci | ||||||
|  |  | ||||||
|       - name: Upload requirements_all_wheels |       - name: Upload requirements_all_wheels | ||||||
|         uses: actions/upload-artifact@v4.6.2 |         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||||
|         with: |         with: | ||||||
|           name: requirements_all_wheels |           name: requirements_all_wheels | ||||||
|           path: ./requirements_all_wheels_*.txt |           path: ./requirements_all_wheels_*.txt | ||||||
| @@ -135,20 +135,20 @@ jobs: | |||||||
|         arch: ${{ fromJson(needs.init.outputs.architectures) }} |         arch: ${{ fromJson(needs.init.outputs.architectures) }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Download env_file |       - name: Download env_file | ||||||
|         uses: actions/download-artifact@v5.0.0 |         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           name: env_file |           name: env_file | ||||||
|  |  | ||||||
|       - name: Download build_constraints |       - name: Download build_constraints | ||||||
|         uses: actions/download-artifact@v5.0.0 |         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           name: build_constraints |           name: build_constraints | ||||||
|  |  | ||||||
|       - name: Download requirements_diff |       - name: Download requirements_diff | ||||||
|         uses: actions/download-artifact@v5.0.0 |         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           name: requirements_diff |           name: requirements_diff | ||||||
|  |  | ||||||
| @@ -158,8 +158,9 @@ jobs: | |||||||
|           sed -i "/uv/d" requirements.txt |           sed -i "/uv/d" requirements.txt | ||||||
|           sed -i "/uv/d" requirements_diff.txt |           sed -i "/uv/d" requirements_diff.txt | ||||||
|  |  | ||||||
|  |       # home-assistant/wheels doesn't support sha pinning | ||||||
|       - name: Build wheels |       - name: Build wheels | ||||||
|         uses: home-assistant/wheels@2025.07.0 |         uses: home-assistant/wheels@2025.09.1 | ||||||
|         with: |         with: | ||||||
|           abi: ${{ matrix.abi }} |           abi: ${{ matrix.abi }} | ||||||
|           tag: musllinux_1_2 |           tag: musllinux_1_2 | ||||||
| @@ -184,25 +185,25 @@ jobs: | |||||||
|         arch: ${{ fromJson(needs.init.outputs.architectures) }} |         arch: ${{ fromJson(needs.init.outputs.architectures) }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Download env_file |       - name: Download env_file | ||||||
|         uses: actions/download-artifact@v5.0.0 |         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           name: env_file |           name: env_file | ||||||
|  |  | ||||||
|       - name: Download build_constraints |       - name: Download build_constraints | ||||||
|         uses: actions/download-artifact@v5.0.0 |         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           name: build_constraints |           name: build_constraints | ||||||
|  |  | ||||||
|       - name: Download requirements_diff |       - name: Download requirements_diff | ||||||
|         uses: actions/download-artifact@v5.0.0 |         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           name: requirements_diff |           name: requirements_diff | ||||||
|  |  | ||||||
|       - name: Download requirements_all_wheels |       - name: Download requirements_all_wheels | ||||||
|         uses: actions/download-artifact@v5.0.0 |         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           name: requirements_all_wheels |           name: requirements_all_wheels | ||||||
|  |  | ||||||
| @@ -218,8 +219,9 @@ jobs: | |||||||
|           sed -i "/uv/d" requirements.txt |           sed -i "/uv/d" requirements.txt | ||||||
|           sed -i "/uv/d" requirements_diff.txt |           sed -i "/uv/d" requirements_diff.txt | ||||||
|  |  | ||||||
|  |       # home-assistant/wheels doesn't support sha pinning | ||||||
|       - name: Build wheels |       - name: Build wheels | ||||||
|         uses: home-assistant/wheels@2025.07.0 |         uses: home-assistant/wheels@2025.09.1 | ||||||
|         with: |         with: | ||||||
|           abi: ${{ matrix.abi }} |           abi: ${{ matrix.abi }} | ||||||
|           tag: musllinux_1_2 |           tag: musllinux_1_2 | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -140,5 +140,5 @@ tmp_cache | |||||||
| pytest_buckets.txt | pytest_buckets.txt | ||||||
|  |  | ||||||
| # AI tooling | # AI tooling | ||||||
| .claude | .claude/settings.local.json | ||||||
|  |  | ||||||
|   | |||||||
| @@ -142,6 +142,7 @@ homeassistant.components.cloud.* | |||||||
| homeassistant.components.co2signal.* | homeassistant.components.co2signal.* | ||||||
| homeassistant.components.comelit.* | homeassistant.components.comelit.* | ||||||
| homeassistant.components.command_line.* | homeassistant.components.command_line.* | ||||||
|  | homeassistant.components.compit.* | ||||||
| homeassistant.components.config.* | homeassistant.components.config.* | ||||||
| homeassistant.components.configurator.* | homeassistant.components.configurator.* | ||||||
| homeassistant.components.cookidoo.* | homeassistant.components.cookidoo.* | ||||||
| @@ -202,6 +203,7 @@ homeassistant.components.feedreader.* | |||||||
| homeassistant.components.file_upload.* | homeassistant.components.file_upload.* | ||||||
| homeassistant.components.filesize.* | homeassistant.components.filesize.* | ||||||
| homeassistant.components.filter.* | homeassistant.components.filter.* | ||||||
|  | homeassistant.components.firefly_iii.* | ||||||
| homeassistant.components.fitbit.* | homeassistant.components.fitbit.* | ||||||
| homeassistant.components.flexit_bacnet.* | homeassistant.components.flexit_bacnet.* | ||||||
| homeassistant.components.flux_led.* | homeassistant.components.flux_led.* | ||||||
| @@ -219,6 +221,7 @@ homeassistant.components.generic_thermostat.* | |||||||
| homeassistant.components.geo_location.* | homeassistant.components.geo_location.* | ||||||
| homeassistant.components.geocaching.* | homeassistant.components.geocaching.* | ||||||
| homeassistant.components.gios.* | homeassistant.components.gios.* | ||||||
|  | homeassistant.components.github.* | ||||||
| homeassistant.components.glances.* | homeassistant.components.glances.* | ||||||
| homeassistant.components.go2rtc.* | homeassistant.components.go2rtc.* | ||||||
| homeassistant.components.goalzero.* | homeassistant.components.goalzero.* | ||||||
| @@ -324,6 +327,7 @@ homeassistant.components.london_underground.* | |||||||
| homeassistant.components.lookin.* | homeassistant.components.lookin.* | ||||||
| homeassistant.components.lovelace.* | homeassistant.components.lovelace.* | ||||||
| homeassistant.components.luftdaten.* | homeassistant.components.luftdaten.* | ||||||
|  | homeassistant.components.lunatone.* | ||||||
| homeassistant.components.madvr.* | homeassistant.components.madvr.* | ||||||
| homeassistant.components.manual.* | homeassistant.components.manual.* | ||||||
| homeassistant.components.mastodon.* | homeassistant.components.mastodon.* | ||||||
| @@ -442,6 +446,7 @@ homeassistant.components.rituals_perfume_genie.* | |||||||
| homeassistant.components.roborock.* | homeassistant.components.roborock.* | ||||||
| homeassistant.components.roku.* | homeassistant.components.roku.* | ||||||
| homeassistant.components.romy.* | homeassistant.components.romy.* | ||||||
|  | homeassistant.components.route_b_smart_meter.* | ||||||
| homeassistant.components.rpi_power.* | homeassistant.components.rpi_power.* | ||||||
| homeassistant.components.rss_feed_template.* | homeassistant.components.rss_feed_template.* | ||||||
| homeassistant.components.russound_rio.* | homeassistant.components.russound_rio.* | ||||||
| @@ -551,6 +556,7 @@ homeassistant.components.vacuum.* | |||||||
| homeassistant.components.vallox.* | homeassistant.components.vallox.* | ||||||
| homeassistant.components.valve.* | homeassistant.components.valve.* | ||||||
| homeassistant.components.velbus.* | homeassistant.components.velbus.* | ||||||
|  | homeassistant.components.vivotek.* | ||||||
| homeassistant.components.vlc_telnet.* | homeassistant.components.vlc_telnet.* | ||||||
| homeassistant.components.vodafone_station.* | homeassistant.components.vodafone_station.* | ||||||
| homeassistant.components.volvo.* | homeassistant.components.volvo.* | ||||||
|   | |||||||
							
								
								
									
										47
									
								
								CODEOWNERS
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										47
									
								
								CODEOWNERS
									
									
									
										generated
									
									
									
								
							| @@ -107,8 +107,8 @@ build.json @home-assistant/supervisor | |||||||
| /homeassistant/components/ambient_station/ @bachya | /homeassistant/components/ambient_station/ @bachya | ||||||
| /tests/components/ambient_station/ @bachya | /tests/components/ambient_station/ @bachya | ||||||
| /homeassistant/components/amcrest/ @flacjacket | /homeassistant/components/amcrest/ @flacjacket | ||||||
| /homeassistant/components/analytics/ @home-assistant/core @ludeeus | /homeassistant/components/analytics/ @home-assistant/core | ||||||
| /tests/components/analytics/ @home-assistant/core @ludeeus | /tests/components/analytics/ @home-assistant/core | ||||||
| /homeassistant/components/analytics_insights/ @joostlek | /homeassistant/components/analytics_insights/ @joostlek | ||||||
| /tests/components/analytics_insights/ @joostlek | /tests/components/analytics_insights/ @joostlek | ||||||
| /homeassistant/components/android_ip_webcam/ @engrbm87 | /homeassistant/components/android_ip_webcam/ @engrbm87 | ||||||
| @@ -292,6 +292,8 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/command_line/ @gjohansson-ST | /tests/components/command_line/ @gjohansson-ST | ||||||
| /homeassistant/components/compensation/ @Petro31 | /homeassistant/components/compensation/ @Petro31 | ||||||
| /tests/components/compensation/ @Petro31 | /tests/components/compensation/ @Petro31 | ||||||
|  | /homeassistant/components/compit/ @Przemko92 | ||||||
|  | /tests/components/compit/ @Przemko92 | ||||||
| /homeassistant/components/config/ @home-assistant/core | /homeassistant/components/config/ @home-assistant/core | ||||||
| /tests/components/config/ @home-assistant/core | /tests/components/config/ @home-assistant/core | ||||||
| /homeassistant/components/configurator/ @home-assistant/core | /homeassistant/components/configurator/ @home-assistant/core | ||||||
| @@ -314,6 +316,8 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/crownstone/ @Crownstone @RicArch97 | /tests/components/crownstone/ @Crownstone @RicArch97 | ||||||
| /homeassistant/components/cups/ @fabaff | /homeassistant/components/cups/ @fabaff | ||||||
| /tests/components/cups/ @fabaff | /tests/components/cups/ @fabaff | ||||||
|  | /homeassistant/components/cync/ @Kinachi249 | ||||||
|  | /tests/components/cync/ @Kinachi249 | ||||||
| /homeassistant/components/daikin/ @fredrike | /homeassistant/components/daikin/ @fredrike | ||||||
| /tests/components/daikin/ @fredrike | /tests/components/daikin/ @fredrike | ||||||
| /homeassistant/components/date/ @home-assistant/core | /homeassistant/components/date/ @home-assistant/core | ||||||
| @@ -408,6 +412,8 @@ build.json @home-assistant/supervisor | |||||||
| /homeassistant/components/egardia/ @jeroenterheerdt | /homeassistant/components/egardia/ @jeroenterheerdt | ||||||
| /homeassistant/components/eheimdigital/ @autinerd | /homeassistant/components/eheimdigital/ @autinerd | ||||||
| /tests/components/eheimdigital/ @autinerd | /tests/components/eheimdigital/ @autinerd | ||||||
|  | /homeassistant/components/ekeybionyx/ @richardpolzer | ||||||
|  | /tests/components/ekeybionyx/ @richardpolzer | ||||||
| /homeassistant/components/electrasmart/ @jafar-atili | /homeassistant/components/electrasmart/ @jafar-atili | ||||||
| /tests/components/electrasmart/ @jafar-atili | /tests/components/electrasmart/ @jafar-atili | ||||||
| /homeassistant/components/electric_kiwi/ @mikey0000 | /homeassistant/components/electric_kiwi/ @mikey0000 | ||||||
| @@ -442,8 +448,6 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/energyzero/ @klaasnicolaas | /tests/components/energyzero/ @klaasnicolaas | ||||||
| /homeassistant/components/enigma2/ @autinerd | /homeassistant/components/enigma2/ @autinerd | ||||||
| /tests/components/enigma2/ @autinerd | /tests/components/enigma2/ @autinerd | ||||||
| /homeassistant/components/enocean/ @bdurrer |  | ||||||
| /tests/components/enocean/ @bdurrer |  | ||||||
| /homeassistant/components/enphase_envoy/ @bdraco @cgarwood @catsmanac | /homeassistant/components/enphase_envoy/ @bdraco @cgarwood @catsmanac | ||||||
| /tests/components/enphase_envoy/ @bdraco @cgarwood @catsmanac | /tests/components/enphase_envoy/ @bdraco @cgarwood @catsmanac | ||||||
| /homeassistant/components/entur_public_transport/ @hfurubotten | /homeassistant/components/entur_public_transport/ @hfurubotten | ||||||
| @@ -488,6 +492,8 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/filesize/ @gjohansson-ST | /tests/components/filesize/ @gjohansson-ST | ||||||
| /homeassistant/components/filter/ @dgomes | /homeassistant/components/filter/ @dgomes | ||||||
| /tests/components/filter/ @dgomes | /tests/components/filter/ @dgomes | ||||||
|  | /homeassistant/components/firefly_iii/ @erwindouna | ||||||
|  | /tests/components/firefly_iii/ @erwindouna | ||||||
| /homeassistant/components/fireservicerota/ @cyberjunky | /homeassistant/components/fireservicerota/ @cyberjunky | ||||||
| /tests/components/fireservicerota/ @cyberjunky | /tests/components/fireservicerota/ @cyberjunky | ||||||
| /homeassistant/components/firmata/ @DaAwesomeP | /homeassistant/components/firmata/ @DaAwesomeP | ||||||
| @@ -772,6 +778,8 @@ build.json @home-assistant/supervisor | |||||||
| /homeassistant/components/iqvia/ @bachya | /homeassistant/components/iqvia/ @bachya | ||||||
| /tests/components/iqvia/ @bachya | /tests/components/iqvia/ @bachya | ||||||
| /homeassistant/components/irish_rail_transport/ @ttroy50 | /homeassistant/components/irish_rail_transport/ @ttroy50 | ||||||
|  | /homeassistant/components/irm_kmi/ @jdejaegh | ||||||
|  | /tests/components/irm_kmi/ @jdejaegh | ||||||
| /homeassistant/components/iron_os/ @tr4nt0r | /homeassistant/components/iron_os/ @tr4nt0r | ||||||
| /tests/components/iron_os/ @tr4nt0r | /tests/components/iron_os/ @tr4nt0r | ||||||
| /homeassistant/components/isal/ @bdraco | /homeassistant/components/isal/ @bdraco | ||||||
| @@ -902,6 +910,8 @@ build.json @home-assistant/supervisor | |||||||
| /homeassistant/components/luci/ @mzdrale | /homeassistant/components/luci/ @mzdrale | ||||||
| /homeassistant/components/luftdaten/ @fabaff @frenck | /homeassistant/components/luftdaten/ @fabaff @frenck | ||||||
| /tests/components/luftdaten/ @fabaff @frenck | /tests/components/luftdaten/ @fabaff @frenck | ||||||
|  | /homeassistant/components/lunatone/ @MoonDevLT | ||||||
|  | /tests/components/lunatone/ @MoonDevLT | ||||||
| /homeassistant/components/lupusec/ @majuss @suaveolent | /homeassistant/components/lupusec/ @majuss @suaveolent | ||||||
| /tests/components/lupusec/ @majuss @suaveolent | /tests/components/lupusec/ @majuss @suaveolent | ||||||
| /homeassistant/components/lutron/ @cdheiser @wilburCForce | /homeassistant/components/lutron/ @cdheiser @wilburCForce | ||||||
| @@ -947,6 +957,8 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/met_eireann/ @DylanGore | /tests/components/met_eireann/ @DylanGore | ||||||
| /homeassistant/components/meteo_france/ @hacf-fr @oncleben31 @Quentame | /homeassistant/components/meteo_france/ @hacf-fr @oncleben31 @Quentame | ||||||
| /tests/components/meteo_france/ @hacf-fr @oncleben31 @Quentame | /tests/components/meteo_france/ @hacf-fr @oncleben31 @Quentame | ||||||
|  | /homeassistant/components/meteo_lt/ @xE1H | ||||||
|  | /tests/components/meteo_lt/ @xE1H | ||||||
| /homeassistant/components/meteoalarm/ @rolfberkenbosch | /homeassistant/components/meteoalarm/ @rolfberkenbosch | ||||||
| /homeassistant/components/meteoclimatic/ @adrianmo | /homeassistant/components/meteoclimatic/ @adrianmo | ||||||
| /tests/components/meteoclimatic/ @adrianmo | /tests/components/meteoclimatic/ @adrianmo | ||||||
| @@ -1017,7 +1029,8 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/nanoleaf/ @milanmeu @joostlek | /tests/components/nanoleaf/ @milanmeu @joostlek | ||||||
| /homeassistant/components/nasweb/ @nasWebio | /homeassistant/components/nasweb/ @nasWebio | ||||||
| /tests/components/nasweb/ @nasWebio | /tests/components/nasweb/ @nasWebio | ||||||
| /homeassistant/components/nederlandse_spoorwegen/ @YarmoM | /homeassistant/components/nederlandse_spoorwegen/ @YarmoM @heindrichpaul | ||||||
|  | /tests/components/nederlandse_spoorwegen/ @YarmoM @heindrichpaul | ||||||
| /homeassistant/components/ness_alarm/ @nickw444 | /homeassistant/components/ness_alarm/ @nickw444 | ||||||
| /tests/components/ness_alarm/ @nickw444 | /tests/components/ness_alarm/ @nickw444 | ||||||
| /homeassistant/components/nest/ @allenporter | /homeassistant/components/nest/ @allenporter | ||||||
| @@ -1052,6 +1065,8 @@ build.json @home-assistant/supervisor | |||||||
| /homeassistant/components/nilu/ @hfurubotten | /homeassistant/components/nilu/ @hfurubotten | ||||||
| /homeassistant/components/nina/ @DeerMaximum | /homeassistant/components/nina/ @DeerMaximum | ||||||
| /tests/components/nina/ @DeerMaximum | /tests/components/nina/ @DeerMaximum | ||||||
|  | /homeassistant/components/nintendo_parental/ @pantherale0 | ||||||
|  | /tests/components/nintendo_parental/ @pantherale0 | ||||||
| /homeassistant/components/nissan_leaf/ @filcole | /homeassistant/components/nissan_leaf/ @filcole | ||||||
| /homeassistant/components/noaa_tides/ @jdelaney72 | /homeassistant/components/noaa_tides/ @jdelaney72 | ||||||
| /homeassistant/components/nobo_hub/ @echoromeo @oyvindwe | /homeassistant/components/nobo_hub/ @echoromeo @oyvindwe | ||||||
| @@ -1183,8 +1198,6 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/plex/ @jjlawren | /tests/components/plex/ @jjlawren | ||||||
| /homeassistant/components/plugwise/ @CoMPaTech @bouwew | /homeassistant/components/plugwise/ @CoMPaTech @bouwew | ||||||
| /tests/components/plugwise/ @CoMPaTech @bouwew | /tests/components/plugwise/ @CoMPaTech @bouwew | ||||||
| /homeassistant/components/plum_lightpad/ @ColinHarrington @prystupa |  | ||||||
| /tests/components/plum_lightpad/ @ColinHarrington @prystupa |  | ||||||
| /homeassistant/components/point/ @fredrike | /homeassistant/components/point/ @fredrike | ||||||
| /tests/components/point/ @fredrike | /tests/components/point/ @fredrike | ||||||
| /homeassistant/components/pooldose/ @lmaertin | /homeassistant/components/pooldose/ @lmaertin | ||||||
| @@ -1327,6 +1340,8 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/roomba/ @pschmitt @cyr-ius @shenxn @Orhideous | /tests/components/roomba/ @pschmitt @cyr-ius @shenxn @Orhideous | ||||||
| /homeassistant/components/roon/ @pavoni | /homeassistant/components/roon/ @pavoni | ||||||
| /tests/components/roon/ @pavoni | /tests/components/roon/ @pavoni | ||||||
|  | /homeassistant/components/route_b_smart_meter/ @SeraphicRav | ||||||
|  | /tests/components/route_b_smart_meter/ @SeraphicRav | ||||||
| /homeassistant/components/rpi_power/ @shenxn @swetoast | /homeassistant/components/rpi_power/ @shenxn @swetoast | ||||||
| /tests/components/rpi_power/ @shenxn @swetoast | /tests/components/rpi_power/ @shenxn @swetoast | ||||||
| /homeassistant/components/rss_feed_template/ @home-assistant/core | /homeassistant/components/rss_feed_template/ @home-assistant/core | ||||||
| @@ -1349,6 +1364,8 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/samsungtv/ @chemelli74 @epenet | /tests/components/samsungtv/ @chemelli74 @epenet | ||||||
| /homeassistant/components/sanix/ @tomaszsluszniak | /homeassistant/components/sanix/ @tomaszsluszniak | ||||||
| /tests/components/sanix/ @tomaszsluszniak | /tests/components/sanix/ @tomaszsluszniak | ||||||
|  | /homeassistant/components/satel_integra/ @Tommatheussen | ||||||
|  | /tests/components/satel_integra/ @Tommatheussen | ||||||
| /homeassistant/components/scene/ @home-assistant/core | /homeassistant/components/scene/ @home-assistant/core | ||||||
| /tests/components/scene/ @home-assistant/core | /tests/components/scene/ @home-assistant/core | ||||||
| /homeassistant/components/schedule/ @home-assistant/core | /homeassistant/components/schedule/ @home-assistant/core | ||||||
| @@ -1396,8 +1413,8 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/sfr_box/ @epenet | /tests/components/sfr_box/ @epenet | ||||||
| /homeassistant/components/sftp_storage/ @maretodoric | /homeassistant/components/sftp_storage/ @maretodoric | ||||||
| /tests/components/sftp_storage/ @maretodoric | /tests/components/sftp_storage/ @maretodoric | ||||||
| /homeassistant/components/sharkiq/ @JeffResc @funkybunch | /homeassistant/components/sharkiq/ @JeffResc @funkybunch @TheOneOgre | ||||||
| /tests/components/sharkiq/ @JeffResc @funkybunch | /tests/components/sharkiq/ @JeffResc @funkybunch @TheOneOgre | ||||||
| /homeassistant/components/shell_command/ @home-assistant/core | /homeassistant/components/shell_command/ @home-assistant/core | ||||||
| /tests/components/shell_command/ @home-assistant/core | /tests/components/shell_command/ @home-assistant/core | ||||||
| /homeassistant/components/shelly/ @bieniu @thecode @chemelli74 @bdraco | /homeassistant/components/shelly/ @bieniu @thecode @chemelli74 @bdraco | ||||||
| @@ -1530,8 +1547,8 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/switchbee/ @jafar-atili | /tests/components/switchbee/ @jafar-atili | ||||||
| /homeassistant/components/switchbot/ @danielhiversen @RenierM26 @murtas @Eloston @dsypniewski @zerzhang | /homeassistant/components/switchbot/ @danielhiversen @RenierM26 @murtas @Eloston @dsypniewski @zerzhang | ||||||
| /tests/components/switchbot/ @danielhiversen @RenierM26 @murtas @Eloston @dsypniewski @zerzhang | /tests/components/switchbot/ @danielhiversen @RenierM26 @murtas @Eloston @dsypniewski @zerzhang | ||||||
| /homeassistant/components/switchbot_cloud/ @SeraphicRav @laurence-presland @Gigatrappeur | /homeassistant/components/switchbot_cloud/ @SeraphicRav @laurence-presland @Gigatrappeur @XiaoLing-git | ||||||
| /tests/components/switchbot_cloud/ @SeraphicRav @laurence-presland @Gigatrappeur | /tests/components/switchbot_cloud/ @SeraphicRav @laurence-presland @Gigatrappeur @XiaoLing-git | ||||||
| /homeassistant/components/switcher_kis/ @thecode @YogevBokobza | /homeassistant/components/switcher_kis/ @thecode @YogevBokobza | ||||||
| /tests/components/switcher_kis/ @thecode @YogevBokobza | /tests/components/switcher_kis/ @thecode @YogevBokobza | ||||||
| /homeassistant/components/switchmate/ @danielhiversen @qiz-li | /homeassistant/components/switchmate/ @danielhiversen @qiz-li | ||||||
| @@ -1676,6 +1693,8 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/uptime_kuma/ @tr4nt0r | /tests/components/uptime_kuma/ @tr4nt0r | ||||||
| /homeassistant/components/uptimerobot/ @ludeeus @chemelli74 | /homeassistant/components/uptimerobot/ @ludeeus @chemelli74 | ||||||
| /tests/components/uptimerobot/ @ludeeus @chemelli74 | /tests/components/uptimerobot/ @ludeeus @chemelli74 | ||||||
|  | /homeassistant/components/usage_prediction/ @home-assistant/core | ||||||
|  | /tests/components/usage_prediction/ @home-assistant/core | ||||||
| /homeassistant/components/usb/ @bdraco | /homeassistant/components/usb/ @bdraco | ||||||
| /tests/components/usb/ @bdraco | /tests/components/usb/ @bdraco | ||||||
| /homeassistant/components/usgs_earthquakes_feed/ @exxamalte | /homeassistant/components/usgs_earthquakes_feed/ @exxamalte | ||||||
| @@ -1705,6 +1724,8 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/vesync/ @markperdue @webdjoe @thegardenmonkey @cdnninja @iprak @sapuseven | /tests/components/vesync/ @markperdue @webdjoe @thegardenmonkey @cdnninja @iprak @sapuseven | ||||||
| /homeassistant/components/vicare/ @CFenner | /homeassistant/components/vicare/ @CFenner | ||||||
| /tests/components/vicare/ @CFenner | /tests/components/vicare/ @CFenner | ||||||
|  | /homeassistant/components/victron_remote_monitoring/ @AndyTempel | ||||||
|  | /tests/components/victron_remote_monitoring/ @AndyTempel | ||||||
| /homeassistant/components/vilfo/ @ManneW | /homeassistant/components/vilfo/ @ManneW | ||||||
| /tests/components/vilfo/ @ManneW | /tests/components/vilfo/ @ManneW | ||||||
| /homeassistant/components/vivotek/ @HarlemSquirrel | /homeassistant/components/vivotek/ @HarlemSquirrel | ||||||
| @@ -1720,8 +1741,8 @@ build.json @home-assistant/supervisor | |||||||
| /tests/components/volumio/ @OnFreund | /tests/components/volumio/ @OnFreund | ||||||
| /homeassistant/components/volvo/ @thomasddn | /homeassistant/components/volvo/ @thomasddn | ||||||
| /tests/components/volvo/ @thomasddn | /tests/components/volvo/ @thomasddn | ||||||
| /homeassistant/components/volvooncall/ @molobrakos | /homeassistant/components/volvooncall/ @molobrakos @svrooij | ||||||
| /tests/components/volvooncall/ @molobrakos | /tests/components/volvooncall/ @molobrakos @svrooij | ||||||
| /homeassistant/components/wake_on_lan/ @ntilley905 | /homeassistant/components/wake_on_lan/ @ntilley905 | ||||||
| /tests/components/wake_on_lan/ @ntilley905 | /tests/components/wake_on_lan/ @ntilley905 | ||||||
| /homeassistant/components/wake_word/ @home-assistant/core @synesthesiam | /homeassistant/components/wake_word/ @home-assistant/core @synesthesiam | ||||||
|   | |||||||
| @@ -34,9 +34,10 @@ WORKDIR /usr/src | |||||||
|  |  | ||||||
| COPY --from=ghcr.io/astral-sh/uv:latest /uv /usr/local/bin/uv | COPY --from=ghcr.io/astral-sh/uv:latest /uv /usr/local/bin/uv | ||||||
|  |  | ||||||
|  | USER vscode | ||||||
|  |  | ||||||
| RUN uv python install 3.13.2 | RUN uv python install 3.13.2 | ||||||
|  |  | ||||||
| USER vscode |  | ||||||
| ENV VIRTUAL_ENV="/home/vscode/.local/ha-venv" | ENV VIRTUAL_ENV="/home/vscode/.local/ha-venv" | ||||||
| RUN uv venv $VIRTUAL_ENV | RUN uv venv $VIRTUAL_ENV | ||||||
| ENV PATH="$VIRTUAL_ENV/bin:$PATH" | ENV PATH="$VIRTUAL_ENV/bin:$PATH" | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								build.yaml
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								build.yaml
									
									
									
									
									
								
							| @@ -1,10 +1,10 @@ | |||||||
| image: ghcr.io/home-assistant/{arch}-homeassistant | image: ghcr.io/home-assistant/{arch}-homeassistant | ||||||
| build_from: | build_from: | ||||||
|   aarch64: ghcr.io/home-assistant/aarch64-homeassistant-base:2025.09.1 |   aarch64: ghcr.io/home-assistant/aarch64-homeassistant-base:2025.10.0 | ||||||
|   armhf: ghcr.io/home-assistant/armhf-homeassistant-base:2025.09.1 |   armhf: ghcr.io/home-assistant/armhf-homeassistant-base:2025.10.0 | ||||||
|   armv7: ghcr.io/home-assistant/armv7-homeassistant-base:2025.09.1 |   armv7: ghcr.io/home-assistant/armv7-homeassistant-base:2025.10.0 | ||||||
|   amd64: ghcr.io/home-assistant/amd64-homeassistant-base:2025.09.1 |   amd64: ghcr.io/home-assistant/amd64-homeassistant-base:2025.10.0 | ||||||
|   i386: ghcr.io/home-assistant/i386-homeassistant-base:2025.09.1 |   i386: ghcr.io/home-assistant/i386-homeassistant-base:2025.10.0 | ||||||
| codenotary: | codenotary: | ||||||
|   signer: notary@home-assistant.io |   signer: notary@home-assistant.io | ||||||
|   base_image: notary@home-assistant.io |   base_image: notary@home-assistant.io | ||||||
|   | |||||||
| @@ -616,34 +616,34 @@ async def async_enable_logging( | |||||||
|         ), |         ), | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     # Log errors to a file if we have write access to file or config dir |     logger = logging.getLogger() | ||||||
|  |     logger.setLevel(logging.INFO if verbose else logging.WARNING) | ||||||
|  |  | ||||||
|     if log_file is None: |     if log_file is None: | ||||||
|         err_log_path = hass.config.path(ERROR_LOG_FILENAME) |         default_log_path = hass.config.path(ERROR_LOG_FILENAME) | ||||||
|  |         if "SUPERVISOR" in os.environ: | ||||||
|  |             _LOGGER.info("Running in Supervisor, not logging to file") | ||||||
|  |             # Rename the default log file if it exists, since previous versions created | ||||||
|  |             # it even on Supervisor | ||||||
|  |             if os.path.isfile(default_log_path): | ||||||
|  |                 with contextlib.suppress(OSError): | ||||||
|  |                     os.rename(default_log_path, f"{default_log_path}.old") | ||||||
|  |             err_log_path = None | ||||||
|  |         else: | ||||||
|  |             err_log_path = default_log_path | ||||||
|     else: |     else: | ||||||
|         err_log_path = os.path.abspath(log_file) |         err_log_path = os.path.abspath(log_file) | ||||||
|  |  | ||||||
|     err_path_exists = os.path.isfile(err_log_path) |     if err_log_path: | ||||||
|     err_dir = os.path.dirname(err_log_path) |  | ||||||
|  |  | ||||||
|     # Check if we can write to the error log if it exists or that |  | ||||||
|     # we can create files in the containing directory if not. |  | ||||||
|     if (err_path_exists and os.access(err_log_path, os.W_OK)) or ( |  | ||||||
|         not err_path_exists and os.access(err_dir, os.W_OK) |  | ||||||
|     ): |  | ||||||
|         err_handler = await hass.async_add_executor_job( |         err_handler = await hass.async_add_executor_job( | ||||||
|             _create_log_file, err_log_path, log_rotate_days |             _create_log_file, err_log_path, log_rotate_days | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         err_handler.setFormatter(logging.Formatter(fmt, datefmt=FORMAT_DATETIME)) |         err_handler.setFormatter(logging.Formatter(fmt, datefmt=FORMAT_DATETIME)) | ||||||
|  |  | ||||||
|         logger = logging.getLogger() |  | ||||||
|         logger.addHandler(err_handler) |         logger.addHandler(err_handler) | ||||||
|         logger.setLevel(logging.INFO if verbose else logging.WARNING) |  | ||||||
|  |  | ||||||
|         # Save the log file location for access by other components. |         # Save the log file location for access by other components. | ||||||
|         hass.data[DATA_LOGGING] = err_log_path |         hass.data[DATA_LOGGING] = err_log_path | ||||||
|     else: |  | ||||||
|         _LOGGER.error("Unable to set up error log %s (access denied)", err_log_path) |  | ||||||
|  |  | ||||||
|     async_activate_log_queue_handler(hass) |     async_activate_log_queue_handler(hass) | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								homeassistant/brands/eltako.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								homeassistant/brands/eltako.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | { | ||||||
|  |   "domain": "eltako", | ||||||
|  |   "name": "Eltako", | ||||||
|  |   "iot_standards": ["matter"] | ||||||
|  | } | ||||||
| @@ -1,5 +0,0 @@ | |||||||
| { |  | ||||||
|   "domain": "ibm", |  | ||||||
|   "name": "IBM", |  | ||||||
|   "integrations": ["watson_iot", "watson_tts"] |  | ||||||
| } |  | ||||||
							
								
								
									
										5
									
								
								homeassistant/brands/konnected.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								homeassistant/brands/konnected.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | { | ||||||
|  |   "domain": "konnected", | ||||||
|  |   "name": "Konnected", | ||||||
|  |   "integrations": ["konnected", "konnected_esphome"] | ||||||
|  | } | ||||||
							
								
								
									
										5
									
								
								homeassistant/brands/level.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								homeassistant/brands/level.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | { | ||||||
|  |   "domain": "level", | ||||||
|  |   "name": "Level", | ||||||
|  |   "iot_standards": ["matter"] | ||||||
|  | } | ||||||
| @@ -8,14 +8,17 @@ import logging | |||||||
| from aioacaia.acaiascale import AcaiaScale | from aioacaia.acaiascale import AcaiaScale | ||||||
| from aioacaia.exceptions import AcaiaDeviceNotFound, AcaiaError | from aioacaia.exceptions import AcaiaDeviceNotFound, AcaiaError | ||||||
|  |  | ||||||
|  | from homeassistant.components.bluetooth import async_get_scanner | ||||||
| from homeassistant.config_entries import ConfigEntry | from homeassistant.config_entries import ConfigEntry | ||||||
| from homeassistant.const import CONF_ADDRESS | from homeassistant.const import CONF_ADDRESS | ||||||
| from homeassistant.core import HomeAssistant | from homeassistant.core import HomeAssistant | ||||||
|  | from homeassistant.helpers.debounce import Debouncer | ||||||
| from homeassistant.helpers.update_coordinator import DataUpdateCoordinator | from homeassistant.helpers.update_coordinator import DataUpdateCoordinator | ||||||
|  |  | ||||||
| from .const import CONF_IS_NEW_STYLE_SCALE | from .const import CONF_IS_NEW_STYLE_SCALE | ||||||
|  |  | ||||||
| SCAN_INTERVAL = timedelta(seconds=15) | SCAN_INTERVAL = timedelta(seconds=15) | ||||||
|  | UPDATE_DEBOUNCE_TIME = 0.2 | ||||||
|  |  | ||||||
| _LOGGER = logging.getLogger(__name__) | _LOGGER = logging.getLogger(__name__) | ||||||
|  |  | ||||||
| @@ -37,11 +40,20 @@ class AcaiaCoordinator(DataUpdateCoordinator[None]): | |||||||
|             config_entry=entry, |             config_entry=entry, | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |         debouncer = Debouncer( | ||||||
|  |             hass=hass, | ||||||
|  |             logger=_LOGGER, | ||||||
|  |             cooldown=UPDATE_DEBOUNCE_TIME, | ||||||
|  |             immediate=True, | ||||||
|  |             function=self.async_update_listeners, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|         self._scale = AcaiaScale( |         self._scale = AcaiaScale( | ||||||
|             address_or_ble_device=entry.data[CONF_ADDRESS], |             address_or_ble_device=entry.data[CONF_ADDRESS], | ||||||
|             name=entry.title, |             name=entry.title, | ||||||
|             is_new_style_scale=entry.data[CONF_IS_NEW_STYLE_SCALE], |             is_new_style_scale=entry.data[CONF_IS_NEW_STYLE_SCALE], | ||||||
|             notify_callback=self.async_update_listeners, |             notify_callback=debouncer.async_schedule_call, | ||||||
|  |             scanner=async_get_scanner(hass), | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|   | |||||||
| @@ -26,5 +26,5 @@ | |||||||
|   "iot_class": "local_push", |   "iot_class": "local_push", | ||||||
|   "loggers": ["aioacaia"], |   "loggers": ["aioacaia"], | ||||||
|   "quality_scale": "platinum", |   "quality_scale": "platinum", | ||||||
|   "requirements": ["aioacaia==0.1.14"] |   "requirements": ["aioacaia==0.1.17"] | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,21 +2,23 @@ | |||||||
|  |  | ||||||
| from __future__ import annotations | from __future__ import annotations | ||||||
|  |  | ||||||
|  | import asyncio | ||||||
| import logging | import logging | ||||||
|  |  | ||||||
| from accuweather import AccuWeather | from accuweather import AccuWeather | ||||||
|  |  | ||||||
| from homeassistant.components.sensor import DOMAIN as SENSOR_PLATFORM | from homeassistant.components.sensor import DOMAIN as SENSOR_PLATFORM | ||||||
| from homeassistant.const import CONF_API_KEY, CONF_NAME, Platform | from homeassistant.const import CONF_API_KEY, Platform | ||||||
| from homeassistant.core import HomeAssistant | from homeassistant.core import HomeAssistant | ||||||
| from homeassistant.helpers import entity_registry as er | from homeassistant.helpers import entity_registry as er | ||||||
| from homeassistant.helpers.aiohttp_client import async_get_clientsession | from homeassistant.helpers.aiohttp_client import async_get_clientsession | ||||||
|  |  | ||||||
| from .const import DOMAIN, UPDATE_INTERVAL_DAILY_FORECAST, UPDATE_INTERVAL_OBSERVATION | from .const import DOMAIN | ||||||
| from .coordinator import ( | from .coordinator import ( | ||||||
|     AccuWeatherConfigEntry, |     AccuWeatherConfigEntry, | ||||||
|     AccuWeatherDailyForecastDataUpdateCoordinator, |     AccuWeatherDailyForecastDataUpdateCoordinator, | ||||||
|     AccuWeatherData, |     AccuWeatherData, | ||||||
|  |     AccuWeatherHourlyForecastDataUpdateCoordinator, | ||||||
|     AccuWeatherObservationDataUpdateCoordinator, |     AccuWeatherObservationDataUpdateCoordinator, | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -28,7 +30,6 @@ PLATFORMS = [Platform.SENSOR, Platform.WEATHER] | |||||||
| async def async_setup_entry(hass: HomeAssistant, entry: AccuWeatherConfigEntry) -> bool: | async def async_setup_entry(hass: HomeAssistant, entry: AccuWeatherConfigEntry) -> bool: | ||||||
|     """Set up AccuWeather as config entry.""" |     """Set up AccuWeather as config entry.""" | ||||||
|     api_key: str = entry.data[CONF_API_KEY] |     api_key: str = entry.data[CONF_API_KEY] | ||||||
|     name: str = entry.data[CONF_NAME] |  | ||||||
|  |  | ||||||
|     location_key = entry.unique_id |     location_key = entry.unique_id | ||||||
|  |  | ||||||
| @@ -41,26 +42,28 @@ async def async_setup_entry(hass: HomeAssistant, entry: AccuWeatherConfigEntry) | |||||||
|         hass, |         hass, | ||||||
|         entry, |         entry, | ||||||
|         accuweather, |         accuweather, | ||||||
|         name, |  | ||||||
|         "observation", |  | ||||||
|         UPDATE_INTERVAL_OBSERVATION, |  | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     coordinator_daily_forecast = AccuWeatherDailyForecastDataUpdateCoordinator( |     coordinator_daily_forecast = AccuWeatherDailyForecastDataUpdateCoordinator( | ||||||
|         hass, |         hass, | ||||||
|         entry, |         entry, | ||||||
|         accuweather, |         accuweather, | ||||||
|         name, |     ) | ||||||
|         "daily forecast", |     coordinator_hourly_forecast = AccuWeatherHourlyForecastDataUpdateCoordinator( | ||||||
|         UPDATE_INTERVAL_DAILY_FORECAST, |         hass, | ||||||
|  |         entry, | ||||||
|  |         accuweather, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     await coordinator_observation.async_config_entry_first_refresh() |     await asyncio.gather( | ||||||
|     await coordinator_daily_forecast.async_config_entry_first_refresh() |         coordinator_observation.async_config_entry_first_refresh(), | ||||||
|  |         coordinator_daily_forecast.async_config_entry_first_refresh(), | ||||||
|  |         coordinator_hourly_forecast.async_config_entry_first_refresh(), | ||||||
|  |     ) | ||||||
|  |  | ||||||
|     entry.runtime_data = AccuWeatherData( |     entry.runtime_data = AccuWeatherData( | ||||||
|         coordinator_observation=coordinator_observation, |         coordinator_observation=coordinator_observation, | ||||||
|         coordinator_daily_forecast=coordinator_daily_forecast, |         coordinator_daily_forecast=coordinator_daily_forecast, | ||||||
|  |         coordinator_hourly_forecast=coordinator_hourly_forecast, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS) |     await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS) | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ | |||||||
| from __future__ import annotations | from __future__ import annotations | ||||||
|  |  | ||||||
| from asyncio import timeout | from asyncio import timeout | ||||||
|  | from collections.abc import Mapping | ||||||
| from typing import Any | from typing import Any | ||||||
|  |  | ||||||
| from accuweather import AccuWeather, ApiError, InvalidApiKeyError, RequestsExceededError | from accuweather import AccuWeather, ApiError, InvalidApiKeyError, RequestsExceededError | ||||||
| @@ -22,6 +23,8 @@ class AccuWeatherFlowHandler(ConfigFlow, domain=DOMAIN): | |||||||
|     """Config flow for AccuWeather.""" |     """Config flow for AccuWeather.""" | ||||||
|  |  | ||||||
|     VERSION = 1 |     VERSION = 1 | ||||||
|  |     _latitude: float | None = None | ||||||
|  |     _longitude: float | None = None | ||||||
|  |  | ||||||
|     async def async_step_user( |     async def async_step_user( | ||||||
|         self, user_input: dict[str, Any] | None = None |         self, user_input: dict[str, Any] | None = None | ||||||
| @@ -74,3 +77,46 @@ class AccuWeatherFlowHandler(ConfigFlow, domain=DOMAIN): | |||||||
|             ), |             ), | ||||||
|             errors=errors, |             errors=errors, | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |     async def async_step_reauth( | ||||||
|  |         self, entry_data: Mapping[str, Any] | ||||||
|  |     ) -> ConfigFlowResult: | ||||||
|  |         """Handle configuration by re-auth.""" | ||||||
|  |         self._latitude = entry_data[CONF_LATITUDE] | ||||||
|  |         self._longitude = entry_data[CONF_LONGITUDE] | ||||||
|  |  | ||||||
|  |         return await self.async_step_reauth_confirm() | ||||||
|  |  | ||||||
|  |     async def async_step_reauth_confirm( | ||||||
|  |         self, user_input: dict[str, Any] | None = None | ||||||
|  |     ) -> ConfigFlowResult: | ||||||
|  |         """Dialog that informs the user that reauth is required.""" | ||||||
|  |         errors: dict[str, str] = {} | ||||||
|  |  | ||||||
|  |         if user_input is not None: | ||||||
|  |             websession = async_get_clientsession(self.hass) | ||||||
|  |             try: | ||||||
|  |                 async with timeout(10): | ||||||
|  |                     accuweather = AccuWeather( | ||||||
|  |                         user_input[CONF_API_KEY], | ||||||
|  |                         websession, | ||||||
|  |                         latitude=self._latitude, | ||||||
|  |                         longitude=self._longitude, | ||||||
|  |                     ) | ||||||
|  |                     await accuweather.async_get_location() | ||||||
|  |             except (ApiError, ClientConnectorError, TimeoutError, ClientError): | ||||||
|  |                 errors["base"] = "cannot_connect" | ||||||
|  |             except InvalidApiKeyError: | ||||||
|  |                 errors["base"] = "invalid_api_key" | ||||||
|  |             except RequestsExceededError: | ||||||
|  |                 errors["base"] = "requests_exceeded" | ||||||
|  |             else: | ||||||
|  |                 return self.async_update_reload_and_abort( | ||||||
|  |                     self._get_reauth_entry(), data_updates=user_input | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |         return self.async_show_form( | ||||||
|  |             step_id="reauth_confirm", | ||||||
|  |             data_schema=vol.Schema({vol.Required(CONF_API_KEY): str}), | ||||||
|  |             errors=errors, | ||||||
|  |         ) | ||||||
|   | |||||||
| @@ -71,3 +71,4 @@ POLLEN_CATEGORY_MAP = { | |||||||
| } | } | ||||||
| UPDATE_INTERVAL_OBSERVATION = timedelta(minutes=10) | UPDATE_INTERVAL_OBSERVATION = timedelta(minutes=10) | ||||||
| UPDATE_INTERVAL_DAILY_FORECAST = timedelta(hours=6) | UPDATE_INTERVAL_DAILY_FORECAST = timedelta(hours=6) | ||||||
|  | UPDATE_INTERVAL_HOURLY_FORECAST = timedelta(minutes=30) | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ | |||||||
| from __future__ import annotations | from __future__ import annotations | ||||||
|  |  | ||||||
| from asyncio import timeout | from asyncio import timeout | ||||||
|  | from collections.abc import Awaitable, Callable | ||||||
| from dataclasses import dataclass | from dataclasses import dataclass | ||||||
| from datetime import timedelta | from datetime import timedelta | ||||||
| import logging | import logging | ||||||
| @@ -12,7 +13,9 @@ from accuweather import AccuWeather, ApiError, InvalidApiKeyError, RequestsExcee | |||||||
| from aiohttp.client_exceptions import ClientConnectorError | from aiohttp.client_exceptions import ClientConnectorError | ||||||
|  |  | ||||||
| from homeassistant.config_entries import ConfigEntry | from homeassistant.config_entries import ConfigEntry | ||||||
|  | from homeassistant.const import CONF_NAME | ||||||
| from homeassistant.core import HomeAssistant | from homeassistant.core import HomeAssistant | ||||||
|  | from homeassistant.exceptions import ConfigEntryAuthFailed | ||||||
| from homeassistant.helpers.device_registry import DeviceEntryType, DeviceInfo | from homeassistant.helpers.device_registry import DeviceEntryType, DeviceInfo | ||||||
| from homeassistant.helpers.update_coordinator import ( | from homeassistant.helpers.update_coordinator import ( | ||||||
|     DataUpdateCoordinator, |     DataUpdateCoordinator, | ||||||
| @@ -20,9 +23,15 @@ from homeassistant.helpers.update_coordinator import ( | |||||||
|     UpdateFailed, |     UpdateFailed, | ||||||
| ) | ) | ||||||
|  |  | ||||||
| from .const import DOMAIN, MANUFACTURER | from .const import ( | ||||||
|  |     DOMAIN, | ||||||
|  |     MANUFACTURER, | ||||||
|  |     UPDATE_INTERVAL_DAILY_FORECAST, | ||||||
|  |     UPDATE_INTERVAL_HOURLY_FORECAST, | ||||||
|  |     UPDATE_INTERVAL_OBSERVATION, | ||||||
|  | ) | ||||||
|  |  | ||||||
| EXCEPTIONS = (ApiError, ClientConnectorError, InvalidApiKeyError, RequestsExceededError) | EXCEPTIONS = (ApiError, ClientConnectorError, RequestsExceededError) | ||||||
|  |  | ||||||
| _LOGGER = logging.getLogger(__name__) | _LOGGER = logging.getLogger(__name__) | ||||||
|  |  | ||||||
| @@ -33,6 +42,7 @@ class AccuWeatherData: | |||||||
|  |  | ||||||
|     coordinator_observation: AccuWeatherObservationDataUpdateCoordinator |     coordinator_observation: AccuWeatherObservationDataUpdateCoordinator | ||||||
|     coordinator_daily_forecast: AccuWeatherDailyForecastDataUpdateCoordinator |     coordinator_daily_forecast: AccuWeatherDailyForecastDataUpdateCoordinator | ||||||
|  |     coordinator_hourly_forecast: AccuWeatherHourlyForecastDataUpdateCoordinator | ||||||
|  |  | ||||||
|  |  | ||||||
| type AccuWeatherConfigEntry = ConfigEntry[AccuWeatherData] | type AccuWeatherConfigEntry = ConfigEntry[AccuWeatherData] | ||||||
| @@ -43,18 +53,18 @@ class AccuWeatherObservationDataUpdateCoordinator( | |||||||
| ): | ): | ||||||
|     """Class to manage fetching AccuWeather data API.""" |     """Class to manage fetching AccuWeather data API.""" | ||||||
|  |  | ||||||
|  |     config_entry: AccuWeatherConfigEntry | ||||||
|  |  | ||||||
|     def __init__( |     def __init__( | ||||||
|         self, |         self, | ||||||
|         hass: HomeAssistant, |         hass: HomeAssistant, | ||||||
|         config_entry: AccuWeatherConfigEntry, |         config_entry: AccuWeatherConfigEntry, | ||||||
|         accuweather: AccuWeather, |         accuweather: AccuWeather, | ||||||
|         name: str, |  | ||||||
|         coordinator_type: str, |  | ||||||
|         update_interval: timedelta, |  | ||||||
|     ) -> None: |     ) -> None: | ||||||
|         """Initialize.""" |         """Initialize.""" | ||||||
|         self.accuweather = accuweather |         self.accuweather = accuweather | ||||||
|         self.location_key = accuweather.location_key |         self.location_key = accuweather.location_key | ||||||
|  |         name = config_entry.data[CONF_NAME] | ||||||
|  |  | ||||||
|         if TYPE_CHECKING: |         if TYPE_CHECKING: | ||||||
|             assert self.location_key is not None |             assert self.location_key is not None | ||||||
| @@ -65,8 +75,8 @@ class AccuWeatherObservationDataUpdateCoordinator( | |||||||
|             hass, |             hass, | ||||||
|             _LOGGER, |             _LOGGER, | ||||||
|             config_entry=config_entry, |             config_entry=config_entry, | ||||||
|             name=f"{name} ({coordinator_type})", |             name=f"{name} (observation)", | ||||||
|             update_interval=update_interval, |             update_interval=UPDATE_INTERVAL_OBSERVATION, | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     async def _async_update_data(self) -> dict[str, Any]: |     async def _async_update_data(self) -> dict[str, Any]: | ||||||
| @@ -80,29 +90,39 @@ class AccuWeatherObservationDataUpdateCoordinator( | |||||||
|                 translation_key="current_conditions_update_error", |                 translation_key="current_conditions_update_error", | ||||||
|                 translation_placeholders={"error": repr(error)}, |                 translation_placeholders={"error": repr(error)}, | ||||||
|             ) from error |             ) from error | ||||||
|  |         except InvalidApiKeyError as err: | ||||||
|  |             raise ConfigEntryAuthFailed( | ||||||
|  |                 translation_domain=DOMAIN, | ||||||
|  |                 translation_key="auth_error", | ||||||
|  |                 translation_placeholders={"entry": self.config_entry.title}, | ||||||
|  |             ) from err | ||||||
|  |  | ||||||
|         _LOGGER.debug("Requests remaining: %d", self.accuweather.requests_remaining) |         _LOGGER.debug("Requests remaining: %d", self.accuweather.requests_remaining) | ||||||
|  |  | ||||||
|         return result |         return result | ||||||
|  |  | ||||||
|  |  | ||||||
| class AccuWeatherDailyForecastDataUpdateCoordinator( | class AccuWeatherForecastDataUpdateCoordinator( | ||||||
|     TimestampDataUpdateCoordinator[list[dict[str, Any]]] |     TimestampDataUpdateCoordinator[list[dict[str, Any]]] | ||||||
| ): | ): | ||||||
|     """Class to manage fetching AccuWeather data API.""" |     """Base class for AccuWeather forecast.""" | ||||||
|  |  | ||||||
|  |     config_entry: AccuWeatherConfigEntry | ||||||
|  |  | ||||||
|     def __init__( |     def __init__( | ||||||
|         self, |         self, | ||||||
|         hass: HomeAssistant, |         hass: HomeAssistant, | ||||||
|         config_entry: AccuWeatherConfigEntry, |         config_entry: AccuWeatherConfigEntry, | ||||||
|         accuweather: AccuWeather, |         accuweather: AccuWeather, | ||||||
|         name: str, |  | ||||||
|         coordinator_type: str, |         coordinator_type: str, | ||||||
|         update_interval: timedelta, |         update_interval: timedelta, | ||||||
|  |         fetch_method: Callable[..., Awaitable[list[dict[str, Any]]]], | ||||||
|     ) -> None: |     ) -> None: | ||||||
|         """Initialize.""" |         """Initialize.""" | ||||||
|         self.accuweather = accuweather |         self.accuweather = accuweather | ||||||
|         self.location_key = accuweather.location_key |         self.location_key = accuweather.location_key | ||||||
|  |         self._fetch_method = fetch_method | ||||||
|  |         name = config_entry.data[CONF_NAME] | ||||||
|  |  | ||||||
|         if TYPE_CHECKING: |         if TYPE_CHECKING: | ||||||
|             assert self.location_key is not None |             assert self.location_key is not None | ||||||
| @@ -118,24 +138,71 @@ class AccuWeatherDailyForecastDataUpdateCoordinator( | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     async def _async_update_data(self) -> list[dict[str, Any]]: |     async def _async_update_data(self) -> list[dict[str, Any]]: | ||||||
|         """Update data via library.""" |         """Update forecast data via library.""" | ||||||
|         try: |         try: | ||||||
|             async with timeout(10): |             async with timeout(10): | ||||||
|                 result = await self.accuweather.async_get_daily_forecast( |                 result = await self._fetch_method(language=self.hass.config.language) | ||||||
|                     language=self.hass.config.language |  | ||||||
|                 ) |  | ||||||
|         except EXCEPTIONS as error: |         except EXCEPTIONS as error: | ||||||
|             raise UpdateFailed( |             raise UpdateFailed( | ||||||
|                 translation_domain=DOMAIN, |                 translation_domain=DOMAIN, | ||||||
|                 translation_key="forecast_update_error", |                 translation_key="forecast_update_error", | ||||||
|                 translation_placeholders={"error": repr(error)}, |                 translation_placeholders={"error": repr(error)}, | ||||||
|             ) from error |             ) from error | ||||||
|  |         except InvalidApiKeyError as err: | ||||||
|  |             raise ConfigEntryAuthFailed( | ||||||
|  |                 translation_domain=DOMAIN, | ||||||
|  |                 translation_key="auth_error", | ||||||
|  |                 translation_placeholders={"entry": self.config_entry.title}, | ||||||
|  |             ) from err | ||||||
|  |  | ||||||
|         _LOGGER.debug("Requests remaining: %d", self.accuweather.requests_remaining) |         _LOGGER.debug("Requests remaining: %d", self.accuweather.requests_remaining) | ||||||
|  |  | ||||||
|         return result |         return result | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AccuWeatherDailyForecastDataUpdateCoordinator( | ||||||
|  |     AccuWeatherForecastDataUpdateCoordinator | ||||||
|  | ): | ||||||
|  |     """Coordinator for daily forecast.""" | ||||||
|  |  | ||||||
|  |     def __init__( | ||||||
|  |         self, | ||||||
|  |         hass: HomeAssistant, | ||||||
|  |         config_entry: AccuWeatherConfigEntry, | ||||||
|  |         accuweather: AccuWeather, | ||||||
|  |     ) -> None: | ||||||
|  |         """Initialize.""" | ||||||
|  |         super().__init__( | ||||||
|  |             hass, | ||||||
|  |             config_entry, | ||||||
|  |             accuweather, | ||||||
|  |             "daily forecast", | ||||||
|  |             UPDATE_INTERVAL_DAILY_FORECAST, | ||||||
|  |             fetch_method=accuweather.async_get_daily_forecast, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AccuWeatherHourlyForecastDataUpdateCoordinator( | ||||||
|  |     AccuWeatherForecastDataUpdateCoordinator | ||||||
|  | ): | ||||||
|  |     """Coordinator for hourly forecast.""" | ||||||
|  |  | ||||||
|  |     def __init__( | ||||||
|  |         self, | ||||||
|  |         hass: HomeAssistant, | ||||||
|  |         config_entry: AccuWeatherConfigEntry, | ||||||
|  |         accuweather: AccuWeather, | ||||||
|  |     ) -> None: | ||||||
|  |         """Initialize.""" | ||||||
|  |         super().__init__( | ||||||
|  |             hass, | ||||||
|  |             config_entry, | ||||||
|  |             accuweather, | ||||||
|  |             "hourly forecast", | ||||||
|  |             UPDATE_INTERVAL_HOURLY_FORECAST, | ||||||
|  |             fetch_method=accuweather.async_get_hourly_forecast, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
| def _get_device_info(location_key: str, name: str) -> DeviceInfo: | def _get_device_info(location_key: str, name: str) -> DeviceInfo: | ||||||
|     """Get device info.""" |     """Get device info.""" | ||||||
|     return DeviceInfo( |     return DeviceInfo( | ||||||
|   | |||||||
| @@ -1,6 +1,9 @@ | |||||||
| { | { | ||||||
|   "entity": { |   "entity": { | ||||||
|     "sensor": { |     "sensor": { | ||||||
|  |       "air_quality": { | ||||||
|  |         "default": "mdi:air-filter" | ||||||
|  |       }, | ||||||
|       "cloud_ceiling": { |       "cloud_ceiling": { | ||||||
|         "default": "mdi:weather-fog" |         "default": "mdi:weather-fog" | ||||||
|       }, |       }, | ||||||
| @@ -34,9 +37,6 @@ | |||||||
|       "thunderstorm_probability_night": { |       "thunderstorm_probability_night": { | ||||||
|         "default": "mdi:weather-lightning" |         "default": "mdi:weather-lightning" | ||||||
|       }, |       }, | ||||||
|       "translation_key": { |  | ||||||
|         "default": "mdi:air-filter" |  | ||||||
|       }, |  | ||||||
|       "tree_pollen": { |       "tree_pollen": { | ||||||
|         "default": "mdi:tree-outline" |         "default": "mdi:tree-outline" | ||||||
|       }, |       }, | ||||||
|   | |||||||
| @@ -7,5 +7,5 @@ | |||||||
|   "integration_type": "service", |   "integration_type": "service", | ||||||
|   "iot_class": "cloud_polling", |   "iot_class": "cloud_polling", | ||||||
|   "loggers": ["accuweather"], |   "loggers": ["accuweather"], | ||||||
|   "requirements": ["accuweather==4.2.1"] |   "requirements": ["accuweather==4.2.2"] | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,6 +7,17 @@ | |||||||
|           "api_key": "[%key:common::config_flow::data::api_key%]", |           "api_key": "[%key:common::config_flow::data::api_key%]", | ||||||
|           "latitude": "[%key:common::config_flow::data::latitude%]", |           "latitude": "[%key:common::config_flow::data::latitude%]", | ||||||
|           "longitude": "[%key:common::config_flow::data::longitude%]" |           "longitude": "[%key:common::config_flow::data::longitude%]" | ||||||
|  |         }, | ||||||
|  |         "data_description": { | ||||||
|  |           "api_key": "API key generated in the AccuWeather APIs portal." | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "reauth_confirm": { | ||||||
|  |         "data": { | ||||||
|  |           "api_key": "[%key:common::config_flow::data::api_key%]" | ||||||
|  |         }, | ||||||
|  |         "data_description": { | ||||||
|  |           "api_key": "[%key:component::accuweather::config::step::user::data_description::api_key%]" | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
| @@ -19,7 +30,8 @@ | |||||||
|       "requests_exceeded": "The allowed number of requests to the AccuWeather API has been exceeded. You have to wait or change the API key." |       "requests_exceeded": "The allowed number of requests to the AccuWeather API has been exceeded. You have to wait or change the API key." | ||||||
|     }, |     }, | ||||||
|     "abort": { |     "abort": { | ||||||
|       "already_configured": "[%key:common::config_flow::abort::already_configured_location%]" |       "already_configured": "[%key:common::config_flow::abort::already_configured_location%]", | ||||||
|  |       "reauth_successful": "[%key:common::config_flow::abort::reauth_successful%]" | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   "entity": { |   "entity": { | ||||||
| @@ -239,6 +251,9 @@ | |||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   "exceptions": { |   "exceptions": { | ||||||
|  |     "auth_error": { | ||||||
|  |       "message": "Authentication failed for {entry}, please update your API key" | ||||||
|  |     }, | ||||||
|     "current_conditions_update_error": { |     "current_conditions_update_error": { | ||||||
|       "message": "An error occurred while retrieving weather current conditions data from the AccuWeather API: {error}" |       "message": "An error occurred while retrieving weather current conditions data from the AccuWeather API: {error}" | ||||||
|     }, |     }, | ||||||
|   | |||||||
| @@ -45,6 +45,7 @@ from .coordinator import ( | |||||||
|     AccuWeatherConfigEntry, |     AccuWeatherConfigEntry, | ||||||
|     AccuWeatherDailyForecastDataUpdateCoordinator, |     AccuWeatherDailyForecastDataUpdateCoordinator, | ||||||
|     AccuWeatherData, |     AccuWeatherData, | ||||||
|  |     AccuWeatherHourlyForecastDataUpdateCoordinator, | ||||||
|     AccuWeatherObservationDataUpdateCoordinator, |     AccuWeatherObservationDataUpdateCoordinator, | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -64,6 +65,7 @@ class AccuWeatherEntity( | |||||||
|     CoordinatorWeatherEntity[ |     CoordinatorWeatherEntity[ | ||||||
|         AccuWeatherObservationDataUpdateCoordinator, |         AccuWeatherObservationDataUpdateCoordinator, | ||||||
|         AccuWeatherDailyForecastDataUpdateCoordinator, |         AccuWeatherDailyForecastDataUpdateCoordinator, | ||||||
|  |         AccuWeatherHourlyForecastDataUpdateCoordinator, | ||||||
|     ] |     ] | ||||||
| ): | ): | ||||||
|     """Define an AccuWeather entity.""" |     """Define an AccuWeather entity.""" | ||||||
| @@ -76,6 +78,7 @@ class AccuWeatherEntity( | |||||||
|         super().__init__( |         super().__init__( | ||||||
|             observation_coordinator=accuweather_data.coordinator_observation, |             observation_coordinator=accuweather_data.coordinator_observation, | ||||||
|             daily_coordinator=accuweather_data.coordinator_daily_forecast, |             daily_coordinator=accuweather_data.coordinator_daily_forecast, | ||||||
|  |             hourly_coordinator=accuweather_data.coordinator_hourly_forecast, | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         self._attr_native_precipitation_unit = UnitOfPrecipitationDepth.MILLIMETERS |         self._attr_native_precipitation_unit = UnitOfPrecipitationDepth.MILLIMETERS | ||||||
| @@ -86,10 +89,13 @@ class AccuWeatherEntity( | |||||||
|         self._attr_unique_id = accuweather_data.coordinator_observation.location_key |         self._attr_unique_id = accuweather_data.coordinator_observation.location_key | ||||||
|         self._attr_attribution = ATTRIBUTION |         self._attr_attribution = ATTRIBUTION | ||||||
|         self._attr_device_info = accuweather_data.coordinator_observation.device_info |         self._attr_device_info = accuweather_data.coordinator_observation.device_info | ||||||
|         self._attr_supported_features = WeatherEntityFeature.FORECAST_DAILY |         self._attr_supported_features = ( | ||||||
|  |             WeatherEntityFeature.FORECAST_DAILY | WeatherEntityFeature.FORECAST_HOURLY | ||||||
|  |         ) | ||||||
|  |  | ||||||
|         self.observation_coordinator = accuweather_data.coordinator_observation |         self.observation_coordinator = accuweather_data.coordinator_observation | ||||||
|         self.daily_coordinator = accuweather_data.coordinator_daily_forecast |         self.daily_coordinator = accuweather_data.coordinator_daily_forecast | ||||||
|  |         self.hourly_coordinator = accuweather_data.coordinator_hourly_forecast | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def condition(self) -> str | None: |     def condition(self) -> str | None: | ||||||
| @@ -207,3 +213,32 @@ class AccuWeatherEntity( | |||||||
|             } |             } | ||||||
|             for item in self.daily_coordinator.data |             for item in self.daily_coordinator.data | ||||||
|         ] |         ] | ||||||
|  |  | ||||||
|  |     @callback | ||||||
|  |     def _async_forecast_hourly(self) -> list[Forecast] | None: | ||||||
|  |         """Return the hourly forecast in native units.""" | ||||||
|  |         return [ | ||||||
|  |             { | ||||||
|  |                 ATTR_FORECAST_TIME: utc_from_timestamp( | ||||||
|  |                     item["EpochDateTime"] | ||||||
|  |                 ).isoformat(), | ||||||
|  |                 ATTR_FORECAST_CLOUD_COVERAGE: item["CloudCover"], | ||||||
|  |                 ATTR_FORECAST_HUMIDITY: item["RelativeHumidity"], | ||||||
|  |                 ATTR_FORECAST_NATIVE_TEMP: item["Temperature"][ATTR_VALUE], | ||||||
|  |                 ATTR_FORECAST_NATIVE_APPARENT_TEMP: item["RealFeelTemperature"][ | ||||||
|  |                     ATTR_VALUE | ||||||
|  |                 ], | ||||||
|  |                 ATTR_FORECAST_NATIVE_PRECIPITATION: item["TotalLiquid"][ATTR_VALUE], | ||||||
|  |                 ATTR_FORECAST_PRECIPITATION_PROBABILITY: item[ | ||||||
|  |                     "PrecipitationProbability" | ||||||
|  |                 ], | ||||||
|  |                 ATTR_FORECAST_NATIVE_WIND_SPEED: item["Wind"][ATTR_SPEED][ATTR_VALUE], | ||||||
|  |                 ATTR_FORECAST_NATIVE_WIND_GUST_SPEED: item["WindGust"][ATTR_SPEED][ | ||||||
|  |                     ATTR_VALUE | ||||||
|  |                 ], | ||||||
|  |                 ATTR_FORECAST_UV_INDEX: item["UVIndex"], | ||||||
|  |                 ATTR_FORECAST_WIND_BEARING: item["Wind"][ATTR_DIRECTION]["Degrees"], | ||||||
|  |                 ATTR_FORECAST_CONDITION: CONDITION_MAP.get(item["WeatherIcon"]), | ||||||
|  |             } | ||||||
|  |             for item in self.hourly_coordinator.data | ||||||
|  |         ] | ||||||
|   | |||||||
| @@ -3,10 +3,8 @@ | |||||||
| import logging | import logging | ||||||
| from typing import Any | from typing import Any | ||||||
|  |  | ||||||
| from aiohttp import web |  | ||||||
| import voluptuous as vol | import voluptuous as vol | ||||||
|  |  | ||||||
| from homeassistant.components.http import KEY_HASS, HomeAssistantView |  | ||||||
| from homeassistant.config_entries import ConfigEntry | from homeassistant.config_entries import ConfigEntry | ||||||
| from homeassistant.const import ATTR_ENTITY_ID, CONF_DESCRIPTION, CONF_SELECTOR | from homeassistant.const import ATTR_ENTITY_ID, CONF_DESCRIPTION, CONF_SELECTOR | ||||||
| from homeassistant.core import ( | from homeassistant.core import ( | ||||||
| @@ -28,7 +26,6 @@ from .const import ( | |||||||
|     ATTR_STRUCTURE, |     ATTR_STRUCTURE, | ||||||
|     ATTR_TASK_NAME, |     ATTR_TASK_NAME, | ||||||
|     DATA_COMPONENT, |     DATA_COMPONENT, | ||||||
|     DATA_IMAGES, |  | ||||||
|     DATA_PREFERENCES, |     DATA_PREFERENCES, | ||||||
|     DOMAIN, |     DOMAIN, | ||||||
|     SERVICE_GENERATE_DATA, |     SERVICE_GENERATE_DATA, | ||||||
| @@ -42,7 +39,6 @@ from .task import ( | |||||||
|     GenDataTaskResult, |     GenDataTaskResult, | ||||||
|     GenImageTask, |     GenImageTask, | ||||||
|     GenImageTaskResult, |     GenImageTaskResult, | ||||||
|     ImageData, |  | ||||||
|     async_generate_data, |     async_generate_data, | ||||||
|     async_generate_image, |     async_generate_image, | ||||||
| ) | ) | ||||||
| @@ -55,7 +51,6 @@ __all__ = [ | |||||||
|     "GenDataTaskResult", |     "GenDataTaskResult", | ||||||
|     "GenImageTask", |     "GenImageTask", | ||||||
|     "GenImageTaskResult", |     "GenImageTaskResult", | ||||||
|     "ImageData", |  | ||||||
|     "async_generate_data", |     "async_generate_data", | ||||||
|     "async_generate_image", |     "async_generate_image", | ||||||
|     "async_setup", |     "async_setup", | ||||||
| @@ -94,10 +89,8 @@ async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool: | |||||||
|     entity_component = EntityComponent[AITaskEntity](_LOGGER, DOMAIN, hass) |     entity_component = EntityComponent[AITaskEntity](_LOGGER, DOMAIN, hass) | ||||||
|     hass.data[DATA_COMPONENT] = entity_component |     hass.data[DATA_COMPONENT] = entity_component | ||||||
|     hass.data[DATA_PREFERENCES] = AITaskPreferences(hass) |     hass.data[DATA_PREFERENCES] = AITaskPreferences(hass) | ||||||
|     hass.data[DATA_IMAGES] = {} |  | ||||||
|     await hass.data[DATA_PREFERENCES].async_load() |     await hass.data[DATA_PREFERENCES].async_load() | ||||||
|     async_setup_http(hass) |     async_setup_http(hass) | ||||||
|     hass.http.register_view(ImageView) |  | ||||||
|     hass.services.async_register( |     hass.services.async_register( | ||||||
|         DOMAIN, |         DOMAIN, | ||||||
|         SERVICE_GENERATE_DATA, |         SERVICE_GENERATE_DATA, | ||||||
| @@ -209,28 +202,3 @@ class AITaskPreferences: | |||||||
|     def as_dict(self) -> dict[str, str | None]: |     def as_dict(self) -> dict[str, str | None]: | ||||||
|         """Get the current preferences.""" |         """Get the current preferences.""" | ||||||
|         return {key: getattr(self, key) for key in self.KEYS} |         return {key: getattr(self, key) for key in self.KEYS} | ||||||
|  |  | ||||||
|  |  | ||||||
| class ImageView(HomeAssistantView): |  | ||||||
|     """View to generated images.""" |  | ||||||
|  |  | ||||||
|     url = f"/api/{DOMAIN}/images/{{filename}}" |  | ||||||
|     name = f"api:{DOMAIN}/images" |  | ||||||
|  |  | ||||||
|     async def get( |  | ||||||
|         self, |  | ||||||
|         request: web.Request, |  | ||||||
|         filename: str, |  | ||||||
|     ) -> web.Response: |  | ||||||
|         """Serve image.""" |  | ||||||
|         hass = request.app[KEY_HASS] |  | ||||||
|         image_storage = hass.data[DATA_IMAGES] |  | ||||||
|         image_data = image_storage.get(filename) |  | ||||||
|  |  | ||||||
|         if image_data is None: |  | ||||||
|             raise web.HTTPNotFound |  | ||||||
|  |  | ||||||
|         return web.Response( |  | ||||||
|             body=image_data.data, |  | ||||||
|             content_type=image_data.mime_type, |  | ||||||
|         ) |  | ||||||
|   | |||||||
| @@ -8,19 +8,19 @@ from typing import TYPE_CHECKING, Final | |||||||
| from homeassistant.util.hass_dict import HassKey | from homeassistant.util.hass_dict import HassKey | ||||||
|  |  | ||||||
| if TYPE_CHECKING: | if TYPE_CHECKING: | ||||||
|  |     from homeassistant.components.media_source import local_source | ||||||
|     from homeassistant.helpers.entity_component import EntityComponent |     from homeassistant.helpers.entity_component import EntityComponent | ||||||
|  |  | ||||||
|     from . import AITaskPreferences |     from . import AITaskPreferences | ||||||
|     from .entity import AITaskEntity |     from .entity import AITaskEntity | ||||||
|     from .task import ImageData |  | ||||||
|  |  | ||||||
| DOMAIN = "ai_task" | DOMAIN = "ai_task" | ||||||
| DATA_COMPONENT: HassKey[EntityComponent[AITaskEntity]] = HassKey(DOMAIN) | DATA_COMPONENT: HassKey[EntityComponent[AITaskEntity]] = HassKey(DOMAIN) | ||||||
| DATA_PREFERENCES: HassKey[AITaskPreferences] = HassKey(f"{DOMAIN}_preferences") | DATA_PREFERENCES: HassKey[AITaskPreferences] = HassKey(f"{DOMAIN}_preferences") | ||||||
| DATA_IMAGES: HassKey[dict[str, ImageData]] = HassKey(f"{DOMAIN}_images") | DATA_MEDIA_SOURCE: HassKey[local_source.LocalSource] = HassKey(f"{DOMAIN}_media_source") | ||||||
|  |  | ||||||
|  | IMAGE_DIR: Final = "image" | ||||||
| IMAGE_EXPIRY_TIME = 60 * 60  # 1 hour | IMAGE_EXPIRY_TIME = 60 * 60  # 1 hour | ||||||
| MAX_IMAGES = 20 |  | ||||||
|  |  | ||||||
| SERVICE_GENERATE_DATA = "generate_data" | SERVICE_GENERATE_DATA = "generate_data" | ||||||
| SERVICE_GENERATE_IMAGE = "generate_image" | SERVICE_GENERATE_IMAGE = "generate_image" | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| { | { | ||||||
|   "domain": "ai_task", |   "domain": "ai_task", | ||||||
|   "name": "AI Task", |   "name": "AI Task", | ||||||
|   "after_dependencies": ["camera", "http"], |   "after_dependencies": ["camera"], | ||||||
|   "codeowners": ["@home-assistant/core"], |   "codeowners": ["@home-assistant/core"], | ||||||
|   "dependencies": ["conversation", "media_source"], |   "dependencies": ["conversation", "media_source"], | ||||||
|   "documentation": "https://www.home-assistant.io/integrations/ai_task", |   "documentation": "https://www.home-assistant.io/integrations/ai_task", | ||||||
|   | |||||||
| @@ -2,89 +2,31 @@ | |||||||
|  |  | ||||||
| from __future__ import annotations | from __future__ import annotations | ||||||
|  |  | ||||||
| from datetime import timedelta | from pathlib import Path | ||||||
| import logging |  | ||||||
|  |  | ||||||
| from homeassistant.components.http.auth import async_sign_path | from homeassistant.components.media_source import MediaSource, local_source | ||||||
| from homeassistant.components.media_player import BrowseError, MediaClass |  | ||||||
| from homeassistant.components.media_source import ( |  | ||||||
|     BrowseMediaSource, |  | ||||||
|     MediaSource, |  | ||||||
|     MediaSourceItem, |  | ||||||
|     PlayMedia, |  | ||||||
|     Unresolvable, |  | ||||||
| ) |  | ||||||
| from homeassistant.core import HomeAssistant | from homeassistant.core import HomeAssistant | ||||||
|  | from homeassistant.exceptions import HomeAssistantError | ||||||
|  |  | ||||||
| from .const import DATA_IMAGES, DOMAIN, IMAGE_EXPIRY_TIME | from .const import DATA_MEDIA_SOURCE, DOMAIN, IMAGE_DIR | ||||||
|  |  | ||||||
| _LOGGER = logging.getLogger(__name__) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| async def async_get_media_source(hass: HomeAssistant) -> ImageMediaSource: | async def async_get_media_source(hass: HomeAssistant) -> MediaSource: | ||||||
|     """Set up image media source.""" |     """Set up local media source.""" | ||||||
|     _LOGGER.debug("Setting up image media source") |     media_dirs = list(hass.config.media_dirs.values()) | ||||||
|     return ImageMediaSource(hass) |  | ||||||
|  |  | ||||||
|  |     if not media_dirs: | ||||||
| class ImageMediaSource(MediaSource): |         raise HomeAssistantError( | ||||||
|     """Provide images as media sources.""" |             "AI Task media source requires at least one media directory configured" | ||||||
|  |  | ||||||
|     name: str = "AI Generated Images" |  | ||||||
|  |  | ||||||
|     def __init__(self, hass: HomeAssistant) -> None: |  | ||||||
|         """Initialize ImageMediaSource.""" |  | ||||||
|         super().__init__(DOMAIN) |  | ||||||
|         self.hass = hass |  | ||||||
|  |  | ||||||
|     async def async_resolve_media(self, item: MediaSourceItem) -> PlayMedia: |  | ||||||
|         """Resolve media to a url.""" |  | ||||||
|         image_storage = self.hass.data[DATA_IMAGES] |  | ||||||
|         image = image_storage.get(item.identifier) |  | ||||||
|  |  | ||||||
|         if image is None: |  | ||||||
|             raise Unresolvable(f"Could not resolve media item: {item.identifier}") |  | ||||||
|  |  | ||||||
|         return PlayMedia( |  | ||||||
|             async_sign_path( |  | ||||||
|                 self.hass, |  | ||||||
|                 f"/api/{DOMAIN}/images/{item.identifier}", |  | ||||||
|                 timedelta(seconds=IMAGE_EXPIRY_TIME or 1800), |  | ||||||
|             ), |  | ||||||
|             image.mime_type, |  | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     async def async_browse_media( |     media_dir = Path(media_dirs[0]) / DOMAIN / IMAGE_DIR | ||||||
|         self, |  | ||||||
|         item: MediaSourceItem, |  | ||||||
|     ) -> BrowseMediaSource: |  | ||||||
|         """Return media.""" |  | ||||||
|         if item.identifier: |  | ||||||
|             raise BrowseError("Unknown item") |  | ||||||
|  |  | ||||||
|         image_storage = self.hass.data[DATA_IMAGES] |     hass.data[DATA_MEDIA_SOURCE] = source = local_source.LocalSource( | ||||||
|  |         hass, | ||||||
|         children = [ |         DOMAIN, | ||||||
|             BrowseMediaSource( |         "AI Generated Images", | ||||||
|                 domain=DOMAIN, |         {IMAGE_DIR: str(media_dir)}, | ||||||
|                 identifier=filename, |         f"/{DOMAIN}", | ||||||
|                 media_class=MediaClass.IMAGE, |  | ||||||
|                 media_content_type=image.mime_type, |  | ||||||
|                 title=image.title or filename, |  | ||||||
|                 can_play=True, |  | ||||||
|                 can_expand=False, |  | ||||||
|             ) |  | ||||||
|             for filename, image in image_storage.items() |  | ||||||
|         ] |  | ||||||
|  |  | ||||||
|         return BrowseMediaSource( |  | ||||||
|             domain=DOMAIN, |  | ||||||
|             identifier=None, |  | ||||||
|             media_class=MediaClass.APP, |  | ||||||
|             media_content_type="", |  | ||||||
|             title="AI Generated Images", |  | ||||||
|             can_play=False, |  | ||||||
|             can_expand=True, |  | ||||||
|             children_media_class=MediaClass.IMAGE, |  | ||||||
|             children=children, |  | ||||||
|     ) |     ) | ||||||
|  |     return source | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ from __future__ import annotations | |||||||
|  |  | ||||||
| from dataclasses import dataclass | from dataclasses import dataclass | ||||||
| from datetime import datetime, timedelta | from datetime import datetime, timedelta | ||||||
| from functools import partial | import io | ||||||
| import mimetypes | import mimetypes | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
| import tempfile | import tempfile | ||||||
| @@ -12,34 +12,33 @@ from typing import Any | |||||||
|  |  | ||||||
| import voluptuous as vol | import voluptuous as vol | ||||||
|  |  | ||||||
| from homeassistant.components import camera, conversation, media_source | from homeassistant.components import camera, conversation, image, media_source | ||||||
| from homeassistant.components.http.auth import async_sign_path | from homeassistant.components.http.auth import async_sign_path | ||||||
| from homeassistant.core import HomeAssistant, ServiceResponse, callback | from homeassistant.core import HomeAssistant, ServiceResponse, callback | ||||||
| from homeassistant.exceptions import HomeAssistantError | from homeassistant.exceptions import HomeAssistantError | ||||||
| from homeassistant.helpers import llm | from homeassistant.helpers import llm | ||||||
| from homeassistant.helpers.chat_session import ChatSession, async_get_chat_session | from homeassistant.helpers.chat_session import ChatSession, async_get_chat_session | ||||||
| from homeassistant.helpers.event import async_call_later |  | ||||||
| from homeassistant.util import RE_SANITIZE_FILENAME, slugify | from homeassistant.util import RE_SANITIZE_FILENAME, slugify | ||||||
|  |  | ||||||
| from .const import ( | from .const import ( | ||||||
|     DATA_COMPONENT, |     DATA_COMPONENT, | ||||||
|     DATA_IMAGES, |     DATA_MEDIA_SOURCE, | ||||||
|     DATA_PREFERENCES, |     DATA_PREFERENCES, | ||||||
|     DOMAIN, |     DOMAIN, | ||||||
|  |     IMAGE_DIR, | ||||||
|     IMAGE_EXPIRY_TIME, |     IMAGE_EXPIRY_TIME, | ||||||
|     MAX_IMAGES, |  | ||||||
|     AITaskEntityFeature, |     AITaskEntityFeature, | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  |  | ||||||
| def _save_camera_snapshot(image: camera.Image) -> Path: | def _save_camera_snapshot(image_data: camera.Image | image.Image) -> Path: | ||||||
|     """Save camera snapshot to temp file.""" |     """Save camera snapshot to temp file.""" | ||||||
|     with tempfile.NamedTemporaryFile( |     with tempfile.NamedTemporaryFile( | ||||||
|         mode="wb", |         mode="wb", | ||||||
|         suffix=mimetypes.guess_extension(image.content_type, False), |         suffix=mimetypes.guess_extension(image_data.content_type, False), | ||||||
|         delete=False, |         delete=False, | ||||||
|     ) as temp_file: |     ) as temp_file: | ||||||
|         temp_file.write(image.content) |         temp_file.write(image_data.content) | ||||||
|         return Path(temp_file.name) |         return Path(temp_file.name) | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -55,26 +54,31 @@ async def _resolve_attachments( | |||||||
|     for attachment in attachments or []: |     for attachment in attachments or []: | ||||||
|         media_content_id = attachment["media_content_id"] |         media_content_id = attachment["media_content_id"] | ||||||
|  |  | ||||||
|         # Special case for camera media sources |         # Special case for certain media sources | ||||||
|         if media_content_id.startswith("media-source://camera/"): |         for integration in camera, image: | ||||||
|             # Extract entity_id from the media content ID |             media_source_prefix = f"media-source://{integration.DOMAIN}/" | ||||||
|             entity_id = media_content_id.removeprefix("media-source://camera/") |             if not media_content_id.startswith(media_source_prefix): | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|             # Get snapshot from camera |             # Extract entity_id from the media content ID | ||||||
|             image = await camera.async_get_image(hass, entity_id) |             entity_id = media_content_id.removeprefix(media_source_prefix) | ||||||
|  |  | ||||||
|  |             # Get snapshot from entity | ||||||
|  |             image_data = await integration.async_get_image(hass, entity_id) | ||||||
|  |  | ||||||
|             temp_filename = await hass.async_add_executor_job( |             temp_filename = await hass.async_add_executor_job( | ||||||
|                 _save_camera_snapshot, image |                 _save_camera_snapshot, image_data | ||||||
|             ) |             ) | ||||||
|             created_files.append(temp_filename) |             created_files.append(temp_filename) | ||||||
|  |  | ||||||
|             resolved_attachments.append( |             resolved_attachments.append( | ||||||
|                 conversation.Attachment( |                 conversation.Attachment( | ||||||
|                     media_content_id=media_content_id, |                     media_content_id=media_content_id, | ||||||
|                     mime_type=image.content_type, |                     mime_type=image_data.content_type, | ||||||
|                     path=temp_filename, |                     path=temp_filename, | ||||||
|                 ) |                 ) | ||||||
|             ) |             ) | ||||||
|  |             break | ||||||
|         else: |         else: | ||||||
|             # Handle regular media sources |             # Handle regular media sources | ||||||
|             media = await media_source.async_resolve_media(hass, media_content_id, None) |             media = await media_source.async_resolve_media(hass, media_content_id, None) | ||||||
| @@ -157,24 +161,6 @@ async def async_generate_data( | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
| def _cleanup_images(image_storage: dict[str, ImageData], num_to_remove: int) -> None: |  | ||||||
|     """Remove old images to keep the storage size under the limit.""" |  | ||||||
|     if num_to_remove <= 0: |  | ||||||
|         return |  | ||||||
|  |  | ||||||
|     if num_to_remove >= len(image_storage): |  | ||||||
|         image_storage.clear() |  | ||||||
|         return |  | ||||||
|  |  | ||||||
|     sorted_images = sorted( |  | ||||||
|         image_storage.items(), |  | ||||||
|         key=lambda item: item[1].timestamp, |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     for filename, _ in sorted_images[:num_to_remove]: |  | ||||||
|         image_storage.pop(filename, None) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| async def async_generate_image( | async def async_generate_image( | ||||||
|     hass: HomeAssistant, |     hass: HomeAssistant, | ||||||
|     *, |     *, | ||||||
| @@ -224,36 +210,34 @@ async def async_generate_image( | |||||||
|     if service_result.get("revised_prompt") is None: |     if service_result.get("revised_prompt") is None: | ||||||
|         service_result["revised_prompt"] = instructions |         service_result["revised_prompt"] = instructions | ||||||
|  |  | ||||||
|     image_storage = hass.data[DATA_IMAGES] |     source = hass.data[DATA_MEDIA_SOURCE] | ||||||
|  |  | ||||||
|     if len(image_storage) + 1 > MAX_IMAGES: |  | ||||||
|         _cleanup_images(image_storage, len(image_storage) + 1 - MAX_IMAGES) |  | ||||||
|  |  | ||||||
|     current_time = datetime.now() |     current_time = datetime.now() | ||||||
|     ext = mimetypes.guess_extension(task_result.mime_type, False) or ".png" |     ext = mimetypes.guess_extension(task_result.mime_type, False) or ".png" | ||||||
|     sanitized_task_name = RE_SANITIZE_FILENAME.sub("", slugify(task_name)) |     sanitized_task_name = RE_SANITIZE_FILENAME.sub("", slugify(task_name)) | ||||||
|     filename = f"{current_time.strftime('%Y-%m-%d_%H%M%S')}_{sanitized_task_name}{ext}" |  | ||||||
|  |  | ||||||
|     image_storage[filename] = ImageData( |     image_file = ImageData( | ||||||
|         data=image_data, |         filename=f"{current_time.strftime('%Y-%m-%d_%H%M%S')}_{sanitized_task_name}{ext}", | ||||||
|         timestamp=int(current_time.timestamp()), |         file=io.BytesIO(image_data), | ||||||
|         mime_type=task_result.mime_type, |         content_type=task_result.mime_type, | ||||||
|         title=service_result["revised_prompt"], |  | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     def _purge_image(filename: str, now: datetime) -> None: |     target_folder = media_source.MediaSourceItem.from_uri( | ||||||
|         """Remove image from storage.""" |         hass, f"media-source://{DOMAIN}/{IMAGE_DIR}", None | ||||||
|         image_storage.pop(filename, None) |     ) | ||||||
|  |  | ||||||
|     if IMAGE_EXPIRY_TIME > 0: |     service_result["media_source_id"] = await source.async_upload_media( | ||||||
|         async_call_later(hass, IMAGE_EXPIRY_TIME, partial(_purge_image, filename)) |         target_folder, image_file | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     item = media_source.MediaSourceItem.from_uri( | ||||||
|  |         hass, service_result["media_source_id"], None | ||||||
|  |     ) | ||||||
|     service_result["url"] = async_sign_path( |     service_result["url"] = async_sign_path( | ||||||
|         hass, |         hass, | ||||||
|         f"/api/{DOMAIN}/images/{filename}", |         (await source.async_resolve_media(item)).url, | ||||||
|         timedelta(seconds=IMAGE_EXPIRY_TIME or 1800), |         timedelta(seconds=IMAGE_EXPIRY_TIME), | ||||||
|     ) |     ) | ||||||
|     service_result["media_source_id"] = f"media-source://{DOMAIN}/images/{filename}" |  | ||||||
|  |  | ||||||
|     return service_result |     return service_result | ||||||
|  |  | ||||||
| @@ -358,20 +342,8 @@ class GenImageTaskResult: | |||||||
|  |  | ||||||
| @dataclass(slots=True) | @dataclass(slots=True) | ||||||
| class ImageData: | class ImageData: | ||||||
|     """Image data for stored generated images.""" |     """Implementation of media_source.local_source.UploadedFile protocol.""" | ||||||
|  |  | ||||||
|     data: bytes |     filename: str | ||||||
|     """Raw image data.""" |     file: io.IOBase | ||||||
|  |     content_type: str | ||||||
|     timestamp: int |  | ||||||
|     """Timestamp when the image was generated, as a Unix timestamp.""" |  | ||||||
|  |  | ||||||
|     mime_type: str |  | ||||||
|     """MIME type of the image.""" |  | ||||||
|  |  | ||||||
|     title: str |  | ||||||
|     """Title of the image, usually the prompt used to generate it.""" |  | ||||||
|  |  | ||||||
|     def __str__(self) -> str: |  | ||||||
|         """Return image data as a string.""" |  | ||||||
|         return f"<ImageData {self.title}: {id(self)}>" |  | ||||||
|   | |||||||
| @@ -1,7 +1,9 @@ | |||||||
| """Airgradient Update platform.""" | """Airgradient Update platform.""" | ||||||
|  |  | ||||||
| from datetime import timedelta | from datetime import timedelta | ||||||
|  | import logging | ||||||
|  |  | ||||||
|  | from airgradient import AirGradientConnectionError | ||||||
| from propcache.api import cached_property | from propcache.api import cached_property | ||||||
|  |  | ||||||
| from homeassistant.components.update import UpdateDeviceClass, UpdateEntity | from homeassistant.components.update import UpdateDeviceClass, UpdateEntity | ||||||
| @@ -13,6 +15,7 @@ from .entity import AirGradientEntity | |||||||
|  |  | ||||||
| PARALLEL_UPDATES = 1 | PARALLEL_UPDATES = 1 | ||||||
| SCAN_INTERVAL = timedelta(hours=1) | SCAN_INTERVAL = timedelta(hours=1) | ||||||
|  | _LOGGER = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  |  | ||||||
| async def async_setup_entry( | async def async_setup_entry( | ||||||
| @@ -31,6 +34,7 @@ class AirGradientUpdate(AirGradientEntity, UpdateEntity): | |||||||
|     """Representation of Airgradient Update.""" |     """Representation of Airgradient Update.""" | ||||||
|  |  | ||||||
|     _attr_device_class = UpdateDeviceClass.FIRMWARE |     _attr_device_class = UpdateDeviceClass.FIRMWARE | ||||||
|  |     _server_unreachable_logged = False | ||||||
|  |  | ||||||
|     def __init__(self, coordinator: AirGradientCoordinator) -> None: |     def __init__(self, coordinator: AirGradientCoordinator) -> None: | ||||||
|         """Initialize the entity.""" |         """Initialize the entity.""" | ||||||
| @@ -47,10 +51,27 @@ class AirGradientUpdate(AirGradientEntity, UpdateEntity): | |||||||
|         """Return the installed version of the entity.""" |         """Return the installed version of the entity.""" | ||||||
|         return self.coordinator.data.measures.firmware_version |         return self.coordinator.data.measures.firmware_version | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def available(self) -> bool: | ||||||
|  |         """Return if entity is available.""" | ||||||
|  |         return super().available and self._attr_available | ||||||
|  |  | ||||||
|     async def async_update(self) -> None: |     async def async_update(self) -> None: | ||||||
|         """Update the entity.""" |         """Update the entity.""" | ||||||
|  |         try: | ||||||
|             self._attr_latest_version = ( |             self._attr_latest_version = ( | ||||||
|                 await self.coordinator.client.get_latest_firmware_version( |                 await self.coordinator.client.get_latest_firmware_version( | ||||||
|                     self.coordinator.serial_number |                     self.coordinator.serial_number | ||||||
|                 ) |                 ) | ||||||
|             ) |             ) | ||||||
|  |         except AirGradientConnectionError: | ||||||
|  |             self._attr_latest_version = None | ||||||
|  |             self._attr_available = False | ||||||
|  |             if not self._server_unreachable_logged: | ||||||
|  |                 _LOGGER.error( | ||||||
|  |                     "Unable to connect to AirGradient server to check for updates" | ||||||
|  |                 ) | ||||||
|  |                 self._server_unreachable_logged = True | ||||||
|  |         else: | ||||||
|  |             self._server_unreachable_logged = False | ||||||
|  |             self._attr_available = True | ||||||
|   | |||||||
| @@ -4,10 +4,18 @@ from __future__ import annotations | |||||||
|  |  | ||||||
| from airos.airos8 import AirOS8 | from airos.airos8 import AirOS8 | ||||||
|  |  | ||||||
| from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME, Platform | from homeassistant.const import ( | ||||||
|  |     CONF_HOST, | ||||||
|  |     CONF_PASSWORD, | ||||||
|  |     CONF_SSL, | ||||||
|  |     CONF_USERNAME, | ||||||
|  |     CONF_VERIFY_SSL, | ||||||
|  |     Platform, | ||||||
|  | ) | ||||||
| from homeassistant.core import HomeAssistant | from homeassistant.core import HomeAssistant | ||||||
| from homeassistant.helpers.aiohttp_client import async_get_clientsession | from homeassistant.helpers.aiohttp_client import async_get_clientsession | ||||||
|  |  | ||||||
|  | from .const import DEFAULT_SSL, DEFAULT_VERIFY_SSL, SECTION_ADVANCED_SETTINGS | ||||||
| from .coordinator import AirOSConfigEntry, AirOSDataUpdateCoordinator | from .coordinator import AirOSConfigEntry, AirOSDataUpdateCoordinator | ||||||
|  |  | ||||||
| _PLATFORMS: list[Platform] = [ | _PLATFORMS: list[Platform] = [ | ||||||
| @@ -21,13 +29,16 @@ async def async_setup_entry(hass: HomeAssistant, entry: AirOSConfigEntry) -> boo | |||||||
|  |  | ||||||
|     # By default airOS 8 comes with self-signed SSL certificates, |     # By default airOS 8 comes with self-signed SSL certificates, | ||||||
|     # with no option in the web UI to change or upload a custom certificate. |     # with no option in the web UI to change or upload a custom certificate. | ||||||
|     session = async_get_clientsession(hass, verify_ssl=False) |     session = async_get_clientsession( | ||||||
|  |         hass, verify_ssl=entry.data[SECTION_ADVANCED_SETTINGS][CONF_VERIFY_SSL] | ||||||
|  |     ) | ||||||
|  |  | ||||||
|     airos_device = AirOS8( |     airos_device = AirOS8( | ||||||
|         host=entry.data[CONF_HOST], |         host=entry.data[CONF_HOST], | ||||||
|         username=entry.data[CONF_USERNAME], |         username=entry.data[CONF_USERNAME], | ||||||
|         password=entry.data[CONF_PASSWORD], |         password=entry.data[CONF_PASSWORD], | ||||||
|         session=session, |         session=session, | ||||||
|  |         use_ssl=entry.data[SECTION_ADVANCED_SETTINGS][CONF_SSL], | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     coordinator = AirOSDataUpdateCoordinator(hass, entry, airos_device) |     coordinator = AirOSDataUpdateCoordinator(hass, entry, airos_device) | ||||||
| @@ -40,6 +51,30 @@ async def async_setup_entry(hass: HomeAssistant, entry: AirOSConfigEntry) -> boo | |||||||
|     return True |     return True | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def async_migrate_entry(hass: HomeAssistant, entry: AirOSConfigEntry) -> bool: | ||||||
|  |     """Migrate old config entry.""" | ||||||
|  |  | ||||||
|  |     if entry.version > 1: | ||||||
|  |         # This means the user has downgraded from a future version | ||||||
|  |         return False | ||||||
|  |  | ||||||
|  |     if entry.version == 1 and entry.minor_version == 1: | ||||||
|  |         new_data = {**entry.data} | ||||||
|  |         advanced_data = { | ||||||
|  |             CONF_SSL: DEFAULT_SSL, | ||||||
|  |             CONF_VERIFY_SSL: DEFAULT_VERIFY_SSL, | ||||||
|  |         } | ||||||
|  |         new_data[SECTION_ADVANCED_SETTINGS] = advanced_data | ||||||
|  |  | ||||||
|  |         hass.config_entries.async_update_entry( | ||||||
|  |             entry, | ||||||
|  |             data=new_data, | ||||||
|  |             minor_version=2, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     return True | ||||||
|  |  | ||||||
|  |  | ||||||
| async def async_unload_entry(hass: HomeAssistant, entry: AirOSConfigEntry) -> bool: | async def async_unload_entry(hass: HomeAssistant, entry: AirOSConfigEntry) -> bool: | ||||||
|     """Unload a config entry.""" |     """Unload a config entry.""" | ||||||
|     return await hass.config_entries.async_unload_platforms(entry, _PLATFORMS) |     return await hass.config_entries.async_unload_platforms(entry, _PLATFORMS) | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ | |||||||
|  |  | ||||||
| from __future__ import annotations | from __future__ import annotations | ||||||
|  |  | ||||||
|  | from collections.abc import Mapping | ||||||
| import logging | import logging | ||||||
| from typing import Any | from typing import Any | ||||||
|  |  | ||||||
| @@ -14,11 +15,23 @@ from airos.exceptions import ( | |||||||
| ) | ) | ||||||
| import voluptuous as vol | import voluptuous as vol | ||||||
|  |  | ||||||
| from homeassistant.config_entries import ConfigFlow, ConfigFlowResult | from homeassistant.config_entries import SOURCE_REAUTH, ConfigFlow, ConfigFlowResult | ||||||
| from homeassistant.const import CONF_HOST, CONF_PASSWORD, CONF_USERNAME | from homeassistant.const import ( | ||||||
|  |     CONF_HOST, | ||||||
|  |     CONF_PASSWORD, | ||||||
|  |     CONF_SSL, | ||||||
|  |     CONF_USERNAME, | ||||||
|  |     CONF_VERIFY_SSL, | ||||||
|  | ) | ||||||
|  | from homeassistant.data_entry_flow import section | ||||||
| from homeassistant.helpers.aiohttp_client import async_get_clientsession | from homeassistant.helpers.aiohttp_client import async_get_clientsession | ||||||
|  | from homeassistant.helpers.selector import ( | ||||||
|  |     TextSelector, | ||||||
|  |     TextSelectorConfig, | ||||||
|  |     TextSelectorType, | ||||||
|  | ) | ||||||
|  |  | ||||||
| from .const import DOMAIN | from .const import DEFAULT_SSL, DEFAULT_VERIFY_SSL, DOMAIN, SECTION_ADVANCED_SETTINGS | ||||||
| from .coordinator import AirOS8 | from .coordinator import AirOS8 | ||||||
|  |  | ||||||
| _LOGGER = logging.getLogger(__name__) | _LOGGER = logging.getLogger(__name__) | ||||||
| @@ -28,6 +41,15 @@ STEP_USER_DATA_SCHEMA = vol.Schema( | |||||||
|         vol.Required(CONF_HOST): str, |         vol.Required(CONF_HOST): str, | ||||||
|         vol.Required(CONF_USERNAME, default="ubnt"): str, |         vol.Required(CONF_USERNAME, default="ubnt"): str, | ||||||
|         vol.Required(CONF_PASSWORD): str, |         vol.Required(CONF_PASSWORD): str, | ||||||
|  |         vol.Required(SECTION_ADVANCED_SETTINGS): section( | ||||||
|  |             vol.Schema( | ||||||
|  |                 { | ||||||
|  |                     vol.Required(CONF_SSL, default=DEFAULT_SSL): bool, | ||||||
|  |                     vol.Required(CONF_VERIFY_SSL, default=DEFAULT_VERIFY_SSL): bool, | ||||||
|  |                 } | ||||||
|  |             ), | ||||||
|  |             {"collapsed": True}, | ||||||
|  |         ), | ||||||
|     } |     } | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -36,23 +58,47 @@ class AirOSConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|     """Handle a config flow for Ubiquiti airOS.""" |     """Handle a config flow for Ubiquiti airOS.""" | ||||||
|  |  | ||||||
|     VERSION = 1 |     VERSION = 1 | ||||||
|  |     MINOR_VERSION = 2 | ||||||
|  |  | ||||||
|  |     def __init__(self) -> None: | ||||||
|  |         """Initialize the config flow.""" | ||||||
|  |         super().__init__() | ||||||
|  |         self.airos_device: AirOS8 | ||||||
|  |         self.errors: dict[str, str] = {} | ||||||
|  |  | ||||||
|     async def async_step_user( |     async def async_step_user( | ||||||
|         self, |         self, user_input: dict[str, Any] | None = None | ||||||
|         user_input: dict[str, Any] | None = None, |  | ||||||
|     ) -> ConfigFlowResult: |     ) -> ConfigFlowResult: | ||||||
|         """Handle the initial step.""" |         """Handle the manual input of host and credentials.""" | ||||||
|         errors: dict[str, str] = {} |         self.errors = {} | ||||||
|         if user_input is not None: |         if user_input is not None: | ||||||
|  |             validated_info = await self._validate_and_get_device_info(user_input) | ||||||
|  |             if validated_info: | ||||||
|  |                 return self.async_create_entry( | ||||||
|  |                     title=validated_info["title"], | ||||||
|  |                     data=validated_info["data"], | ||||||
|  |                 ) | ||||||
|  |         return self.async_show_form( | ||||||
|  |             step_id="user", data_schema=STEP_USER_DATA_SCHEMA, errors=self.errors | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     async def _validate_and_get_device_info( | ||||||
|  |         self, config_data: dict[str, Any] | ||||||
|  |     ) -> dict[str, Any] | None: | ||||||
|  |         """Validate user input with the device API.""" | ||||||
|         # By default airOS 8 comes with self-signed SSL certificates, |         # By default airOS 8 comes with self-signed SSL certificates, | ||||||
|         # with no option in the web UI to change or upload a custom certificate. |         # with no option in the web UI to change or upload a custom certificate. | ||||||
|             session = async_get_clientsession(self.hass, verify_ssl=False) |         session = async_get_clientsession( | ||||||
|  |             self.hass, | ||||||
|  |             verify_ssl=config_data[SECTION_ADVANCED_SETTINGS][CONF_VERIFY_SSL], | ||||||
|  |         ) | ||||||
|  |  | ||||||
|         airos_device = AirOS8( |         airos_device = AirOS8( | ||||||
|                 host=user_input[CONF_HOST], |             host=config_data[CONF_HOST], | ||||||
|                 username=user_input[CONF_USERNAME], |             username=config_data[CONF_USERNAME], | ||||||
|                 password=user_input[CONF_PASSWORD], |             password=config_data[CONF_PASSWORD], | ||||||
|             session=session, |             session=session, | ||||||
|  |             use_ssl=config_data[SECTION_ADVANCED_SETTINGS][CONF_SSL], | ||||||
|         ) |         ) | ||||||
|         try: |         try: | ||||||
|             await airos_device.login() |             await airos_device.login() | ||||||
| @@ -62,21 +108,59 @@ class AirOSConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|             AirOSConnectionSetupError, |             AirOSConnectionSetupError, | ||||||
|             AirOSDeviceConnectionError, |             AirOSDeviceConnectionError, | ||||||
|         ): |         ): | ||||||
|                 errors["base"] = "cannot_connect" |             self.errors["base"] = "cannot_connect" | ||||||
|         except (AirOSConnectionAuthenticationError, AirOSDataMissingError): |         except (AirOSConnectionAuthenticationError, AirOSDataMissingError): | ||||||
|                 errors["base"] = "invalid_auth" |             self.errors["base"] = "invalid_auth" | ||||||
|         except AirOSKeyDataMissingError: |         except AirOSKeyDataMissingError: | ||||||
|                 errors["base"] = "key_data_missing" |             self.errors["base"] = "key_data_missing" | ||||||
|         except Exception: |         except Exception: | ||||||
|                 _LOGGER.exception("Unexpected exception") |             _LOGGER.exception("Unexpected exception during credential validation") | ||||||
|                 errors["base"] = "unknown" |             self.errors["base"] = "unknown" | ||||||
|         else: |         else: | ||||||
|             await self.async_set_unique_id(airos_data.derived.mac) |             await self.async_set_unique_id(airos_data.derived.mac) | ||||||
|  |  | ||||||
|  |             if self.source == SOURCE_REAUTH: | ||||||
|  |                 self._abort_if_unique_id_mismatch() | ||||||
|  |             else: | ||||||
|                 self._abort_if_unique_id_configured() |                 self._abort_if_unique_id_configured() | ||||||
|                 return self.async_create_entry( |  | ||||||
|                     title=airos_data.host.hostname, data=user_input |             return {"title": airos_data.host.hostname, "data": config_data} | ||||||
|  |  | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |     async def async_step_reauth( | ||||||
|  |         self, | ||||||
|  |         user_input: Mapping[str, Any], | ||||||
|  |     ) -> ConfigFlowResult: | ||||||
|  |         """Perform reauthentication upon an API authentication error.""" | ||||||
|  |         return await self.async_step_reauth_confirm(user_input) | ||||||
|  |  | ||||||
|  |     async def async_step_reauth_confirm( | ||||||
|  |         self, | ||||||
|  |         user_input: Mapping[str, Any], | ||||||
|  |     ) -> ConfigFlowResult: | ||||||
|  |         """Perform reauthentication upon an API authentication error.""" | ||||||
|  |         self.errors = {} | ||||||
|  |  | ||||||
|  |         if user_input: | ||||||
|  |             validate_data = {**self._get_reauth_entry().data, **user_input} | ||||||
|  |             if await self._validate_and_get_device_info(config_data=validate_data): | ||||||
|  |                 return self.async_update_reload_and_abort( | ||||||
|  |                     self._get_reauth_entry(), | ||||||
|  |                     data_updates=validate_data, | ||||||
|                 ) |                 ) | ||||||
|  |  | ||||||
|         return self.async_show_form( |         return self.async_show_form( | ||||||
|             step_id="user", data_schema=STEP_USER_DATA_SCHEMA, errors=errors |             step_id="reauth_confirm", | ||||||
|  |             data_schema=vol.Schema( | ||||||
|  |                 { | ||||||
|  |                     vol.Required(CONF_PASSWORD): TextSelector( | ||||||
|  |                         TextSelectorConfig( | ||||||
|  |                             type=TextSelectorType.PASSWORD, | ||||||
|  |                             autocomplete="current-password", | ||||||
|  |                         ) | ||||||
|  |                     ), | ||||||
|  |                 } | ||||||
|  |             ), | ||||||
|  |             errors=self.errors, | ||||||
|         ) |         ) | ||||||
|   | |||||||
| @@ -7,3 +7,8 @@ DOMAIN = "airos" | |||||||
| SCAN_INTERVAL = timedelta(minutes=1) | SCAN_INTERVAL = timedelta(minutes=1) | ||||||
|  |  | ||||||
| MANUFACTURER = "Ubiquiti" | MANUFACTURER = "Ubiquiti" | ||||||
|  |  | ||||||
|  | DEFAULT_VERIFY_SSL = False | ||||||
|  | DEFAULT_SSL = True | ||||||
|  |  | ||||||
|  | SECTION_ADVANCED_SETTINGS = "advanced_settings" | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ from airos.exceptions import ( | |||||||
|  |  | ||||||
| from homeassistant.config_entries import ConfigEntry | from homeassistant.config_entries import ConfigEntry | ||||||
| from homeassistant.core import HomeAssistant | from homeassistant.core import HomeAssistant | ||||||
| from homeassistant.exceptions import ConfigEntryError | from homeassistant.exceptions import ConfigEntryAuthFailed | ||||||
| from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed | from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed | ||||||
|  |  | ||||||
| from .const import DOMAIN, SCAN_INTERVAL | from .const import DOMAIN, SCAN_INTERVAL | ||||||
| @@ -47,9 +47,9 @@ class AirOSDataUpdateCoordinator(DataUpdateCoordinator[AirOS8Data]): | |||||||
|         try: |         try: | ||||||
|             await self.airos_device.login() |             await self.airos_device.login() | ||||||
|             return await self.airos_device.status() |             return await self.airos_device.status() | ||||||
|         except (AirOSConnectionAuthenticationError,) as err: |         except AirOSConnectionAuthenticationError as err: | ||||||
|             _LOGGER.exception("Error authenticating with airOS device") |             _LOGGER.exception("Error authenticating with airOS device") | ||||||
|             raise ConfigEntryError( |             raise ConfigEntryAuthFailed( | ||||||
|                 translation_domain=DOMAIN, translation_key="invalid_auth" |                 translation_domain=DOMAIN, translation_key="invalid_auth" | ||||||
|             ) from err |             ) from err | ||||||
|         except ( |         except ( | ||||||
|   | |||||||
| @@ -2,11 +2,11 @@ | |||||||
|  |  | ||||||
| from __future__ import annotations | from __future__ import annotations | ||||||
|  |  | ||||||
| from homeassistant.const import CONF_HOST | from homeassistant.const import CONF_HOST, CONF_SSL | ||||||
| from homeassistant.helpers.device_registry import CONNECTION_NETWORK_MAC, DeviceInfo | from homeassistant.helpers.device_registry import CONNECTION_NETWORK_MAC, DeviceInfo | ||||||
| from homeassistant.helpers.update_coordinator import CoordinatorEntity | from homeassistant.helpers.update_coordinator import CoordinatorEntity | ||||||
|  |  | ||||||
| from .const import DOMAIN, MANUFACTURER | from .const import DOMAIN, MANUFACTURER, SECTION_ADVANCED_SETTINGS | ||||||
| from .coordinator import AirOSDataUpdateCoordinator | from .coordinator import AirOSDataUpdateCoordinator | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -20,9 +20,14 @@ class AirOSEntity(CoordinatorEntity[AirOSDataUpdateCoordinator]): | |||||||
|         super().__init__(coordinator) |         super().__init__(coordinator) | ||||||
|  |  | ||||||
|         airos_data = self.coordinator.data |         airos_data = self.coordinator.data | ||||||
|  |         url_schema = ( | ||||||
|  |             "https" | ||||||
|  |             if coordinator.config_entry.data[SECTION_ADVANCED_SETTINGS][CONF_SSL] | ||||||
|  |             else "http" | ||||||
|  |         ) | ||||||
|  |  | ||||||
|         configuration_url: str | None = ( |         configuration_url: str | None = ( | ||||||
|             f"https://{coordinator.config_entry.data[CONF_HOST]}" |             f"{url_schema}://{coordinator.config_entry.data[CONF_HOST]}" | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         self._attr_device_info = DeviceInfo( |         self._attr_device_info = DeviceInfo( | ||||||
|   | |||||||
| @@ -6,5 +6,5 @@ | |||||||
|   "documentation": "https://www.home-assistant.io/integrations/airos", |   "documentation": "https://www.home-assistant.io/integrations/airos", | ||||||
|   "iot_class": "local_polling", |   "iot_class": "local_polling", | ||||||
|   "quality_scale": "bronze", |   "quality_scale": "bronze", | ||||||
|   "requirements": ["airos==0.5.1"] |   "requirements": ["airos==0.5.5"] | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,6 +2,14 @@ | |||||||
|   "config": { |   "config": { | ||||||
|     "flow_title": "Ubiquiti airOS device", |     "flow_title": "Ubiquiti airOS device", | ||||||
|     "step": { |     "step": { | ||||||
|  |       "reauth_confirm": { | ||||||
|  |         "data": { | ||||||
|  |           "password": "[%key:common::config_flow::data::password%]" | ||||||
|  |         }, | ||||||
|  |         "data_description": { | ||||||
|  |           "password": "[%key:component::airos::config::step::user::data_description::password%]" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "user": { |       "user": { | ||||||
|         "data": { |         "data": { | ||||||
|           "host": "[%key:common::config_flow::data::host%]", |           "host": "[%key:common::config_flow::data::host%]", | ||||||
| @@ -12,6 +20,18 @@ | |||||||
|           "host": "IP address or hostname of the airOS device", |           "host": "IP address or hostname of the airOS device", | ||||||
|           "username": "Administrator username for the airOS device, normally 'ubnt'", |           "username": "Administrator username for the airOS device, normally 'ubnt'", | ||||||
|           "password": "Password configured through the UISP app or web interface" |           "password": "Password configured through the UISP app or web interface" | ||||||
|  |         }, | ||||||
|  |         "sections": { | ||||||
|  |           "advanced_settings": { | ||||||
|  |             "data": { | ||||||
|  |               "ssl": "Use HTTPS", | ||||||
|  |               "verify_ssl": "[%key:common::config_flow::data::verify_ssl%]" | ||||||
|  |             }, | ||||||
|  |             "data_description": { | ||||||
|  |               "ssl": "Whether the connection should be encrypted (required for most devices)", | ||||||
|  |               "verify_ssl": "Whether the certificate should be verified when using HTTPS. This should be off for self-signed certificates" | ||||||
|  |             } | ||||||
|  |           } | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
| @@ -22,7 +42,9 @@ | |||||||
|       "unknown": "[%key:common::config_flow::error::unknown%]" |       "unknown": "[%key:common::config_flow::error::unknown%]" | ||||||
|     }, |     }, | ||||||
|     "abort": { |     "abort": { | ||||||
|       "already_configured": "[%key:common::config_flow::abort::already_configured_device%]" |       "already_configured": "[%key:common::config_flow::abort::already_configured_device%]", | ||||||
|  |       "reauth_successful": "[%key:common::config_flow::abort::reauth_successful%]", | ||||||
|  |       "unique_id_mismatch": "Re-authentication should be used for the same device not a new one" | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   "entity": { |   "entity": { | ||||||
|   | |||||||
| @@ -23,6 +23,10 @@ STEP_USER_DATA_SCHEMA = vol.Schema( | |||||||
|     } |     } | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | URL_API_INTEGRATION = { | ||||||
|  |     "url": "https://dashboard.airthings.com/integrations/api-integration" | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| class AirthingsConfigFlow(ConfigFlow, domain=DOMAIN): | class AirthingsConfigFlow(ConfigFlow, domain=DOMAIN): | ||||||
|     """Handle a config flow for Airthings.""" |     """Handle a config flow for Airthings.""" | ||||||
| @@ -37,11 +41,7 @@ class AirthingsConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|             return self.async_show_form( |             return self.async_show_form( | ||||||
|                 step_id="user", |                 step_id="user", | ||||||
|                 data_schema=STEP_USER_DATA_SCHEMA, |                 data_schema=STEP_USER_DATA_SCHEMA, | ||||||
|                 description_placeholders={ |                 description_placeholders=URL_API_INTEGRATION, | ||||||
|                     "url": ( |  | ||||||
|                         "https://dashboard.airthings.com/integrations/api-integration" |  | ||||||
|                     ), |  | ||||||
|                 }, |  | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         errors = {} |         errors = {} | ||||||
| @@ -65,5 +65,8 @@ class AirthingsConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|             return self.async_create_entry(title="Airthings", data=user_input) |             return self.async_create_entry(title="Airthings", data=user_input) | ||||||
|  |  | ||||||
|         return self.async_show_form( |         return self.async_show_form( | ||||||
|             step_id="user", data_schema=STEP_USER_DATA_SCHEMA, errors=errors |             step_id="user", | ||||||
|  |             data_schema=STEP_USER_DATA_SCHEMA, | ||||||
|  |             errors=errors, | ||||||
|  |             description_placeholders=URL_API_INTEGRATION, | ||||||
|         ) |         ) | ||||||
|   | |||||||
| @@ -4,10 +4,10 @@ | |||||||
|       "user": { |       "user": { | ||||||
|         "data": { |         "data": { | ||||||
|           "id": "ID", |           "id": "ID", | ||||||
|           "secret": "Secret", |           "secret": "Secret" | ||||||
|  |         }, | ||||||
|         "description": "Log in at {url} to find your credentials" |         "description": "Log in at {url} to find your credentials" | ||||||
|       } |       } | ||||||
|       } |  | ||||||
|     }, |     }, | ||||||
|     "error": { |     "error": { | ||||||
|       "cannot_connect": "[%key:common::config_flow::error::cannot_connect%]", |       "cannot_connect": "[%key:common::config_flow::error::cannot_connect%]", | ||||||
|   | |||||||
| @@ -6,8 +6,13 @@ import dataclasses | |||||||
| import logging | import logging | ||||||
| from typing import Any | from typing import Any | ||||||
|  |  | ||||||
| from airthings_ble import AirthingsBluetoothDeviceData, AirthingsDevice | from airthings_ble import ( | ||||||
|  |     AirthingsBluetoothDeviceData, | ||||||
|  |     AirthingsDevice, | ||||||
|  |     UnsupportedDeviceError, | ||||||
|  | ) | ||||||
| from bleak import BleakError | from bleak import BleakError | ||||||
|  | from habluetooth import BluetoothServiceInfoBleak | ||||||
| import voluptuous as vol | import voluptuous as vol | ||||||
|  |  | ||||||
| from homeassistant.components import bluetooth | from homeassistant.components import bluetooth | ||||||
| @@ -27,6 +32,7 @@ SERVICE_UUIDS = [ | |||||||
|     "b42e4a8e-ade7-11e4-89d3-123b93f75cba", |     "b42e4a8e-ade7-11e4-89d3-123b93f75cba", | ||||||
|     "b42e1c08-ade7-11e4-89d3-123b93f75cba", |     "b42e1c08-ade7-11e4-89d3-123b93f75cba", | ||||||
|     "b42e3882-ade7-11e4-89d3-123b93f75cba", |     "b42e3882-ade7-11e4-89d3-123b93f75cba", | ||||||
|  |     "b42e90a2-ade7-11e4-89d3-123b93f75cba", | ||||||
| ] | ] | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -37,6 +43,7 @@ class Discovery: | |||||||
|     name: str |     name: str | ||||||
|     discovery_info: BluetoothServiceInfo |     discovery_info: BluetoothServiceInfo | ||||||
|     device: AirthingsDevice |     device: AirthingsDevice | ||||||
|  |     data: AirthingsBluetoothDeviceData | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_name(device: AirthingsDevice) -> str: | def get_name(device: AirthingsDevice) -> str: | ||||||
| @@ -44,7 +51,7 @@ def get_name(device: AirthingsDevice) -> str: | |||||||
|  |  | ||||||
|     name = device.friendly_name() |     name = device.friendly_name() | ||||||
|     if identifier := device.identifier: |     if identifier := device.identifier: | ||||||
|         name += f" ({identifier})" |         name += f" ({device.model.value}{identifier})" | ||||||
|     return name |     return name | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -62,8 +69,8 @@ class AirthingsConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|         self._discovered_device: Discovery | None = None |         self._discovered_device: Discovery | None = None | ||||||
|         self._discovered_devices: dict[str, Discovery] = {} |         self._discovered_devices: dict[str, Discovery] = {} | ||||||
|  |  | ||||||
|     async def _get_device_data( |     async def _get_device( | ||||||
|         self, discovery_info: BluetoothServiceInfo |         self, data: AirthingsBluetoothDeviceData, discovery_info: BluetoothServiceInfo | ||||||
|     ) -> AirthingsDevice: |     ) -> AirthingsDevice: | ||||||
|         ble_device = bluetooth.async_ble_device_from_address( |         ble_device = bluetooth.async_ble_device_from_address( | ||||||
|             self.hass, discovery_info.address |             self.hass, discovery_info.address | ||||||
| @@ -72,10 +79,8 @@ class AirthingsConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|             _LOGGER.debug("no ble_device in _get_device_data") |             _LOGGER.debug("no ble_device in _get_device_data") | ||||||
|             raise AirthingsDeviceUpdateError("No ble_device") |             raise AirthingsDeviceUpdateError("No ble_device") | ||||||
|  |  | ||||||
|         airthings = AirthingsBluetoothDeviceData(_LOGGER) |  | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             data = await airthings.update_device(ble_device) |             device = await data.update_device(ble_device) | ||||||
|         except BleakError as err: |         except BleakError as err: | ||||||
|             _LOGGER.error( |             _LOGGER.error( | ||||||
|                 "Error connecting to and getting data from %s: %s", |                 "Error connecting to and getting data from %s: %s", | ||||||
| @@ -83,12 +88,15 @@ class AirthingsConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|                 err, |                 err, | ||||||
|             ) |             ) | ||||||
|             raise AirthingsDeviceUpdateError("Failed getting device data") from err |             raise AirthingsDeviceUpdateError("Failed getting device data") from err | ||||||
|  |         except UnsupportedDeviceError: | ||||||
|  |             _LOGGER.debug("Skipping unsupported device: %s", discovery_info.name) | ||||||
|  |             raise | ||||||
|         except Exception as err: |         except Exception as err: | ||||||
|             _LOGGER.error( |             _LOGGER.error( | ||||||
|                 "Unknown error occurred from %s: %s", discovery_info.address, err |                 "Unknown error occurred from %s: %s", discovery_info.address, err | ||||||
|             ) |             ) | ||||||
|             raise |             raise | ||||||
|         return data |         return device | ||||||
|  |  | ||||||
|     async def async_step_bluetooth( |     async def async_step_bluetooth( | ||||||
|         self, discovery_info: BluetoothServiceInfo |         self, discovery_info: BluetoothServiceInfo | ||||||
| @@ -98,17 +106,21 @@ class AirthingsConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|         await self.async_set_unique_id(discovery_info.address) |         await self.async_set_unique_id(discovery_info.address) | ||||||
|         self._abort_if_unique_id_configured() |         self._abort_if_unique_id_configured() | ||||||
|  |  | ||||||
|  |         data = AirthingsBluetoothDeviceData(logger=_LOGGER) | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             device = await self._get_device_data(discovery_info) |             device = await self._get_device(data=data, discovery_info=discovery_info) | ||||||
|         except AirthingsDeviceUpdateError: |         except AirthingsDeviceUpdateError: | ||||||
|             return self.async_abort(reason="cannot_connect") |             return self.async_abort(reason="cannot_connect") | ||||||
|  |         except UnsupportedDeviceError: | ||||||
|  |             return self.async_abort(reason="unsupported_device") | ||||||
|         except Exception: |         except Exception: | ||||||
|             _LOGGER.exception("Unknown error occurred") |             _LOGGER.exception("Unknown error occurred") | ||||||
|             return self.async_abort(reason="unknown") |             return self.async_abort(reason="unknown") | ||||||
|  |  | ||||||
|         name = get_name(device) |         name = get_name(device) | ||||||
|         self.context["title_placeholders"] = {"name": name} |         self.context["title_placeholders"] = {"name": name} | ||||||
|         self._discovered_device = Discovery(name, discovery_info, device) |         self._discovered_device = Discovery(name, discovery_info, device, data=data) | ||||||
|  |  | ||||||
|         return await self.async_step_bluetooth_confirm() |         return await self.async_step_bluetooth_confirm() | ||||||
|  |  | ||||||
| @@ -117,6 +129,12 @@ class AirthingsConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|     ) -> ConfigFlowResult: |     ) -> ConfigFlowResult: | ||||||
|         """Confirm discovery.""" |         """Confirm discovery.""" | ||||||
|         if user_input is not None: |         if user_input is not None: | ||||||
|  |             if ( | ||||||
|  |                 self._discovered_device is not None | ||||||
|  |                 and self._discovered_device.device.firmware.need_firmware_upgrade | ||||||
|  |             ): | ||||||
|  |                 return self.async_abort(reason="firmware_upgrade_required") | ||||||
|  |  | ||||||
|             return self.async_create_entry( |             return self.async_create_entry( | ||||||
|                 title=self.context["title_placeholders"]["name"], data={} |                 title=self.context["title_placeholders"]["name"], data={} | ||||||
|             ) |             ) | ||||||
| @@ -137,6 +155,9 @@ class AirthingsConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|             self._abort_if_unique_id_configured() |             self._abort_if_unique_id_configured() | ||||||
|             discovery = self._discovered_devices[address] |             discovery = self._discovered_devices[address] | ||||||
|  |  | ||||||
|  |             if discovery.device.firmware.need_firmware_upgrade: | ||||||
|  |                 return self.async_abort(reason="firmware_upgrade_required") | ||||||
|  |  | ||||||
|             self.context["title_placeholders"] = { |             self.context["title_placeholders"] = { | ||||||
|                 "name": discovery.name, |                 "name": discovery.name, | ||||||
|             } |             } | ||||||
| @@ -146,32 +167,53 @@ class AirthingsConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|             return self.async_create_entry(title=discovery.name, data={}) |             return self.async_create_entry(title=discovery.name, data={}) | ||||||
|  |  | ||||||
|         current_addresses = self._async_current_ids(include_ignore=False) |         current_addresses = self._async_current_ids(include_ignore=False) | ||||||
|  |         devices: list[BluetoothServiceInfoBleak] = [] | ||||||
|         for discovery_info in async_discovered_service_info(self.hass): |         for discovery_info in async_discovered_service_info(self.hass): | ||||||
|             address = discovery_info.address |             address = discovery_info.address | ||||||
|             if address in current_addresses or address in self._discovered_devices: |             if address in current_addresses or address in self._discovered_devices: | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             if MFCT_ID not in discovery_info.manufacturer_data: |             if MFCT_ID not in discovery_info.manufacturer_data: | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             if not any(uuid in SERVICE_UUIDS for uuid in discovery_info.service_uuids): |             if not any(uuid in SERVICE_UUIDS for uuid in discovery_info.service_uuids): | ||||||
|  |                 _LOGGER.debug( | ||||||
|  |                     "Skipping unsupported device: %s (%s)", discovery_info.name, address | ||||||
|  |                 ) | ||||||
|                 continue |                 continue | ||||||
|  |             devices.append(discovery_info) | ||||||
|  |  | ||||||
|  |         for discovery_info in devices: | ||||||
|  |             address = discovery_info.address | ||||||
|  |             data = AirthingsBluetoothDeviceData(logger=_LOGGER) | ||||||
|             try: |             try: | ||||||
|                 device = await self._get_device_data(discovery_info) |                 device = await self._get_device(data, discovery_info) | ||||||
|             except AirthingsDeviceUpdateError: |             except AirthingsDeviceUpdateError: | ||||||
|                 return self.async_abort(reason="cannot_connect") |                 _LOGGER.error( | ||||||
|  |                     "Error connecting to and getting data from %s (%s)", | ||||||
|  |                     discovery_info.name, | ||||||
|  |                     discovery_info.address, | ||||||
|  |                 ) | ||||||
|  |                 continue | ||||||
|  |             except UnsupportedDeviceError: | ||||||
|  |                 _LOGGER.debug( | ||||||
|  |                     "Skipping unsupported device: %s (%s)", | ||||||
|  |                     discovery_info.name, | ||||||
|  |                     discovery_info.address, | ||||||
|  |                 ) | ||||||
|  |                 continue | ||||||
|             except Exception: |             except Exception: | ||||||
|                 _LOGGER.exception("Unknown error occurred") |                 _LOGGER.exception("Unknown error occurred") | ||||||
|                 return self.async_abort(reason="unknown") |                 return self.async_abort(reason="unknown") | ||||||
|             name = get_name(device) |             name = get_name(device) | ||||||
|             self._discovered_devices[address] = Discovery(name, discovery_info, device) |             _LOGGER.debug("Discovered Airthings device: %s (%s)", name, address) | ||||||
|  |             self._discovered_devices[address] = Discovery( | ||||||
|  |                 name, discovery_info, device, data | ||||||
|  |             ) | ||||||
|  |  | ||||||
|         if not self._discovered_devices: |         if not self._discovered_devices: | ||||||
|             return self.async_abort(reason="no_devices_found") |             return self.async_abort(reason="no_devices_found") | ||||||
|  |  | ||||||
|         titles = { |         titles = { | ||||||
|             address: discovery.device.name |             address: get_name(discovery.device) | ||||||
|             for (address, discovery) in self._discovered_devices.items() |             for (address, discovery) in self._discovered_devices.items() | ||||||
|         } |         } | ||||||
|         return self.async_show_form( |         return self.async_show_form( | ||||||
|   | |||||||
| @@ -17,6 +17,10 @@ | |||||||
|     { |     { | ||||||
|       "manufacturer_id": 820, |       "manufacturer_id": 820, | ||||||
|       "service_uuid": "b42e3882-ade7-11e4-89d3-123b93f75cba" |       "service_uuid": "b42e3882-ade7-11e4-89d3-123b93f75cba" | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "manufacturer_id": 820, | ||||||
|  |       "service_uuid": "b42e90a2-ade7-11e4-89d3-123b93f75cba" | ||||||
|     } |     } | ||||||
|   ], |   ], | ||||||
|   "codeowners": ["@vincegio", "@LaStrada"], |   "codeowners": ["@vincegio", "@LaStrada"], | ||||||
| @@ -24,5 +28,5 @@ | |||||||
|   "dependencies": ["bluetooth_adapters"], |   "dependencies": ["bluetooth_adapters"], | ||||||
|   "documentation": "https://www.home-assistant.io/integrations/airthings_ble", |   "documentation": "https://www.home-assistant.io/integrations/airthings_ble", | ||||||
|   "iot_class": "local_polling", |   "iot_class": "local_polling", | ||||||
|   "requirements": ["airthings-ble==0.9.2"] |   "requirements": ["airthings-ble==1.1.1"] | ||||||
| } | } | ||||||
|   | |||||||
| @@ -16,10 +16,12 @@ from homeassistant.components.sensor import ( | |||||||
| from homeassistant.const import ( | from homeassistant.const import ( | ||||||
|     CONCENTRATION_PARTS_PER_BILLION, |     CONCENTRATION_PARTS_PER_BILLION, | ||||||
|     CONCENTRATION_PARTS_PER_MILLION, |     CONCENTRATION_PARTS_PER_MILLION, | ||||||
|  |     LIGHT_LUX, | ||||||
|     PERCENTAGE, |     PERCENTAGE, | ||||||
|     EntityCategory, |     EntityCategory, | ||||||
|     Platform, |     Platform, | ||||||
|     UnitOfPressure, |     UnitOfPressure, | ||||||
|  |     UnitOfSoundPressure, | ||||||
|     UnitOfTemperature, |     UnitOfTemperature, | ||||||
| ) | ) | ||||||
| from homeassistant.core import HomeAssistant, callback | from homeassistant.core import HomeAssistant, callback | ||||||
| @@ -112,8 +114,25 @@ SENSORS_MAPPING_TEMPLATE: dict[str, SensorEntityDescription] = { | |||||||
|         state_class=SensorStateClass.MEASUREMENT, |         state_class=SensorStateClass.MEASUREMENT, | ||||||
|         suggested_display_precision=0, |         suggested_display_precision=0, | ||||||
|     ), |     ), | ||||||
|  |     "lux": SensorEntityDescription( | ||||||
|  |         key="lux", | ||||||
|  |         device_class=SensorDeviceClass.ILLUMINANCE, | ||||||
|  |         native_unit_of_measurement=LIGHT_LUX, | ||||||
|  |         state_class=SensorStateClass.MEASUREMENT, | ||||||
|  |         suggested_display_precision=0, | ||||||
|  |     ), | ||||||
|  |     "noise": SensorEntityDescription( | ||||||
|  |         key="noise", | ||||||
|  |         translation_key="ambient_noise", | ||||||
|  |         device_class=SensorDeviceClass.SOUND_PRESSURE, | ||||||
|  |         native_unit_of_measurement=UnitOfSoundPressure.WEIGHTED_DECIBEL_A, | ||||||
|  |         state_class=SensorStateClass.MEASUREMENT, | ||||||
|  |         suggested_display_precision=0, | ||||||
|  |     ), | ||||||
| } | } | ||||||
|  |  | ||||||
|  | PARALLEL_UPDATES = 0 | ||||||
|  |  | ||||||
|  |  | ||||||
| @callback | @callback | ||||||
| def async_migrate(hass: HomeAssistant, address: str, sensor_name: str) -> None: | def async_migrate(hass: HomeAssistant, address: str, sensor_name: str) -> None: | ||||||
|   | |||||||
| @@ -6,6 +6,9 @@ | |||||||
|         "description": "[%key:component::bluetooth::config::step::user::description%]", |         "description": "[%key:component::bluetooth::config::step::user::description%]", | ||||||
|         "data": { |         "data": { | ||||||
|           "address": "[%key:common::config_flow::data::device%]" |           "address": "[%key:common::config_flow::data::device%]" | ||||||
|  |         }, | ||||||
|  |         "data_description": { | ||||||
|  |           "address": "The Airthings devices discovered via Bluetooth." | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "bluetooth_confirm": { |       "bluetooth_confirm": { | ||||||
| @@ -17,6 +20,8 @@ | |||||||
|       "already_in_progress": "[%key:common::config_flow::abort::already_in_progress%]", |       "already_in_progress": "[%key:common::config_flow::abort::already_in_progress%]", | ||||||
|       "already_configured": "[%key:common::config_flow::abort::already_configured_device%]", |       "already_configured": "[%key:common::config_flow::abort::already_configured_device%]", | ||||||
|       "cannot_connect": "[%key:common::config_flow::error::cannot_connect%]", |       "cannot_connect": "[%key:common::config_flow::error::cannot_connect%]", | ||||||
|  |       "firmware_upgrade_required": "Your device requires a firmware upgrade. Please use the Airthings app (Android/iOS) to upgrade it.", | ||||||
|  |       "unsupported_device": "Unsupported device", | ||||||
|       "unknown": "[%key:common::config_flow::error::unknown%]" |       "unknown": "[%key:common::config_flow::error::unknown%]" | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
| @@ -36,6 +41,9 @@ | |||||||
|       }, |       }, | ||||||
|       "illuminance": { |       "illuminance": { | ||||||
|         "name": "[%key:component::sensor::entity_component::illuminance::name%]" |         "name": "[%key:component::sensor::entity_component::illuminance::name%]" | ||||||
|  |       }, | ||||||
|  |       "ambient_noise": { | ||||||
|  |         "name": "Ambient noise" | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -2,17 +2,14 @@ | |||||||
|  |  | ||||||
| from airtouch4pyapi import AirTouch | from airtouch4pyapi import AirTouch | ||||||
|  |  | ||||||
| from homeassistant.config_entries import ConfigEntry |  | ||||||
| from homeassistant.const import CONF_HOST, Platform | from homeassistant.const import CONF_HOST, Platform | ||||||
| from homeassistant.core import HomeAssistant | from homeassistant.core import HomeAssistant | ||||||
| from homeassistant.exceptions import ConfigEntryNotReady | from homeassistant.exceptions import ConfigEntryNotReady | ||||||
|  |  | ||||||
| from .coordinator import AirtouchDataUpdateCoordinator | from .coordinator import AirTouch4ConfigEntry, AirtouchDataUpdateCoordinator | ||||||
|  |  | ||||||
| PLATFORMS = [Platform.CLIMATE] | PLATFORMS = [Platform.CLIMATE] | ||||||
|  |  | ||||||
| type AirTouch4ConfigEntry = ConfigEntry[AirtouchDataUpdateCoordinator] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| async def async_setup_entry(hass: HomeAssistant, entry: AirTouch4ConfigEntry) -> bool: | async def async_setup_entry(hass: HomeAssistant, entry: AirTouch4ConfigEntry) -> bool: | ||||||
|     """Set up AirTouch4 from a config entry.""" |     """Set up AirTouch4 from a config entry.""" | ||||||
| @@ -22,7 +19,7 @@ async def async_setup_entry(hass: HomeAssistant, entry: AirTouch4ConfigEntry) -> | |||||||
|     info = airtouch.GetAcs() |     info = airtouch.GetAcs() | ||||||
|     if not info: |     if not info: | ||||||
|         raise ConfigEntryNotReady |         raise ConfigEntryNotReady | ||||||
|     coordinator = AirtouchDataUpdateCoordinator(hass, airtouch) |     coordinator = AirtouchDataUpdateCoordinator(hass, entry, airtouch) | ||||||
|     await coordinator.async_config_entry_first_refresh() |     await coordinator.async_config_entry_first_refresh() | ||||||
|     entry.runtime_data = coordinator |     entry.runtime_data = coordinator | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,26 +2,34 @@ | |||||||
|  |  | ||||||
| import logging | import logging | ||||||
|  |  | ||||||
|  | from airtouch4pyapi import AirTouch | ||||||
| from airtouch4pyapi.airtouch import AirTouchStatus | from airtouch4pyapi.airtouch import AirTouchStatus | ||||||
|  |  | ||||||
| from homeassistant.components.climate import SCAN_INTERVAL | from homeassistant.components.climate import SCAN_INTERVAL | ||||||
|  | from homeassistant.config_entries import ConfigEntry | ||||||
|  | from homeassistant.core import HomeAssistant | ||||||
| from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed | from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed | ||||||
|  |  | ||||||
| from .const import DOMAIN | from .const import DOMAIN | ||||||
|  |  | ||||||
| _LOGGER = logging.getLogger(__name__) | _LOGGER = logging.getLogger(__name__) | ||||||
|  |  | ||||||
|  | type AirTouch4ConfigEntry = ConfigEntry[AirtouchDataUpdateCoordinator] | ||||||
|  |  | ||||||
|  |  | ||||||
| class AirtouchDataUpdateCoordinator(DataUpdateCoordinator): | class AirtouchDataUpdateCoordinator(DataUpdateCoordinator): | ||||||
|     """Class to manage fetching Airtouch data.""" |     """Class to manage fetching Airtouch data.""" | ||||||
|  |  | ||||||
|     def __init__(self, hass, airtouch): |     def __init__( | ||||||
|  |         self, hass: HomeAssistant, entry: AirTouch4ConfigEntry, airtouch: AirTouch | ||||||
|  |     ) -> None: | ||||||
|         """Initialize global Airtouch data updater.""" |         """Initialize global Airtouch data updater.""" | ||||||
|         self.airtouch = airtouch |         self.airtouch = airtouch | ||||||
|  |  | ||||||
|         super().__init__( |         super().__init__( | ||||||
|             hass, |             hass, | ||||||
|             _LOGGER, |             _LOGGER, | ||||||
|  |             config_entry=entry, | ||||||
|             name=DOMAIN, |             name=DOMAIN, | ||||||
|             update_interval=SCAN_INTERVAL, |             update_interval=SCAN_INTERVAL, | ||||||
|         ) |         ) | ||||||
|   | |||||||
| @@ -6,17 +6,19 @@ from collections.abc import Callable | |||||||
| from dataclasses import dataclass | from dataclasses import dataclass | ||||||
| from typing import Any, Final | from typing import Any, Final | ||||||
|  |  | ||||||
| from aioairzone.common import GrilleAngle, OperationMode, SleepTimeout | from aioairzone.common import GrilleAngle, OperationMode, QAdapt, SleepTimeout | ||||||
| from aioairzone.const import ( | from aioairzone.const import ( | ||||||
|     API_COLD_ANGLE, |     API_COLD_ANGLE, | ||||||
|     API_HEAT_ANGLE, |     API_HEAT_ANGLE, | ||||||
|     API_MODE, |     API_MODE, | ||||||
|  |     API_Q_ADAPT, | ||||||
|     API_SLEEP, |     API_SLEEP, | ||||||
|     AZD_COLD_ANGLE, |     AZD_COLD_ANGLE, | ||||||
|     AZD_HEAT_ANGLE, |     AZD_HEAT_ANGLE, | ||||||
|     AZD_MASTER, |     AZD_MASTER, | ||||||
|     AZD_MODE, |     AZD_MODE, | ||||||
|     AZD_MODES, |     AZD_MODES, | ||||||
|  |     AZD_Q_ADAPT, | ||||||
|     AZD_SLEEP, |     AZD_SLEEP, | ||||||
|     AZD_ZONES, |     AZD_ZONES, | ||||||
| ) | ) | ||||||
| @@ -65,6 +67,14 @@ SLEEP_DICT: Final[dict[str, int]] = { | |||||||
|     "90m": SleepTimeout.SLEEP_90, |     "90m": SleepTimeout.SLEEP_90, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | Q_ADAPT_DICT: Final[dict[str, int]] = { | ||||||
|  |     "standard": QAdapt.STANDARD, | ||||||
|  |     "power": QAdapt.POWER, | ||||||
|  |     "silence": QAdapt.SILENCE, | ||||||
|  |     "minimum": QAdapt.MINIMUM, | ||||||
|  |     "maximum": QAdapt.MAXIMUM, | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| def main_zone_options( | def main_zone_options( | ||||||
|     zone_data: dict[str, Any], |     zone_data: dict[str, Any], | ||||||
| @@ -83,6 +93,14 @@ MAIN_ZONE_SELECT_TYPES: Final[tuple[AirzoneSelectDescription, ...]] = ( | |||||||
|         options_fn=main_zone_options, |         options_fn=main_zone_options, | ||||||
|         translation_key="modes", |         translation_key="modes", | ||||||
|     ), |     ), | ||||||
|  |     AirzoneSelectDescription( | ||||||
|  |         api_param=API_Q_ADAPT, | ||||||
|  |         entity_category=EntityCategory.CONFIG, | ||||||
|  |         key=AZD_Q_ADAPT, | ||||||
|  |         options=list(Q_ADAPT_DICT), | ||||||
|  |         options_dict=Q_ADAPT_DICT, | ||||||
|  |         translation_key="q_adapt", | ||||||
|  |     ), | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -63,6 +63,16 @@ | |||||||
|           "stop": "Stop" |           "stop": "Stop" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|  |       "q_adapt": { | ||||||
|  |         "name": "Q-Adapt", | ||||||
|  |         "state": { | ||||||
|  |           "standard": "Standard", | ||||||
|  |           "power": "Power", | ||||||
|  |           "silence": "Silence", | ||||||
|  |           "minimum": "Minimum", | ||||||
|  |           "maximum": "Maximum" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "sleep_times": { |       "sleep_times": { | ||||||
|         "name": "Sleep", |         "name": "Sleep", | ||||||
|         "state": { |         "state": { | ||||||
|   | |||||||
| @@ -3,7 +3,6 @@ | |||||||
| from __future__ import annotations | from __future__ import annotations | ||||||
|  |  | ||||||
| from genie_partner_sdk.client import AladdinConnectClient | from genie_partner_sdk.client import AladdinConnectClient | ||||||
| from genie_partner_sdk.model import GarageDoor |  | ||||||
|  |  | ||||||
| from homeassistant.const import Platform | from homeassistant.const import Platform | ||||||
| from homeassistant.core import HomeAssistant | from homeassistant.core import HomeAssistant | ||||||
| @@ -36,22 +35,7 @@ async def async_setup_entry( | |||||||
|         api.AsyncConfigEntryAuth(aiohttp_client.async_get_clientsession(hass), session) |         api.AsyncConfigEntryAuth(aiohttp_client.async_get_clientsession(hass), session) | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     sdk_doors = await client.get_doors() |     doors = await client.get_doors() | ||||||
|  |  | ||||||
|     # Convert SDK GarageDoor objects to integration GarageDoor objects |  | ||||||
|     doors = [ |  | ||||||
|         GarageDoor( |  | ||||||
|             { |  | ||||||
|                 "device_id": door.device_id, |  | ||||||
|                 "door_number": door.door_number, |  | ||||||
|                 "name": door.name, |  | ||||||
|                 "status": door.status, |  | ||||||
|                 "link_status": door.link_status, |  | ||||||
|                 "battery_level": door.battery_level, |  | ||||||
|             } |  | ||||||
|         ) |  | ||||||
|         for door in sdk_doors |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     entry.runtime_data = { |     entry.runtime_data = { | ||||||
|         door.unique_id: AladdinConnectCoordinator(hass, entry, client, door) |         door.unique_id: AladdinConnectCoordinator(hass, entry, client, door) | ||||||
|   | |||||||
| @@ -22,6 +22,17 @@ class OAuth2FlowHandler( | |||||||
|     VERSION = CONFIG_FLOW_VERSION |     VERSION = CONFIG_FLOW_VERSION | ||||||
|     MINOR_VERSION = CONFIG_FLOW_MINOR_VERSION |     MINOR_VERSION = CONFIG_FLOW_MINOR_VERSION | ||||||
|  |  | ||||||
|  |     async def async_step_user( | ||||||
|  |         self, user_input: dict[str, Any] | None = None | ||||||
|  |     ) -> ConfigFlowResult: | ||||||
|  |         """Check we have the cloud integration set up.""" | ||||||
|  |         if "cloud" not in self.hass.config.components: | ||||||
|  |             return self.async_abort( | ||||||
|  |                 reason="cloud_not_enabled", | ||||||
|  |                 description_placeholders={"default_config": "default_config"}, | ||||||
|  |             ) | ||||||
|  |         return await super().async_step_user(user_input) | ||||||
|  |  | ||||||
|     async def async_step_reauth( |     async def async_step_reauth( | ||||||
|         self, user_input: Mapping[str, Any] |         self, user_input: Mapping[str, Any] | ||||||
|     ) -> ConfigFlowResult: |     ) -> ConfigFlowResult: | ||||||
|   | |||||||
| @@ -41,4 +41,10 @@ class AladdinConnectCoordinator(DataUpdateCoordinator[GarageDoor]): | |||||||
|     async def _async_update_data(self) -> GarageDoor: |     async def _async_update_data(self) -> GarageDoor: | ||||||
|         """Fetch data from the Aladdin Connect API.""" |         """Fetch data from the Aladdin Connect API.""" | ||||||
|         await self.client.update_door(self.data.device_id, self.data.door_number) |         await self.client.update_door(self.data.device_id, self.data.door_number) | ||||||
|  |         self.data.status = self.client.get_door_status( | ||||||
|  |             self.data.device_id, self.data.door_number | ||||||
|  |         ) | ||||||
|  |         self.data.battery_level = self.client.get_battery_status( | ||||||
|  |             self.data.device_id, self.data.door_number | ||||||
|  |         ) | ||||||
|         return self.data |         return self.data | ||||||
|   | |||||||
| @@ -49,7 +49,9 @@ class AladdinCoverEntity(AladdinConnectEntity, CoverEntity): | |||||||
|     @property |     @property | ||||||
|     def is_closed(self) -> bool | None: |     def is_closed(self) -> bool | None: | ||||||
|         """Update is closed attribute.""" |         """Update is closed attribute.""" | ||||||
|         return self.coordinator.data.status == "closed" |         if (status := self.coordinator.data.status) is None: | ||||||
|  |             return None | ||||||
|  |         return status == "closed" | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def is_closing(self) -> bool | None: |     def is_closing(self) -> bool | None: | ||||||
|   | |||||||
| @@ -12,5 +12,5 @@ | |||||||
|   "documentation": "https://www.home-assistant.io/integrations/aladdin_connect", |   "documentation": "https://www.home-assistant.io/integrations/aladdin_connect", | ||||||
|   "integration_type": "hub", |   "integration_type": "hub", | ||||||
|   "iot_class": "cloud_polling", |   "iot_class": "cloud_polling", | ||||||
|   "requirements": ["genie-partner-sdk==1.0.10"] |   "requirements": ["genie-partner-sdk==1.0.11"] | ||||||
| } | } | ||||||
|   | |||||||
| @@ -24,7 +24,8 @@ | |||||||
|       "no_url_available": "[%key:common::config_flow::abort::oauth2_no_url_available%]", |       "no_url_available": "[%key:common::config_flow::abort::oauth2_no_url_available%]", | ||||||
|       "user_rejected_authorize": "[%key:common::config_flow::abort::oauth2_user_rejected_authorize%]", |       "user_rejected_authorize": "[%key:common::config_flow::abort::oauth2_user_rejected_authorize%]", | ||||||
|       "reauth_successful": "[%key:common::config_flow::abort::reauth_successful%]", |       "reauth_successful": "[%key:common::config_flow::abort::reauth_successful%]", | ||||||
|       "wrong_account": "You are authenticated with a different account than the one set up. Please authenticate with the configured account." |       "wrong_account": "You are authenticated with a different account than the one set up. Please authenticate with the configured account.", | ||||||
|  |       "cloud_not_enabled": "Please make sure you run Home Assistant with `{default_config}` enabled in your configuration.yaml." | ||||||
|     }, |     }, | ||||||
|     "create_entry": { |     "create_entry": { | ||||||
|       "default": "[%key:common::config_flow::create_entry::authenticated%]" |       "default": "[%key:common::config_flow::create_entry::authenticated%]" | ||||||
|   | |||||||
| @@ -2,10 +2,9 @@ | |||||||
|  |  | ||||||
| from __future__ import annotations | from __future__ import annotations | ||||||
|  |  | ||||||
| import asyncio |  | ||||||
| from datetime import timedelta | from datetime import timedelta | ||||||
| import logging | import logging | ||||||
| from typing import TYPE_CHECKING, Any, Final, final | from typing import Any, Final, final | ||||||
|  |  | ||||||
| from propcache.api import cached_property | from propcache.api import cached_property | ||||||
| import voluptuous as vol | import voluptuous as vol | ||||||
| @@ -28,8 +27,6 @@ from homeassistant.helpers import config_validation as cv | |||||||
| from homeassistant.helpers.config_validation import make_entity_service_schema | from homeassistant.helpers.config_validation import make_entity_service_schema | ||||||
| from homeassistant.helpers.entity import Entity, EntityDescription | from homeassistant.helpers.entity import Entity, EntityDescription | ||||||
| from homeassistant.helpers.entity_component import EntityComponent | from homeassistant.helpers.entity_component import EntityComponent | ||||||
| from homeassistant.helpers.entity_platform import EntityPlatform |  | ||||||
| from homeassistant.helpers.frame import ReportBehavior, report_usage |  | ||||||
| from homeassistant.helpers.typing import ConfigType | from homeassistant.helpers.typing import ConfigType | ||||||
| from homeassistant.util.hass_dict import HassKey | from homeassistant.util.hass_dict import HassKey | ||||||
|  |  | ||||||
| @@ -149,68 +146,11 @@ class AlarmControlPanelEntity(Entity, cached_properties=CACHED_PROPERTIES_WITH_A | |||||||
|     ) |     ) | ||||||
|     _alarm_control_panel_option_default_code: str | None = None |     _alarm_control_panel_option_default_code: str | None = None | ||||||
|  |  | ||||||
|     __alarm_legacy_state: bool = False |  | ||||||
|  |  | ||||||
|     def __init_subclass__(cls, **kwargs: Any) -> None: |  | ||||||
|         """Post initialisation processing.""" |  | ||||||
|         super().__init_subclass__(**kwargs) |  | ||||||
|         if any(method in cls.__dict__ for method in ("_attr_state", "state")): |  | ||||||
|             # Integrations should use the 'alarm_state' property instead of |  | ||||||
|             # setting the state directly. |  | ||||||
|             cls.__alarm_legacy_state = True |  | ||||||
|  |  | ||||||
|     def __setattr__(self, name: str, value: Any, /) -> None: |  | ||||||
|         """Set attribute. |  | ||||||
|  |  | ||||||
|         Deprecation warning if setting '_attr_state' directly |  | ||||||
|         unless already reported. |  | ||||||
|         """ |  | ||||||
|         if name == "_attr_state": |  | ||||||
|             self._report_deprecated_alarm_state_handling() |  | ||||||
|         return super().__setattr__(name, value) |  | ||||||
|  |  | ||||||
|     @callback |  | ||||||
|     def add_to_platform_start( |  | ||||||
|         self, |  | ||||||
|         hass: HomeAssistant, |  | ||||||
|         platform: EntityPlatform, |  | ||||||
|         parallel_updates: asyncio.Semaphore | None, |  | ||||||
|     ) -> None: |  | ||||||
|         """Start adding an entity to a platform.""" |  | ||||||
|         super().add_to_platform_start(hass, platform, parallel_updates) |  | ||||||
|         if self.__alarm_legacy_state: |  | ||||||
|             self._report_deprecated_alarm_state_handling() |  | ||||||
|  |  | ||||||
|     @callback |  | ||||||
|     def _report_deprecated_alarm_state_handling(self) -> None: |  | ||||||
|         """Report on deprecated handling of alarm state. |  | ||||||
|  |  | ||||||
|         Integrations should implement alarm_state instead of using state directly. |  | ||||||
|         """ |  | ||||||
|         report_usage( |  | ||||||
|             "is setting state directly." |  | ||||||
|             f" Entity {self.entity_id} ({type(self)}) should implement the 'alarm_state'" |  | ||||||
|             " property and return its state using the AlarmControlPanelState enum", |  | ||||||
|             core_integration_behavior=ReportBehavior.ERROR, |  | ||||||
|             custom_integration_behavior=ReportBehavior.LOG, |  | ||||||
|             breaks_in_ha_version="2025.11", |  | ||||||
|             integration_domain=self.platform.platform_name if self.platform else None, |  | ||||||
|             exclude_integrations={DOMAIN}, |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     @final |     @final | ||||||
|     @property |     @property | ||||||
|     def state(self) -> str | None: |     def state(self) -> str | None: | ||||||
|         """Return the current state.""" |         """Return the current state.""" | ||||||
|         if (alarm_state := self.alarm_state) is not None: |         return self.alarm_state | ||||||
|             return alarm_state |  | ||||||
|         if self._attr_state is not None: |  | ||||||
|             # Backwards compatibility for integrations that set state directly |  | ||||||
|             # Should be removed in 2025.11 |  | ||||||
|             if TYPE_CHECKING: |  | ||||||
|                 assert isinstance(self._attr_state, str) |  | ||||||
|             return self._attr_state |  | ||||||
|         return None |  | ||||||
|  |  | ||||||
|     @cached_property |     @cached_property | ||||||
|     def alarm_state(self) -> AlarmControlPanelState | None: |     def alarm_state(self) -> AlarmControlPanelState | None: | ||||||
|   | |||||||
| @@ -1472,10 +1472,10 @@ class AlexaModeController(AlexaCapability): | |||||||
|             # Return state instead of position when using ModeController. |             # Return state instead of position when using ModeController. | ||||||
|             mode = self.entity.state |             mode = self.entity.state | ||||||
|             if mode in ( |             if mode in ( | ||||||
|                 cover.STATE_OPEN, |                 cover.CoverState.OPEN, | ||||||
|                 cover.STATE_OPENING, |                 cover.CoverState.OPENING, | ||||||
|                 cover.STATE_CLOSED, |                 cover.CoverState.CLOSED, | ||||||
|                 cover.STATE_CLOSING, |                 cover.CoverState.CLOSING, | ||||||
|                 STATE_UNKNOWN, |                 STATE_UNKNOWN, | ||||||
|             ): |             ): | ||||||
|                 return f"{cover.ATTR_POSITION}.{mode}" |                 return f"{cover.ATTR_POSITION}.{mode}" | ||||||
| @@ -1594,11 +1594,11 @@ class AlexaModeController(AlexaCapability): | |||||||
|                 ["Position", AlexaGlobalCatalog.SETTING_OPENING], False |                 ["Position", AlexaGlobalCatalog.SETTING_OPENING], False | ||||||
|             ) |             ) | ||||||
|             self._resource.add_mode( |             self._resource.add_mode( | ||||||
|                 f"{cover.ATTR_POSITION}.{cover.STATE_OPEN}", |                 f"{cover.ATTR_POSITION}.{cover.CoverState.OPEN}", | ||||||
|                 [AlexaGlobalCatalog.VALUE_OPEN], |                 [AlexaGlobalCatalog.VALUE_OPEN], | ||||||
|             ) |             ) | ||||||
|             self._resource.add_mode( |             self._resource.add_mode( | ||||||
|                 f"{cover.ATTR_POSITION}.{cover.STATE_CLOSED}", |                 f"{cover.ATTR_POSITION}.{cover.CoverState.CLOSED}", | ||||||
|                 [AlexaGlobalCatalog.VALUE_CLOSE], |                 [AlexaGlobalCatalog.VALUE_CLOSE], | ||||||
|             ) |             ) | ||||||
|             self._resource.add_mode( |             self._resource.add_mode( | ||||||
| @@ -1651,22 +1651,22 @@ class AlexaModeController(AlexaCapability): | |||||||
|                 raise_labels.append(AlexaSemantics.ACTION_OPEN) |                 raise_labels.append(AlexaSemantics.ACTION_OPEN) | ||||||
|                 self._semantics.add_states_to_value( |                 self._semantics.add_states_to_value( | ||||||
|                     [AlexaSemantics.STATES_CLOSED], |                     [AlexaSemantics.STATES_CLOSED], | ||||||
|                     f"{cover.ATTR_POSITION}.{cover.STATE_CLOSED}", |                     f"{cover.ATTR_POSITION}.{cover.CoverState.CLOSED}", | ||||||
|                 ) |                 ) | ||||||
|                 self._semantics.add_states_to_value( |                 self._semantics.add_states_to_value( | ||||||
|                     [AlexaSemantics.STATES_OPEN], |                     [AlexaSemantics.STATES_OPEN], | ||||||
|                     f"{cover.ATTR_POSITION}.{cover.STATE_OPEN}", |                     f"{cover.ATTR_POSITION}.{cover.CoverState.OPEN}", | ||||||
|                 ) |                 ) | ||||||
|  |  | ||||||
|             self._semantics.add_action_to_directive( |             self._semantics.add_action_to_directive( | ||||||
|                 lower_labels, |                 lower_labels, | ||||||
|                 "SetMode", |                 "SetMode", | ||||||
|                 {"mode": f"{cover.ATTR_POSITION}.{cover.STATE_CLOSED}"}, |                 {"mode": f"{cover.ATTR_POSITION}.{cover.CoverState.CLOSED}"}, | ||||||
|             ) |             ) | ||||||
|             self._semantics.add_action_to_directive( |             self._semantics.add_action_to_directive( | ||||||
|                 raise_labels, |                 raise_labels, | ||||||
|                 "SetMode", |                 "SetMode", | ||||||
|                 {"mode": f"{cover.ATTR_POSITION}.{cover.STATE_OPEN}"}, |                 {"mode": f"{cover.ATTR_POSITION}.{cover.CoverState.OPEN}"}, | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|             return self._semantics.serialize_semantics() |             return self._semantics.serialize_semantics() | ||||||
|   | |||||||
| @@ -1261,9 +1261,9 @@ async def async_api_set_mode( | |||||||
|     elif instance == f"{cover.DOMAIN}.{cover.ATTR_POSITION}": |     elif instance == f"{cover.DOMAIN}.{cover.ATTR_POSITION}": | ||||||
|         position = mode.split(".")[1] |         position = mode.split(".")[1] | ||||||
|  |  | ||||||
|         if position == cover.STATE_CLOSED: |         if position == cover.CoverState.CLOSED: | ||||||
|             service = cover.SERVICE_CLOSE_COVER |             service = cover.SERVICE_CLOSE_COVER | ||||||
|         elif position == cover.STATE_OPEN: |         elif position == cover.CoverState.OPEN: | ||||||
|             service = cover.SERVICE_OPEN_COVER |             service = cover.SERVICE_OPEN_COVER | ||||||
|         elif position == "custom": |         elif position == "custom": | ||||||
|             service = cover.SERVICE_STOP_COVER |             service = cover.SERVICE_STOP_COVER | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ from aioamazondevices.api import AmazonDevice | |||||||
| from aioamazondevices.const import SENSOR_STATE_OFF | from aioamazondevices.const import SENSOR_STATE_OFF | ||||||
|  |  | ||||||
| from homeassistant.components.binary_sensor import ( | from homeassistant.components.binary_sensor import ( | ||||||
|  |     DOMAIN as BINARY_SENSOR_DOMAIN, | ||||||
|     BinarySensorDeviceClass, |     BinarySensorDeviceClass, | ||||||
|     BinarySensorEntity, |     BinarySensorEntity, | ||||||
|     BinarySensorEntityDescription, |     BinarySensorEntityDescription, | ||||||
| @@ -17,9 +18,12 @@ from homeassistant.components.binary_sensor import ( | |||||||
| from homeassistant.const import EntityCategory | from homeassistant.const import EntityCategory | ||||||
| from homeassistant.core import HomeAssistant | from homeassistant.core import HomeAssistant | ||||||
| from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | ||||||
|  | import homeassistant.helpers.entity_registry as er | ||||||
|  |  | ||||||
|  | from .const import _LOGGER, DOMAIN | ||||||
| from .coordinator import AmazonConfigEntry | from .coordinator import AmazonConfigEntry | ||||||
| from .entity import AmazonEntity | from .entity import AmazonEntity | ||||||
|  | from .utils import async_update_unique_id | ||||||
|  |  | ||||||
| # Coordinator is used to centralize the data updates | # Coordinator is used to centralize the data updates | ||||||
| PARALLEL_UPDATES = 0 | PARALLEL_UPDATES = 0 | ||||||
| @@ -31,6 +35,7 @@ class AmazonBinarySensorEntityDescription(BinarySensorEntityDescription): | |||||||
|  |  | ||||||
|     is_on_fn: Callable[[AmazonDevice, str], bool] |     is_on_fn: Callable[[AmazonDevice, str], bool] | ||||||
|     is_supported: Callable[[AmazonDevice, str], bool] = lambda device, key: True |     is_supported: Callable[[AmazonDevice, str], bool] = lambda device, key: True | ||||||
|  |     is_available_fn: Callable[[AmazonDevice, str], bool] = lambda device, key: True | ||||||
|  |  | ||||||
|  |  | ||||||
| BINARY_SENSORS: Final = ( | BINARY_SENSORS: Final = ( | ||||||
| @@ -40,47 +45,52 @@ BINARY_SENSORS: Final = ( | |||||||
|         entity_category=EntityCategory.DIAGNOSTIC, |         entity_category=EntityCategory.DIAGNOSTIC, | ||||||
|         is_on_fn=lambda device, _: device.online, |         is_on_fn=lambda device, _: device.online, | ||||||
|     ), |     ), | ||||||
|  |     AmazonBinarySensorEntityDescription( | ||||||
|  |         key="detectionState", | ||||||
|  |         device_class=BinarySensorDeviceClass.MOTION, | ||||||
|  |         is_on_fn=lambda device, key: bool( | ||||||
|  |             device.sensors[key].value != SENSOR_STATE_OFF | ||||||
|  |         ), | ||||||
|  |         is_supported=lambda device, key: device.sensors.get(key) is not None, | ||||||
|  |         is_available_fn=lambda device, key: ( | ||||||
|  |             device.online | ||||||
|  |             and (sensor := device.sensors.get(key)) is not None | ||||||
|  |             and sensor.error is False | ||||||
|  |         ), | ||||||
|  |     ), | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | DEPRECATED_BINARY_SENSORS: Final = ( | ||||||
|     AmazonBinarySensorEntityDescription( |     AmazonBinarySensorEntityDescription( | ||||||
|         key="bluetooth", |         key="bluetooth", | ||||||
|         entity_category=EntityCategory.DIAGNOSTIC, |         entity_category=EntityCategory.DIAGNOSTIC, | ||||||
|         translation_key="bluetooth", |         translation_key="bluetooth", | ||||||
|         is_on_fn=lambda device, _: device.bluetooth_state, |         is_on_fn=lambda device, key: False, | ||||||
|     ), |     ), | ||||||
|     AmazonBinarySensorEntityDescription( |     AmazonBinarySensorEntityDescription( | ||||||
|         key="babyCryDetectionState", |         key="babyCryDetectionState", | ||||||
|         translation_key="baby_cry_detection", |         translation_key="baby_cry_detection", | ||||||
|         is_on_fn=lambda device, key: (device.sensors[key].value != SENSOR_STATE_OFF), |         is_on_fn=lambda device, key: False, | ||||||
|         is_supported=lambda device, key: device.sensors.get(key) is not None, |  | ||||||
|     ), |     ), | ||||||
|     AmazonBinarySensorEntityDescription( |     AmazonBinarySensorEntityDescription( | ||||||
|         key="beepingApplianceDetectionState", |         key="beepingApplianceDetectionState", | ||||||
|         translation_key="beeping_appliance_detection", |         translation_key="beeping_appliance_detection", | ||||||
|         is_on_fn=lambda device, key: (device.sensors[key].value != SENSOR_STATE_OFF), |         is_on_fn=lambda device, key: False, | ||||||
|         is_supported=lambda device, key: device.sensors.get(key) is not None, |  | ||||||
|     ), |     ), | ||||||
|     AmazonBinarySensorEntityDescription( |     AmazonBinarySensorEntityDescription( | ||||||
|         key="coughDetectionState", |         key="coughDetectionState", | ||||||
|         translation_key="cough_detection", |         translation_key="cough_detection", | ||||||
|         is_on_fn=lambda device, key: (device.sensors[key].value != SENSOR_STATE_OFF), |         is_on_fn=lambda device, key: False, | ||||||
|         is_supported=lambda device, key: device.sensors.get(key) is not None, |  | ||||||
|     ), |     ), | ||||||
|     AmazonBinarySensorEntityDescription( |     AmazonBinarySensorEntityDescription( | ||||||
|         key="dogBarkDetectionState", |         key="dogBarkDetectionState", | ||||||
|         translation_key="dog_bark_detection", |         translation_key="dog_bark_detection", | ||||||
|         is_on_fn=lambda device, key: (device.sensors[key].value != SENSOR_STATE_OFF), |         is_on_fn=lambda device, key: False, | ||||||
|         is_supported=lambda device, key: device.sensors.get(key) is not None, |  | ||||||
|     ), |  | ||||||
|     AmazonBinarySensorEntityDescription( |  | ||||||
|         key="humanPresenceDetectionState", |  | ||||||
|         device_class=BinarySensorDeviceClass.MOTION, |  | ||||||
|         is_on_fn=lambda device, key: (device.sensors[key].value != SENSOR_STATE_OFF), |  | ||||||
|         is_supported=lambda device, key: device.sensors.get(key) is not None, |  | ||||||
|     ), |     ), | ||||||
|     AmazonBinarySensorEntityDescription( |     AmazonBinarySensorEntityDescription( | ||||||
|         key="waterSoundsDetectionState", |         key="waterSoundsDetectionState", | ||||||
|         translation_key="water_sounds_detection", |         translation_key="water_sounds_detection", | ||||||
|         is_on_fn=lambda device, key: (device.sensors[key].value != SENSOR_STATE_OFF), |         is_on_fn=lambda device, key: False, | ||||||
|         is_supported=lambda device, key: device.sensors.get(key) is not None, |  | ||||||
|     ), |     ), | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -94,12 +104,45 @@ async def async_setup_entry( | |||||||
|  |  | ||||||
|     coordinator = entry.runtime_data |     coordinator = entry.runtime_data | ||||||
|  |  | ||||||
|  |     entity_registry = er.async_get(hass) | ||||||
|  |  | ||||||
|  |     # Replace unique id for "detectionState" binary sensor | ||||||
|  |     await async_update_unique_id( | ||||||
|  |         hass, | ||||||
|  |         coordinator, | ||||||
|  |         BINARY_SENSOR_DOMAIN, | ||||||
|  |         "humanPresenceDetectionState", | ||||||
|  |         "detectionState", | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     # Clean up deprecated sensors | ||||||
|  |     for sensor_desc in DEPRECATED_BINARY_SENSORS: | ||||||
|  |         for serial_num in coordinator.data: | ||||||
|  |             unique_id = f"{serial_num}-{sensor_desc.key}" | ||||||
|  |             if entity_id := entity_registry.async_get_entity_id( | ||||||
|  |                 BINARY_SENSOR_DOMAIN, DOMAIN, unique_id | ||||||
|  |             ): | ||||||
|  |                 _LOGGER.debug("Removing deprecated entity %s", entity_id) | ||||||
|  |                 entity_registry.async_remove(entity_id) | ||||||
|  |  | ||||||
|  |     known_devices: set[str] = set() | ||||||
|  |  | ||||||
|  |     def _check_device() -> None: | ||||||
|  |         current_devices = set(coordinator.data) | ||||||
|  |         new_devices = current_devices - known_devices | ||||||
|  |         if new_devices: | ||||||
|  |             known_devices.update(new_devices) | ||||||
|             async_add_entities( |             async_add_entities( | ||||||
|                 AmazonBinarySensorEntity(coordinator, serial_num, sensor_desc) |                 AmazonBinarySensorEntity(coordinator, serial_num, sensor_desc) | ||||||
|                 for sensor_desc in BINARY_SENSORS |                 for sensor_desc in BINARY_SENSORS | ||||||
|         for serial_num in coordinator.data |                 for serial_num in new_devices | ||||||
|         if sensor_desc.is_supported(coordinator.data[serial_num], sensor_desc.key) |                 if sensor_desc.is_supported( | ||||||
|  |                     coordinator.data[serial_num], sensor_desc.key | ||||||
|                 ) |                 ) | ||||||
|  |             ) | ||||||
|  |  | ||||||
|  |     _check_device() | ||||||
|  |     entry.async_on_unload(coordinator.async_add_listener(_check_device)) | ||||||
|  |  | ||||||
|  |  | ||||||
| class AmazonBinarySensorEntity(AmazonEntity, BinarySensorEntity): | class AmazonBinarySensorEntity(AmazonEntity, BinarySensorEntity): | ||||||
| @@ -113,3 +156,13 @@ class AmazonBinarySensorEntity(AmazonEntity, BinarySensorEntity): | |||||||
|         return self.entity_description.is_on_fn( |         return self.entity_description.is_on_fn( | ||||||
|             self.device, self.entity_description.key |             self.device, self.entity_description.key | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def available(self) -> bool: | ||||||
|  |         """Return if entity is available.""" | ||||||
|  |         return ( | ||||||
|  |             self.entity_description.is_available_fn( | ||||||
|  |                 self.device, self.entity_description.key | ||||||
|  |             ) | ||||||
|  |             and super().available | ||||||
|  |         ) | ||||||
|   | |||||||
| @@ -64,7 +64,7 @@ class AmazonDevicesConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|                 data = await validate_input(self.hass, user_input) |                 data = await validate_input(self.hass, user_input) | ||||||
|             except CannotConnect: |             except CannotConnect: | ||||||
|                 errors["base"] = "cannot_connect" |                 errors["base"] = "cannot_connect" | ||||||
|             except (CannotAuthenticate, TypeError): |             except CannotAuthenticate: | ||||||
|                 errors["base"] = "invalid_auth" |                 errors["base"] = "invalid_auth" | ||||||
|             except CannotRetrieveData: |             except CannotRetrieveData: | ||||||
|                 errors["base"] = "cannot_retrieve_data" |                 errors["base"] = "cannot_retrieve_data" | ||||||
| @@ -112,7 +112,7 @@ class AmazonDevicesConfigFlow(ConfigFlow, domain=DOMAIN): | |||||||
|                 ) |                 ) | ||||||
|             except CannotConnect: |             except CannotConnect: | ||||||
|                 errors["base"] = "cannot_connect" |                 errors["base"] = "cannot_connect" | ||||||
|             except (CannotAuthenticate, TypeError): |             except CannotAuthenticate: | ||||||
|                 errors["base"] = "invalid_auth" |                 errors["base"] = "invalid_auth" | ||||||
|             except CannotRetrieveData: |             except CannotRetrieveData: | ||||||
|                 errors["base"] = "cannot_retrieve_data" |                 errors["base"] = "cannot_retrieve_data" | ||||||
|   | |||||||
| @@ -68,7 +68,7 @@ class AmazonDevicesCoordinator(DataUpdateCoordinator[dict[str, AmazonDevice]]): | |||||||
|                 translation_key="cannot_retrieve_data_with_error", |                 translation_key="cannot_retrieve_data_with_error", | ||||||
|                 translation_placeholders={"error": repr(err)}, |                 translation_placeholders={"error": repr(err)}, | ||||||
|             ) from err |             ) from err | ||||||
|         except (CannotAuthenticate, TypeError) as err: |         except CannotAuthenticate as err: | ||||||
|             raise ConfigEntryAuthFailed( |             raise ConfigEntryAuthFailed( | ||||||
|                 translation_domain=DOMAIN, |                 translation_domain=DOMAIN, | ||||||
|                 translation_key="invalid_auth", |                 translation_key="invalid_auth", | ||||||
|   | |||||||
| @@ -60,7 +60,5 @@ def build_device_data(device: AmazonDevice) -> dict[str, Any]: | |||||||
|         "online": device.online, |         "online": device.online, | ||||||
|         "serial number": device.serial_number, |         "serial number": device.serial_number, | ||||||
|         "software version": device.software_version, |         "software version": device.software_version, | ||||||
|         "do not disturb": device.do_not_disturb, |         "sensors": device.sensors, | ||||||
|         "response style": device.response_style, |  | ||||||
|         "bluetooth state": device.bluetooth_state, |  | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,44 +1,4 @@ | |||||||
| { | { | ||||||
|   "entity": { |  | ||||||
|     "binary_sensor": { |  | ||||||
|       "bluetooth": { |  | ||||||
|         "default": "mdi:bluetooth-off", |  | ||||||
|         "state": { |  | ||||||
|           "on": "mdi:bluetooth" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "baby_cry_detection": { |  | ||||||
|         "default": "mdi:account-voice-off", |  | ||||||
|         "state": { |  | ||||||
|           "on": "mdi:account-voice" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "beeping_appliance_detection": { |  | ||||||
|         "default": "mdi:bell-off", |  | ||||||
|         "state": { |  | ||||||
|           "on": "mdi:bell-ring" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "cough_detection": { |  | ||||||
|         "default": "mdi:blur-off", |  | ||||||
|         "state": { |  | ||||||
|           "on": "mdi:blur" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "dog_bark_detection": { |  | ||||||
|         "default": "mdi:dog-side-off", |  | ||||||
|         "state": { |  | ||||||
|           "on": "mdi:dog-side" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "water_sounds_detection": { |  | ||||||
|         "default": "mdi:water-pump-off", |  | ||||||
|         "state": { |  | ||||||
|           "on": "mdi:water-pump" |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   }, |  | ||||||
|   "services": { |   "services": { | ||||||
|     "send_sound": { |     "send_sound": { | ||||||
|       "service": "mdi:cast-audio" |       "service": "mdi:cast-audio" | ||||||
|   | |||||||
| @@ -7,6 +7,6 @@ | |||||||
|   "integration_type": "hub", |   "integration_type": "hub", | ||||||
|   "iot_class": "cloud_polling", |   "iot_class": "cloud_polling", | ||||||
|   "loggers": ["aioamazondevices"], |   "loggers": ["aioamazondevices"], | ||||||
|   "quality_scale": "silver", |   "quality_scale": "platinum", | ||||||
|   "requirements": ["aioamazondevices==6.0.0"] |   "requirements": ["aioamazondevices==6.4.0"] | ||||||
| } | } | ||||||
|   | |||||||
| @@ -57,14 +57,24 @@ async def async_setup_entry( | |||||||
|  |  | ||||||
|     coordinator = entry.runtime_data |     coordinator = entry.runtime_data | ||||||
|  |  | ||||||
|  |     known_devices: set[str] = set() | ||||||
|  |  | ||||||
|  |     def _check_device() -> None: | ||||||
|  |         current_devices = set(coordinator.data) | ||||||
|  |         new_devices = current_devices - known_devices | ||||||
|  |         if new_devices: | ||||||
|  |             known_devices.update(new_devices) | ||||||
|             async_add_entities( |             async_add_entities( | ||||||
|                 AmazonNotifyEntity(coordinator, serial_num, sensor_desc) |                 AmazonNotifyEntity(coordinator, serial_num, sensor_desc) | ||||||
|                 for sensor_desc in NOTIFY |                 for sensor_desc in NOTIFY | ||||||
|         for serial_num in coordinator.data |                 for serial_num in new_devices | ||||||
|                 if sensor_desc.subkey in coordinator.data[serial_num].capabilities |                 if sensor_desc.subkey in coordinator.data[serial_num].capabilities | ||||||
|                 and sensor_desc.is_supported(coordinator.data[serial_num]) |                 and sensor_desc.is_supported(coordinator.data[serial_num]) | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|  |     _check_device() | ||||||
|  |     entry.async_on_unload(coordinator.async_add_listener(_check_device)) | ||||||
|  |  | ||||||
|  |  | ||||||
| class AmazonNotifyEntity(AmazonEntity, NotifyEntity): | class AmazonNotifyEntity(AmazonEntity, NotifyEntity): | ||||||
|     """Binary sensor notify platform.""" |     """Binary sensor notify platform.""" | ||||||
|   | |||||||
| @@ -53,7 +53,7 @@ rules: | |||||||
|   docs-supported-functions: done |   docs-supported-functions: done | ||||||
|   docs-troubleshooting: done |   docs-troubleshooting: done | ||||||
|   docs-use-cases: done |   docs-use-cases: done | ||||||
|   dynamic-devices: todo |   dynamic-devices: done | ||||||
|   entity-category: done |   entity-category: done | ||||||
|   entity-device-class: done |   entity-device-class: done | ||||||
|   entity-disabled-by-default: done |   entity-disabled-by-default: done | ||||||
|   | |||||||
| @@ -31,15 +31,20 @@ class AmazonSensorEntityDescription(SensorEntityDescription): | |||||||
|     """Amazon Devices sensor entity description.""" |     """Amazon Devices sensor entity description.""" | ||||||
|  |  | ||||||
|     native_unit_of_measurement_fn: Callable[[AmazonDevice, str], str] | None = None |     native_unit_of_measurement_fn: Callable[[AmazonDevice, str], str] | None = None | ||||||
|  |     is_available_fn: Callable[[AmazonDevice, str], bool] = lambda device, key: ( | ||||||
|  |         device.online | ||||||
|  |         and (sensor := device.sensors.get(key)) is not None | ||||||
|  |         and sensor.error is False | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
| SENSORS: Final = ( | SENSORS: Final = ( | ||||||
|     AmazonSensorEntityDescription( |     AmazonSensorEntityDescription( | ||||||
|         key="temperature", |         key="temperature", | ||||||
|         device_class=SensorDeviceClass.TEMPERATURE, |         device_class=SensorDeviceClass.TEMPERATURE, | ||||||
|         native_unit_of_measurement_fn=lambda device, _key: ( |         native_unit_of_measurement_fn=lambda device, key: ( | ||||||
|             UnitOfTemperature.CELSIUS |             UnitOfTemperature.CELSIUS | ||||||
|             if device.sensors[_key].scale == "CELSIUS" |             if key in device.sensors and device.sensors[key].scale == "CELSIUS" | ||||||
|             else UnitOfTemperature.FAHRENHEIT |             else UnitOfTemperature.FAHRENHEIT | ||||||
|         ), |         ), | ||||||
|         state_class=SensorStateClass.MEASUREMENT, |         state_class=SensorStateClass.MEASUREMENT, | ||||||
| @@ -62,13 +67,23 @@ async def async_setup_entry( | |||||||
|  |  | ||||||
|     coordinator = entry.runtime_data |     coordinator = entry.runtime_data | ||||||
|  |  | ||||||
|  |     known_devices: set[str] = set() | ||||||
|  |  | ||||||
|  |     def _check_device() -> None: | ||||||
|  |         current_devices = set(coordinator.data) | ||||||
|  |         new_devices = current_devices - known_devices | ||||||
|  |         if new_devices: | ||||||
|  |             known_devices.update(new_devices) | ||||||
|             async_add_entities( |             async_add_entities( | ||||||
|                 AmazonSensorEntity(coordinator, serial_num, sensor_desc) |                 AmazonSensorEntity(coordinator, serial_num, sensor_desc) | ||||||
|                 for sensor_desc in SENSORS |                 for sensor_desc in SENSORS | ||||||
|         for serial_num in coordinator.data |                 for serial_num in new_devices | ||||||
|                 if coordinator.data[serial_num].sensors.get(sensor_desc.key) is not None |                 if coordinator.data[serial_num].sensors.get(sensor_desc.key) is not None | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|  |     _check_device() | ||||||
|  |     entry.async_on_unload(coordinator.async_add_listener(_check_device)) | ||||||
|  |  | ||||||
|  |  | ||||||
| class AmazonSensorEntity(AmazonEntity, SensorEntity): | class AmazonSensorEntity(AmazonEntity, SensorEntity): | ||||||
|     """Sensor device.""" |     """Sensor device.""" | ||||||
| @@ -89,3 +104,13 @@ class AmazonSensorEntity(AmazonEntity, SensorEntity): | |||||||
|     def native_value(self) -> StateType: |     def native_value(self) -> StateType: | ||||||
|         """Return the state of the sensor.""" |         """Return the state of the sensor.""" | ||||||
|         return self.device.sensors[self.entity_description.key].value |         return self.device.sensors[self.entity_description.key].value | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def available(self) -> bool: | ||||||
|  |         """Return if entity is available.""" | ||||||
|  |         return ( | ||||||
|  |             self.entity_description.is_available_fn( | ||||||
|  |                 self.device, self.entity_description.key | ||||||
|  |             ) | ||||||
|  |             and super().available | ||||||
|  |         ) | ||||||
|   | |||||||
| @@ -58,26 +58,6 @@ | |||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   "entity": { |   "entity": { | ||||||
|     "binary_sensor": { |  | ||||||
|       "bluetooth": { |  | ||||||
|         "name": "Bluetooth" |  | ||||||
|       }, |  | ||||||
|       "baby_cry_detection": { |  | ||||||
|         "name": "Baby crying" |  | ||||||
|       }, |  | ||||||
|       "beeping_appliance_detection": { |  | ||||||
|         "name": "Beeping appliance" |  | ||||||
|       }, |  | ||||||
|       "cough_detection": { |  | ||||||
|         "name": "Coughing" |  | ||||||
|       }, |  | ||||||
|       "dog_bark_detection": { |  | ||||||
|         "name": "Dog barking" |  | ||||||
|       }, |  | ||||||
|       "water_sounds_detection": { |  | ||||||
|         "name": "Water sounds" |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|     "notify": { |     "notify": { | ||||||
|       "speak": { |       "speak": { | ||||||
|         "name": "Speak" |         "name": "Speak" | ||||||
|   | |||||||
| @@ -8,13 +8,21 @@ from typing import TYPE_CHECKING, Any, Final | |||||||
|  |  | ||||||
| from aioamazondevices.api import AmazonDevice | from aioamazondevices.api import AmazonDevice | ||||||
|  |  | ||||||
| from homeassistant.components.switch import SwitchEntity, SwitchEntityDescription | from homeassistant.components.switch import ( | ||||||
|  |     DOMAIN as SWITCH_DOMAIN, | ||||||
|  |     SwitchEntity, | ||||||
|  |     SwitchEntityDescription, | ||||||
|  | ) | ||||||
| from homeassistant.core import HomeAssistant | from homeassistant.core import HomeAssistant | ||||||
| from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | ||||||
|  |  | ||||||
| from .coordinator import AmazonConfigEntry | from .coordinator import AmazonConfigEntry | ||||||
| from .entity import AmazonEntity | from .entity import AmazonEntity | ||||||
| from .utils import alexa_api_call | from .utils import ( | ||||||
|  |     alexa_api_call, | ||||||
|  |     async_remove_dnd_from_virtual_group, | ||||||
|  |     async_update_unique_id, | ||||||
|  | ) | ||||||
|  |  | ||||||
| PARALLEL_UPDATES = 1 | PARALLEL_UPDATES = 1 | ||||||
|  |  | ||||||
| @@ -24,16 +32,19 @@ class AmazonSwitchEntityDescription(SwitchEntityDescription): | |||||||
|     """Alexa Devices switch entity description.""" |     """Alexa Devices switch entity description.""" | ||||||
|  |  | ||||||
|     is_on_fn: Callable[[AmazonDevice], bool] |     is_on_fn: Callable[[AmazonDevice], bool] | ||||||
|     subkey: str |     is_available_fn: Callable[[AmazonDevice, str], bool] = lambda device, key: ( | ||||||
|  |         device.online | ||||||
|  |         and (sensor := device.sensors.get(key)) is not None | ||||||
|  |         and sensor.error is False | ||||||
|  |     ) | ||||||
|     method: str |     method: str | ||||||
|  |  | ||||||
|  |  | ||||||
| SWITCHES: Final = ( | SWITCHES: Final = ( | ||||||
|     AmazonSwitchEntityDescription( |     AmazonSwitchEntityDescription( | ||||||
|         key="do_not_disturb", |         key="dnd", | ||||||
|         subkey="AUDIO_PLAYER", |  | ||||||
|         translation_key="do_not_disturb", |         translation_key="do_not_disturb", | ||||||
|         is_on_fn=lambda _device: _device.do_not_disturb, |         is_on_fn=lambda device: bool(device.sensors["dnd"].value), | ||||||
|         method="set_do_not_disturb", |         method="set_do_not_disturb", | ||||||
|     ), |     ), | ||||||
| ) | ) | ||||||
| @@ -48,13 +59,31 @@ async def async_setup_entry( | |||||||
|  |  | ||||||
|     coordinator = entry.runtime_data |     coordinator = entry.runtime_data | ||||||
|  |  | ||||||
|  |     # Replace unique id for "DND" switch and remove from Speaker Group | ||||||
|  |     await async_update_unique_id( | ||||||
|  |         hass, coordinator, SWITCH_DOMAIN, "do_not_disturb", "dnd" | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |     # Remove DND switch from virtual groups | ||||||
|  |     await async_remove_dnd_from_virtual_group(hass, coordinator) | ||||||
|  |  | ||||||
|  |     known_devices: set[str] = set() | ||||||
|  |  | ||||||
|  |     def _check_device() -> None: | ||||||
|  |         current_devices = set(coordinator.data) | ||||||
|  |         new_devices = current_devices - known_devices | ||||||
|  |         if new_devices: | ||||||
|  |             known_devices.update(new_devices) | ||||||
|             async_add_entities( |             async_add_entities( | ||||||
|                 AmazonSwitchEntity(coordinator, serial_num, switch_desc) |                 AmazonSwitchEntity(coordinator, serial_num, switch_desc) | ||||||
|                 for switch_desc in SWITCHES |                 for switch_desc in SWITCHES | ||||||
|         for serial_num in coordinator.data |                 for serial_num in new_devices | ||||||
|         if switch_desc.subkey in coordinator.data[serial_num].capabilities |                 if switch_desc.key in coordinator.data[serial_num].sensors | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|  |     _check_device() | ||||||
|  |     entry.async_on_unload(coordinator.async_add_listener(_check_device)) | ||||||
|  |  | ||||||
|  |  | ||||||
| class AmazonSwitchEntity(AmazonEntity, SwitchEntity): | class AmazonSwitchEntity(AmazonEntity, SwitchEntity): | ||||||
|     """Switch device.""" |     """Switch device.""" | ||||||
| @@ -84,3 +113,13 @@ class AmazonSwitchEntity(AmazonEntity, SwitchEntity): | |||||||
|     def is_on(self) -> bool: |     def is_on(self) -> bool: | ||||||
|         """Return True if switch is on.""" |         """Return True if switch is on.""" | ||||||
|         return self.entity_description.is_on_fn(self.device) |         return self.entity_description.is_on_fn(self.device) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def available(self) -> bool: | ||||||
|  |         """Return if entity is available.""" | ||||||
|  |         return ( | ||||||
|  |             self.entity_description.is_available_fn( | ||||||
|  |                 self.device, self.entity_description.key | ||||||
|  |             ) | ||||||
|  |             and super().available | ||||||
|  |         ) | ||||||
|   | |||||||
| @@ -4,11 +4,16 @@ from collections.abc import Awaitable, Callable, Coroutine | |||||||
| from functools import wraps | from functools import wraps | ||||||
| from typing import Any, Concatenate | from typing import Any, Concatenate | ||||||
|  |  | ||||||
|  | from aioamazondevices.const import SPEAKER_GROUP_FAMILY | ||||||
| from aioamazondevices.exceptions import CannotConnect, CannotRetrieveData | from aioamazondevices.exceptions import CannotConnect, CannotRetrieveData | ||||||
|  |  | ||||||
|  | from homeassistant.components.switch import DOMAIN as SWITCH_DOMAIN | ||||||
|  | from homeassistant.core import HomeAssistant | ||||||
| from homeassistant.exceptions import HomeAssistantError | from homeassistant.exceptions import HomeAssistantError | ||||||
|  | import homeassistant.helpers.entity_registry as er | ||||||
|  |  | ||||||
| from .const import DOMAIN | from .const import _LOGGER, DOMAIN | ||||||
|  | from .coordinator import AmazonDevicesCoordinator | ||||||
| from .entity import AmazonEntity | from .entity import AmazonEntity | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -38,3 +43,41 @@ def alexa_api_call[_T: AmazonEntity, **_P]( | |||||||
|             ) from err |             ) from err | ||||||
|  |  | ||||||
|     return cmd_wrapper |     return cmd_wrapper | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def async_update_unique_id( | ||||||
|  |     hass: HomeAssistant, | ||||||
|  |     coordinator: AmazonDevicesCoordinator, | ||||||
|  |     domain: str, | ||||||
|  |     old_key: str, | ||||||
|  |     new_key: str, | ||||||
|  | ) -> None: | ||||||
|  |     """Update unique id for entities created with old format.""" | ||||||
|  |     entity_registry = er.async_get(hass) | ||||||
|  |  | ||||||
|  |     for serial_num in coordinator.data: | ||||||
|  |         unique_id = f"{serial_num}-{old_key}" | ||||||
|  |         if entity_id := entity_registry.async_get_entity_id(domain, DOMAIN, unique_id): | ||||||
|  |             _LOGGER.debug("Updating unique_id for %s", entity_id) | ||||||
|  |             new_unique_id = unique_id.replace(old_key, new_key) | ||||||
|  |  | ||||||
|  |             # Update the registry with the new unique_id | ||||||
|  |             entity_registry.async_update_entity(entity_id, new_unique_id=new_unique_id) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def async_remove_dnd_from_virtual_group( | ||||||
|  |     hass: HomeAssistant, | ||||||
|  |     coordinator: AmazonDevicesCoordinator, | ||||||
|  | ) -> None: | ||||||
|  |     """Remove entity DND from virtual group.""" | ||||||
|  |     entity_registry = er.async_get(hass) | ||||||
|  |  | ||||||
|  |     for serial_num in coordinator.data: | ||||||
|  |         unique_id = f"{serial_num}-do_not_disturb" | ||||||
|  |         entity_id = entity_registry.async_get_entity_id( | ||||||
|  |             DOMAIN, SWITCH_DOMAIN, unique_id | ||||||
|  |         ) | ||||||
|  |         is_group = coordinator.data[serial_num].device_family == SPEAKER_GROUP_FAMILY | ||||||
|  |         if entity_id and is_group: | ||||||
|  |             entity_registry.async_remove(entity_id) | ||||||
|  |             _LOGGER.debug("Removed DND switch from virtual group %s", entity_id) | ||||||
|   | |||||||
| @@ -65,6 +65,31 @@ SENSOR_DESCRIPTIONS = [ | |||||||
|         suggested_display_precision=2, |         suggested_display_precision=2, | ||||||
|         translation_placeholders={"sensor_name": "BME280"}, |         translation_placeholders={"sensor_name": "BME280"}, | ||||||
|     ), |     ), | ||||||
|  |     AltruistSensorEntityDescription( | ||||||
|  |         device_class=SensorDeviceClass.HUMIDITY, | ||||||
|  |         key="BME680_humidity", | ||||||
|  |         translation_key="humidity", | ||||||
|  |         native_unit_of_measurement=PERCENTAGE, | ||||||
|  |         suggested_display_precision=2, | ||||||
|  |         translation_placeholders={"sensor_name": "BME680"}, | ||||||
|  |     ), | ||||||
|  |     AltruistSensorEntityDescription( | ||||||
|  |         device_class=SensorDeviceClass.PRESSURE, | ||||||
|  |         key="BME680_pressure", | ||||||
|  |         translation_key="pressure", | ||||||
|  |         native_unit_of_measurement=UnitOfPressure.PA, | ||||||
|  |         suggested_unit_of_measurement=UnitOfPressure.MMHG, | ||||||
|  |         suggested_display_precision=0, | ||||||
|  |         translation_placeholders={"sensor_name": "BME680"}, | ||||||
|  |     ), | ||||||
|  |     AltruistSensorEntityDescription( | ||||||
|  |         device_class=SensorDeviceClass.TEMPERATURE, | ||||||
|  |         key="BME680_temperature", | ||||||
|  |         translation_key="temperature", | ||||||
|  |         native_unit_of_measurement=UnitOfTemperature.CELSIUS, | ||||||
|  |         suggested_display_precision=2, | ||||||
|  |         translation_placeholders={"sensor_name": "BME680"}, | ||||||
|  |     ), | ||||||
|     AltruistSensorEntityDescription( |     AltruistSensorEntityDescription( | ||||||
|         device_class=SensorDeviceClass.PRESSURE, |         device_class=SensorDeviceClass.PRESSURE, | ||||||
|         key="BMP_pressure", |         key="BMP_pressure", | ||||||
|   | |||||||
| @@ -41,7 +41,7 @@ def async_setup_services(hass: HomeAssistant) -> None: | |||||||
|         if call.data.get(ATTR_ENTITY_ID) == ENTITY_MATCH_NONE: |         if call.data.get(ATTR_ENTITY_ID) == ENTITY_MATCH_NONE: | ||||||
|             return [] |             return [] | ||||||
|  |  | ||||||
|         call_ids = await async_extract_entity_ids(hass, call) |         call_ids = await async_extract_entity_ids(call) | ||||||
|         entity_ids = [] |         entity_ids = [] | ||||||
|         for entity_id in hass.data[DATA_AMCREST][CAMERAS]: |         for entity_id in hass.data[DATA_AMCREST][CAMERAS]: | ||||||
|             if entity_id not in call_ids: |             if entity_id not in call_ids: | ||||||
|   | |||||||
| @@ -12,10 +12,25 @@ from homeassistant.helpers.event import async_call_later, async_track_time_inter | |||||||
| from homeassistant.helpers.typing import ConfigType | from homeassistant.helpers.typing import ConfigType | ||||||
| from homeassistant.util.hass_dict import HassKey | from homeassistant.util.hass_dict import HassKey | ||||||
|  |  | ||||||
| from .analytics import Analytics | from .analytics import ( | ||||||
|  |     Analytics, | ||||||
|  |     AnalyticsInput, | ||||||
|  |     AnalyticsModifications, | ||||||
|  |     DeviceAnalyticsModifications, | ||||||
|  |     EntityAnalyticsModifications, | ||||||
|  |     async_devices_payload, | ||||||
|  | ) | ||||||
| from .const import ATTR_ONBOARDED, ATTR_PREFERENCES, DOMAIN, INTERVAL, PREFERENCE_SCHEMA | from .const import ATTR_ONBOARDED, ATTR_PREFERENCES, DOMAIN, INTERVAL, PREFERENCE_SCHEMA | ||||||
| from .http import AnalyticsDevicesView | from .http import AnalyticsDevicesView | ||||||
|  |  | ||||||
|  | __all__ = [ | ||||||
|  |     "AnalyticsInput", | ||||||
|  |     "AnalyticsModifications", | ||||||
|  |     "DeviceAnalyticsModifications", | ||||||
|  |     "EntityAnalyticsModifications", | ||||||
|  |     "async_devices_payload", | ||||||
|  | ] | ||||||
|  |  | ||||||
| CONFIG_SCHEMA = cv.empty_config_schema(DOMAIN) | CONFIG_SCHEMA = cv.empty_config_schema(DOMAIN) | ||||||
|  |  | ||||||
| DATA_COMPONENT: HassKey[Analytics] = HassKey(DOMAIN) | DATA_COMPONENT: HassKey[Analytics] = HassKey(DOMAIN) | ||||||
|   | |||||||
| @@ -4,9 +4,10 @@ from __future__ import annotations | |||||||
|  |  | ||||||
| import asyncio | import asyncio | ||||||
| from asyncio import timeout | from asyncio import timeout | ||||||
| from dataclasses import asdict as dataclass_asdict, dataclass | from collections.abc import Awaitable, Callable, Iterable, Mapping | ||||||
|  | from dataclasses import asdict as dataclass_asdict, dataclass, field | ||||||
| from datetime import datetime | from datetime import datetime | ||||||
| from typing import Any | from typing import Any, Protocol | ||||||
| import uuid | import uuid | ||||||
|  |  | ||||||
| import aiohttp | import aiohttp | ||||||
| @@ -35,11 +36,14 @@ from homeassistant.exceptions import HomeAssistantError | |||||||
| from homeassistant.helpers import device_registry as dr, entity_registry as er | from homeassistant.helpers import device_registry as dr, entity_registry as er | ||||||
| from homeassistant.helpers.aiohttp_client import async_get_clientsession | from homeassistant.helpers.aiohttp_client import async_get_clientsession | ||||||
| from homeassistant.helpers.hassio import is_hassio | from homeassistant.helpers.hassio import is_hassio | ||||||
|  | from homeassistant.helpers.singleton import singleton | ||||||
| from homeassistant.helpers.storage import Store | from homeassistant.helpers.storage import Store | ||||||
| from homeassistant.helpers.system_info import async_get_system_info | from homeassistant.helpers.system_info import async_get_system_info | ||||||
|  | from homeassistant.helpers.typing import UNDEFINED | ||||||
| from homeassistant.loader import ( | from homeassistant.loader import ( | ||||||
|     Integration, |     Integration, | ||||||
|     IntegrationNotFound, |     IntegrationNotFound, | ||||||
|  |     async_get_integration, | ||||||
|     async_get_integrations, |     async_get_integrations, | ||||||
| ) | ) | ||||||
| from homeassistant.setup import async_get_loaded_integrations | from homeassistant.setup import async_get_loaded_integrations | ||||||
| @@ -75,12 +79,115 @@ from .const import ( | |||||||
|     ATTR_USER_COUNT, |     ATTR_USER_COUNT, | ||||||
|     ATTR_UUID, |     ATTR_UUID, | ||||||
|     ATTR_VERSION, |     ATTR_VERSION, | ||||||
|  |     DOMAIN, | ||||||
|     LOGGER, |     LOGGER, | ||||||
|     PREFERENCE_SCHEMA, |     PREFERENCE_SCHEMA, | ||||||
|     STORAGE_KEY, |     STORAGE_KEY, | ||||||
|     STORAGE_VERSION, |     STORAGE_VERSION, | ||||||
| ) | ) | ||||||
|  |  | ||||||
|  | DATA_ANALYTICS_MODIFIERS = "analytics_modifiers" | ||||||
|  |  | ||||||
|  | type AnalyticsModifier = Callable[ | ||||||
|  |     [HomeAssistant, AnalyticsInput], Awaitable[AnalyticsModifications] | ||||||
|  | ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @singleton(DATA_ANALYTICS_MODIFIERS) | ||||||
|  | def _async_get_modifiers( | ||||||
|  |     hass: HomeAssistant, | ||||||
|  | ) -> dict[str, AnalyticsModifier | None]: | ||||||
|  |     """Return the analytics modifiers.""" | ||||||
|  |     return {} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass | ||||||
|  | class AnalyticsInput: | ||||||
|  |     """Analytics input for a single integration. | ||||||
|  |  | ||||||
|  |     This is sent to integrations that implement the platform. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     device_ids: Iterable[str] = field(default_factory=list) | ||||||
|  |     entity_ids: Iterable[str] = field(default_factory=list) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass | ||||||
|  | class AnalyticsModifications: | ||||||
|  |     """Analytics config for a single integration. | ||||||
|  |  | ||||||
|  |     This is used by integrations that implement the platform. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     remove: bool = False | ||||||
|  |     devices: Mapping[str, DeviceAnalyticsModifications] | None = None | ||||||
|  |     entities: Mapping[str, EntityAnalyticsModifications] | None = None | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass | ||||||
|  | class DeviceAnalyticsModifications: | ||||||
|  |     """Analytics config for a single device. | ||||||
|  |  | ||||||
|  |     This is used by integrations that implement the platform. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     remove: bool = False | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @dataclass | ||||||
|  | class EntityAnalyticsModifications: | ||||||
|  |     """Analytics config for a single entity. | ||||||
|  |  | ||||||
|  |     This is used by integrations that implement the platform. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     remove: bool = False | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AnalyticsPlatformProtocol(Protocol): | ||||||
|  |     """Define the format of analytics platforms.""" | ||||||
|  |  | ||||||
|  |     async def async_modify_analytics( | ||||||
|  |         self, | ||||||
|  |         hass: HomeAssistant, | ||||||
|  |         analytics_input: AnalyticsInput, | ||||||
|  |     ) -> AnalyticsModifications: | ||||||
|  |         """Modify the analytics.""" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def _async_get_analytics_platform( | ||||||
|  |     hass: HomeAssistant, domain: str | ||||||
|  | ) -> AnalyticsPlatformProtocol | None: | ||||||
|  |     """Get analytics platform.""" | ||||||
|  |     try: | ||||||
|  |         integration = await async_get_integration(hass, domain) | ||||||
|  |     except IntegrationNotFound: | ||||||
|  |         return None | ||||||
|  |     try: | ||||||
|  |         return await integration.async_get_platform(DOMAIN) | ||||||
|  |     except ImportError: | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def _async_get_modifier( | ||||||
|  |     hass: HomeAssistant, domain: str | ||||||
|  | ) -> AnalyticsModifier | None: | ||||||
|  |     """Get analytics modifier.""" | ||||||
|  |     modifiers = _async_get_modifiers(hass) | ||||||
|  |     modifier = modifiers.get(domain, UNDEFINED) | ||||||
|  |  | ||||||
|  |     if modifier is not UNDEFINED: | ||||||
|  |         return modifier | ||||||
|  |  | ||||||
|  |     platform = await _async_get_analytics_platform(hass, domain) | ||||||
|  |     if platform is None: | ||||||
|  |         modifiers[domain] = None | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |     modifier = getattr(platform, "async_modify_analytics", None) | ||||||
|  |     modifiers[domain] = modifier | ||||||
|  |     return modifier | ||||||
|  |  | ||||||
|  |  | ||||||
| def gen_uuid() -> str: | def gen_uuid() -> str: | ||||||
|     """Generate a new UUID.""" |     """Generate a new UUID.""" | ||||||
| @@ -393,17 +500,22 @@ def _domains_from_yaml_config(yaml_configuration: dict[str, Any]) -> set[str]: | |||||||
|     return domains |     return domains | ||||||
|  |  | ||||||
|  |  | ||||||
| async def async_devices_payload(hass: HomeAssistant) -> dict: | DEFAULT_ANALYTICS_CONFIG = AnalyticsModifications() | ||||||
|  | DEFAULT_DEVICE_ANALYTICS_CONFIG = DeviceAnalyticsModifications() | ||||||
|  | DEFAULT_ENTITY_ANALYTICS_CONFIG = EntityAnalyticsModifications() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def async_devices_payload(hass: HomeAssistant) -> dict:  # noqa: C901 | ||||||
|     """Return detailed information about entities and devices.""" |     """Return detailed information about entities and devices.""" | ||||||
|     integrations_info: dict[str, dict[str, Any]] = {} |  | ||||||
|  |  | ||||||
|     dev_reg = dr.async_get(hass) |     dev_reg = dr.async_get(hass) | ||||||
|  |     ent_reg = er.async_get(hass) | ||||||
|  |  | ||||||
|     # We need to refer to other devices, for example in `via_device` field. |     integration_inputs: dict[str, tuple[list[str], list[str]]] = {} | ||||||
|     # We don't however send the original device ids outside of Home Assistant, |     integration_configs: dict[str, AnalyticsModifications] = {} | ||||||
|     # instead we refer to devices by (integration_domain, index_in_integration_device_list). |  | ||||||
|     device_id_mapping: dict[str, tuple[str, int]] = {} |  | ||||||
|  |  | ||||||
|  |     removed_devices: set[str] = set() | ||||||
|  |  | ||||||
|  |     # Get device list | ||||||
|     for device_entry in dev_reg.devices.values(): |     for device_entry in dev_reg.devices.values(): | ||||||
|         if not device_entry.primary_config_entry: |         if not device_entry.primary_config_entry: | ||||||
|             continue |             continue | ||||||
| @@ -415,18 +527,108 @@ async def async_devices_payload(hass: HomeAssistant) -> dict: | |||||||
|         if config_entry is None: |         if config_entry is None: | ||||||
|             continue |             continue | ||||||
|  |  | ||||||
|  |         if device_entry.entry_type is dr.DeviceEntryType.SERVICE: | ||||||
|  |             removed_devices.add(device_entry.id) | ||||||
|  |             continue | ||||||
|  |  | ||||||
|         integration_domain = config_entry.domain |         integration_domain = config_entry.domain | ||||||
|  |  | ||||||
|  |         integration_input = integration_inputs.setdefault(integration_domain, ([], [])) | ||||||
|  |         integration_input[0].append(device_entry.id) | ||||||
|  |  | ||||||
|  |     # Get entity list | ||||||
|  |     for entity_entry in ent_reg.entities.values(): | ||||||
|  |         integration_domain = entity_entry.platform | ||||||
|  |  | ||||||
|  |         integration_input = integration_inputs.setdefault(integration_domain, ([], [])) | ||||||
|  |         integration_input[1].append(entity_entry.entity_id) | ||||||
|  |  | ||||||
|  |     integrations = { | ||||||
|  |         domain: integration | ||||||
|  |         for domain, integration in ( | ||||||
|  |             await async_get_integrations(hass, integration_inputs.keys()) | ||||||
|  |         ).items() | ||||||
|  |         if isinstance(integration, Integration) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     # Filter out custom integrations and integrations that are not device or hub type | ||||||
|  |     integration_inputs = { | ||||||
|  |         domain: integration_info | ||||||
|  |         for domain, integration_info in integration_inputs.items() | ||||||
|  |         if (integration := integrations.get(domain)) is not None | ||||||
|  |         and integration.is_built_in | ||||||
|  |         and integration.manifest.get("integration_type") in ("device", "hub") | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     # Call integrations that implement the analytics platform | ||||||
|  |     for integration_domain, integration_input in integration_inputs.items(): | ||||||
|  |         if ( | ||||||
|  |             modifier := await _async_get_modifier(hass, integration_domain) | ||||||
|  |         ) is not None: | ||||||
|  |             try: | ||||||
|  |                 integration_config = await modifier( | ||||||
|  |                     hass, AnalyticsInput(*integration_input) | ||||||
|  |                 ) | ||||||
|  |             except Exception as err:  # noqa: BLE001 | ||||||
|  |                 LOGGER.exception( | ||||||
|  |                     "Calling async_modify_analytics for integration '%s' failed: %s", | ||||||
|  |                     integration_domain, | ||||||
|  |                     err, | ||||||
|  |                 ) | ||||||
|  |                 integration_configs[integration_domain] = AnalyticsModifications( | ||||||
|  |                     remove=True | ||||||
|  |                 ) | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             if not isinstance(integration_config, AnalyticsModifications): | ||||||
|  |                 LOGGER.error(  # type: ignore[unreachable] | ||||||
|  |                     "Calling async_modify_analytics for integration '%s' did not return an AnalyticsConfig", | ||||||
|  |                     integration_domain, | ||||||
|  |                 ) | ||||||
|  |                 integration_configs[integration_domain] = AnalyticsModifications( | ||||||
|  |                     remove=True | ||||||
|  |                 ) | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             integration_configs[integration_domain] = integration_config | ||||||
|  |  | ||||||
|  |     integrations_info: dict[str, dict[str, Any]] = {} | ||||||
|  |  | ||||||
|  |     # We need to refer to other devices, for example in `via_device` field. | ||||||
|  |     # We don't however send the original device ids outside of Home Assistant, | ||||||
|  |     # instead we refer to devices by (integration_domain, index_in_integration_device_list). | ||||||
|  |     device_id_mapping: dict[str, tuple[str, int]] = {} | ||||||
|  |  | ||||||
|  |     # Fill out information about devices | ||||||
|  |     for integration_domain, integration_input in integration_inputs.items(): | ||||||
|  |         integration_config = integration_configs.get( | ||||||
|  |             integration_domain, DEFAULT_ANALYTICS_CONFIG | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         if integration_config.remove: | ||||||
|  |             continue | ||||||
|  |  | ||||||
|         integration_info = integrations_info.setdefault( |         integration_info = integrations_info.setdefault( | ||||||
|             integration_domain, {"devices": [], "entities": []} |             integration_domain, {"devices": [], "entities": []} | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         devices_info = integration_info["devices"] |         devices_info = integration_info["devices"] | ||||||
|  |  | ||||||
|         device_id_mapping[device_entry.id] = (integration_domain, len(devices_info)) |         for device_id in integration_input[0]: | ||||||
|  |             device_config = DEFAULT_DEVICE_ANALYTICS_CONFIG | ||||||
|  |             if integration_config.devices is not None: | ||||||
|  |                 device_config = integration_config.devices.get(device_id, device_config) | ||||||
|  |  | ||||||
|  |             if device_config.remove: | ||||||
|  |                 removed_devices.add(device_id) | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             device_entry = dev_reg.devices[device_id] | ||||||
|  |  | ||||||
|  |             device_id_mapping[device_id] = (integration_domain, len(devices_info)) | ||||||
|  |  | ||||||
|             devices_info.append( |             devices_info.append( | ||||||
|                 { |                 { | ||||||
|                 "entities": [], |  | ||||||
|                     "entry_type": device_entry.entry_type, |                     "entry_type": device_entry.entry_type, | ||||||
|                     "has_configuration_url": device_entry.configuration_url is not None, |                     "has_configuration_url": device_entry.configuration_url is not None, | ||||||
|                     "hw_version": device_entry.hw_version, |                     "hw_version": device_entry.hw_version, | ||||||
| @@ -435,6 +637,7 @@ async def async_devices_payload(hass: HomeAssistant) -> dict: | |||||||
|                     "model_id": device_entry.model_id, |                     "model_id": device_entry.model_id, | ||||||
|                     "sw_version": device_entry.sw_version, |                     "sw_version": device_entry.sw_version, | ||||||
|                     "via_device": device_entry.via_device_id, |                     "via_device": device_entry.via_device_id, | ||||||
|  |                     "entities": [], | ||||||
|                 } |                 } | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
| @@ -445,10 +648,15 @@ async def async_devices_payload(hass: HomeAssistant) -> dict: | |||||||
|                 continue |                 continue | ||||||
|             device_info["via_device"] = device_id_mapping.get(device_info["via_device"]) |             device_info["via_device"] = device_id_mapping.get(device_info["via_device"]) | ||||||
|  |  | ||||||
|     ent_reg = er.async_get(hass) |     # Fill out information about entities | ||||||
|  |     for integration_domain, integration_input in integration_inputs.items(): | ||||||
|  |         integration_config = integration_configs.get( | ||||||
|  |             integration_domain, DEFAULT_ANALYTICS_CONFIG | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         if integration_config.remove: | ||||||
|  |             continue | ||||||
|  |  | ||||||
|     for entity_entry in ent_reg.entities.values(): |  | ||||||
|         integration_domain = entity_entry.platform |  | ||||||
|         integration_info = integrations_info.setdefault( |         integration_info = integrations_info.setdefault( | ||||||
|             integration_domain, {"devices": [], "entities": []} |             integration_domain, {"devices": [], "entities": []} | ||||||
|         ) |         ) | ||||||
| @@ -456,17 +664,30 @@ async def async_devices_payload(hass: HomeAssistant) -> dict: | |||||||
|         devices_info = integration_info["devices"] |         devices_info = integration_info["devices"] | ||||||
|         entities_info = integration_info["entities"] |         entities_info = integration_info["entities"] | ||||||
|  |  | ||||||
|         entity_state = hass.states.get(entity_entry.entity_id) |         for entity_id in integration_input[1]: | ||||||
|  |             entity_config = DEFAULT_ENTITY_ANALYTICS_CONFIG | ||||||
|  |             if integration_config.entities is not None: | ||||||
|  |                 entity_config = integration_config.entities.get( | ||||||
|  |                     entity_id, entity_config | ||||||
|  |                 ) | ||||||
|  |  | ||||||
|  |             if entity_config.remove: | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|  |             entity_entry = ent_reg.entities[entity_id] | ||||||
|  |  | ||||||
|  |             entity_state = hass.states.get(entity_id) | ||||||
|  |  | ||||||
|             entity_info = { |             entity_info = { | ||||||
|                 # LIMITATION: `assumed_state` can be overridden by users; |                 # LIMITATION: `assumed_state` can be overridden by users; | ||||||
|                 # we should replace it with the original value in the future. |                 # we should replace it with the original value in the future. | ||||||
|                 # It is also not present, if entity is not in the state machine, |                 # It is also not present, if entity is not in the state machine, | ||||||
|                 # which can happen for disabled entities. |                 # which can happen for disabled entities. | ||||||
|             "assumed_state": entity_state.attributes.get(ATTR_ASSUMED_STATE, False) |                 "assumed_state": ( | ||||||
|  |                     entity_state.attributes.get(ATTR_ASSUMED_STATE, False) | ||||||
|                     if entity_state is not None |                     if entity_state is not None | ||||||
|             else None, |                     else None | ||||||
|             "capabilities": entity_entry.capabilities, |                 ), | ||||||
|                 "domain": entity_entry.domain, |                 "domain": entity_entry.domain, | ||||||
|                 "entity_category": entity_entry.entity_category, |                 "entity_category": entity_entry.entity_category, | ||||||
|                 "has_entity_name": entity_entry.has_entity_name, |                 "has_entity_name": entity_entry.has_entity_name, | ||||||
| @@ -476,33 +697,20 @@ async def async_devices_payload(hass: HomeAssistant) -> dict: | |||||||
|                 "unit_of_measurement": entity_entry.unit_of_measurement, |                 "unit_of_measurement": entity_entry.unit_of_measurement, | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  |             if (device_id_ := entity_entry.device_id) is not None: | ||||||
|  |                 if device_id_ in removed_devices: | ||||||
|  |                     # The device was removed, so we remove the entity too | ||||||
|  |                     continue | ||||||
|  |  | ||||||
|                 if ( |                 if ( | ||||||
|             ((device_id := entity_entry.device_id) is not None) |                     new_device_id := device_id_mapping.get(device_id_) | ||||||
|             and ((new_device_id := device_id_mapping.get(device_id)) is not None) |                 ) is not None and (new_device_id[0] == integration_domain): | ||||||
|             and (new_device_id[0] == integration_domain) |  | ||||||
|         ): |  | ||||||
|                     device_info = devices_info[new_device_id[1]] |                     device_info = devices_info[new_device_id[1]] | ||||||
|                     device_info["entities"].append(entity_info) |                     device_info["entities"].append(entity_info) | ||||||
|         else: |                     continue | ||||||
|  |  | ||||||
|             entities_info.append(entity_info) |             entities_info.append(entity_info) | ||||||
|  |  | ||||||
|     integrations = { |  | ||||||
|         domain: integration |  | ||||||
|         for domain, integration in ( |  | ||||||
|             await async_get_integrations(hass, integrations_info.keys()) |  | ||||||
|         ).items() |  | ||||||
|         if isinstance(integration, Integration) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     for domain, integration_info in integrations_info.items(): |  | ||||||
|         if integration := integrations.get(domain): |  | ||||||
|             integration_info["is_custom_integration"] = not integration.is_built_in |  | ||||||
|             # Include version for custom integrations |  | ||||||
|             if not integration.is_built_in and integration.version: |  | ||||||
|                 integration_info["custom_integration_version"] = str( |  | ||||||
|                     integration.version |  | ||||||
|                 ) |  | ||||||
|  |  | ||||||
|     return { |     return { | ||||||
|         "version": "home-assistant:1", |         "version": "home-assistant:1", | ||||||
|         "home_assistant": HA_VERSION, |         "home_assistant": HA_VERSION, | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
|   "domain": "analytics", |   "domain": "analytics", | ||||||
|   "name": "Analytics", |   "name": "Analytics", | ||||||
|   "after_dependencies": ["energy", "hassio", "recorder"], |   "after_dependencies": ["energy", "hassio", "recorder"], | ||||||
|   "codeowners": ["@home-assistant/core", "@ludeeus"], |   "codeowners": ["@home-assistant/core"], | ||||||
|   "dependencies": ["api", "websocket_api", "http"], |   "dependencies": ["api", "websocket_api", "http"], | ||||||
|   "documentation": "https://www.home-assistant.io/integrations/analytics", |   "documentation": "https://www.home-assistant.io/integrations/analytics", | ||||||
|   "integration_type": "system", |   "integration_type": "system", | ||||||
|   | |||||||
| @@ -33,9 +33,11 @@ from homeassistant.const import ( | |||||||
| ) | ) | ||||||
| from homeassistant.core import Event, HomeAssistant | from homeassistant.core import Event, HomeAssistant | ||||||
| from homeassistant.exceptions import ConfigEntryNotReady | from homeassistant.exceptions import ConfigEntryNotReady | ||||||
|  | from homeassistant.helpers import config_validation as cv | ||||||
| from homeassistant.helpers.device_registry import format_mac | from homeassistant.helpers.device_registry import format_mac | ||||||
| from homeassistant.helpers.dispatcher import async_dispatcher_send | from homeassistant.helpers.dispatcher import async_dispatcher_send | ||||||
| from homeassistant.helpers.storage import STORAGE_DIR | from homeassistant.helpers.storage import STORAGE_DIR | ||||||
|  | from homeassistant.helpers.typing import ConfigType | ||||||
|  |  | ||||||
| from .const import ( | from .const import ( | ||||||
|     CONF_ADB_SERVER_IP, |     CONF_ADB_SERVER_IP, | ||||||
| @@ -46,10 +48,12 @@ from .const import ( | |||||||
|     DEFAULT_ADB_SERVER_PORT, |     DEFAULT_ADB_SERVER_PORT, | ||||||
|     DEVICE_ANDROIDTV, |     DEVICE_ANDROIDTV, | ||||||
|     DEVICE_FIRETV, |     DEVICE_FIRETV, | ||||||
|  |     DOMAIN, | ||||||
|     PROP_ETHMAC, |     PROP_ETHMAC, | ||||||
|     PROP_WIFIMAC, |     PROP_WIFIMAC, | ||||||
|     SIGNAL_CONFIG_ENTITY, |     SIGNAL_CONFIG_ENTITY, | ||||||
| ) | ) | ||||||
|  | from .services import async_setup_services | ||||||
|  |  | ||||||
| ADB_PYTHON_EXCEPTIONS: tuple = ( | ADB_PYTHON_EXCEPTIONS: tuple = ( | ||||||
|     AdbTimeoutError, |     AdbTimeoutError, | ||||||
| @@ -63,6 +67,8 @@ ADB_PYTHON_EXCEPTIONS: tuple = ( | |||||||
| ) | ) | ||||||
| ADB_TCP_EXCEPTIONS: tuple = (ConnectionResetError, RuntimeError) | ADB_TCP_EXCEPTIONS: tuple = (ConnectionResetError, RuntimeError) | ||||||
|  |  | ||||||
|  | CONFIG_SCHEMA = cv.config_entry_only_config_schema(DOMAIN) | ||||||
|  |  | ||||||
| PLATFORMS = [Platform.MEDIA_PLAYER, Platform.REMOTE] | PLATFORMS = [Platform.MEDIA_PLAYER, Platform.REMOTE] | ||||||
| RELOAD_OPTIONS = [CONF_STATE_DETECTION_RULES] | RELOAD_OPTIONS = [CONF_STATE_DETECTION_RULES] | ||||||
|  |  | ||||||
| @@ -188,6 +194,12 @@ async def async_migrate_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: | |||||||
|     return True |     return True | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool: | ||||||
|  |     """Set up the Android TV / Fire TV integration.""" | ||||||
|  |     async_setup_services(hass) | ||||||
|  |     return True | ||||||
|  |  | ||||||
|  |  | ||||||
| async def async_setup_entry(hass: HomeAssistant, entry: AndroidTVConfigEntry) -> bool: | async def async_setup_entry(hass: HomeAssistant, entry: AndroidTVConfigEntry) -> bool: | ||||||
|     """Set up Android Debug Bridge platform.""" |     """Set up Android Debug Bridge platform.""" | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,7 +8,6 @@ import logging | |||||||
|  |  | ||||||
| from androidtv.constants import APPS, KEYS | from androidtv.constants import APPS, KEYS | ||||||
| from androidtv.setup_async import AndroidTVAsync, FireTVAsync | from androidtv.setup_async import AndroidTVAsync, FireTVAsync | ||||||
| import voluptuous as vol |  | ||||||
|  |  | ||||||
| from homeassistant.components import persistent_notification | from homeassistant.components import persistent_notification | ||||||
| from homeassistant.components.media_player import ( | from homeassistant.components.media_player import ( | ||||||
| @@ -17,9 +16,7 @@ from homeassistant.components.media_player import ( | |||||||
|     MediaPlayerEntityFeature, |     MediaPlayerEntityFeature, | ||||||
|     MediaPlayerState, |     MediaPlayerState, | ||||||
| ) | ) | ||||||
| from homeassistant.const import ATTR_COMMAND |  | ||||||
| from homeassistant.core import HomeAssistant | from homeassistant.core import HomeAssistant | ||||||
| from homeassistant.helpers import config_validation as cv, entity_platform |  | ||||||
| from homeassistant.helpers.dispatcher import async_dispatcher_connect | from homeassistant.helpers.dispatcher import async_dispatcher_connect | ||||||
| from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | ||||||
| from homeassistant.util.dt import utcnow | from homeassistant.util.dt import utcnow | ||||||
| @@ -39,19 +36,10 @@ from .const import ( | |||||||
|     SIGNAL_CONFIG_ENTITY, |     SIGNAL_CONFIG_ENTITY, | ||||||
| ) | ) | ||||||
| from .entity import AndroidTVEntity, adb_decorator | from .entity import AndroidTVEntity, adb_decorator | ||||||
|  | from .services import ATTR_ADB_RESPONSE, ATTR_HDMI_INPUT, SERVICE_LEARN_SENDEVENT | ||||||
|  |  | ||||||
| _LOGGER = logging.getLogger(__name__) | _LOGGER = logging.getLogger(__name__) | ||||||
|  |  | ||||||
| ATTR_ADB_RESPONSE = "adb_response" |  | ||||||
| ATTR_DEVICE_PATH = "device_path" |  | ||||||
| ATTR_HDMI_INPUT = "hdmi_input" |  | ||||||
| ATTR_LOCAL_PATH = "local_path" |  | ||||||
|  |  | ||||||
| SERVICE_ADB_COMMAND = "adb_command" |  | ||||||
| SERVICE_DOWNLOAD = "download" |  | ||||||
| SERVICE_LEARN_SENDEVENT = "learn_sendevent" |  | ||||||
| SERVICE_UPLOAD = "upload" |  | ||||||
|  |  | ||||||
| # Translate from `AndroidTV` / `FireTV` reported state to HA state. | # Translate from `AndroidTV` / `FireTV` reported state to HA state. | ||||||
| ANDROIDTV_STATES = { | ANDROIDTV_STATES = { | ||||||
|     "off": MediaPlayerState.OFF, |     "off": MediaPlayerState.OFF, | ||||||
| @@ -77,32 +65,6 @@ async def async_setup_entry( | |||||||
|         ] |         ] | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     platform = entity_platform.async_get_current_platform() |  | ||||||
|     platform.async_register_entity_service( |  | ||||||
|         SERVICE_ADB_COMMAND, |  | ||||||
|         {vol.Required(ATTR_COMMAND): cv.string}, |  | ||||||
|         "adb_command", |  | ||||||
|     ) |  | ||||||
|     platform.async_register_entity_service( |  | ||||||
|         SERVICE_LEARN_SENDEVENT, None, "learn_sendevent" |  | ||||||
|     ) |  | ||||||
|     platform.async_register_entity_service( |  | ||||||
|         SERVICE_DOWNLOAD, |  | ||||||
|         { |  | ||||||
|             vol.Required(ATTR_DEVICE_PATH): cv.string, |  | ||||||
|             vol.Required(ATTR_LOCAL_PATH): cv.string, |  | ||||||
|         }, |  | ||||||
|         "service_download", |  | ||||||
|     ) |  | ||||||
|     platform.async_register_entity_service( |  | ||||||
|         SERVICE_UPLOAD, |  | ||||||
|         { |  | ||||||
|             vol.Required(ATTR_DEVICE_PATH): cv.string, |  | ||||||
|             vol.Required(ATTR_LOCAL_PATH): cv.string, |  | ||||||
|         }, |  | ||||||
|         "service_upload", |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class ADBDevice(AndroidTVEntity, MediaPlayerEntity): | class ADBDevice(AndroidTVEntity, MediaPlayerEntity): | ||||||
|     """Representation of an Android or Fire TV device.""" |     """Representation of an Android or Fire TV device.""" | ||||||
|   | |||||||
							
								
								
									
										66
									
								
								homeassistant/components/androidtv/services.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								homeassistant/components/androidtv/services.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | |||||||
|  | """Services for Android/Fire TV devices.""" | ||||||
|  |  | ||||||
|  | from __future__ import annotations | ||||||
|  |  | ||||||
|  | import voluptuous as vol | ||||||
|  |  | ||||||
|  | from homeassistant.components.media_player import DOMAIN as MEDIA_PLAYER_DOMAIN | ||||||
|  | from homeassistant.const import ATTR_COMMAND | ||||||
|  | from homeassistant.core import HomeAssistant, callback | ||||||
|  | from homeassistant.helpers import config_validation as cv, service | ||||||
|  |  | ||||||
|  | from .const import DOMAIN | ||||||
|  |  | ||||||
|  | ATTR_ADB_RESPONSE = "adb_response" | ||||||
|  | ATTR_DEVICE_PATH = "device_path" | ||||||
|  | ATTR_HDMI_INPUT = "hdmi_input" | ||||||
|  | ATTR_LOCAL_PATH = "local_path" | ||||||
|  |  | ||||||
|  | SERVICE_ADB_COMMAND = "adb_command" | ||||||
|  | SERVICE_DOWNLOAD = "download" | ||||||
|  | SERVICE_LEARN_SENDEVENT = "learn_sendevent" | ||||||
|  | SERVICE_UPLOAD = "upload" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @callback | ||||||
|  | def async_setup_services(hass: HomeAssistant) -> None: | ||||||
|  |     """Register the Android TV / Fire TV services.""" | ||||||
|  |  | ||||||
|  |     service.async_register_platform_entity_service( | ||||||
|  |         hass, | ||||||
|  |         DOMAIN, | ||||||
|  |         SERVICE_ADB_COMMAND, | ||||||
|  |         entity_domain=MEDIA_PLAYER_DOMAIN, | ||||||
|  |         schema={vol.Required(ATTR_COMMAND): cv.string}, | ||||||
|  |         func="adb_command", | ||||||
|  |     ) | ||||||
|  |     service.async_register_platform_entity_service( | ||||||
|  |         hass, | ||||||
|  |         DOMAIN, | ||||||
|  |         SERVICE_LEARN_SENDEVENT, | ||||||
|  |         entity_domain=MEDIA_PLAYER_DOMAIN, | ||||||
|  |         schema=None, | ||||||
|  |         func="learn_sendevent", | ||||||
|  |     ) | ||||||
|  |     service.async_register_platform_entity_service( | ||||||
|  |         hass, | ||||||
|  |         DOMAIN, | ||||||
|  |         SERVICE_DOWNLOAD, | ||||||
|  |         entity_domain=MEDIA_PLAYER_DOMAIN, | ||||||
|  |         schema={ | ||||||
|  |             vol.Required(ATTR_DEVICE_PATH): cv.string, | ||||||
|  |             vol.Required(ATTR_LOCAL_PATH): cv.string, | ||||||
|  |         }, | ||||||
|  |         func="service_download", | ||||||
|  |     ) | ||||||
|  |     service.async_register_platform_entity_service( | ||||||
|  |         hass, | ||||||
|  |         DOMAIN, | ||||||
|  |         SERVICE_UPLOAD, | ||||||
|  |         entity_domain=MEDIA_PLAYER_DOMAIN, | ||||||
|  |         schema={ | ||||||
|  |             vol.Required(ATTR_DEVICE_PATH): cv.string, | ||||||
|  |             vol.Required(ATTR_LOCAL_PATH): cv.string, | ||||||
|  |         }, | ||||||
|  |         func="service_upload", | ||||||
|  |     ) | ||||||
| @@ -19,9 +19,8 @@ CONF_THINKING_BUDGET = "thinking_budget" | |||||||
| RECOMMENDED_THINKING_BUDGET = 0 | RECOMMENDED_THINKING_BUDGET = 0 | ||||||
| MIN_THINKING_BUDGET = 1024 | MIN_THINKING_BUDGET = 1024 | ||||||
|  |  | ||||||
| THINKING_MODELS = [ | NON_THINKING_MODELS = [ | ||||||
|     "claude-3-7-sonnet", |     "claude-3-5",  # Both sonnet and haiku | ||||||
|     "claude-sonnet-4-0", |     "claude-3-opus", | ||||||
|     "claude-opus-4-0", |     "claude-3-haiku", | ||||||
|     "claude-opus-4-1", |  | ||||||
| ] | ] | ||||||
|   | |||||||
| @@ -51,11 +51,11 @@ from .const import ( | |||||||
|     DOMAIN, |     DOMAIN, | ||||||
|     LOGGER, |     LOGGER, | ||||||
|     MIN_THINKING_BUDGET, |     MIN_THINKING_BUDGET, | ||||||
|  |     NON_THINKING_MODELS, | ||||||
|     RECOMMENDED_CHAT_MODEL, |     RECOMMENDED_CHAT_MODEL, | ||||||
|     RECOMMENDED_MAX_TOKENS, |     RECOMMENDED_MAX_TOKENS, | ||||||
|     RECOMMENDED_TEMPERATURE, |     RECOMMENDED_TEMPERATURE, | ||||||
|     RECOMMENDED_THINKING_BUDGET, |     RECOMMENDED_THINKING_BUDGET, | ||||||
|     THINKING_MODELS, |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| # Max number of back and forth with the LLM to generate a response | # Max number of back and forth with the LLM to generate a response | ||||||
| @@ -364,7 +364,7 @@ class AnthropicBaseLLMEntity(Entity): | |||||||
|         if tools: |         if tools: | ||||||
|             model_args["tools"] = tools |             model_args["tools"] = tools | ||||||
|         if ( |         if ( | ||||||
|             model.startswith(tuple(THINKING_MODELS)) |             not model.startswith(tuple(NON_THINKING_MODELS)) | ||||||
|             and thinking_budget >= MIN_THINKING_BUDGET |             and thinking_budget >= MIN_THINKING_BUDGET | ||||||
|         ): |         ): | ||||||
|             model_args["thinking"] = ThinkingConfigEnabledParam( |             model_args["thinking"] = ThinkingConfigEnabledParam( | ||||||
|   | |||||||
| @@ -8,5 +8,5 @@ | |||||||
|   "documentation": "https://www.home-assistant.io/integrations/anthropic", |   "documentation": "https://www.home-assistant.io/integrations/anthropic", | ||||||
|   "integration_type": "service", |   "integration_type": "service", | ||||||
|   "iot_class": "cloud_polling", |   "iot_class": "cloud_polling", | ||||||
|   "requirements": ["anthropic==0.62.0"] |   "requirements": ["anthropic==0.69.0"] | ||||||
| } | } | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ from .coordinator import ( | |||||||
|     AOSmithStatusCoordinator, |     AOSmithStatusCoordinator, | ||||||
| ) | ) | ||||||
|  |  | ||||||
| PLATFORMS: list[Platform] = [Platform.SENSOR, Platform.WATER_HEATER] | PLATFORMS: list[Platform] = [Platform.SELECT, Platform.SENSOR, Platform.WATER_HEATER] | ||||||
|  |  | ||||||
|  |  | ||||||
| async def async_setup_entry(hass: HomeAssistant, entry: AOSmithConfigEntry) -> bool: | async def async_setup_entry(hass: HomeAssistant, entry: AOSmithConfigEntry) -> bool: | ||||||
|   | |||||||
| @@ -1,5 +1,10 @@ | |||||||
| { | { | ||||||
|   "entity": { |   "entity": { | ||||||
|  |     "select": { | ||||||
|  |       "hot_water_plus_level": { | ||||||
|  |         "default": "mdi:water-plus" | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "sensor": { |     "sensor": { | ||||||
|       "hot_water_availability": { |       "hot_water_availability": { | ||||||
|         "default": "mdi:water-thermometer" |         "default": "mdi:water-thermometer" | ||||||
|   | |||||||
							
								
								
									
										70
									
								
								homeassistant/components/aosmith/select.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								homeassistant/components/aosmith/select.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,70 @@ | |||||||
|  | """The select platform for the A. O. Smith integration.""" | ||||||
|  |  | ||||||
|  | from homeassistant.components.select import SelectEntity | ||||||
|  | from homeassistant.core import HomeAssistant | ||||||
|  | from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | ||||||
|  |  | ||||||
|  | from . import AOSmithConfigEntry | ||||||
|  | from .coordinator import AOSmithStatusCoordinator | ||||||
|  | from .entity import AOSmithStatusEntity | ||||||
|  |  | ||||||
|  | HWP_LEVEL_HA_TO_AOSMITH = { | ||||||
|  |     "off": 0, | ||||||
|  |     "level1": 1, | ||||||
|  |     "level2": 2, | ||||||
|  |     "level3": 3, | ||||||
|  | } | ||||||
|  | HWP_LEVEL_AOSMITH_TO_HA = {value: key for key, value in HWP_LEVEL_HA_TO_AOSMITH.items()} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | async def async_setup_entry( | ||||||
|  |     hass: HomeAssistant, | ||||||
|  |     entry: AOSmithConfigEntry, | ||||||
|  |     async_add_entities: AddConfigEntryEntitiesCallback, | ||||||
|  | ) -> None: | ||||||
|  |     """Set up A. O. Smith select platform.""" | ||||||
|  |     data = entry.runtime_data | ||||||
|  |  | ||||||
|  |     async_add_entities( | ||||||
|  |         AOSmithHotWaterPlusSelectEntity(data.status_coordinator, device.junction_id) | ||||||
|  |         for device in data.status_coordinator.data.values() | ||||||
|  |         if device.supports_hot_water_plus | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class AOSmithHotWaterPlusSelectEntity(AOSmithStatusEntity, SelectEntity): | ||||||
|  |     """Class for the Hot Water+ select entity.""" | ||||||
|  |  | ||||||
|  |     _attr_translation_key = "hot_water_plus_level" | ||||||
|  |     _attr_options = list(HWP_LEVEL_HA_TO_AOSMITH) | ||||||
|  |  | ||||||
|  |     def __init__(self, coordinator: AOSmithStatusCoordinator, junction_id: str) -> None: | ||||||
|  |         """Initialize the entity.""" | ||||||
|  |         super().__init__(coordinator, junction_id) | ||||||
|  |         self._attr_unique_id = f"hot_water_plus_level_{junction_id}" | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def suggested_object_id(self) -> str | None: | ||||||
|  |         """Override the suggested object id to make '+' get converted to 'plus' in the entity id.""" | ||||||
|  |         return "hot_water_plus_level" | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def current_option(self) -> str | None: | ||||||
|  |         """Return the current Hot Water+ mode.""" | ||||||
|  |         hot_water_plus_level = self.device.status.hot_water_plus_level | ||||||
|  |         return ( | ||||||
|  |             None | ||||||
|  |             if hot_water_plus_level is None | ||||||
|  |             else HWP_LEVEL_AOSMITH_TO_HA.get(hot_water_plus_level) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     async def async_select_option(self, option: str) -> None: | ||||||
|  |         """Set the Hot Water+ mode.""" | ||||||
|  |         aosmith_hwp_level = HWP_LEVEL_HA_TO_AOSMITH[option] | ||||||
|  |         await self.client.update_mode( | ||||||
|  |             junction_id=self.junction_id, | ||||||
|  |             mode=self.device.status.current_mode, | ||||||
|  |             hot_water_plus_level=aosmith_hwp_level, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         await self.coordinator.async_request_refresh() | ||||||
| @@ -26,6 +26,17 @@ | |||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   "entity": { |   "entity": { | ||||||
|  |     "select": { | ||||||
|  |       "hot_water_plus_level": { | ||||||
|  |         "name": "Hot Water+ level", | ||||||
|  |         "state": { | ||||||
|  |           "off": "[%key:common::state::off%]", | ||||||
|  |           "level1": "Level 1", | ||||||
|  |           "level2": "Level 2", | ||||||
|  |           "level3": "Level 3" | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|     "sensor": { |     "sensor": { | ||||||
|       "hot_water_availability": { |       "hot_water_availability": { | ||||||
|         "name": "Hot water availability" |         "name": "Hot water availability" | ||||||
|   | |||||||
| @@ -7,5 +7,5 @@ | |||||||
|   "iot_class": "local_polling", |   "iot_class": "local_polling", | ||||||
|   "loggers": ["apcaccess"], |   "loggers": ["apcaccess"], | ||||||
|   "quality_scale": "platinum", |   "quality_scale": "platinum", | ||||||
|   "requirements": ["aioapcaccess==0.4.2"] |   "requirements": ["aioapcaccess==1.0.0"] | ||||||
| } | } | ||||||
|   | |||||||
| @@ -395,6 +395,7 @@ SENSORS: dict[str, SensorEntityDescription] = { | |||||||
|     "upsmode": SensorEntityDescription( |     "upsmode": SensorEntityDescription( | ||||||
|         key="upsmode", |         key="upsmode", | ||||||
|         translation_key="ups_mode", |         translation_key="ups_mode", | ||||||
|  |         entity_category=EntityCategory.DIAGNOSTIC, | ||||||
|     ), |     ), | ||||||
|     "upsname": SensorEntityDescription( |     "upsname": SensorEntityDescription( | ||||||
|         key="upsname", |         key="upsname", | ||||||
| @@ -466,7 +467,10 @@ async def async_setup_entry( | |||||||
|     # periodical (or manual) self test since last daemon restart. It might not be available |     # periodical (or manual) self test since last daemon restart. It might not be available | ||||||
|     # when we set up the integration, and we do not know if it would ever be available. Here we |     # when we set up the integration, and we do not know if it would ever be available. Here we | ||||||
|     # add it anyway and mark it as unknown initially. |     # add it anyway and mark it as unknown initially. | ||||||
|     for resource in available_resources | {LAST_S_TEST}: |     # | ||||||
|  |     # We also sort the resources to ensure the order of entities created is deterministic since | ||||||
|  |     # "APCMODEL" and "MODEL" resources map to the same "Model" name. | ||||||
|  |     for resource in sorted(available_resources | {LAST_S_TEST}): | ||||||
|         if resource not in SENSORS: |         if resource not in SENSORS: | ||||||
|             _LOGGER.warning("Invalid resource from APCUPSd: %s", resource.upper()) |             _LOGGER.warning("Invalid resource from APCUPSd: %s", resource.upper()) | ||||||
|             continue |             continue | ||||||
|   | |||||||
| @@ -7,6 +7,8 @@ from typing import Any | |||||||
| from pyaprilaire.const import Attribute | from pyaprilaire.const import Attribute | ||||||
|  |  | ||||||
| from homeassistant.components.climate import ( | from homeassistant.components.climate import ( | ||||||
|  |     ATTR_TARGET_TEMP_HIGH, | ||||||
|  |     ATTR_TARGET_TEMP_LOW, | ||||||
|     FAN_AUTO, |     FAN_AUTO, | ||||||
|     FAN_ON, |     FAN_ON, | ||||||
|     PRESET_AWAY, |     PRESET_AWAY, | ||||||
| @@ -16,7 +18,12 @@ from homeassistant.components.climate import ( | |||||||
|     HVACAction, |     HVACAction, | ||||||
|     HVACMode, |     HVACMode, | ||||||
| ) | ) | ||||||
| from homeassistant.const import PRECISION_HALVES, PRECISION_WHOLE, UnitOfTemperature | from homeassistant.const import ( | ||||||
|  |     ATTR_TEMPERATURE, | ||||||
|  |     PRECISION_HALVES, | ||||||
|  |     PRECISION_WHOLE, | ||||||
|  |     UnitOfTemperature, | ||||||
|  | ) | ||||||
| from homeassistant.core import HomeAssistant | from homeassistant.core import HomeAssistant | ||||||
| from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | ||||||
|  |  | ||||||
| @@ -232,15 +239,15 @@ class AprilaireClimate(BaseAprilaireEntity, ClimateEntity): | |||||||
|         cool_setpoint = 0 |         cool_setpoint = 0 | ||||||
|         heat_setpoint = 0 |         heat_setpoint = 0 | ||||||
|  |  | ||||||
|         if temperature := kwargs.get("temperature"): |         if temperature := kwargs.get(ATTR_TEMPERATURE): | ||||||
|             if self.coordinator.data.get(Attribute.MODE) == 3: |             if self.coordinator.data.get(Attribute.MODE) == 3: | ||||||
|                 cool_setpoint = temperature |                 cool_setpoint = temperature | ||||||
|             else: |             else: | ||||||
|                 heat_setpoint = temperature |                 heat_setpoint = temperature | ||||||
|         else: |         else: | ||||||
|             if target_temp_low := kwargs.get("target_temp_low"): |             if target_temp_low := kwargs.get(ATTR_TARGET_TEMP_LOW): | ||||||
|                 heat_setpoint = target_temp_low |                 heat_setpoint = target_temp_low | ||||||
|             if target_temp_high := kwargs.get("target_temp_high"): |             if target_temp_high := kwargs.get(ATTR_TARGET_TEMP_HIGH): | ||||||
|                 cool_setpoint = target_temp_high |                 cool_setpoint = target_temp_high | ||||||
|  |  | ||||||
|         if cool_setpoint == 0 and heat_setpoint == 0: |         if cool_setpoint == 0 and heat_setpoint == 0: | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								homeassistant/components/assist_pipeline/acknowledge.mp3
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								homeassistant/components/assist_pipeline/acknowledge.mp3
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -1,5 +1,7 @@ | |||||||
| """Constants for the Assist pipeline integration.""" | """Constants for the Assist pipeline integration.""" | ||||||
|  |  | ||||||
|  | from pathlib import Path | ||||||
|  |  | ||||||
| DOMAIN = "assist_pipeline" | DOMAIN = "assist_pipeline" | ||||||
|  |  | ||||||
| DATA_CONFIG = f"{DOMAIN}.config" | DATA_CONFIG = f"{DOMAIN}.config" | ||||||
| @@ -23,3 +25,5 @@ SAMPLES_PER_CHUNK = SAMPLE_RATE // (1000 // MS_PER_CHUNK)  # 10 ms @ 16Khz | |||||||
| BYTES_PER_CHUNK = SAMPLES_PER_CHUNK * SAMPLE_WIDTH * SAMPLE_CHANNELS  # 16-bit | BYTES_PER_CHUNK = SAMPLES_PER_CHUNK * SAMPLE_WIDTH * SAMPLE_CHANNELS  # 16-bit | ||||||
|  |  | ||||||
| OPTION_PREFERRED = "preferred" | OPTION_PREFERRED = "preferred" | ||||||
|  |  | ||||||
|  | ACKNOWLEDGE_PATH = Path(__file__).parent / "acknowledge.mp3" | ||||||
|   | |||||||
| @@ -23,7 +23,12 @@ from homeassistant.components import conversation, stt, tts, wake_word, websocke | |||||||
| from homeassistant.const import ATTR_SUPPORTED_FEATURES, MATCH_ALL | from homeassistant.const import ATTR_SUPPORTED_FEATURES, MATCH_ALL | ||||||
| from homeassistant.core import Context, HomeAssistant, callback | from homeassistant.core import Context, HomeAssistant, callback | ||||||
| from homeassistant.exceptions import HomeAssistantError | from homeassistant.exceptions import HomeAssistantError | ||||||
| from homeassistant.helpers import chat_session, intent | from homeassistant.helpers import ( | ||||||
|  |     chat_session, | ||||||
|  |     device_registry as dr, | ||||||
|  |     entity_registry as er, | ||||||
|  |     intent, | ||||||
|  | ) | ||||||
| from homeassistant.helpers.collection import ( | from homeassistant.helpers.collection import ( | ||||||
|     CHANGE_UPDATED, |     CHANGE_UPDATED, | ||||||
|     CollectionError, |     CollectionError, | ||||||
| @@ -45,6 +50,7 @@ from homeassistant.util.limited_size_dict import LimitedSizeDict | |||||||
|  |  | ||||||
| from .audio_enhancer import AudioEnhancer, EnhancedAudioChunk, MicroVadSpeexEnhancer | from .audio_enhancer import AudioEnhancer, EnhancedAudioChunk, MicroVadSpeexEnhancer | ||||||
| from .const import ( | from .const import ( | ||||||
|  |     ACKNOWLEDGE_PATH, | ||||||
|     BYTES_PER_CHUNK, |     BYTES_PER_CHUNK, | ||||||
|     CONF_DEBUG_RECORDING_DIR, |     CONF_DEBUG_RECORDING_DIR, | ||||||
|     DATA_CONFIG, |     DATA_CONFIG, | ||||||
| @@ -113,6 +119,7 @@ PIPELINE_FIELDS: VolDictType = { | |||||||
|     vol.Required("wake_word_entity"): vol.Any(str, None), |     vol.Required("wake_word_entity"): vol.Any(str, None), | ||||||
|     vol.Required("wake_word_id"): vol.Any(str, None), |     vol.Required("wake_word_id"): vol.Any(str, None), | ||||||
|     vol.Optional("prefer_local_intents"): bool, |     vol.Optional("prefer_local_intents"): bool, | ||||||
|  |     vol.Optional("acknowledge_media_id"): str, | ||||||
| } | } | ||||||
|  |  | ||||||
| STORED_PIPELINE_RUNS = 10 | STORED_PIPELINE_RUNS = 10 | ||||||
| @@ -1066,8 +1073,11 @@ class PipelineRun: | |||||||
|         intent_input: str, |         intent_input: str, | ||||||
|         conversation_id: str, |         conversation_id: str, | ||||||
|         conversation_extra_system_prompt: str | None, |         conversation_extra_system_prompt: str | None, | ||||||
|     ) -> str: |     ) -> tuple[str, bool]: | ||||||
|         """Run intent recognition portion of pipeline. Returns text to speak.""" |         """Run intent recognition portion of pipeline. | ||||||
|  |  | ||||||
|  |         Returns (speech, all_targets_in_satellite_area). | ||||||
|  |         """ | ||||||
|         if self.intent_agent is None or self._conversation_data is None: |         if self.intent_agent is None or self._conversation_data is None: | ||||||
|             raise RuntimeError("Recognize intent was not prepared") |             raise RuntimeError("Recognize intent was not prepared") | ||||||
|  |  | ||||||
| @@ -1116,6 +1126,7 @@ class PipelineRun: | |||||||
|  |  | ||||||
|             agent_id = self.intent_agent.id |             agent_id = self.intent_agent.id | ||||||
|             processed_locally = agent_id == conversation.HOME_ASSISTANT_AGENT |             processed_locally = agent_id == conversation.HOME_ASSISTANT_AGENT | ||||||
|  |             all_targets_in_satellite_area = False | ||||||
|             intent_response: intent.IntentResponse | None = None |             intent_response: intent.IntentResponse | None = None | ||||||
|             if not processed_locally and not self._intent_agent_only: |             if not processed_locally and not self._intent_agent_only: | ||||||
|                 # Sentence triggers override conversation agent |                 # Sentence triggers override conversation agent | ||||||
| @@ -1290,6 +1301,19 @@ class PipelineRun: | |||||||
|                     if tts_input_stream and self._streamed_response_text: |                     if tts_input_stream and self._streamed_response_text: | ||||||
|                         tts_input_stream.put_nowait(None) |                         tts_input_stream.put_nowait(None) | ||||||
|  |  | ||||||
|  |                 if agent_id == conversation.HOME_ASSISTANT_AGENT: | ||||||
|  |                     # Check if all targeted entities were in the same area as | ||||||
|  |                     # the satellite device. | ||||||
|  |                     # If so, the satellite should respond with an acknowledge beep | ||||||
|  |                     # instead of a full response. | ||||||
|  |                     all_targets_in_satellite_area = ( | ||||||
|  |                         self._get_all_targets_in_satellite_area( | ||||||
|  |                             conversation_result.response, | ||||||
|  |                             self._satellite_id, | ||||||
|  |                             self._device_id, | ||||||
|  |                         ) | ||||||
|  |                     ) | ||||||
|  |  | ||||||
|         except Exception as src_error: |         except Exception as src_error: | ||||||
|             _LOGGER.exception("Unexpected error during intent recognition") |             _LOGGER.exception("Unexpected error during intent recognition") | ||||||
|             raise IntentRecognitionError( |             raise IntentRecognitionError( | ||||||
| @@ -1312,7 +1336,68 @@ class PipelineRun: | |||||||
|         if conversation_result.continue_conversation: |         if conversation_result.continue_conversation: | ||||||
|             self._conversation_data.continue_conversation_agent = agent_id |             self._conversation_data.continue_conversation_agent = agent_id | ||||||
|  |  | ||||||
|         return speech |         return (speech, all_targets_in_satellite_area) | ||||||
|  |  | ||||||
|  |     def _get_all_targets_in_satellite_area( | ||||||
|  |         self, | ||||||
|  |         intent_response: intent.IntentResponse, | ||||||
|  |         satellite_id: str | None, | ||||||
|  |         device_id: str | None, | ||||||
|  |     ) -> bool: | ||||||
|  |         """Return true if all targeted entities were in the same area as the device.""" | ||||||
|  |         if ( | ||||||
|  |             intent_response.response_type != intent.IntentResponseType.ACTION_DONE | ||||||
|  |             or not intent_response.matched_states | ||||||
|  |         ): | ||||||
|  |             return False | ||||||
|  |  | ||||||
|  |         entity_registry = er.async_get(self.hass) | ||||||
|  |         device_registry = dr.async_get(self.hass) | ||||||
|  |  | ||||||
|  |         area_id: str | None = None | ||||||
|  |  | ||||||
|  |         if ( | ||||||
|  |             satellite_id is not None | ||||||
|  |             and (target_entity_entry := entity_registry.async_get(satellite_id)) | ||||||
|  |             is not None | ||||||
|  |         ): | ||||||
|  |             area_id = target_entity_entry.area_id | ||||||
|  |             device_id = target_entity_entry.device_id | ||||||
|  |  | ||||||
|  |         if area_id is None: | ||||||
|  |             if device_id is None: | ||||||
|  |                 return False | ||||||
|  |  | ||||||
|  |             device_entry = device_registry.async_get(device_id) | ||||||
|  |             if device_entry is None: | ||||||
|  |                 return False | ||||||
|  |  | ||||||
|  |             area_id = device_entry.area_id | ||||||
|  |             if area_id is None: | ||||||
|  |                 return False | ||||||
|  |  | ||||||
|  |         for state in intent_response.matched_states: | ||||||
|  |             target_entity_entry = entity_registry.async_get(state.entity_id) | ||||||
|  |             if target_entity_entry is None: | ||||||
|  |                 return False | ||||||
|  |  | ||||||
|  |             target_area_id = target_entity_entry.area_id | ||||||
|  |             if target_area_id is None: | ||||||
|  |                 if target_entity_entry.device_id is None: | ||||||
|  |                     return False | ||||||
|  |  | ||||||
|  |                 target_device_entry = device_registry.async_get( | ||||||
|  |                     target_entity_entry.device_id | ||||||
|  |                 ) | ||||||
|  |                 if target_device_entry is None: | ||||||
|  |                     return False | ||||||
|  |  | ||||||
|  |                 target_area_id = target_device_entry.area_id | ||||||
|  |  | ||||||
|  |             if target_area_id != area_id: | ||||||
|  |                 return False | ||||||
|  |  | ||||||
|  |         return True | ||||||
|  |  | ||||||
|     async def prepare_text_to_speech(self) -> None: |     async def prepare_text_to_speech(self) -> None: | ||||||
|         """Prepare text-to-speech.""" |         """Prepare text-to-speech.""" | ||||||
| @@ -1350,7 +1435,9 @@ class PipelineRun: | |||||||
|                 ), |                 ), | ||||||
|             ) from err |             ) from err | ||||||
|  |  | ||||||
|     async def text_to_speech(self, tts_input: str) -> None: |     async def text_to_speech( | ||||||
|  |         self, tts_input: str, override_media_path: Path | None = None | ||||||
|  |     ) -> None: | ||||||
|         """Run text-to-speech portion of pipeline.""" |         """Run text-to-speech portion of pipeline.""" | ||||||
|         assert self.tts_stream is not None |         assert self.tts_stream is not None | ||||||
|  |  | ||||||
| @@ -1362,11 +1449,14 @@ class PipelineRun: | |||||||
|                     "language": self.pipeline.tts_language, |                     "language": self.pipeline.tts_language, | ||||||
|                     "voice": self.pipeline.tts_voice, |                     "voice": self.pipeline.tts_voice, | ||||||
|                     "tts_input": tts_input, |                     "tts_input": tts_input, | ||||||
|  |                     "acknowledge_override": override_media_path is not None, | ||||||
|                 }, |                 }, | ||||||
|             ) |             ) | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         if not self._streamed_response_text: |         if override_media_path: | ||||||
|  |             self.tts_stream.async_override_result(override_media_path) | ||||||
|  |         elif not self._streamed_response_text: | ||||||
|             self.tts_stream.async_set_message(tts_input) |             self.tts_stream.async_set_message(tts_input) | ||||||
|  |  | ||||||
|         tts_output = { |         tts_output = { | ||||||
| @@ -1664,16 +1754,20 @@ class PipelineInput: | |||||||
|  |  | ||||||
|             if self.run.end_stage != PipelineStage.STT: |             if self.run.end_stage != PipelineStage.STT: | ||||||
|                 tts_input = self.tts_input |                 tts_input = self.tts_input | ||||||
|  |                 all_targets_in_satellite_area = False | ||||||
|  |  | ||||||
|                 if current_stage == PipelineStage.INTENT: |                 if current_stage == PipelineStage.INTENT: | ||||||
|                     # intent-recognition |                     # intent-recognition | ||||||
|                     assert intent_input is not None |                     assert intent_input is not None | ||||||
|                     tts_input = await self.run.recognize_intent( |                     ( | ||||||
|  |                         tts_input, | ||||||
|  |                         all_targets_in_satellite_area, | ||||||
|  |                     ) = await self.run.recognize_intent( | ||||||
|                         intent_input, |                         intent_input, | ||||||
|                         self.session.conversation_id, |                         self.session.conversation_id, | ||||||
|                         self.conversation_extra_system_prompt, |                         self.conversation_extra_system_prompt, | ||||||
|                     ) |                     ) | ||||||
|                     if tts_input.strip(): |                     if all_targets_in_satellite_area or tts_input.strip(): | ||||||
|                         current_stage = PipelineStage.TTS |                         current_stage = PipelineStage.TTS | ||||||
|                     else: |                     else: | ||||||
|                         # Skip TTS |                         # Skip TTS | ||||||
| @@ -1682,6 +1776,12 @@ class PipelineInput: | |||||||
|                 if self.run.end_stage != PipelineStage.INTENT: |                 if self.run.end_stage != PipelineStage.INTENT: | ||||||
|                     # text-to-speech |                     # text-to-speech | ||||||
|                     if current_stage == PipelineStage.TTS: |                     if current_stage == PipelineStage.TTS: | ||||||
|  |                         if all_targets_in_satellite_area: | ||||||
|  |                             # Use acknowledge media instead of full response | ||||||
|  |                             await self.run.text_to_speech( | ||||||
|  |                                 tts_input or "", override_media_path=ACKNOWLEDGE_PATH | ||||||
|  |                             ) | ||||||
|  |                         else: | ||||||
|                             assert tts_input is not None |                             assert tts_input is not None | ||||||
|                             await self.run.text_to_speech(tts_input) |                             await self.run.text_to_speech(tts_input) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ | |||||||
| from __future__ import annotations | from __future__ import annotations | ||||||
|  |  | ||||||
| from collections.abc import Iterable | from collections.abc import Iterable | ||||||
|  | from dataclasses import replace | ||||||
|  |  | ||||||
| from homeassistant.components.select import SelectEntity, SelectEntityDescription | from homeassistant.components.select import SelectEntity, SelectEntityDescription | ||||||
| from homeassistant.const import EntityCategory, Platform | from homeassistant.const import EntityCategory, Platform | ||||||
| @@ -64,15 +65,36 @@ class AssistPipelineSelect(SelectEntity, restore_state.RestoreEntity): | |||||||
|         translation_key="pipeline", |         translation_key="pipeline", | ||||||
|         entity_category=EntityCategory.CONFIG, |         entity_category=EntityCategory.CONFIG, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     _attr_should_poll = False |     _attr_should_poll = False | ||||||
|     _attr_current_option = OPTION_PREFERRED |     _attr_current_option = OPTION_PREFERRED | ||||||
|     _attr_options = [OPTION_PREFERRED] |     _attr_options = [OPTION_PREFERRED] | ||||||
|  |  | ||||||
|     def __init__(self, hass: HomeAssistant, domain: str, unique_id_prefix: str) -> None: |     def __init__( | ||||||
|  |         self, | ||||||
|  |         hass: HomeAssistant, | ||||||
|  |         domain: str, | ||||||
|  |         unique_id_prefix: str, | ||||||
|  |         index: int = 0, | ||||||
|  |     ) -> None: | ||||||
|         """Initialize a pipeline selector.""" |         """Initialize a pipeline selector.""" | ||||||
|  |         if index < 1: | ||||||
|  |             # Keep compatibility | ||||||
|  |             key_suffix = "" | ||||||
|  |             placeholder = "" | ||||||
|  |         else: | ||||||
|  |             key_suffix = f"_{index + 1}" | ||||||
|  |             placeholder = f" {index + 1}" | ||||||
|  |  | ||||||
|  |         self.entity_description = replace( | ||||||
|  |             self.entity_description, | ||||||
|  |             key=f"pipeline{key_suffix}", | ||||||
|  |             translation_placeholders={"index": placeholder}, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|         self._domain = domain |         self._domain = domain | ||||||
|         self._unique_id_prefix = unique_id_prefix |         self._unique_id_prefix = unique_id_prefix | ||||||
|         self._attr_unique_id = f"{unique_id_prefix}-pipeline" |         self._attr_unique_id = f"{unique_id_prefix}-{self.entity_description.key}" | ||||||
|         self.hass = hass |         self.hass = hass | ||||||
|         self._update_options() |         self._update_options() | ||||||
|  |  | ||||||
| @@ -87,7 +109,7 @@ class AssistPipelineSelect(SelectEntity, restore_state.RestoreEntity): | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         state = await self.async_get_last_state() |         state = await self.async_get_last_state() | ||||||
|         if state is not None and state.state in self.options: |         if (state is not None) and (state.state in self.options): | ||||||
|             self._attr_current_option = state.state |             self._attr_current_option = state.state | ||||||
|  |  | ||||||
|         if self.registry_entry and (device_id := self.registry_entry.device_id): |         if self.registry_entry and (device_id := self.registry_entry.device_id): | ||||||
| @@ -97,7 +119,7 @@ class AssistPipelineSelect(SelectEntity, restore_state.RestoreEntity): | |||||||
|  |  | ||||||
|             def cleanup() -> None: |             def cleanup() -> None: | ||||||
|                 """Clean up registered device.""" |                 """Clean up registered device.""" | ||||||
|                 pipeline_data.pipeline_devices.pop(device_id) |                 pipeline_data.pipeline_devices.pop(device_id, None) | ||||||
|  |  | ||||||
|             self.async_on_remove(cleanup) |             self.async_on_remove(cleanup) | ||||||
|  |  | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user