mirror of
				https://github.com/home-assistant/supervisor.git
				synced 2025-10-26 03:59:35 +00:00 
			
		
		
		
	Compare commits
	
		
			1026 Commits
		
	
	
		
			faster_bac
			...
			2025.09.3
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ![github-actions[bot]](/assets/img/avatar_default.png)  | e9cac9db06 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 67c15678c6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b0145a8507 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9f6b154097 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 90c0d014db | ||
|   | fabfe760fb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 092013e457 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e13f216b2e | ||
|   | 97c7686b95 | ||
|   | 42f93d0176 | ||
|   | ed7155604c | ||
|   | 595e33ac68 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ae70ffd1b2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 17cb18a371 | ||
|   | 9f5bebd0eb | ||
|   | c712d3cc53 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 46fc5c8aa1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8b23383e26 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c1ccb00946 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5693a5be0d | ||
|   | 01911a44cd | ||
|   | 857dae7736 | ||
|   | d2ddd9579c | ||
|   | ac9947d599 | ||
|   | 2e22e1e884 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e7f3573e32 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b26451a59a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4e882f7c76 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5fa50ccf05 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3891df5266 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5aad32c15b | ||
|   | 4a40490af7 | ||
|   | 0a46e030f5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bd00f90304 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 819f097f01 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4513592993 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7e526a26af | ||
|   | b3af22f048 | ||
|   | bbb9469c1c | ||
|   | 859c32a706 | ||
|   | 87fc84c65c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e38ca5acb4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 09cd8eede2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d1c537b280 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e6785d6a89 | ||
|   | 59e051ad93 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3397def8b9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b832edc10d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f69071878c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e065ba6081 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 38611ad12f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8beb66d46c | ||
|   | c277f3cad6 | ||
|   | 236c39cbb0 | ||
|   | 7ed83a15fe | ||
|   | a3a5f6ba98 | ||
|   | 8d3ededf2f | ||
|   | 3d62c9afb1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ef313d1fb5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cae31637ae | ||
|   | 9392d10625 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5ce62f324f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f84d514958 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3c39f2f785 | ||
|   | 30db72df78 | ||
|   | 00a78f372b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b69546f2c1 | ||
|   | 78be155b94 | ||
|   | 9900dfc8ca | ||
|   | 3a1ebc9d37 | ||
|   | 580c3273dc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b889f94ca4 | ||
|   | 2d12920b35 | ||
|   | 8a95113ebd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3fc1abf661 | ||
|   | 207b665e1d | ||
|   | 1fb15772d7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9740de7a83 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8e8d77d90c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dbce22bd08 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 192d446888 | ||
|   | d95ca401ec | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 07d8fd006a | ||
|   | b49ce96df8 | ||
|   | 4109c15a36 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d0e2778255 | ||
|   | 014082eda8 | ||
|   | 2324b70084 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 43f20fe24f | ||
|   | 8ef5eae22a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e5dd09ab6b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3f2db956cb | ||
|   | 603df92618 | ||
|   | 8a82b98e5b | ||
|   | 07dd0b7394 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cf0a85a4b1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9924165cd3 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 91392a5443 | ||
|   | fd205ce2ef | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9ec56d9266 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 886b1bd281 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ee0474edf5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f173489e69 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cee495bde3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 59104a4438 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e4eaeb91cd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e61d88779d | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 0513ea0438 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 030927dc01 | ||
|   | cad14bf46e | ||
|   | 5d851ad747 | ||
|   | 528032fb36 | ||
|   | 3b093200e3 | ||
|   | 15ba1a3c94 | ||
|   | 8e4a87c751 | ||
|   | fdde95d849 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 65e5a36aa7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bd62602cde | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f9bcc273f8 | ||
|   | 059b161f4f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f11eb6b35a | ||
|   | 9bee58a8b1 | ||
|   | 8a1e6b0895 | ||
|   | f150d1b287 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 628a18c6b8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 74e43411e5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e6b0d4144c | ||
|   | 033896480d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 478e00c0fe | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6f2ba7d68c | ||
|   | 22afa60f55 | ||
|   | 9f2fda5dc7 | ||
|   | 27b092aed0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3af13cb7e2 | ||
|   | 6871ea4b81 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cf77ab2290 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ceeffa3284 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 31f2f70cd9 | ||
|   | deac85bddb | ||
|   | 7dcf5ba631 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a004830131 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a8cc6c416d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 74b26642b0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5e26ab5f4a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a841cb8282 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3b1b03c8a7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 680428f304 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f34128c37e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2ed0682b34 | ||
|   | fbb0915ef8 | ||
|   | 780ae1e15c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c617358855 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b679c4f4d8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c946c421f2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | aeabf7ea25 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 365b838abf | ||
|   | 99c040520e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | eefe2f2e06 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a366e36b37 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 27a2fde9e1 | ||
|   | 9a0f530a2f | ||
|   | baf9695cf7 | ||
|   | 7873c457d5 | ||
|   | cbc48c381f | ||
|   | 11e37011bd | ||
|   | cfda559a90 | ||
|   | 806bd9f52c | ||
|   | 953f7d01d7 | ||
|   | 381e719a0e | ||
|   | 296071067d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8336537f51 | ||
|   | 5c90a00263 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1f2bf77784 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9aa4f381b8 | ||
|   | ae036ceffe | ||
|   | f0ea0d4a44 | ||
|   | abc44946bb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3e20a0937d | ||
|   | 6cebf52249 | ||
|   | bc57deb474 | ||
|   | 38750d74a8 | ||
|   | d1c1a2d418 | ||
|   | cf32f036c0 | ||
|   | b8852872fe | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 779f47e25d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | be8b36b560 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8378d434d4 | ||
|   | 0b79e09bc0 | ||
|   | d747a59696 | ||
|   | 3ee7c082ec | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3f921e50b3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0370320f75 | ||
|   | 1e19e26ef3 | ||
|   | e1a18eeba8 | ||
|   | b030879efd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dfa1602ac6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bbda943583 | ||
|   | aea15b65b7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5c04249e41 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 456cec7ed1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 52a519e55c | ||
|   | fcb20d0ae8 | ||
|   | 9b3f2b17bd | ||
|   | 3d026b9534 | ||
|   | 0e8ace949a | ||
|   | 1fe6f8ad99 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9ef2352d12 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2543bcae29 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ad9de9f73c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a5556651ae | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ac28deff6d | ||
|   | 82ee4bc441 | ||
|   | bdbd09733a | ||
|   | d5b5a328d7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 52b24e177f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e10c58c424 | ||
|   | 9682870c2c | ||
|   | fd0b894d6a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 697515b81f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d912c234fa | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e8445ae8f2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6710439ce5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 95eec03c91 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9b686a2d9a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 063d69da90 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | baaf04981f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bdb25a7ff8 | ||
|   | ad2d6a3156 | ||
|   | 42f885595e | ||
|   | 2a88cb9339 | ||
|   | 4d1a5e2dc2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 705e76abe3 | ||
|   | 7f54383147 | ||
|   | 63fde3b410 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5285e60cd3 | ||
|   | 2a1e32bb36 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a2251e0729 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1efee641ba | ||
|   | bbb8fa0b92 | ||
|   | 7593f857e8 | ||
|   | 87232cf1e4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9e6a4d65cd | ||
|   | c80fbd77c8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a452969ffe | ||
|   | 89fa5c9c7a | ||
|   | 73069b628e | ||
|   | 8251b6c61c | ||
|   | 1faf529b42 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 86c016b35d | ||
|   | 4f35759fe3 | ||
|   | 3b575eedba | ||
|   | 6e6fe5ba39 | ||
|   | b5a7e521ae | ||
|   | bac7c21fe8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2eb9ec20d6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 406348c068 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5e3f4e8ff3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 31a67bc642 | ||
|   | d0d11db7b1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cbf4b4e27e | ||
|   | c855eaab52 | ||
|   | 6bac751c4c | ||
|   | da0ae75e8e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 154aeaee87 | ||
|   | b9bbb99f37 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ff849ce692 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 24456efb6b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0cd9d04e63 | ||
|   | 39bd20c0e7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 481bbc5be8 | ||
|   | 36da382af3 | ||
|   | 85f8107b60 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2e44e6494f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cd1cc66c77 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b76a1f58ea | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3fcd254d25 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3dff2abe65 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ba91be1367 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 25f93cd338 | ||
|   | 9b0044edd6 | ||
|   | 9915c21243 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 657cb56fb9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1b384cebc9 | ||
|   | 61089c3507 | ||
|   | bc9e3eb95b | ||
|   | c1b45406d6 | ||
|   | 8e714072c2 | ||
|   | 88087046de | ||
|   | 53393afe8d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4b5bcece64 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0e7e4f8b42 | ||
|   | 9470f44840 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0e55e6e67b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6116425265 | ||
|   | de497cdc19 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 88b41e80bb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 876afdb26e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9d062c8ed0 | ||
|   | 122b73202b | ||
|   | 5d07dd2c42 | ||
|   | adfb433f57 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 198af54d1e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c3e63a5669 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8f27958e20 | ||
|   | 6fad7d14e1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f7317134e3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9d8db27701 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7da3a34304 | ||
|   | d413e0dcb9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 542ab0411c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 999789f7ce | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | de105f8cb7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b37b0ff744 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | db330ab58a | ||
|   | 4a00caa2e8 | ||
|   | 59a7e9519d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dedf5df5ad | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d09b686269 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9b8f03fa00 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2a3d0fdf61 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | eaae40718b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5a88128cec | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 62b3259d9c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5e05af26a8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c5186101d3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 86cf083902 | ||
|   | 5c1f7ed18d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d051cbcafb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 798af687cf | ||
|   | 01a682cfaa | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 67b9a44160 | ||
|   | 8fe17d9270 | ||
|   | 0a684bdb12 | ||
|   | 9222a3c9c0 | ||
|   | 92cadb4c55 | ||
|   | 8b3bf547d7 | ||
|   | 81fc15d6ac | ||
|   | 63b507a589 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | af9b1e5b1e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 062103ae24 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 48807a65dd | ||
|   | 0636e49fe2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 543d6efec4 | ||
|   | 80f7f07341 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ec721c41c1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 03ca32ced4 | ||
|   | cb16a34401 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d756fd7e14 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c559bd47c3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a2b3427be9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6a2d7bad03 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cfdefbf043 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d7e3dc41ff | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9afb50242b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 52b02d1235 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 84bc72d485 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bd772bb28a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fd2c7c3cc3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a7f139d3e1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8a45e0fd85 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 52290b485b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 525d0fd8ea | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 40c83f4c1e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 99088ad880 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 37c077205a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ac5f9dcb59 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6a9269c052 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | de615bfc1d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3ee639b133 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 632e569347 | ||
|   | cc74831113 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 78c6868ad3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f5f6e8b659 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c91a815cca | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1efe01c21f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c54ff06e0f | ||
|   | 5facf4e790 | ||
|   | 34752466d5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 20ea71f7ff | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ac27e3ac0d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b31e3ce234 | ||
|   | e1c9c8b786 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 23e03a95f4 | ||
|   | a2b8df0a6a | ||
|   | 6ef4f3cc67 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1fb4d1cc11 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 65b1729314 | ||
|   | c7e3d86e2d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5d06ebe430 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5aba616ba4 | ||
|   | 767f435090 | ||
|   | 26024053ed | ||
|   | 324b059970 | ||
|   | 76e916a07e | ||
|   | 582b128ad9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c01d788c4c | ||
|   | 8fb66bcf18 | ||
|   | fdd96ae21c | ||
|   | 1355ef192d | ||
|   | f8bab20728 | ||
|   | 9a3702bc1a | ||
|   | a7c6699f6a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fa7626f83a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 84b265a2e0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | debcafa962 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4634ef82c6 | ||
|   | 5b18fb6b12 | ||
|   | d42ec12ae8 | ||
|   | 86133f8ecd | ||
|   | 12c951f62d | ||
|   | fcb3e2eb55 | ||
|   | 176e511180 | ||
|   | 696dcf6149 | ||
|   | 8030b346e0 | ||
|   | 53d97ce0c6 | ||
|   | 77523f7bec | ||
|   | f4d69f1811 | ||
|   | cf5a0dc548 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a8cc3ae6ef | ||
|   | 362bd8fd21 | ||
|   | 2274de969f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dfed251c7a | ||
|   | 151d4bdd73 | ||
|   | c5d4ebcd48 | ||
|   | 0ad559adcd | ||
|   | 39f5b91f12 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ddee79d209 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ff111253d5 | ||
|   | 31193abb7b | ||
|   | ae266e1692 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c315a15816 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3bd732147c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ddbde93a6d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6db11a8ade | ||
|   | 42e78408a7 | ||
|   | 15e8940c7f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 644ec45ded | ||
|   | a8d2743f56 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0acef4a6e6 | ||
|   | 5733db94aa | ||
|   | da8c6cf111 | ||
|   | 802ee25a8b | ||
|   | ce8b107f1e | ||
|   | 32936e5de0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c35746c3e1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 392dd9f904 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | d8f792950b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1f6cdc3018 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 616f1903b7 | ||
|   | 997a51fc42 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cda6325be4 | ||
|   | c8cc6fe003 | ||
|   | 34939cfe52 | ||
|   | 37bc703bbb | ||
|   | 5f8e41b441 | ||
|   | 606db3585c | ||
|   | 4054749eb2 | ||
|   | ad5827d33f | ||
|   | 249464e928 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3bc55c054a | ||
|   | 4c108eea64 | ||
|   | 9b2dbd634d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2cb2a48184 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ed5a0b511e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1475dcb50b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5cd7f6fd84 | ||
|   | 52cc17fa3f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fa6949f4e4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 63a4cee770 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7aed0c1b0d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | de592a6ef4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ff7086c0d0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ef0352ecd6 | ||
|   | 7348745049 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 2078044062 | ||
|   | d254937590 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9a8e52d1fc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6e7fac5493 | ||
|   | 129a37a1f4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 01382e774e | ||
|   | 9164d35615 | ||
|   | 58df65541c | ||
|   | 4c04f364a3 | ||
|   | 7f39538231 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | be98e0c0f4 | ||
|   | 9491b1ff89 | ||
|   | 30cbb039d0 | ||
|   | 1aabca9489 | ||
|   | 28a87db515 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 05b648629f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d1d8446480 | ||
|   | 8b897ba537 | ||
|   | c8f1b222c0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 257e2ceb82 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 67a27cae40 | ||
|   | 8ff9c08e82 | ||
|   | 1b0aa30881 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2a8d2d2b48 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 44bd787276 | ||
|   | 690f1c07a7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8e185a8413 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1f7df73964 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a10afc45b1 | ||
|   | 61a2101d8a | ||
|   | 088832c253 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a545b680b3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 805017eabf | ||
|   | b7412b0679 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fff3bfd01e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5f165a79ba | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0d3acd1aca | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 463f196472 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 52d5df6778 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ce75c85e65 | ||
|   | 12fd61142d | ||
|   | 0073227785 | ||
|   | 89a215cc1f | ||
|   | b2aece8208 | ||
|   | 600bf91c4f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | da6bdfa795 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5d4894a1ba | ||
|   | d4c047bd01 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6183b9719c | ||
|   | f02d67ee47 | ||
|   | bd156ebb53 | ||
|   | b07236b544 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5928a31fc4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3a71ea7003 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 96900b1f1b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 65b39661a6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 18251ae8ae | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c418e0ea76 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 74b009ccd7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d2631bf398 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c62358d851 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e3af04701a | ||
|   | c2f6e319f2 | ||
|   | 61b37877be | ||
|   | e72c5a037b | ||
|   | 578383411c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dbd37d6575 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c7cf1e7593 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c06fb069ab | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b6c2259bd7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d0b7cc8ab3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0f77021bcc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b44e6d8cd3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dfe9e94f87 | ||
|   | 53ccc5249a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5993818c16 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a631dea01a | ||
|   | c5b85b2831 | ||
|   | 3c1920e4e1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ca6ae7f4ce | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 031ad0dbe6 | ||
|   | d8101ddba8 | ||
|   | de68868788 | ||
|   | 90590ae2de | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5e6bef7189 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7ab5555087 | ||
|   | 02ceb713ea | ||
|   | 774aef74e8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 045454b597 | ||
|   | 829193fe84 | ||
|   | 1f893117cc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9008009727 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3bf3bffabf | ||
|   | d44e995aed | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5a22599b93 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ae60e947f3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8115fd98bc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3201061ada | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b68caecbce | ||
|   | 5e780293c7 | ||
|   | 6e32144e9a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9b52fee0a3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7af4b17430 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4195c0fb33 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8fe1cfbb20 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 623c532c9e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3a904383af | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 28299affef | ||
|   | 11ca772ada | ||
|   | 42e704d563 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ec7241c0fd | ||
|   | d11d59dd92 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7a55f58a5f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0b5b5f7fd4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 56f3d384d6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 29117bb90b | ||
|   | 5519f6a53b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a45d507bee | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0a663b5c27 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0f1fed525c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 209cddc843 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4e0de93096 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3b6c5d5d33 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0843971e95 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 12d7496cd1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ed34348c80 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fefb83558a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 93a0ae4030 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5394cff296 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ca3e6da943 | ||
|   | 756a5f8836 | ||
|   | a8e7bb670e | ||
|   | 687d7652a0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9f414ee9da | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 67c2f8eb83 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c033d5ce8d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fd056f3840 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e3488b8a08 | ||
|   | e1e5d3a8f2 | ||
|   | 473662e56d | ||
|   | b29bc23487 | ||
|   | 54817ef562 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dd8abf738e | ||
|   | 55e58d39d9 | ||
|   | ac5ce4cc9e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2525467a2e | ||
|   | 81066aab83 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 93f4b24e72 | ||
|   | 9a07ff7fc4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1a278f2590 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 93472ed6dd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dcaf2653b8 | ||
|   | 0714d7845a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8f2269d871 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c7487e004d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 09d3edf526 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9c99bf368f | ||
|   | 6f196c9dea | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fcac17f335 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f5a026cdd8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c6488c1ee3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f47d0d2867 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 96df335b36 | ||
|   | cc9a931baa | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 95c638991d | ||
|   | e2ada42001 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 22e50b4ace | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 334484de7f | ||
|   | 180a7c3990 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d5f33de808 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6539f0df6f | ||
|   | 1504278223 | ||
|   | 9f3767b23d | ||
|   | e0d7985369 | ||
|   | 2968a5717c | ||
|   | e2b25fe7ce | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8601f5c49a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 42279461e0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 409447d6ca | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5b313db49d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d64618600d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1ee01b1d5e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | af590202c3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 12ca2fb624 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ea95f83742 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e4d4da601c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0582f6fd39 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f254af8326 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3333770246 | ||
|   | ee5ded29ac | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f530db98ff | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 911f9d661f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9935eac146 | ||
|   | eae2c9e221 | ||
|   | 1a67fe8a83 | ||
|   | 3af565267b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d09460a971 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c65329442a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 48430dfa28 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 70e2de372d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 75784480ab | ||
|   | 8a70ba841d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 77733829d7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d4b67f1946 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 51ab138bb1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b81413c8b2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2ec33c6ef3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 68b2c38c7c | ||
|   | 1ca22799d1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 549dddcb11 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 131af90469 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c7c39da7c6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8310c426f0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bb8f91e39a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a359b9a3d5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e130ebad1f | ||
|   | f5b996b66c | ||
|   | 05e0c7c3ab | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6c1203e4bf | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5fbcaa8edd | ||
|   | 00d217b5f7 | ||
|   | c0e35376f3 | ||
|   | 2be84e1282 | ||
|   | 08f10c96ef | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 12f8ccdf02 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d63e78cf34 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 65d97ca924 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5770cafea9 | ||
|   | 0177cd9528 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 91a8fae9b5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f16a4ce3ef | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 306f63c75b | ||
|   | 2a0312318d | ||
|   | 695a23a454 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7366673eea | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 53fa0fe215 | ||
|   | 1ba621be60 | ||
|   | 5117364625 | ||
|   | 986b92aee4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 12d26b05af | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e6c9704505 | ||
|   | 8ab396d77c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8438448843 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 362edb9a61 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1ff53e1853 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cfd28dbb5c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cbec558289 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ca3a2937d0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3e67fc12c5 | ||
|   | f6faa18409 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 21ae2c2e54 | ||
|   | eb3986bea2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5d6738ced8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2f2fecddf2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 218ba3601e | ||
|   | 4c3f60c44b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cb85e5e464 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5b46235872 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 70f675ac82 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bf0c714ea4 | ||
|   | c95df56e8d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5f3d851954 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 10c69dcdae | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bdd81ce3a9 | ||
|   | 17ee234be4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 61034dfa7b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 185cd362fb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e2ca357774 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3dea7fc4e8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 01ba591bc9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 640b7d46e3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d6560c51ee | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3e9b1938c6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 44ce8de71f | ||
|   | 0bbd15bfda | ||
|   | 591b9a4d87 | ||
|   | 5ee7d16687 | ||
|   | 4ab4350c58 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4ea7133fa8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 627d67f9d0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | eb37655598 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 19b62dd0d4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b2ad1ceea3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c1545b5b78 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2c2f04ba85 | ||
|   | 77e7bf51b7 | ||
|   | a42d71dcef | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1ff0432f4d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 54afd6e1c8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 458c493a74 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8ac8ecb17e | ||
|   | eac167067e | ||
|   | aa7f4aafeb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d2183fa12b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 928f32bb4f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cbe21303c4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 94987c04b8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d4ba46a846 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1a22d83895 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6b73bf5c28 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c9c9451c36 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1882d448ea | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2f11c9c9e3 | ||
|   | 02bdc4b555 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1a1ee50d9d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 50dc09d1a9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 130efd340c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 00bc13c049 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3caad67f61 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 13783f0d4a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | eae97ba3f4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 134dad7357 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1c4d2e8dec | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f2d7be3aac | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d06edb2dd6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7fa15b334a | ||
|   | ffb4e2d6d7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bd8047ae9c | ||
|   | 49bc0624af | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5e1d764eb3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0064d93d75 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5a838ecfe7 | ||
|   | c37b5effd7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ca7f3e8acb | ||
|   | b0cdb91d5e | ||
|   | 4829eb8ae1 | ||
|   | 1bb814b793 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 918fcb7d62 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bbfd899564 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 12c4d9da87 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6b4fd9b6b8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 07c22f4a60 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 252e1e2ac0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b684c8673e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 547f42439d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c51ceb000f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4cbede1bc8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5eac8c7780 | ||
|   | ab78d87304 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 09166e3867 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8a5c813cdd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4200622f43 | ||
|   | c4452a85b4 | ||
|   | e57de4a3c1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9fd2c91c55 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fbd70013a8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8d18f3e66e | ||
|   | 5f5754e860 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 974c882b9a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a9ea90096b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 45c72c426e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4e5b75fe19 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3cd617e68f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ddff02f73b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b59347b3d3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1dc769076f | ||
|   | f150a19c0f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c4bc1e3824 | ||
|   | eca99b69db | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 043af72847 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 05c7b6c639 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3385c99f1f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 895117f857 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9e3135e2de | ||
|   | 9a1c517437 | ||
|   | c0c0c4b7ad | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | be6e39fed0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b384921ee0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0d05a6eae3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 430aef68c6 | ||
|   | eac6070e12 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6693b7c2e6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7898c3e433 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 420ecd064e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4289be53f8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 29b41b564e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 998eb69583 | ||
|   | 8ebc097ff4 | ||
|   | c05984ca49 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1a700c3013 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a9c92cdec8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | da8b938d5b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 71e91328f1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6356be4c52 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e26e5440b6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fecfbd1a3e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c00d6dfc76 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 85be66d90d | ||
|   | 1ac506b391 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f7738b77de | ||
|   | 824037bb7d | ||
|   | 221292ad14 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 16f8c75e9f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 90a37079f1 | ||
|   | 798092af5e | ||
|   | 2a622a929d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ca8eeaa68c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d1b8ac1249 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3f629c4d60 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3fa910e68b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e3cf2989c9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 136b2f402d | ||
|   | 8d18d2d9c6 | ||
|   | f18213361a | ||
|   | 18d9d32bca | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1246e429c9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 77bc46bc37 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ce16963c94 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a70e8cfe58 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ba922a1aaa | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b09230a884 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f1cb9ca08e | ||
|   | 06513e88c6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b4a79bd068 | ||
|   | dfd8fe84e0 | ||
|   | 4857c2e243 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7d384f6160 | ||
|   | 672a7621f9 | ||
|   | f0e2fb3f57 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8c3a520512 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 22e50d56db | ||
|   | a0735f3585 | ||
|   | 50a2e8fde3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 55ed63cc79 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 97e9dfff3f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 501c9579fb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f9aedadee6 | ||
|   | c3c17b2bc3 | ||
|   | a894c4589e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 56a8a1b5a1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | be3f7a6c37 | ||
|   | 906e400ab7 | ||
|   | a9265afd4c | ||
|   | d26058ac80 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ebd1f30606 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c78e077649 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 07619223b0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 25c326ec6c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | df167b94c2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3730908881 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 975dc1bc11 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 31409f0c32 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b19273227b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f89179fb03 | ||
|   | 90c971f9f1 | ||
|   | d685780a4a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b6bc8b7b7c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 92daba898f | ||
|   | 138843591e | ||
|   | 0814552b2a | ||
|   | 0e0fadd72d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5426bd4392 | ||
|   | 3520a65099 | ||
|   | b15a5c2c87 | ||
|   | a8af04ff82 | ||
|   | 2148de45a0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c4143dacee | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a8025e77b3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dd1e76be93 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 36f997959a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c1faed163a | ||
|   | 9ca927dbe7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 02c6011818 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2e96b16396 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 53b8de6c1c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | daea9f893c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d1b5b1734c | ||
|   | 74a5899626 | ||
|   | 202ebf6d4e | ||
|   | 2c7b417e25 | ||
|   | bb5e138134 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3a2c3e2f84 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d5be0c34ac | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ea5431ef2b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9c4cdcd11f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e5ef6333e4 | ||
|   | 98779a48b1 | ||
|   | 9d4848ee77 | ||
|   | 5126820619 | ||
|   | 8b5c808e8c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9c75996c40 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d524778e42 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 52d4bc660e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8884696a6c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d493ccde28 | ||
|   | 1ececaaaa2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 91b48ad432 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f3fe40a19f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cf4b29c425 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4344e14a9d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | df935ec423 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e7f9f7504e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5721b2353a | ||
|   | c9de846d0e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a598108c26 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5467aa399d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | da052b074a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 90c035edd0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fc4eb44a24 | ||
|   | a71111b378 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 52e0c7e484 | ||
|   | e32970f191 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 897cc36017 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d79c575860 | ||
|   | 1f19f84edd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 27c37b8b84 | ||
|   | 06a5dd3153 | ||
|   | b5bf270d22 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8e71d69a64 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 06edb6f8a8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | dca82ec0a1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9c82ce4103 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8a23a9eb1b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e1b7e515df | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c8ff335ed7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5736da8ab7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 060bba4dce | ||
|   | 4c573991d2 | ||
|   | 7fd6dce55f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1861d756e9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c36c041f5e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c3d877bdd2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1242030d4a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1626e74608 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b1b913777f | ||
|   | 190894010c | ||
|   | 765265723c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7e20502379 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 366fc30e9d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | aa91788a69 | ||
|   | 375789b019 | ||
|   | 140b769a42 | ||
|   | 88d718271d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6ed26cdd1f | ||
|   | d1851fa607 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e846157c52 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e190bb4c1a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 137fbe7acd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9ccdb2ae3a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f5f7515744 | ||
|   | ddadbec7e3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d24543e103 | ||
|   | f80c4c9565 | ||
|   | 480b383782 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d3efd4c24b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 67a0acffa2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 41b07da399 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a6ce55d5b5 | ||
|   | 98c01fe1b3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 51df986222 | ||
|   | 9c625f93a5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7101d47e2e | 
| @@ -1,38 +1,51 @@ | ||||
| { | ||||
|   "name": "Supervisor dev", | ||||
|   "image": "ghcr.io/home-assistant/devcontainer:supervisor", | ||||
|   "image": "ghcr.io/home-assistant/devcontainer:2-supervisor", | ||||
|   "containerEnv": { | ||||
|     "WORKSPACE_DIRECTORY": "${containerWorkspaceFolder}" | ||||
|   }, | ||||
|   "remoteEnv": { | ||||
|     "PATH": "${containerEnv:VIRTUAL_ENV}/bin:${containerEnv:PATH}" | ||||
|   }, | ||||
|   "appPort": ["9123:8123", "7357:4357"], | ||||
|   "postCreateCommand": "bash devcontainer_bootstrap", | ||||
|   "postCreateCommand": "bash devcontainer_setup", | ||||
|   "postStartCommand": "bash devcontainer_bootstrap", | ||||
|   "runArgs": ["-e", "GIT_EDITOR=code --wait", "--privileged"], | ||||
|   "customizations": { | ||||
|     "vscode": { | ||||
|       "extensions": [ | ||||
|         "ms-python.python", | ||||
|         "charliermarsh.ruff", | ||||
|         "ms-python.pylint", | ||||
|         "ms-python.vscode-pylance", | ||||
|         "visualstudioexptteam.vscodeintellicode", | ||||
|         "esbenp.prettier-vscode" | ||||
|         "redhat.vscode-yaml", | ||||
|         "esbenp.prettier-vscode", | ||||
|         "GitHub.vscode-pull-request-github" | ||||
|       ], | ||||
|       "settings": { | ||||
|         "python.defaultInterpreterPath": "/home/vscode/.local/ha-venv/bin/python", | ||||
|         "python.pythonPath": "/home/vscode/.local/ha-venv/bin/python", | ||||
|         "python.terminal.activateEnvInCurrentTerminal": true, | ||||
|         "python.testing.pytestArgs": ["--no-cov"], | ||||
|         "pylint.importStrategy": "fromEnvironment", | ||||
|         "editor.formatOnPaste": false, | ||||
|         "editor.formatOnSave": true, | ||||
|         "editor.formatOnType": true, | ||||
|         "files.trimTrailingWhitespace": true, | ||||
|         "terminal.integrated.profiles.linux": { | ||||
|           "zsh": { | ||||
|             "path": "/usr/bin/zsh" | ||||
|           } | ||||
|         }, | ||||
|         "terminal.integrated.defaultProfile.linux": "zsh", | ||||
|         "editor.formatOnPaste": false, | ||||
|         "editor.formatOnSave": true, | ||||
|         "editor.formatOnType": true, | ||||
|         "files.trimTrailingWhitespace": true, | ||||
|         "python.pythonPath": "/usr/local/bin/python3", | ||||
|         "python.formatting.provider": "black", | ||||
|         "python.formatting.blackArgs": ["--target-version", "py312"], | ||||
|         "python.formatting.blackPath": "/usr/local/bin/black" | ||||
|         "[python]": { | ||||
|           "editor.defaultFormatter": "charliermarsh.ruff" | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   }, | ||||
|   "mounts": ["type=volume,target=/var/lib/docker"] | ||||
|   "mounts": [ | ||||
|     "type=volume,target=/var/lib/docker", | ||||
|     "type=volume,target=/mnt/supervisor" | ||||
|   ] | ||||
| } | ||||
|   | ||||
							
								
								
									
										69
									
								
								.github/ISSUE_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										69
									
								
								.github/ISSUE_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -1,69 +0,0 @@ | ||||
| --- | ||||
| name: Report a bug with the Supervisor on a supported System | ||||
| about: Report an issue related to the Home Assistant Supervisor. | ||||
| labels: bug | ||||
| --- | ||||
|  | ||||
| <!-- READ THIS FIRST: | ||||
| - If you need additional help with this template please refer to https://www.home-assistant.io/help/reporting_issues/ | ||||
| - This is for bugs only. Feature and enhancement requests should go in our community forum: https://community.home-assistant.io/c/feature-requests | ||||
| - Provide as many details as possible. Paste logs, configuration sample and code into the backticks. Do not delete any text from this template! | ||||
| - If you have a problem with an add-on, make an issue in it's repository. | ||||
| --> | ||||
|  | ||||
| <!-- | ||||
| Important: You can only fill a bug repport for an supported system! If you run an unsupported installation. This report would be closed without comment. | ||||
| --> | ||||
|  | ||||
| ### Describe the issue | ||||
|  | ||||
| <!-- Provide as many details as possible. --> | ||||
|  | ||||
| ### Steps to reproduce | ||||
|  | ||||
| <!-- What do you do to encounter the issue. --> | ||||
|  | ||||
| 1. ... | ||||
| 2. ... | ||||
| 3. ... | ||||
|  | ||||
| ### Enviroment details | ||||
|  | ||||
| <!-- You can find these details in the system tab of the supervisor panel, or by using the `ha` CLI. --> | ||||
|  | ||||
| - **Operating System:**: xxx | ||||
| - **Supervisor version:**: xxx | ||||
| - **Home Assistant version**: xxx | ||||
|  | ||||
| ### Supervisor logs | ||||
|  | ||||
| <details> | ||||
| <summary>Supervisor logs</summary> | ||||
| <!-- | ||||
| - Frontend -> Supervisor -> System | ||||
| - Or use this command: ha supervisor logs | ||||
| - Logs are more than just errors, even if you don't think it's important, it is. | ||||
| --> | ||||
|  | ||||
| ``` | ||||
| Paste supervisor logs here | ||||
|  | ||||
| ``` | ||||
|  | ||||
| </details> | ||||
|  | ||||
| ### System Information | ||||
|  | ||||
| <details> | ||||
| <summary>System Information</summary> | ||||
| <!-- | ||||
| - Use this command: ha info | ||||
| --> | ||||
|  | ||||
| ``` | ||||
| Paste system info here | ||||
|  | ||||
| ``` | ||||
|  | ||||
| </details> | ||||
|  | ||||
							
								
								
									
										20
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,6 +1,5 @@ | ||||
| name: Bug Report Form | ||||
| name: Report an issue with Home Assistant Supervisor | ||||
| description: Report an issue related to the Home Assistant Supervisor. | ||||
| labels: bug | ||||
| body: | ||||
|   - type: markdown | ||||
|     attributes: | ||||
| @@ -9,7 +8,7 @@ body: | ||||
|  | ||||
|         If you have a feature or enhancement request, please use the [feature request][fr] section of our [Community Forum][fr]. | ||||
|  | ||||
|         [fr]: https://community.home-assistant.io/c/feature-requests | ||||
|         [fr]: https://github.com/orgs/home-assistant/discussions | ||||
|   - type: textarea | ||||
|     validations: | ||||
|       required: true | ||||
| @@ -26,7 +25,7 @@ body: | ||||
|     attributes: | ||||
|       label: What type of installation are you running? | ||||
|       description: > | ||||
|         If you don't know, can be found in [Settings -> System -> Repairs -> System Information](https://my.home-assistant.io/redirect/system_health/). | ||||
|         If you don't know, can be found in [Settings -> System -> Repairs -> (three dot menu) -> System Information](https://my.home-assistant.io/redirect/system_health/). | ||||
|         It is listed as the `Installation Type` value. | ||||
|       options: | ||||
|         - Home Assistant OS | ||||
| @@ -72,20 +71,21 @@ body: | ||||
|     validations: | ||||
|       required: true | ||||
|     attributes: | ||||
|       label: System Health information | ||||
|       label: System information | ||||
|       description: > | ||||
|         System Health information can be found in the top right menu in [Settings -> System -> Repairs](https://my.home-assistant.io/redirect/repairs/). | ||||
|         The System information can be found in [Settings -> System -> Repairs -> (three dot menu) -> System Information](https://my.home-assistant.io/redirect/system_health/). | ||||
|         Click the copy button at the bottom of the pop-up and paste it here. | ||||
|          | ||||
|  | ||||
|         [](https://my.home-assistant.io/redirect/system_health/) | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: Supervisor diagnostics | ||||
|       placeholder: "drag-and-drop the diagnostics data file here (do not copy-and-paste the content)" | ||||
|       description: >- | ||||
|         Supervisor diagnostics can be found in [Settings -> Integrations](https://my.home-assistant.io/redirect/integrations/). | ||||
|         Find the card that says `Home Assistant Supervisor`, open its menu and select 'Download diagnostics'. | ||||
|          | ||||
|         Supervisor diagnostics can be found in [Settings -> Devices & services](https://my.home-assistant.io/redirect/integrations/). | ||||
|         Find the card that says `Home Assistant Supervisor`, open it, and select the three dot menu of the Supervisor integration entry | ||||
|         and select 'Download diagnostics'. | ||||
|  | ||||
|         **Please drag-and-drop the downloaded file into the textbox below. Do not copy and paste its contents.** | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @@ -13,7 +13,7 @@ contact_links: | ||||
|     about: Our documentation has its own issue tracker. Please report issues with the website there. | ||||
|  | ||||
|   - name: Request a feature for the Supervisor | ||||
|     url: https://community.home-assistant.io/c/feature-requests | ||||
|     url: https://github.com/orgs/home-assistant/discussions | ||||
|     about: Request an new feature for the Supervisor. | ||||
|  | ||||
|   - name: I have a question or need support | ||||
|   | ||||
							
								
								
									
										53
									
								
								.github/ISSUE_TEMPLATE/task.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								.github/ISSUE_TEMPLATE/task.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | ||||
| name: Task | ||||
| description: For staff only - Create a task | ||||
| type: Task | ||||
| body: | ||||
|   - type: markdown | ||||
|     attributes: | ||||
|       value: | | ||||
|         ## ⚠️ RESTRICTED ACCESS | ||||
|  | ||||
|         **This form is restricted to Open Home Foundation staff and authorized contributors only.** | ||||
|  | ||||
|         If you are a community member wanting to contribute, please: | ||||
|         - For bug reports: Use the [bug report form](https://github.com/home-assistant/supervisor/issues/new?template=bug_report.yml) | ||||
|         - For feature requests: Submit to [Feature Requests](https://github.com/orgs/home-assistant/discussions) | ||||
|  | ||||
|         --- | ||||
|  | ||||
|         ### For authorized contributors | ||||
|  | ||||
|         Use this form to create tasks for development work, improvements, or other actionable items that need to be tracked. | ||||
|   - type: textarea | ||||
|     id: description | ||||
|     attributes: | ||||
|       label: Description | ||||
|       description: | | ||||
|         Provide a clear and detailed description of the task that needs to be accomplished. | ||||
|  | ||||
|         Be specific about what needs to be done, why it's important, and any constraints or requirements. | ||||
|       placeholder: | | ||||
|         Describe the task, including: | ||||
|         - What needs to be done | ||||
|         - Why this task is needed | ||||
|         - Expected outcome | ||||
|         - Any constraints or requirements | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: textarea | ||||
|     id: additional_context | ||||
|     attributes: | ||||
|       label: Additional context | ||||
|       description: | | ||||
|         Any additional information, links, research, or context that would be helpful. | ||||
|  | ||||
|         Include links to related issues, research, prototypes, roadmap opportunities etc. | ||||
|       placeholder: | | ||||
|         - Roadmap opportunity: [link] | ||||
|         - Epic: [link] | ||||
|         - Feature request: [link] | ||||
|         - Technical design documents: [link] | ||||
|         - Prototype/mockup: [link] | ||||
|         - Dependencies: [links] | ||||
|     validations: | ||||
|       required: false | ||||
							
								
								
									
										9
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -38,6 +38,7 @@ | ||||
| - This PR is related to issue: | ||||
| - Link to documentation pull request: | ||||
| - Link to cli pull request: | ||||
| - Link to client library pull request: | ||||
|  | ||||
| ## Checklist | ||||
|  | ||||
| @@ -52,12 +53,14 @@ | ||||
| - [ ] Local tests pass. **Your PR cannot be merged unless tests pass** | ||||
| - [ ] There is no commented out code in this PR. | ||||
| - [ ] I have followed the [development checklist][dev-checklist] | ||||
| - [ ] The code has been formatted using Black (`black --fast supervisor tests`) | ||||
| - [ ] The code has been formatted using Ruff (`ruff format supervisor tests`) | ||||
| - [ ] Tests have been added to verify that the new code works. | ||||
|  | ||||
| If API endpoints of add-on configuration are added/changed: | ||||
| If API endpoints or add-on configuration are added/changed: | ||||
|  | ||||
| - [ ] Documentation added/updated for [developers.home-assistant.io][docs-repository] | ||||
| - [ ] [CLI][cli-repository] updated (if necessary) | ||||
| - [ ] [Client library][client-library-repository] updated (if necessary) | ||||
|  | ||||
| <!-- | ||||
|   Thank you for contributing <3 | ||||
| @@ -67,3 +70,5 @@ If API endpoints of add-on configuration are added/changed: | ||||
|  | ||||
| [dev-checklist]: https://developers.home-assistant.io/docs/en/development_checklist.html | ||||
| [docs-repository]: https://github.com/home-assistant/developers.home-assistant | ||||
| [cli-repository]: https://github.com/home-assistant/cli | ||||
| [client-library-repository]: https://github.com/home-assistant-libs/python-supervisor-client/ | ||||
|   | ||||
							
								
								
									
										288
									
								
								.github/copilot-instructions.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										288
									
								
								.github/copilot-instructions.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,288 @@ | ||||
| # GitHub Copilot & Claude Code Instructions | ||||
|  | ||||
| This repository contains the Home Assistant Supervisor, a Python 3 based container | ||||
| orchestration and management system for Home Assistant. | ||||
|  | ||||
| ## Supervisor Capabilities & Features | ||||
|  | ||||
| ### Architecture Overview | ||||
|  | ||||
| Home Assistant Supervisor is a Python-based container orchestration system that | ||||
| communicates with the Docker daemon to manage containerized components. It is tightly | ||||
| integrated with the underlying Operating System and core Operating System components | ||||
| through D-Bus. | ||||
|  | ||||
| **Managed Components:** | ||||
| - **Home Assistant Core**: The main home automation application running in its own | ||||
|   container (also provides the web interface) | ||||
| - **Add-ons**: Third-party applications and services (each add-on runs in its own | ||||
|   container) | ||||
| - **Plugins**: Built-in system services like DNS, Audio, CLI, Multicast, and Observer | ||||
| - **Host System Integration**: OS-level operations and hardware access via D-Bus | ||||
| - **Container Networking**: Internal Docker network management and external | ||||
|   connectivity | ||||
| - **Storage & Backup**: Data persistence and backup management across all containers | ||||
|  | ||||
| **Key Dependencies:** | ||||
| - **Docker Engine**: Required for all container operations | ||||
| - **D-Bus**: System-level communication with the host OS | ||||
| - **systemd**: Service management for host system operations | ||||
| - **NetworkManager**: Network configuration and management | ||||
|  | ||||
| ### Add-on System | ||||
|  | ||||
| **Add-on Architecture**: Add-ons are containerized applications available through | ||||
| add-on stores. Each store contains multiple add-ons, and each add-on includes metadata | ||||
| that tells Supervisor the version, startup configuration (permissions), and available | ||||
| user configurable options. Add-on metadata typically references a container image that | ||||
| Supervisor fetches during installation. If not, the Supervisor builds the container | ||||
| image from a Dockerfile. | ||||
|  | ||||
| **Built-in Stores**: Supervisor comes with several pre-configured stores: | ||||
| - **Core Add-ons**: Official add-ons maintained by the Home Assistant team | ||||
| - **Community Add-ons**: Popular third-party add-ons repository | ||||
| - **ESPHome**: Add-ons for ESPHome ecosystem integration | ||||
| - **Music Assistant**: Audio and music-related add-ons | ||||
| - **Local Development**: Local folder for testing custom add-ons during development | ||||
|  | ||||
| **Store Management**: Stores are Git-based repositories that are periodically updated. | ||||
| When updates are available, users receive notifications. | ||||
|  | ||||
| **Add-on Lifecycle**: | ||||
| - **Installation**: Supervisor fetches or builds container images based on add-on | ||||
|   metadata | ||||
| - **Configuration**: Schema-validated options with integrated UI management | ||||
| - **Runtime**: Full container lifecycle management, health monitoring | ||||
| - **Updates**: Automatic or manual version management | ||||
|  | ||||
| ### Update System | ||||
|  | ||||
| **Core Components**: Supervisor, Home Assistant Core, HAOS, and built-in plugins | ||||
| receive version information from a central JSON file fetched from | ||||
| `https://version.home-assistant.io/{channel}.json`. The `Updater` class handles | ||||
| fetching this data, validating signatures, and updating internal version tracking. | ||||
|  | ||||
| **Update Channels**: Three channels (`stable`/`beta`/`dev`) determine which version | ||||
| JSON file is fetched, allowing users to opt into different release streams. | ||||
|  | ||||
| **Add-on Updates**: Add-on version information comes from store repository updates, not | ||||
| the central JSON file. When repositories are refreshed via the store system, add-ons | ||||
| compare their local versions against repository versions to determine update | ||||
| availability. | ||||
|  | ||||
| ### Backup & Recovery System | ||||
|  | ||||
| **Backup Capabilities**: | ||||
| - **Full Backups**: Complete system state capture including all add-ons, | ||||
|   configuration, and data | ||||
| - **Partial Backups**: Selective backup of specific components (Home Assistant, | ||||
|   add-ons, folders) | ||||
| - **Encrypted Backups**: Optional backup encryption with user-provided passwords | ||||
| - **Multiple Storage Locations**: Local storage and remote backup destinations | ||||
|  | ||||
| **Recovery Features**: | ||||
| - **One-click Restore**: Simple restoration from backup files | ||||
| - **Selective Restore**: Choose specific components to restore | ||||
| - **Automatic Recovery**: Self-healing for common system issues | ||||
|  | ||||
| --- | ||||
|  | ||||
| ## Supervisor Development | ||||
|  | ||||
| ### Python Requirements | ||||
|  | ||||
| - **Compatibility**: Python 3.13+ | ||||
| - **Language Features**: Use modern Python features: | ||||
|   - Type hints with `typing` module | ||||
|   - f-strings (preferred over `%` or `.format()`) | ||||
|   - Dataclasses and enum classes | ||||
|   - Async/await patterns | ||||
|   - Pattern matching where appropriate | ||||
|  | ||||
| ### Code Quality Standards | ||||
|  | ||||
| - **Formatting**: Ruff | ||||
| - **Linting**: PyLint and Ruff   | ||||
| - **Type Checking**: MyPy | ||||
| - **Testing**: pytest with asyncio support | ||||
| - **Language**: American English for all code, comments, and documentation | ||||
|  | ||||
| ### Code Organization | ||||
|  | ||||
| **Core Structure**: | ||||
| ``` | ||||
| supervisor/ | ||||
| ├── __init__.py           # Package initialization | ||||
| ├── const.py             # Constants and enums | ||||
| ├── coresys.py           # Core system management | ||||
| ├── bootstrap.py         # System initialization | ||||
| ├── exceptions.py        # Custom exception classes | ||||
| ├── api/                 # REST API endpoints | ||||
| ├── addons/              # Add-on management | ||||
| ├── backups/             # Backup system | ||||
| ├── docker/              # Docker integration | ||||
| ├── host/                # Host system interface | ||||
| ├── homeassistant/       # Home Assistant Core management | ||||
| ├── dbus/                # D-Bus system integration | ||||
| ├── hardware/            # Hardware detection and management | ||||
| ├── plugins/             # Plugin system | ||||
| ├── resolution/          # Issue detection and resolution | ||||
| ├── security/            # Security management | ||||
| ├── services/            # Service discovery and management | ||||
| ├── store/               # Add-on store management | ||||
| └── utils/               # Utility functions | ||||
| ``` | ||||
|  | ||||
| **Shared Constants**: Use constants from `supervisor/const.py` instead of hardcoding | ||||
| values. Define new constants following existing patterns and group related constants | ||||
| together. | ||||
|  | ||||
| ### Supervisor Architecture Patterns | ||||
|  | ||||
| **CoreSysAttributes Inheritance Pattern**: Nearly all major classes in Supervisor | ||||
| inherit from `CoreSysAttributes`, providing access to the centralized system state | ||||
| via `self.coresys` and convenient `sys_*` properties. | ||||
|  | ||||
| ```python | ||||
| # Standard Supervisor class pattern | ||||
| class MyManager(CoreSysAttributes): | ||||
|     """Manage my functionality.""" | ||||
|      | ||||
|     def __init__(self, coresys: CoreSys): | ||||
|         """Initialize manager.""" | ||||
|         self.coresys: CoreSys = coresys | ||||
|         self._component: MyComponent = MyComponent(coresys) | ||||
|      | ||||
|     @property | ||||
|     def component(self) -> MyComponent: | ||||
|         """Return component handler.""" | ||||
|         return self._component | ||||
|      | ||||
|     # Access system components via inherited properties | ||||
|     async def do_something(self): | ||||
|         await self.sys_docker.containers.get("my_container") | ||||
|         self.sys_bus.fire_event(BusEvent.MY_EVENT, {"data": "value"}) | ||||
| ``` | ||||
|  | ||||
| **Key Inherited Properties from CoreSysAttributes**: | ||||
| - `self.sys_docker` - Docker API access | ||||
| - `self.sys_run_in_executor()` - Execute blocking operations | ||||
| - `self.sys_create_task()` - Create async tasks | ||||
| - `self.sys_bus` - Event bus for system events | ||||
| - `self.sys_config` - System configuration | ||||
| - `self.sys_homeassistant` - Home Assistant Core management | ||||
| - `self.sys_addons` - Add-on management | ||||
| - `self.sys_host` - Host system access | ||||
| - `self.sys_dbus` - D-Bus system interface | ||||
|  | ||||
| **Load Pattern**: Many components implement a `load()` method which effectively | ||||
| initialize the component from external sources (containers, files, D-Bus services). | ||||
|  | ||||
| ### API Development | ||||
|  | ||||
| **REST API Structure**: | ||||
| - **Base Path**: `/api/` for all endpoints | ||||
| - **Authentication**: Bearer token authentication | ||||
| - **Consistent Response Format**: `{"result": "ok", "data": {...}}` or | ||||
|   `{"result": "error", "message": "..."}` | ||||
| - **Validation**: Use voluptuous schemas with `api_validate()` | ||||
|  | ||||
| **Use `@api_process` Decorator**: This decorator handles all standard error handling | ||||
| and response formatting automatically. The decorator catches `APIError`, `HassioError`, | ||||
| and other exceptions, returning appropriate HTTP responses. | ||||
|  | ||||
| ```python | ||||
| from ..api.utils import api_process, api_validate | ||||
|  | ||||
| @api_process | ||||
| async def backup_full(self, request: web.Request) -> dict[str, Any]: | ||||
|     """Create full backup.""" | ||||
|     body = await api_validate(SCHEMA_BACKUP_FULL, request) | ||||
|     job = await self.sys_backups.do_backup_full(**body) | ||||
|     return {ATTR_JOB_ID: job.uuid} | ||||
| ``` | ||||
|  | ||||
| ### Docker Integration | ||||
|  | ||||
| - **Container Management**: Use Supervisor's Docker manager instead of direct | ||||
|   Docker API | ||||
| - **Networking**: Supervisor manages internal Docker networks with predefined IP | ||||
|   ranges | ||||
| - **Security**: AppArmor profiles, capability restrictions, and user namespace | ||||
|   isolation | ||||
| - **Health Checks**: Implement health monitoring for all managed containers | ||||
|  | ||||
| ### D-Bus Integration | ||||
|  | ||||
| - **Use dbus-fast**: Async D-Bus library for system integration | ||||
| - **Service Management**: systemd, NetworkManager, hostname management | ||||
| - **Error Handling**: Wrap D-Bus exceptions in Supervisor-specific exceptions | ||||
|  | ||||
| ### Async Programming | ||||
|  | ||||
| - **All I/O operations must be async**: File operations, network calls, subprocess | ||||
|   execution | ||||
| - **Use asyncio patterns**: Prefer `asyncio.gather()` over sequential awaits | ||||
| - **Executor jobs**: Use `self.sys_run_in_executor()` for blocking operations | ||||
| - **Two-phase initialization**: `__init__` for sync setup, `post_init()` for async | ||||
|   initialization | ||||
|  | ||||
| ### Testing | ||||
|  | ||||
| - **Location**: `tests/` directory with module mirroring | ||||
| - **Fixtures**: Extensive use of pytest fixtures for CoreSys setup | ||||
| - **Mocking**: Mock external dependencies (Docker, D-Bus, network calls) | ||||
| - **Coverage**: Minimum 90% test coverage, 100% for security-sensitive code | ||||
|  | ||||
| ### Error Handling | ||||
|  | ||||
| - **Custom Exceptions**: Defined in `exceptions.py` with clear inheritance hierarchy | ||||
| - **Error Propagation**: Use `from` clause for exception chaining | ||||
| - **API Errors**: Use `APIError` with appropriate HTTP status codes | ||||
|  | ||||
| ### Security Considerations | ||||
|  | ||||
| - **Container Security**: AppArmor profiles mandatory for add-ons, minimal | ||||
|   capabilities | ||||
| - **Authentication**: Token-based API authentication with role-based access | ||||
| - **Data Protection**: Backup encryption, secure secret management, comprehensive | ||||
|   input validation | ||||
|  | ||||
| ### Development Commands | ||||
|  | ||||
| ```bash | ||||
| # Run tests, adjust paths as necessary | ||||
| pytest -qsx tests/ | ||||
|  | ||||
| # Linting and formatting | ||||
| ruff check supervisor/ | ||||
| ruff format supervisor/ | ||||
|  | ||||
| # Type checking | ||||
| mypy --ignore-missing-imports supervisor/ | ||||
|  | ||||
| # Pre-commit hooks | ||||
| pre-commit run --all-files | ||||
| ``` | ||||
|  | ||||
| Always run the pre-commit hooks at the end of code editing. | ||||
|  | ||||
| ### Common Patterns to Follow | ||||
|  | ||||
| **✅ Use These Patterns**: | ||||
| - Inherit from `CoreSysAttributes` for system access | ||||
| - Use `@api_process` decorator for API endpoints | ||||
| - Use `self.sys_run_in_executor()` for blocking operations | ||||
| - Access Docker via `self.sys_docker` not direct Docker API | ||||
| - Use constants from `const.py` instead of hardcoding | ||||
| - Store types in (per-module) `const.py` (e.g. supervisor/store/const.py) | ||||
|  | ||||
| **❌ Avoid These Patterns**: | ||||
| - Direct Docker API usage - use Supervisor's Docker manager | ||||
| - Blocking operations in async context (use asyncio alternatives) | ||||
| - Hardcoded values - use constants from `const.py` | ||||
| - Manual error handling in API endpoints - let `@api_process` handle it | ||||
|  | ||||
| This guide provides the foundation for contributing to Home Assistant Supervisor. | ||||
| Follow these patterns and guidelines to ensure code quality, security, and | ||||
| maintainability. | ||||
							
								
								
									
										33
									
								
								.github/workflows/builder.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										33
									
								
								.github/workflows/builder.yml
									
									
									
									
										vendored
									
									
								
							| @@ -33,7 +33,7 @@ on: | ||||
|       - setup.py | ||||
|  | ||||
| env: | ||||
|   DEFAULT_PYTHON: "3.12" | ||||
|   DEFAULT_PYTHON: "3.13" | ||||
|   BUILD_NAME: supervisor | ||||
|   BUILD_TYPE: supervisor | ||||
|  | ||||
| @@ -53,7 +53,7 @@ jobs: | ||||
|       requirements: ${{ steps.requirements.outputs.changed }} | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
| @@ -70,7 +70,7 @@ jobs: | ||||
|       - name: Get changed files | ||||
|         id: changed_files | ||||
|         if: steps.version.outputs.publish == 'false' | ||||
|         uses: masesgroup/retrieve-changed-files@v3.0.0 | ||||
|         uses: masesgroup/retrieve-changed-files@491e80760c0e28d36ca6240a27b1ccb8e1402c13 # v3.0.0 | ||||
|  | ||||
|       - name: Check if requirements files changed | ||||
|         id: requirements | ||||
| @@ -92,7 +92,7 @@ jobs: | ||||
|         arch: ${{ fromJson(needs.init.outputs.architectures) }} | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
| @@ -104,11 +104,12 @@ jobs: | ||||
|             echo "CARGO_NET_GIT_FETCH_WITH_CLI=true" | ||||
|           ) > .env_file | ||||
|  | ||||
|       # home-assistant/wheels doesn't support sha pinning | ||||
|       - name: Build wheels | ||||
|         if: needs.init.outputs.requirements == 'true' | ||||
|         uses: home-assistant/wheels@2024.01.0 | ||||
|         uses: home-assistant/wheels@2025.09.1 | ||||
|         with: | ||||
|           abi: cp312 | ||||
|           abi: cp313 | ||||
|           tag: musllinux_1_2 | ||||
|           arch: ${{ matrix.arch }} | ||||
|           wheels-key: ${{ secrets.WHEELS_KEY }} | ||||
| @@ -125,20 +126,20 @@ jobs: | ||||
|  | ||||
|       - name: Set up Python ${{ env.DEFAULT_PYTHON }} | ||||
|         if: needs.init.outputs.publish == 'true' | ||||
|         uses: actions/setup-python@v5.0.0 | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON }} | ||||
|  | ||||
|       - name: Install Cosign | ||||
|         if: needs.init.outputs.publish == 'true' | ||||
|         uses: sigstore/cosign-installer@v3.3.0 | ||||
|         uses: sigstore/cosign-installer@d7543c93d881b35a8faa02e8e3605f69b7a1ce62 # v3.10.0 | ||||
|         with: | ||||
|           cosign-release: "v2.0.2" | ||||
|           cosign-release: "v2.5.3" | ||||
|  | ||||
|       - name: Install dirhash and calc hash | ||||
|         if: needs.init.outputs.publish == 'true' | ||||
|         run: | | ||||
|           pip3 install dirhash | ||||
|           pip3 install setuptools dirhash | ||||
|           dir_hash="$(dirhash "${{ github.workspace }}/supervisor" -a sha256 --match "*.py")" | ||||
|           echo "${dir_hash}" > rootfs/supervisor.sha256 | ||||
|  | ||||
| @@ -149,7 +150,7 @@ jobs: | ||||
|  | ||||
|       - name: Login to GitHub Container Registry | ||||
|         if: needs.init.outputs.publish == 'true' | ||||
|         uses: docker/login-action@v3.0.0 | ||||
|         uses: docker/login-action@184bdaa0721073962dff0199f1fb9940f07167d1 # v3.5.0 | ||||
|         with: | ||||
|           registry: ghcr.io | ||||
|           username: ${{ github.repository_owner }} | ||||
| @@ -159,8 +160,9 @@ jobs: | ||||
|         if: needs.init.outputs.publish == 'false' | ||||
|         run: echo "BUILD_ARGS=--test" >> $GITHUB_ENV | ||||
|  | ||||
|       # home-assistant/builder doesn't support sha pinning | ||||
|       - name: Build supervisor | ||||
|         uses: home-assistant/builder@2024.01.0 | ||||
|         uses: home-assistant/builder@2025.09.0 | ||||
|         with: | ||||
|           args: | | ||||
|             $BUILD_ARGS \ | ||||
| @@ -178,7 +180,7 @@ jobs: | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         if: needs.init.outputs.publish == 'true' | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|  | ||||
|       - name: Initialize git | ||||
|         if: needs.init.outputs.publish == 'true' | ||||
| @@ -203,11 +205,12 @@ jobs: | ||||
|     timeout-minutes: 60 | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|  | ||||
|       # home-assistant/builder doesn't support sha pinning | ||||
|       - name: Build the Supervisor | ||||
|         if: needs.init.outputs.publish != 'true' | ||||
|         uses: home-assistant/builder@2024.01.0 | ||||
|         uses: home-assistant/builder@2025.09.0 | ||||
|         with: | ||||
|           args: | | ||||
|             --test \ | ||||
|   | ||||
							
								
								
									
										362
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										362
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -8,8 +8,9 @@ on: | ||||
|   pull_request: ~ | ||||
|  | ||||
| env: | ||||
|   DEFAULT_PYTHON: "3.12" | ||||
|   DEFAULT_PYTHON: "3.13" | ||||
|   PRE_COMMIT_CACHE: ~/.cache/pre-commit | ||||
|   MYPY_CACHE_VERSION: 1 | ||||
|  | ||||
| concurrency: | ||||
|   group: "${{ github.workflow }}-${{ github.ref }}" | ||||
| @@ -25,15 +26,15 @@ jobs: | ||||
|     name: Prepare Python dependencies | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python | ||||
|         id: python | ||||
|         uses: actions/setup-python@v5.0.0 | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON }} | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -47,7 +48,7 @@ jobs: | ||||
|           pip install -r requirements.txt -r requirements_tests.txt | ||||
|       - name: Restore pre-commit environment from cache | ||||
|         id: cache-precommit | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: ${{ env.PRE_COMMIT_CACHE }} | ||||
|           lookup-only: true | ||||
| @@ -61,21 +62,21 @@ jobs: | ||||
|           . venv/bin/activate | ||||
|           pre-commit install-hooks | ||||
|  | ||||
|   lint-black: | ||||
|     name: Check black | ||||
|   lint-ruff-format: | ||||
|     name: Check ruff-format | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.0.0 | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         id: python | ||||
|         with: | ||||
|           python-version: ${{ needs.prepare.outputs.python-version }} | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -85,10 +86,67 @@ jobs: | ||||
|         run: | | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Run black | ||||
|       - name: Restore pre-commit environment from cache | ||||
|         id: cache-precommit | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: ${{ env.PRE_COMMIT_CACHE }} | ||||
|           key: | | ||||
|             ${{ runner.os }}-pre-commit-${{ hashFiles('.pre-commit-config.yaml') }} | ||||
|       - name: Fail job if cache restore failed | ||||
|         if: steps.cache-venv.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Run ruff-format | ||||
|         run: | | ||||
|           . venv/bin/activate | ||||
|           black --target-version py312 --check supervisor tests setup.py | ||||
|           pre-commit run --hook-stage manual ruff-format --all-files --show-diff-on-failure | ||||
|         env: | ||||
|           RUFF_OUTPUT_FORMAT: github | ||||
|  | ||||
|   lint-ruff: | ||||
|     name: Check ruff | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         id: python | ||||
|         with: | ||||
|           python-version: ${{ needs.prepare.outputs.python-version }} | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
|             ${{ runner.os }}-venv-${{ needs.prepare.outputs.python-version }}-${{ hashFiles('requirements.txt') }}-${{ hashFiles('requirements_tests.txt') }} | ||||
|       - name: Fail job if Python cache restore failed | ||||
|         if: steps.cache-venv.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Restore pre-commit environment from cache | ||||
|         id: cache-precommit | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: ${{ env.PRE_COMMIT_CACHE }} | ||||
|           key: | | ||||
|             ${{ runner.os }}-pre-commit-${{ hashFiles('.pre-commit-config.yaml') }} | ||||
|       - name: Fail job if cache restore failed | ||||
|         if: steps.cache-venv.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Run ruff | ||||
|         run: | | ||||
|           . venv/bin/activate | ||||
|           pre-commit run --hook-stage manual ruff --all-files --show-diff-on-failure | ||||
|         env: | ||||
|           RUFF_OUTPUT_FORMAT: github | ||||
|  | ||||
|   lint-dockerfile: | ||||
|     name: Check Dockerfile | ||||
| @@ -96,7 +154,7 @@ jobs: | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Register hadolint problem matcher | ||||
|         run: | | ||||
|           echo "::add-matcher::.github/workflows/matchers/hadolint.json" | ||||
| @@ -111,15 +169,15 @@ jobs: | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.0.0 | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         id: python | ||||
|         with: | ||||
|           python-version: ${{ needs.prepare.outputs.python-version }} | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -131,7 +189,7 @@ jobs: | ||||
|           exit 1 | ||||
|       - name: Restore pre-commit environment from cache | ||||
|         id: cache-precommit | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: ${{ env.PRE_COMMIT_CACHE }} | ||||
|           key: | | ||||
| @@ -149,94 +207,21 @@ jobs: | ||||
|           . venv/bin/activate | ||||
|           pre-commit run --hook-stage manual check-executables-have-shebangs --all-files | ||||
|  | ||||
|   lint-flake8: | ||||
|     name: Check flake8 | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.0.0 | ||||
|         id: python | ||||
|         with: | ||||
|           python-version: ${{ needs.prepare.outputs.python-version }} | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
|             ${{ runner.os }}-venv-${{ needs.prepare.outputs.python-version }}-${{ hashFiles('requirements.txt') }}-${{ hashFiles('requirements_tests.txt') }} | ||||
|       - name: Fail job if Python cache restore failed | ||||
|         if: steps.cache-venv.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Register flake8 problem matcher | ||||
|         run: | | ||||
|           echo "::add-matcher::.github/workflows/matchers/flake8.json" | ||||
|       - name: Run flake8 | ||||
|         run: | | ||||
|           . venv/bin/activate | ||||
|           flake8 supervisor tests | ||||
|  | ||||
|   lint-isort: | ||||
|     name: Check isort | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.0.0 | ||||
|         id: python | ||||
|         with: | ||||
|           python-version: ${{ needs.prepare.outputs.python-version }} | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
|             ${{ runner.os }}-venv-${{ needs.prepare.outputs.python-version }}-${{ hashFiles('requirements.txt') }}-${{ hashFiles('requirements_tests.txt') }} | ||||
|       - name: Fail job if Python cache restore failed | ||||
|         if: steps.cache-venv.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Restore pre-commit environment from cache | ||||
|         id: cache-precommit | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         with: | ||||
|           path: ${{ env.PRE_COMMIT_CACHE }} | ||||
|           key: | | ||||
|             ${{ runner.os }}-pre-commit-${{ hashFiles('.pre-commit-config.yaml') }} | ||||
|       - name: Fail job if cache restore failed | ||||
|         if: steps.cache-venv.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Run isort | ||||
|         run: | | ||||
|           . venv/bin/activate | ||||
|           pre-commit run --hook-stage manual isort --all-files --show-diff-on-failure | ||||
|  | ||||
|   lint-json: | ||||
|     name: Check JSON | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.0.0 | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         id: python | ||||
|         with: | ||||
|           python-version: ${{ needs.prepare.outputs.python-version }} | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -248,7 +233,7 @@ jobs: | ||||
|           exit 1 | ||||
|       - name: Restore pre-commit environment from cache | ||||
|         id: cache-precommit | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: ${{ env.PRE_COMMIT_CACHE }} | ||||
|           key: | | ||||
| @@ -272,92 +257,15 @@ jobs: | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.0.0 | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         id: python | ||||
|         with: | ||||
|           python-version: ${{ needs.prepare.outputs.python-version }} | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
|             ${{ runner.os }}-venv-${{ needs.prepare.outputs.python-version }}-${{ hashFiles('requirements.txt') }}-${{ hashFiles('requirements_tests.txt') }} | ||||
|       - name: Fail job if Python cache restore failed | ||||
|         if: steps.cache-venv.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Register pylint problem matcher | ||||
|         run: | | ||||
|           echo "::add-matcher::.github/workflows/matchers/pylint.json" | ||||
|       - name: Run pylint | ||||
|         run: | | ||||
|           . venv/bin/activate | ||||
|           pylint supervisor tests | ||||
|  | ||||
|   lint-pyupgrade: | ||||
|     name: Check pyupgrade | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.0.0 | ||||
|         id: python | ||||
|         with: | ||||
|           python-version: ${{ needs.prepare.outputs.python-version }} | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
|             ${{ runner.os }}-venv-${{ needs.prepare.outputs.python-version }}-${{ hashFiles('requirements.txt') }}-${{ hashFiles('requirements_tests.txt') }} | ||||
|       - name: Fail job if Python cache restore failed | ||||
|         if: steps.cache-venv.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Restore pre-commit environment from cache | ||||
|         id: cache-precommit | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         with: | ||||
|           path: ${{ env.PRE_COMMIT_CACHE }} | ||||
|           key: | | ||||
|             ${{ runner.os }}-pre-commit-${{ hashFiles('.pre-commit-config.yaml') }} | ||||
|       - name: Fail job if cache restore failed | ||||
|         if: steps.cache-venv.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Run pyupgrade | ||||
|         run: | | ||||
|           . venv/bin/activate | ||||
|           pre-commit run --hook-stage manual pyupgrade --all-files --show-diff-on-failure | ||||
|  | ||||
|   pytest: | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: prepare | ||||
|     name: Run tests Python ${{ needs.prepare.outputs.python-version }} | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.0.0 | ||||
|         id: python | ||||
|         with: | ||||
|           python-version: ${{ needs.prepare.outputs.python-version }} | ||||
|       - name: Install Cosign | ||||
|         uses: sigstore/cosign-installer@v3.3.0 | ||||
|         with: | ||||
|           cosign-release: "v2.0.2" | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -370,7 +278,93 @@ jobs: | ||||
|       - name: Install additional system dependencies | ||||
|         run: | | ||||
|           sudo apt-get update | ||||
|           sudo apt-get install -y --no-install-recommends libpulse0 libudev1 dbus dbus-x11 | ||||
|           sudo apt-get install -y --no-install-recommends libpulse0 | ||||
|       - name: Register pylint problem matcher | ||||
|         run: | | ||||
|           echo "::add-matcher::.github/workflows/matchers/pylint.json" | ||||
|       - name: Run pylint | ||||
|         run: | | ||||
|           . venv/bin/activate | ||||
|           pylint supervisor tests | ||||
|  | ||||
|   mypy: | ||||
|     name: Check mypy | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         id: python | ||||
|         with: | ||||
|           python-version: ${{ needs.prepare.outputs.python-version }} | ||||
|       - name: Generate partial mypy restore key | ||||
|         id: generate-mypy-key | ||||
|         run: | | ||||
|           mypy_version=$(cat requirements_test.txt | grep mypy | cut -d '=' -f 3) | ||||
|           echo "version=$mypy_version" >> $GITHUB_OUTPUT | ||||
|           echo "key=mypy-${{ env.MYPY_CACHE_VERSION }}-$mypy_version-$(date -u '+%Y-%m-%dT%H:%M:%s')" >> $GITHUB_OUTPUT | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: >- | ||||
|             ${{ runner.os }}-venv-${{ needs.prepare.outputs.python-version }}-${{ hashFiles('requirements.txt') }}-${{ hashFiles('requirements_tests.txt') }} | ||||
|       - name: Fail job if Python cache restore failed | ||||
|         if: steps.cache-venv.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Restore mypy cache | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: .mypy_cache | ||||
|           key: >- | ||||
|             ${{ runner.os }}-mypy-${{ needs.prepare.outputs.python-version }}-${{ steps.generate-mypy-key.outputs.key }} | ||||
|           restore-keys: >- | ||||
|             ${{ runner.os }}-venv-${{ needs.prepare.outputs.python-version }}-mypy-${{ env.MYPY_CACHE_VERSION }}-${{ steps.generate-mypy-key.outputs.version }} | ||||
|       - name: Register mypy problem matcher | ||||
|         run: | | ||||
|           echo "::add-matcher::.github/workflows/matchers/mypy.json" | ||||
|       - name: Run mypy | ||||
|         run: | | ||||
|           . venv/bin/activate | ||||
|           mypy --ignore-missing-imports supervisor | ||||
|  | ||||
|   pytest: | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: prepare | ||||
|     name: Run tests Python ${{ needs.prepare.outputs.python-version }} | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         id: python | ||||
|         with: | ||||
|           python-version: ${{ needs.prepare.outputs.python-version }} | ||||
|       - name: Install Cosign | ||||
|         uses: sigstore/cosign-installer@d7543c93d881b35a8faa02e8e3605f69b7a1ce62 # v3.10.0 | ||||
|         with: | ||||
|           cosign-release: "v2.5.3" | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
|             ${{ runner.os }}-venv-${{ needs.prepare.outputs.python-version }}-${{ hashFiles('requirements.txt') }}-${{ hashFiles('requirements_tests.txt') }} | ||||
|       - name: Fail job if Python cache restore failed | ||||
|         if: steps.cache-venv.outputs.cache-hit != 'true' | ||||
|         run: | | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Install additional system dependencies | ||||
|         run: | | ||||
|           sudo apt-get update | ||||
|           sudo apt-get install -y --no-install-recommends libpulse0 libudev1 dbus-daemon | ||||
|       - name: Register Python problem matcher | ||||
|         run: | | ||||
|           echo "::add-matcher::.github/workflows/matchers/python.json" | ||||
| @@ -392,10 +386,11 @@ jobs: | ||||
|             -o console_output_style=count \ | ||||
|             tests | ||||
|       - name: Upload coverage artifact | ||||
|         uses: actions/upload-artifact@v4.0.0 | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         with: | ||||
|           name: coverage-${{ matrix.python-version }} | ||||
|           name: coverage | ||||
|           path: .coverage | ||||
|           include-hidden-files: true | ||||
|  | ||||
|   coverage: | ||||
|     name: Process test coverage | ||||
| @@ -403,15 +398,15 @@ jobs: | ||||
|     needs: ["pytest", "prepare"] | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.0.0 | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         id: python | ||||
|         with: | ||||
|           python-version: ${{ needs.prepare.outputs.python-version }} | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@v3.3.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -422,7 +417,10 @@ jobs: | ||||
|           echo "Failed to restore Python virtual environment from cache" | ||||
|           exit 1 | ||||
|       - name: Download all coverage artifacts | ||||
|         uses: actions/download-artifact@v4.1.1 | ||||
|         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||
|         with: | ||||
|           name: coverage | ||||
|           path: coverage/ | ||||
|       - name: Combine coverage results | ||||
|         run: | | ||||
|           . venv/bin/activate | ||||
| @@ -430,4 +428,4 @@ jobs: | ||||
|           coverage report | ||||
|           coverage xml | ||||
|       - name: Upload coverage to Codecov | ||||
|         uses: codecov/codecov-action@v3.1.4 | ||||
|         uses: codecov/codecov-action@5a1091511ad55cbe89839c7260b706298ca349f7 # v5.5.1 | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,7 +9,7 @@ jobs: | ||||
|   lock: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: dessant/lock-threads@v5.0.1 | ||||
|       - uses: dessant/lock-threads@1bf7ec25051fe7c00bdd17e6a7cf3d7bfb7dc771 # v5.0.1 | ||||
|         with: | ||||
|           github-token: ${{ github.token }} | ||||
|           issue-inactive-days: "30" | ||||
|   | ||||
							
								
								
									
										30
									
								
								.github/workflows/matchers/flake8.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										30
									
								
								.github/workflows/matchers/flake8.json
									
									
									
									
										vendored
									
									
								
							| @@ -1,30 +0,0 @@ | ||||
| { | ||||
|   "problemMatcher": [ | ||||
|     { | ||||
|       "owner": "flake8-error", | ||||
|       "severity": "error", | ||||
|       "pattern": [ | ||||
|         { | ||||
|           "regexp": "^(.*):(\\d+):(\\d+):\\s(E\\d{3}\\s.*)$", | ||||
|           "file": 1, | ||||
|           "line": 2, | ||||
|           "column": 3, | ||||
|           "message": 4 | ||||
|         } | ||||
|       ] | ||||
|     }, | ||||
|     { | ||||
|       "owner": "flake8-warning", | ||||
|       "severity": "warning", | ||||
|       "pattern": [ | ||||
|         { | ||||
|           "regexp": "^(.*):(\\d+):(\\d+):\\s([CDFNW]\\d{3}\\s.*)$", | ||||
|           "file": 1, | ||||
|           "line": 2, | ||||
|           "column": 3, | ||||
|           "message": 4 | ||||
|         } | ||||
|       ] | ||||
|     } | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										16
									
								
								.github/workflows/matchers/mypy.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								.github/workflows/matchers/mypy.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| { | ||||
|     "problemMatcher": [ | ||||
|         { | ||||
|             "owner": "mypy", | ||||
|             "pattern": [ | ||||
|                 { | ||||
|                     "regexp": "^(.+):(\\d+):\\s(error|warning):\\s(.+)$", | ||||
|                     "file": 1, | ||||
|                     "line": 2, | ||||
|                     "severity": 3, | ||||
|                     "message": 4 | ||||
|                 } | ||||
|             ] | ||||
|         } | ||||
|     ] | ||||
| } | ||||
							
								
								
									
										4
									
								
								.github/workflows/release-drafter.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/release-drafter.yml
									
									
									
									
										vendored
									
									
								
							| @@ -11,7 +11,7 @@ jobs: | ||||
|     name: Release Drafter | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
| @@ -36,7 +36,7 @@ jobs: | ||||
|           echo "version=$datepre.$newpost" >> "$GITHUB_OUTPUT" | ||||
|  | ||||
|       - name: Run Release Drafter | ||||
|         uses: release-drafter/release-drafter@v5.25.0 | ||||
|         uses: release-drafter/release-drafter@b1476f6e6eb133afa41ed8589daba6dc69b4d3f5 # v6.1.0 | ||||
|         with: | ||||
|           tag: ${{ steps.version.outputs.version }} | ||||
|           name: ${{ steps.version.outputs.version }} | ||||
|   | ||||
							
								
								
									
										58
									
								
								.github/workflows/restrict-task-creation.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								.github/workflows/restrict-task-creation.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | ||||
| name: Restrict task creation | ||||
|  | ||||
| # yamllint disable-line rule:truthy | ||||
| on: | ||||
|   issues: | ||||
|     types: [opened] | ||||
|  | ||||
| jobs: | ||||
|   check-authorization: | ||||
|     runs-on: ubuntu-latest | ||||
|     # Only run if this is a Task issue type (from the issue form) | ||||
|     if: github.event.issue.type.name == 'Task' | ||||
|     steps: | ||||
|       - name: Check if user is authorized | ||||
|         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | ||||
|         with: | ||||
|           script: | | ||||
|             const issueAuthor = context.payload.issue.user.login; | ||||
|  | ||||
|             // Check if user is an organization member | ||||
|             try { | ||||
|               await github.rest.orgs.checkMembershipForUser({ | ||||
|                 org: 'home-assistant', | ||||
|                 username: issueAuthor | ||||
|               }); | ||||
|               console.log(`✅ ${issueAuthor} is an organization member`); | ||||
|               return; // Authorized | ||||
|             } catch (error) { | ||||
|               console.log(`❌ ${issueAuthor} is not authorized to create Task issues`); | ||||
|             } | ||||
|  | ||||
|             // Close the issue with a comment | ||||
|             await github.rest.issues.createComment({ | ||||
|               owner: context.repo.owner, | ||||
|               repo: context.repo.repo, | ||||
|               issue_number: context.issue.number, | ||||
|               body: `Hi @${issueAuthor}, thank you for your contribution!\n\n` + | ||||
|                     `Task issues are restricted to Open Home Foundation staff and authorized contributors.\n\n` + | ||||
|                     `If you would like to:\n` + | ||||
|                     `- Report a bug: Please use the [bug report form](https://github.com/home-assistant/supervisor/issues/new?template=bug_report.yml)\n` + | ||||
|                     `- Request a feature: Please submit to [Feature Requests](https://github.com/orgs/home-assistant/discussions)\n\n` + | ||||
|                     `If you believe you should have access to create Task issues, please contact the maintainers.` | ||||
|             }); | ||||
|  | ||||
|             await github.rest.issues.update({ | ||||
|               owner: context.repo.owner, | ||||
|               repo: context.repo.repo, | ||||
|               issue_number: context.issue.number, | ||||
|               state: 'closed' | ||||
|             }); | ||||
|  | ||||
|             // Add a label to indicate this was auto-closed | ||||
|             await github.rest.issues.addLabels({ | ||||
|               owner: context.repo.owner, | ||||
|               repo: context.repo.repo, | ||||
|               issue_number: context.issue.number, | ||||
|               labels: ['auto-closed'] | ||||
|             }); | ||||
							
								
								
									
										4
									
								
								.github/workflows/sentry.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/sentry.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -10,9 +10,9 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.1.1 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Sentry Release | ||||
|         uses: getsentry/action-release@v1.6.0 | ||||
|         uses: getsentry/action-release@4f502acc1df792390abe36f2dcb03612ef144818 # v3.3.0 | ||||
|         env: | ||||
|           SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }} | ||||
|           SENTRY_ORG: ${{ secrets.SENTRY_ORG }} | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,7 +9,7 @@ jobs: | ||||
|   stale: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/stale@v9.0.0 | ||||
|       - uses: actions/stale@3a9db7e6a41a89f618792c92c0e97cc736e1b13f # v10.0.0 | ||||
|         with: | ||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           days-before-stale: 30 | ||||
|   | ||||
							
								
								
									
										82
									
								
								.github/workflows/update_frontend.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								.github/workflows/update_frontend.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,82 @@ | ||||
| name: Update frontend | ||||
|  | ||||
| on: | ||||
|   schedule: # once a day | ||||
|     - cron: "0 0 * * *" | ||||
|   workflow_dispatch: | ||||
|  | ||||
| jobs: | ||||
|   check-version: | ||||
|     runs-on: ubuntu-latest | ||||
|     outputs: | ||||
|       skip: ${{ steps.check_version.outputs.skip || steps.check_existing_pr.outputs.skip }} | ||||
|       current_version: ${{ steps.check_version.outputs.current_version }} | ||||
|       latest_version: ${{ steps.latest_frontend_version.outputs.latest_tag }} | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Get latest frontend release | ||||
|         id: latest_frontend_version | ||||
|         uses: abatilo/release-info-action@32cb932219f1cee3fc4f4a298fd65ead5d35b661 # v1.3.3 | ||||
|         with: | ||||
|           owner: home-assistant | ||||
|           repo: frontend | ||||
|       - name: Check if version is up to date | ||||
|         id: check_version | ||||
|         run: | | ||||
|           current_version="$(cat .ha-frontend-version)" | ||||
|           latest_version="${{ steps.latest_frontend_version.outputs.latest_tag }}" | ||||
|           echo "current_version=${current_version}" >> $GITHUB_OUTPUT | ||||
|           echo "LATEST_VERSION=${latest_version}" >> $GITHUB_ENV | ||||
|           if [[ ! "$current_version" < "$latest_version" ]]; then | ||||
|             echo "Frontend version is up to date" | ||||
|             echo "skip=true" >> $GITHUB_OUTPUT | ||||
|           fi | ||||
|       - name: Check if there is no open PR with this version | ||||
|         if: steps.check_version.outputs.skip != 'true' | ||||
|         id: check_existing_pr | ||||
|         env: | ||||
|           GH_TOKEN: ${{ github.token }} | ||||
|         run: | | ||||
|           PR=$(gh pr list --state open --base main --json title --search "Update frontend to version $LATEST_VERSION") | ||||
|           if [[ "$PR" != "[]" ]]; then | ||||
|             echo "Skipping - There is already a PR open for version $LATEST_VERSION" | ||||
|             echo "skip=true" >> $GITHUB_OUTPUT | ||||
|           fi | ||||
|   create-pr: | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: check-version | ||||
|     if: needs.check-version.outputs.skip != 'true' | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Clear www folder | ||||
|         run: | | ||||
|           rm -rf supervisor/api/panel/* | ||||
|       - name: Update version file | ||||
|         run: | | ||||
|           echo "${{ needs.check-version.outputs.latest_version }}" > .ha-frontend-version | ||||
|       - name: Download release assets | ||||
|         uses: robinraju/release-downloader@daf26c55d821e836577a15f77d86ddc078948b05 # v1.12 | ||||
|         with: | ||||
|           repository: 'home-assistant/frontend' | ||||
|           tag: ${{ needs.check-version.outputs.latest_version }} | ||||
|           fileName: home_assistant_frontend_supervisor-${{ needs.check-version.outputs.latest_version }}.tar.gz | ||||
|           extract: true | ||||
|           out-file-path: supervisor/api/panel/ | ||||
|       - name: Remove release assets archive | ||||
|         run: | | ||||
|           rm -f supervisor/api/panel/home_assistant_frontend_supervisor-*.tar.gz | ||||
|       - name: Create PR | ||||
|         uses: peter-evans/create-pull-request@271a8d0340265f705b14b6d32b9829c1cb33d45e # v7.0.8 | ||||
|         with: | ||||
|           commit-message: "Update frontend to version ${{ needs.check-version.outputs.latest_version }}" | ||||
|           branch: autoupdate-frontend | ||||
|           base: main | ||||
|           draft: true | ||||
|           sign-commits: true | ||||
|           title: "Update frontend to version ${{ needs.check-version.outputs.latest_version }}" | ||||
|           body: > | ||||
|             Update frontend from ${{ needs.check-version.outputs.current_version }} to | ||||
|             [${{ needs.check-version.outputs.latest_version }}](https://github.com/home-assistant/frontend/releases/tag/${{ needs.check-version.outputs.latest_version }}) | ||||
|  | ||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -100,3 +100,6 @@ ENV/ | ||||
| # mypy | ||||
| /.mypy_cache/* | ||||
| /.dmypy.json | ||||
|  | ||||
| # Mac | ||||
| .DS_Store | ||||
							
								
								
									
										4
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +0,0 @@ | ||||
| [submodule "home-assistant-polymer"] | ||||
| 	path = home-assistant-polymer | ||||
| 	url = https://github.com/home-assistant/home-assistant-polymer | ||||
| 	branch = dev | ||||
							
								
								
									
										1
									
								
								.ha-frontend-version
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.ha-frontend-version
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| 20250925.1 | ||||
| @@ -1,34 +1,27 @@ | ||||
| repos: | ||||
|   - repo: https://github.com/psf/black | ||||
|     rev: 23.12.1 | ||||
|   - repo: https://github.com/astral-sh/ruff-pre-commit | ||||
|     rev: v0.11.10 | ||||
|     hooks: | ||||
|       - id: black | ||||
|       - id: ruff | ||||
|         args: | ||||
|           - --safe | ||||
|           - --quiet | ||||
|           - --target-version | ||||
|           - py312 | ||||
|           - --fix | ||||
|       - id: ruff-format | ||||
|         files: ^((supervisor|tests)/.+)?[^/]+\.py$ | ||||
|   - repo: https://github.com/PyCQA/flake8 | ||||
|     rev: 7.0.0 | ||||
|     hooks: | ||||
|       - id: flake8 | ||||
|         additional_dependencies: | ||||
|           - flake8-docstrings==1.7.0 | ||||
|           - pydocstyle==6.3.0 | ||||
|         files: ^(supervisor|script|tests)/.+\.py$ | ||||
|   - repo: https://github.com/pre-commit/pre-commit-hooks | ||||
|     rev: v4.5.0 | ||||
|     rev: v5.0.0 | ||||
|     hooks: | ||||
|       - id: check-executables-have-shebangs | ||||
|         stages: [manual] | ||||
|       - id: check-json | ||||
|   - repo: https://github.com/PyCQA/isort | ||||
|     rev: 5.13.2 | ||||
|   - repo: local | ||||
|     hooks: | ||||
|       - id: isort | ||||
|   - repo: https://github.com/asottile/pyupgrade | ||||
|     rev: v3.15.0 | ||||
|     hooks: | ||||
|       - id: pyupgrade | ||||
|         args: [--py312-plus] | ||||
|       # Run mypy through our wrapper script in order to get the possible | ||||
|       # pyenv and/or virtualenv activated; it may not have been e.g. if | ||||
|       # committing from a GUI tool that was not launched from an activated | ||||
|       # shell. | ||||
|       - id: mypy | ||||
|         name: mypy | ||||
|         entry: script/run-in-env.sh mypy --ignore-missing-imports | ||||
|         language: script | ||||
|         types_or: [python, pyi] | ||||
|         files: ^supervisor/.+\.(py|pyi)$ | ||||
|   | ||||
							
								
								
									
										18
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										18
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							| @@ -58,9 +58,23 @@ | ||||
|       "problemMatcher": [] | ||||
|     }, | ||||
|     { | ||||
|       "label": "Flake8", | ||||
|       "label": "Ruff Check", | ||||
|       "type": "shell", | ||||
|       "command": "flake8 supervisor tests", | ||||
|       "command": "ruff check --fix supervisor tests", | ||||
|       "group": { | ||||
|         "kind": "test", | ||||
|         "isDefault": true | ||||
|       }, | ||||
|       "presentation": { | ||||
|         "reveal": "always", | ||||
|         "panel": "new" | ||||
|       }, | ||||
|       "problemMatcher": [] | ||||
|     }, | ||||
|     { | ||||
|       "label": "Ruff Format", | ||||
|       "type": "shell", | ||||
|       "command": "ruff format supervisor tests", | ||||
|       "group": { | ||||
|         "kind": "test", | ||||
|         "isDefault": true | ||||
|   | ||||
							
								
								
									
										20
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -4,11 +4,13 @@ FROM ${BUILD_FROM} | ||||
| ENV \ | ||||
|     S6_SERVICES_GRACETIME=10000 \ | ||||
|     SUPERVISOR_API=http://localhost \ | ||||
|     CRYPTOGRAPHY_OPENSSL_NO_LEGACY=1 | ||||
|     CRYPTOGRAPHY_OPENSSL_NO_LEGACY=1 \ | ||||
|     UV_SYSTEM_PYTHON=true | ||||
|  | ||||
| ARG \ | ||||
|     COSIGN_VERSION \ | ||||
|     BUILD_ARCH | ||||
|     BUILD_ARCH \ | ||||
|     QEMU_CPU | ||||
|  | ||||
| # Install base | ||||
| WORKDIR /usr/src | ||||
| @@ -26,20 +28,24 @@ RUN \ | ||||
|         yaml \ | ||||
|     \ | ||||
|     && curl -Lso /usr/bin/cosign "https://github.com/home-assistant/cosign/releases/download/${COSIGN_VERSION}/cosign_${BUILD_ARCH}" \ | ||||
|     && chmod a+x /usr/bin/cosign | ||||
|     && chmod a+x /usr/bin/cosign \ | ||||
|     && pip3 install uv==0.8.9 | ||||
|  | ||||
| # Install requirements | ||||
| COPY requirements.txt . | ||||
| RUN \ | ||||
|     export MAKEFLAGS="-j$(nproc)" \ | ||||
|     && pip3 install --only-binary=:all: \ | ||||
|         -r ./requirements.txt \ | ||||
|     if [ "${BUILD_ARCH}" = "i386" ]; then \ | ||||
|         setarch="linux32"; \ | ||||
|     else \ | ||||
|         setarch=""; \ | ||||
|     fi \ | ||||
|     && ${setarch} uv pip install --compile-bytecode --no-cache --no-build -r requirements.txt \ | ||||
|     && rm -f requirements.txt | ||||
|  | ||||
| # Install Home Assistant Supervisor | ||||
| COPY . supervisor | ||||
| RUN \ | ||||
|     pip3 install -e ./supervisor \ | ||||
|     uv pip install --no-cache -e ./supervisor \ | ||||
|     && python3 -m compileall ./supervisor/supervisor | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -30,3 +30,5 @@ Releases are done in 3 stages (channels) with this structure: | ||||
|  | ||||
| [development]: https://developers.home-assistant.io/docs/supervisor/development | ||||
| [stable]: https://github.com/home-assistant/version/blob/master/stable.json | ||||
|  | ||||
| [](https://www.openhomefoundation.org/) | ||||
|   | ||||
							
								
								
									
										12
									
								
								build.yaml
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								build.yaml
									
									
									
									
									
								
							| @@ -1,10 +1,10 @@ | ||||
| image: ghcr.io/home-assistant/{arch}-hassio-supervisor | ||||
| build_from: | ||||
|   aarch64: ghcr.io/home-assistant/aarch64-base-python:3.12-alpine3.18 | ||||
|   armhf: ghcr.io/home-assistant/armhf-base-python:3.12-alpine3.18 | ||||
|   armv7: ghcr.io/home-assistant/armv7-base-python:3.12-alpine3.18 | ||||
|   amd64: ghcr.io/home-assistant/amd64-base-python:3.12-alpine3.18 | ||||
|   i386: ghcr.io/home-assistant/i386-base-python:3.12-alpine3.18 | ||||
|   aarch64: ghcr.io/home-assistant/aarch64-base-python:3.13-alpine3.22 | ||||
|   armhf: ghcr.io/home-assistant/armhf-base-python:3.13-alpine3.22 | ||||
|   armv7: ghcr.io/home-assistant/armv7-base-python:3.13-alpine3.22 | ||||
|   amd64: ghcr.io/home-assistant/amd64-base-python:3.13-alpine3.22 | ||||
|   i386: ghcr.io/home-assistant/i386-base-python:3.13-alpine3.22 | ||||
| codenotary: | ||||
|   signer: notary@home-assistant.io | ||||
|   base_image: notary@home-assistant.io | ||||
| @@ -12,7 +12,7 @@ cosign: | ||||
|   base_identity: https://github.com/home-assistant/docker-base/.* | ||||
|   identity: https://github.com/home-assistant/supervisor/.* | ||||
| args: | ||||
|   COSIGN_VERSION: 2.0.2 | ||||
|   COSIGN_VERSION: 2.5.3 | ||||
| labels: | ||||
|   io.hass.type: supervisor | ||||
|   org.opencontainers.image.title: Home Assistant Supervisor | ||||
|   | ||||
 Submodule home-assistant-polymer deleted from 9d457d52e8
									
								
							
							
								
								
									
										298
									
								
								pyproject.toml
									
									
									
									
									
								
							
							
						
						
									
										298
									
								
								pyproject.toml
									
									
									
									
									
								
							| @@ -1,5 +1,5 @@ | ||||
| [build-system] | ||||
| requires = ["setuptools~=68.0.0", "wheel~=0.40.0"] | ||||
| requires = ["setuptools~=80.9.0", "wheel~=0.46.1"] | ||||
| build-backend = "setuptools.build_meta" | ||||
|  | ||||
| [project] | ||||
| @@ -12,7 +12,7 @@ authors = [ | ||||
|     { name = "The Home Assistant Authors", email = "hello@home-assistant.io" }, | ||||
| ] | ||||
| keywords = ["docker", "home-assistant", "api"] | ||||
| requires-python = ">=3.12.0" | ||||
| requires-python = ">=3.13.0" | ||||
|  | ||||
| [project.urls] | ||||
| "Homepage" = "https://www.home-assistant.io/" | ||||
| @@ -31,7 +31,7 @@ include-package-data = true | ||||
| include = ["supervisor*"] | ||||
|  | ||||
| [tool.pylint.MAIN] | ||||
| py-version = "3.11" | ||||
| py-version = "3.13" | ||||
| # Use a conservative default here; 2 should speed up most setups and not hurt | ||||
| # any too bad. Override on command line as appropriate. | ||||
| jobs = 2 | ||||
| @@ -44,7 +44,7 @@ good-names = ["id", "i", "j", "k", "ex", "Run", "_", "fp", "T", "os"] | ||||
|  | ||||
| [tool.pylint."MESSAGES CONTROL"] | ||||
| # Reasons disabled: | ||||
| # format - handled by black | ||||
| # format - handled by ruff | ||||
| # abstract-method - with intro of async there are always methods missing | ||||
| # cyclic-import - doesn't test if both import on load | ||||
| # duplicate-code - unavoidable | ||||
| @@ -71,6 +71,136 @@ disable = [ | ||||
|     "too-many-statements", | ||||
|     "unused-argument", | ||||
|     "consider-using-with", | ||||
|  | ||||
|     # Handled by ruff | ||||
|     # Ref: <https://github.com/astral-sh/ruff/issues/970> | ||||
|     "await-outside-async",    # PLE1142 | ||||
|     "bad-str-strip-call",     # PLE1310 | ||||
|     "bad-string-format-type", # PLE1307 | ||||
|     "bidirectional-unicode",  # PLE2502 | ||||
|     "continue-in-finally",    # PLE0116 | ||||
|     "duplicate-bases",        # PLE0241 | ||||
|     "format-needs-mapping",   # F502 | ||||
|     "function-redefined",     # F811 | ||||
|     # Needed because ruff does not understand type of __all__ generated by a function | ||||
|     # "invalid-all-format", # PLE0605 | ||||
|     "invalid-all-object",                 # PLE0604 | ||||
|     "invalid-character-backspace",        # PLE2510 | ||||
|     "invalid-character-esc",              # PLE2513 | ||||
|     "invalid-character-nul",              # PLE2514 | ||||
|     "invalid-character-sub",              # PLE2512 | ||||
|     "invalid-character-zero-width-space", # PLE2515 | ||||
|     "logging-too-few-args",               # PLE1206 | ||||
|     "logging-too-many-args",              # PLE1205 | ||||
|     "missing-format-string-key",          # F524 | ||||
|     "mixed-format-string",                # F506 | ||||
|     "no-method-argument",                 # N805 | ||||
|     "no-self-argument",                   # N805 | ||||
|     "nonexistent-operator",               # B002 | ||||
|     "nonlocal-without-binding",           # PLE0117 | ||||
|     "not-in-loop",                        # F701, F702 | ||||
|     "notimplemented-raised",              # F901 | ||||
|     "return-in-init",                     # PLE0101 | ||||
|     "return-outside-function",            # F706 | ||||
|     "syntax-error",                       # E999 | ||||
|     "too-few-format-args",                # F524 | ||||
|     "too-many-format-args",               # F522 | ||||
|     "too-many-star-expressions",          # F622 | ||||
|     "truncated-format-string",            # F501 | ||||
|     "undefined-all-variable",             # F822 | ||||
|     "undefined-variable",                 # F821 | ||||
|     "used-prior-global-declaration",      # PLE0118 | ||||
|     "yield-inside-async-function",        # PLE1700 | ||||
|     "yield-outside-function",             # F704 | ||||
|     "anomalous-backslash-in-string",      # W605 | ||||
|     "assert-on-string-literal",           # PLW0129 | ||||
|     "assert-on-tuple",                    # F631 | ||||
|     "bad-format-string",                  # W1302, F | ||||
|     "bad-format-string-key",              # W1300, F | ||||
|     "bare-except",                        # E722 | ||||
|     "binary-op-exception",                # PLW0711 | ||||
|     "cell-var-from-loop",                 # B023 | ||||
|     # "dangerous-default-value", # B006, ruff catches new occurrences, needs more work | ||||
|     "duplicate-except",                     # B014 | ||||
|     "duplicate-key",                        # F601 | ||||
|     "duplicate-string-formatting-argument", # F | ||||
|     "duplicate-value",                      # F | ||||
|     "eval-used",                            # PGH001 | ||||
|     "exec-used",                            # S102 | ||||
|     # "expression-not-assigned", # B018, ruff catches new occurrences, needs more work | ||||
|     "f-string-without-interpolation",      # F541 | ||||
|     "forgotten-debug-statement",           # T100 | ||||
|     "format-string-without-interpolation", # F | ||||
|     # "global-statement", # PLW0603, ruff catches new occurrences, needs more work | ||||
|     "global-variable-not-assigned",  # PLW0602 | ||||
|     "implicit-str-concat",           # ISC001 | ||||
|     "import-self",                   # PLW0406 | ||||
|     "inconsistent-quotes",           # Q000 | ||||
|     "invalid-envvar-default",        # PLW1508 | ||||
|     "keyword-arg-before-vararg",     # B026 | ||||
|     "logging-format-interpolation",  # G | ||||
|     "logging-fstring-interpolation", # G | ||||
|     "logging-not-lazy",              # G | ||||
|     "misplaced-future",              # F404 | ||||
|     "named-expr-without-context",    # PLW0131 | ||||
|     "nested-min-max",                # PLW3301 | ||||
|     # "pointless-statement", # B018, ruff catches new occurrences, needs more work | ||||
|     "raise-missing-from", # TRY200 | ||||
|     # "redefined-builtin", # A001, ruff is way more stricter, needs work | ||||
|     "try-except-raise",               # TRY203 | ||||
|     "unused-argument",                # ARG001, we don't use it | ||||
|     "unused-format-string-argument",  #F507 | ||||
|     "unused-format-string-key",       # F504 | ||||
|     "unused-import",                  # F401 | ||||
|     "unused-variable",                # F841 | ||||
|     "useless-else-on-loop",           # PLW0120 | ||||
|     "wildcard-import",                # F403 | ||||
|     "bad-classmethod-argument",       # N804 | ||||
|     "consider-iterating-dictionary",  # SIM118 | ||||
|     "empty-docstring",                # D419 | ||||
|     "invalid-name",                   # N815 | ||||
|     "line-too-long",                  # E501, disabled globally | ||||
|     "missing-class-docstring",        # D101 | ||||
|     "missing-final-newline",          # W292 | ||||
|     "missing-function-docstring",     # D103 | ||||
|     "missing-module-docstring",       # D100 | ||||
|     "multiple-imports",               #E401 | ||||
|     "singleton-comparison",           # E711, E712 | ||||
|     "subprocess-run-check",           # PLW1510 | ||||
|     "superfluous-parens",             # UP034 | ||||
|     "ungrouped-imports",              # I001 | ||||
|     "unidiomatic-typecheck",          # E721 | ||||
|     "unnecessary-direct-lambda-call", # PLC3002 | ||||
|     "unnecessary-lambda-assignment",  # PLC3001 | ||||
|     "unneeded-not",                   # SIM208 | ||||
|     "useless-import-alias",           # PLC0414 | ||||
|     "wrong-import-order",             # I001 | ||||
|     "wrong-import-position",          # E402 | ||||
|     "comparison-of-constants",        # PLR0133 | ||||
|     "comparison-with-itself",         # PLR0124 | ||||
|     # "consider-alternative-union-syntax", # UP007, typing extension | ||||
|     "consider-merging-isinstance", # PLR1701 | ||||
|     # "consider-using-alias",              # UP006, typing extension | ||||
|     "consider-using-dict-comprehension", # C402 | ||||
|     "consider-using-generator",          # C417 | ||||
|     "consider-using-get",                # SIM401 | ||||
|     "consider-using-set-comprehension",  # C401 | ||||
|     "consider-using-sys-exit",           # PLR1722 | ||||
|     "consider-using-ternary",            # SIM108 | ||||
|     "literal-comparison",                # F632 | ||||
|     "property-with-parameters",          # PLR0206 | ||||
|     "super-with-arguments",              # UP008 | ||||
|     "too-many-branches",                 # PLR0912 | ||||
|     "too-many-return-statements",        # PLR0911 | ||||
|     "too-many-statements",               # PLR0915 | ||||
|     "trailing-comma-tuple",              # COM818 | ||||
|     "unnecessary-comprehension",         # C416 | ||||
|     "use-a-generator",                   # C417 | ||||
|     "use-dict-literal",                  # C406 | ||||
|     "use-list-literal",                  # C405 | ||||
|     "useless-object-inheritance",        # UP004 | ||||
|     "useless-return",                    # PLR1711 | ||||
|     # "no-self-use", # PLR6301  # Optional plugin, not enabled | ||||
| ] | ||||
|  | ||||
| [tool.pylint.REPORTS] | ||||
| @@ -85,28 +215,162 @@ expected-line-ending-format = "LF" | ||||
| [tool.pylint.EXCEPTIONS] | ||||
| overgeneral-exceptions = ["builtins.BaseException", "builtins.Exception"] | ||||
|  | ||||
| [tool.pylint.DESIGN] | ||||
| max-positional-arguments = 10 | ||||
|  | ||||
| [tool.pytest.ini_options] | ||||
| testpaths = ["tests"] | ||||
| norecursedirs = [".git"] | ||||
| log_format = "%(asctime)s.%(msecs)03d %(levelname)-8s %(threadName)s %(name)s:%(filename)s:%(lineno)s %(message)s" | ||||
| log_date_format = "%Y-%m-%d %H:%M:%S" | ||||
| asyncio_default_fixture_loop_scope = "function" | ||||
| asyncio_mode = "auto" | ||||
| filterwarnings = [ | ||||
|     "error", | ||||
|     "ignore:pkg_resources is deprecated as an API:DeprecationWarning:dirhash", | ||||
|     "ignore::pytest.PytestUnraisableExceptionWarning", | ||||
| ] | ||||
| markers = [ | ||||
|     "no_mock_init_websession: disable the autouse mock of init_websession for this test", | ||||
| ] | ||||
|  | ||||
| [tool.isort] | ||||
| multi_line_output = 3 | ||||
| include_trailing_comma = true | ||||
| force_grid_wrap = 0 | ||||
| line_length = 88 | ||||
| indent = "    " | ||||
| force_sort_within_sections = true | ||||
| sections = ["FUTURE", "STDLIB", "THIRDPARTY", "FIRSTPARTY", "LOCALFOLDER"] | ||||
| default_section = "THIRDPARTY" | ||||
| forced_separate = "tests" | ||||
| combine_as_imports = true | ||||
| use_parentheses = true | ||||
| known_first_party = ["supervisor", "tests"] | ||||
| [tool.ruff] | ||||
| lint.select = [ | ||||
|     "B002",    # Python does not support the unary prefix increment | ||||
|     "B007",    # Loop control variable {name} not used within loop body | ||||
|     "B014",    # Exception handler with duplicate exception | ||||
|     "B023",    # Function definition does not bind loop variable {name} | ||||
|     "B026",    # Star-arg unpacking after a keyword argument is strongly discouraged | ||||
|     "B904",    # Use raise from to specify exception cause | ||||
|     "C",       # complexity | ||||
|     "COM818",  # Trailing comma on bare tuple prohibited | ||||
|     "D",       # docstrings | ||||
|     "DTZ003",  # Use datetime.now(tz=) instead of datetime.utcnow() | ||||
|     "DTZ004",  # Use datetime.fromtimestamp(ts, tz=) instead of datetime.utcfromtimestamp(ts) | ||||
|     "E",       # pycodestyle | ||||
|     "F",       # pyflakes/autoflake | ||||
|     "G",       # flake8-logging-format | ||||
|     "I",       # isort | ||||
|     "ICN001",  # import concentions; {name} should be imported as {asname} | ||||
|     "N804",    # First argument of a class method should be named cls | ||||
|     "N805",    # First argument of a method should be named self | ||||
|     "N815",    # Variable {name} in class scope should not be mixedCase | ||||
|     "PGH004",  # Use specific rule codes when using noqa | ||||
|     "PLC0414", # Useless import alias. Import alias does not rename original package. | ||||
|     "PLC",     # pylint | ||||
|     "PLE",     # pylint | ||||
|     "PLR",     # pylint | ||||
|     "PLW",     # pylint | ||||
|     "Q000",    # Double quotes found but single quotes preferred | ||||
|     "RUF006",  # Store a reference to the return value of asyncio.create_task | ||||
|     "S102",    # Use of exec detected | ||||
|     "S103",    # bad-file-permissions | ||||
|     "S108",    # hardcoded-temp-file | ||||
|     "S306",    # suspicious-mktemp-usage | ||||
|     "S307",    # suspicious-eval-usage | ||||
|     "S313",    # suspicious-xmlc-element-tree-usage | ||||
|     "S314",    # suspicious-xml-element-tree-usage | ||||
|     "S315",    # suspicious-xml-expat-reader-usage | ||||
|     "S316",    # suspicious-xml-expat-builder-usage | ||||
|     "S317",    # suspicious-xml-sax-usage | ||||
|     "S318",    # suspicious-xml-mini-dom-usage | ||||
|     "S319",    # suspicious-xml-pull-dom-usage | ||||
|     "S601",    # paramiko-call | ||||
|     "S602",    # subprocess-popen-with-shell-equals-true | ||||
|     "S604",    # call-with-shell-equals-true | ||||
|     "S608",    # hardcoded-sql-expression | ||||
|     "S609",    # unix-command-wildcard-injection | ||||
|     "SIM105",  # Use contextlib.suppress({exception}) instead of try-except-pass | ||||
|     "SIM117",  # Merge with-statements that use the same scope | ||||
|     "SIM118",  # Use {key} in {dict} instead of {key} in {dict}.keys() | ||||
|     "SIM201",  # Use {left} != {right} instead of not {left} == {right} | ||||
|     "SIM208",  # Use {expr} instead of not (not {expr}) | ||||
|     "SIM212",  # Use {a} if {a} else {b} instead of {b} if not {a} else {a} | ||||
|     "SIM300",  # Yoda conditions. Use 'age == 42' instead of '42 == age'. | ||||
|     "SIM401",  # Use get from dict with default instead of an if block | ||||
|     "T100",    # Trace found: {name} used | ||||
|     "T20",     # flake8-print | ||||
|     "TID251",  # Banned imports | ||||
|     "TRY004",  # Prefer TypeError exception for invalid type | ||||
|     "TRY203",  # Remove exception handler; error is immediately re-raised | ||||
|     "UP",      # pyupgrade | ||||
|     "W",       # pycodestyle | ||||
| ] | ||||
|  | ||||
| lint.ignore = [ | ||||
|     "D202", # No blank lines allowed after function docstring | ||||
|     "D203", # 1 blank line required before class docstring | ||||
|     "D213", # Multi-line docstring summary should start at the second line | ||||
|     "D406", # Section name should end with a newline | ||||
|     "D407", # Section name underlining | ||||
|     "E501", # line too long | ||||
|     "E731", # do not assign a lambda expression, use a def | ||||
|  | ||||
|     # Ignore ignored, as the rule is now back in preview/nursery, which cannot | ||||
|     # be ignored anymore without warnings. | ||||
|     # https://github.com/astral-sh/ruff/issues/7491 | ||||
|     # "PLC1901", # Lots of false positives | ||||
|  | ||||
|     # False positives https://github.com/astral-sh/ruff/issues/5386 | ||||
|     "PLC0208", # Use a sequence type instead of a `set` when iterating over values | ||||
|     "PLR0911", # Too many return statements ({returns} > {max_returns}) | ||||
|     "PLR0912", # Too many branches ({branches} > {max_branches}) | ||||
|     "PLR0913", # Too many arguments to function call ({c_args} > {max_args}) | ||||
|     "PLR0915", # Too many statements ({statements} > {max_statements}) | ||||
|     "PLR2004", # Magic value used in comparison, consider replacing {value} with a constant variable | ||||
|     "PLW2901", # Outer {outer_kind} variable {name} overwritten by inner {inner_kind} target | ||||
|     "UP006",   # keep type annotation style as is | ||||
|     "UP007",   # keep type annotation style as is | ||||
|     # Ignored due to performance: https://github.com/charliermarsh/ruff/issues/2923 | ||||
|     "UP038", # Use `X | Y` in `isinstance` call instead of `(X, Y)` | ||||
|  | ||||
|     # May conflict with the formatter, https://docs.astral.sh/ruff/formatter/#conflicting-lint-rules | ||||
|     "W191", | ||||
|     "E111", | ||||
|     "E114", | ||||
|     "E117", | ||||
|     "D206", | ||||
|     "D300", | ||||
|     "Q000", | ||||
|     "Q001", | ||||
|     "Q002", | ||||
|     "Q003", | ||||
|     "COM812", | ||||
|     "COM819", | ||||
|     "ISC001", | ||||
|     "ISC002", | ||||
|  | ||||
|     # Disabled because ruff does not understand type of __all__ generated by a function | ||||
|     "PLE0605", | ||||
| ] | ||||
|  | ||||
| [tool.ruff.lint.flake8-import-conventions.extend-aliases] | ||||
| voluptuous = "vol" | ||||
|  | ||||
| [tool.ruff.lint.flake8-pytest-style] | ||||
| fixture-parentheses = false | ||||
|  | ||||
| [tool.ruff.lint.flake8-tidy-imports.banned-api] | ||||
| "pytz".msg = "use zoneinfo instead" | ||||
|  | ||||
| [tool.ruff.lint.isort] | ||||
| force-sort-within-sections = true | ||||
| section-order = [ | ||||
|     "future", | ||||
|     "standard-library", | ||||
|     "third-party", | ||||
|     "first-party", | ||||
|     "local-folder", | ||||
| ] | ||||
| forced-separate = ["tests"] | ||||
| known-first-party = ["supervisor", "tests"] | ||||
| combine-as-imports = true | ||||
| split-on-trailing-comma = false | ||||
|  | ||||
| [tool.ruff.lint.per-file-ignores] | ||||
|  | ||||
| # DBus Service Mocks must use typing and names understood by dbus-fast | ||||
| "tests/dbus_service_mocks/*.py" = ["F722", "F821", "N815"] | ||||
|  | ||||
| [tool.ruff.lint.mccabe] | ||||
| max-complexity = 25 | ||||
|   | ||||
| @@ -1,28 +1,30 @@ | ||||
| aiodns==3.1.1 | ||||
| aiohttp==3.9.1 | ||||
| aiohttp-fast-url-dispatcher==0.3.0 | ||||
| async_timeout==4.0.3 | ||||
| aiodns==3.5.0 | ||||
| aiohttp==3.12.15 | ||||
| atomicwrites-homeassistant==1.4.1 | ||||
| attrs==23.2.0 | ||||
| awesomeversion==23.11.0 | ||||
| attrs==25.3.0 | ||||
| awesomeversion==25.8.0 | ||||
| blockbuster==1.5.25 | ||||
| brotli==1.1.0 | ||||
| ciso8601==2.3.1 | ||||
| colorlog==6.8.0 | ||||
| cpe==1.2.1 | ||||
| cryptography==41.0.7 | ||||
| debugpy==1.8.0 | ||||
| deepmerge==1.1.1 | ||||
| dirhash==0.2.1 | ||||
| docker==7.0.0 | ||||
| ciso8601==2.3.3 | ||||
| colorlog==6.9.0 | ||||
| cpe==1.3.1 | ||||
| cryptography==46.0.1 | ||||
| debugpy==1.8.17 | ||||
| deepmerge==2.0 | ||||
| dirhash==0.5.0 | ||||
| docker==7.1.0 | ||||
| faust-cchardet==2.1.19 | ||||
| gitpython==3.1.41 | ||||
| jinja2==3.1.3 | ||||
| orjson==3.9.10 | ||||
| pulsectl==23.5.2 | ||||
| pyudev==0.24.1 | ||||
| PyYAML==6.0.1 | ||||
| securetar==2023.12.0 | ||||
| sentry-sdk==1.39.2 | ||||
| voluptuous==0.14.1 | ||||
| dbus-fast==2.21.0 | ||||
| typing_extensions==4.9.0 | ||||
| gitpython==3.1.45 | ||||
| jinja2==3.1.6 | ||||
| log-rate-limit==1.4.2 | ||||
| orjson==3.11.3 | ||||
| pulsectl==24.12.0 | ||||
| pyudev==0.24.3 | ||||
| PyYAML==6.0.3 | ||||
| requests==2.32.5 | ||||
| securetar==2025.2.1 | ||||
| sentry-sdk==2.39.0 | ||||
| setuptools==80.9.0 | ||||
| voluptuous==0.15.2 | ||||
| dbus-fast==2.44.3 | ||||
| zlib-fast==0.2.1 | ||||
|   | ||||
| @@ -1,16 +1,16 @@ | ||||
| black==23.12.1 | ||||
| coverage==7.4.0 | ||||
| flake8-docstrings==1.7.0 | ||||
| flake8==7.0.0 | ||||
| pre-commit==3.6.0 | ||||
| pydocstyle==6.3.0 | ||||
| pylint==3.0.3 | ||||
| pytest-aiohttp==1.0.5 | ||||
| pytest-asyncio==0.23.3 | ||||
| pytest-cov==4.1.0 | ||||
| pytest-timeout==2.2.0 | ||||
| pytest==7.4.4 | ||||
| pyupgrade==3.15.0 | ||||
| time-machine==2.13.0 | ||||
| typing_extensions==4.9.0 | ||||
| urllib3==2.1.0 | ||||
| astroid==3.3.11 | ||||
| coverage==7.10.7 | ||||
| mypy==1.18.2 | ||||
| pre-commit==4.3.0 | ||||
| pylint==3.3.8 | ||||
| pytest-aiohttp==1.1.0 | ||||
| pytest-asyncio==0.25.2 | ||||
| pytest-cov==7.0.0 | ||||
| pytest-timeout==2.4.0 | ||||
| pytest==8.4.2 | ||||
| ruff==0.13.2 | ||||
| time-machine==2.19.0 | ||||
| types-docker==7.1.0.20250916 | ||||
| types-pyyaml==6.0.12.20250915 | ||||
| types-requests==2.32.4.20250913 | ||||
| urllib3==2.5.0 | ||||
|   | ||||
							
								
								
									
										30
									
								
								script/run-in-env.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										30
									
								
								script/run-in-env.sh
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| #!/usr/bin/env sh | ||||
| set -eu | ||||
|  | ||||
| # Used in venv activate script. | ||||
| # Would be an error if undefined. | ||||
| OSTYPE="${OSTYPE-}" | ||||
|  | ||||
| # Activate pyenv and virtualenv if present, then run the specified command | ||||
|  | ||||
| # pyenv, pyenv-virtualenv | ||||
| if [ -s .python-version ]; then | ||||
|     PYENV_VERSION=$(head -n 1 .python-version) | ||||
|     export PYENV_VERSION | ||||
| fi | ||||
|  | ||||
| if [ -n "${VIRTUAL_ENV-}" ] && [ -f "${VIRTUAL_ENV}/bin/activate" ]; then | ||||
|   . "${VIRTUAL_ENV}/bin/activate" | ||||
| else | ||||
|   # other common virtualenvs | ||||
|   my_path=$(git rev-parse --show-toplevel) | ||||
|  | ||||
|   for venv in venv .venv .; do | ||||
|     if [ -f "${my_path}/${venv}/bin/activate" ]; then | ||||
|       . "${my_path}/${venv}/bin/activate" | ||||
|       break | ||||
|     fi | ||||
|   done | ||||
| fi | ||||
|  | ||||
| exec "$@" | ||||
| @@ -1,30 +0,0 @@ | ||||
| #!/bin/bash | ||||
| source "/etc/supervisor_scripts/common" | ||||
|  | ||||
| set -e | ||||
|  | ||||
| # Update frontend | ||||
| git submodule update --init --recursive --remote | ||||
|  | ||||
| [ -s "$NVM_DIR/nvm.sh" ] && \. "$NVM_DIR/nvm.sh" | ||||
| cd home-assistant-polymer | ||||
| nvm install | ||||
| script/bootstrap | ||||
|  | ||||
| # Download translations | ||||
| start_docker | ||||
| ./script/translations_download | ||||
|  | ||||
| # build frontend | ||||
| cd hassio | ||||
| ./script/build_hassio | ||||
|  | ||||
| # Copy frontend | ||||
| rm -rf ../../supervisor/api/panel/* | ||||
| cp -rf build/* ../../supervisor/api/panel/ | ||||
|  | ||||
| # Reset frontend git | ||||
| cd .. | ||||
| git reset --hard HEAD | ||||
|  | ||||
| stop_docker | ||||
							
								
								
									
										17
									
								
								setup.cfg
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								setup.cfg
									
									
									
									
									
								
							| @@ -1,17 +0,0 @@ | ||||
| [flake8] | ||||
| exclude = .venv,.git,.tox,docs,venv,bin,lib,deps,build | ||||
| doctests = True | ||||
| max-line-length = 88 | ||||
| # E501: line too long | ||||
| # W503: Line break occurred before a binary operator | ||||
| # E203: Whitespace before ':' | ||||
| # D202 No blank lines allowed after function docstring | ||||
| # W504 line break after binary operator | ||||
| ignore = | ||||
|     E501, | ||||
|     W503, | ||||
|     E203, | ||||
|     D202, | ||||
|     W504 | ||||
| per-file-ignores = | ||||
|     tests/dbus_service_mocks/*.py: F821,F722 | ||||
							
								
								
									
										3
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										3
									
								
								setup.py
									
									
									
									
									
								
							| @@ -1,4 +1,5 @@ | ||||
| """Home Assistant Supervisor setup.""" | ||||
|  | ||||
| from pathlib import Path | ||||
| import re | ||||
|  | ||||
| @@ -18,7 +19,7 @@ def _get_supervisor_version(): | ||||
|     for line in CONSTANTS.split("/n"): | ||||
|         if match := RE_SUPERVISOR_VERSION.match(line): | ||||
|             return match.group(1) | ||||
|     return "99.9.9dev" | ||||
|     return "9999.09.9.dev9999" | ||||
|  | ||||
|  | ||||
| setup( | ||||
|   | ||||
| @@ -1,12 +1,22 @@ | ||||
| """Main file for Supervisor.""" | ||||
|  | ||||
| import asyncio | ||||
| from concurrent.futures import ThreadPoolExecutor | ||||
| import logging | ||||
| from pathlib import Path | ||||
| import sys | ||||
|  | ||||
| from supervisor import bootstrap | ||||
| from supervisor.utils.logging import activate_log_queue_handler | ||||
| import zlib_fast | ||||
|  | ||||
| # Enable fast zlib before importing supervisor | ||||
| zlib_fast.enable() | ||||
|  | ||||
| # pylint: disable=wrong-import-position | ||||
| from supervisor import bootstrap  # noqa: E402 | ||||
| from supervisor.utils.blockbuster import BlockBusterManager  # noqa: E402 | ||||
| from supervisor.utils.logging import activate_log_queue_handler  # noqa: E402 | ||||
|  | ||||
| # pylint: enable=wrong-import-position | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
|  | ||||
| @@ -44,10 +54,11 @@ if __name__ == "__main__": | ||||
|     _LOGGER.info("Initializing Supervisor setup") | ||||
|     coresys = loop.run_until_complete(bootstrap.initialize_coresys()) | ||||
|     loop.set_debug(coresys.config.debug) | ||||
|     if coresys.config.detect_blocking_io: | ||||
|         BlockBusterManager.activate() | ||||
|     loop.run_until_complete(coresys.core.connect()) | ||||
|  | ||||
|     bootstrap.supervisor_debugger(coresys) | ||||
|     bootstrap.migrate_system_env(coresys) | ||||
|     loop.run_until_complete(bootstrap.supervisor_debugger(coresys)) | ||||
|  | ||||
|     # Signal health startup for container | ||||
|     run_os_startup_check_cleanup() | ||||
| @@ -55,8 +66,28 @@ if __name__ == "__main__": | ||||
|     _LOGGER.info("Setting up Supervisor") | ||||
|     loop.run_until_complete(coresys.core.setup()) | ||||
|  | ||||
|     loop.call_soon_threadsafe(loop.create_task, coresys.core.start()) | ||||
|     loop.call_soon_threadsafe(bootstrap.reg_signal, loop, coresys) | ||||
|     # Create startup task that can be cancelled gracefully | ||||
|     startup_task = loop.create_task(coresys.core.start()) | ||||
|  | ||||
|     def shutdown_handler() -> None: | ||||
|         """Handle shutdown signals gracefully during startup.""" | ||||
|         if not startup_task.done(): | ||||
|             _LOGGER.warning("Supervisor startup interrupted by shutdown signal") | ||||
|             startup_task.cancel() | ||||
|  | ||||
|         coresys.create_task(coresys.core.stop()) | ||||
|  | ||||
|     bootstrap.register_signal_handlers(loop, shutdown_handler) | ||||
|  | ||||
|     try: | ||||
|         loop.run_until_complete(startup_task) | ||||
|     except asyncio.CancelledError: | ||||
|         _LOGGER.warning("Supervisor startup cancelled") | ||||
|     except Exception as err:  # pylint: disable=broad-except | ||||
|         # Supervisor itself is running at this point, just something didn't | ||||
|         # start as expected. Log with traceback to get more insights for | ||||
|         # such cases. | ||||
|         _LOGGER.critical("Supervisor start failed: %s", err, exc_info=True) | ||||
|  | ||||
|     try: | ||||
|         _LOGGER.info("Running Supervisor") | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,9 +1,10 @@ | ||||
| """Supervisor add-on build environment.""" | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| from functools import cached_property | ||||
| from pathlib import Path | ||||
| from typing import TYPE_CHECKING | ||||
| from typing import TYPE_CHECKING, Any | ||||
|  | ||||
| from awesomeversion import AwesomeVersion | ||||
|  | ||||
| @@ -14,6 +15,7 @@ from ..const import ( | ||||
|     ATTR_SQUASH, | ||||
|     FILE_SUFFIX_CONFIGURATION, | ||||
|     META_ADDON, | ||||
|     SOCKET_DOCKER, | ||||
| ) | ||||
| from ..coresys import CoreSys, CoreSysAttributes | ||||
| from ..docker.interface import MAP_ARCH | ||||
| @@ -22,7 +24,7 @@ from ..utils.common import FileConfiguration, find_one_filetype | ||||
| from .validate import SCHEMA_BUILD_CONFIG | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from . import AnyAddon | ||||
|     from .manager import AnyAddon | ||||
|  | ||||
|  | ||||
| class AddonBuild(FileConfiguration, CoreSysAttributes): | ||||
| @@ -33,23 +35,36 @@ class AddonBuild(FileConfiguration, CoreSysAttributes): | ||||
|         self.coresys: CoreSys = coresys | ||||
|         self.addon = addon | ||||
|  | ||||
|         # Search for build file later in executor | ||||
|         super().__init__(None, SCHEMA_BUILD_CONFIG) | ||||
|  | ||||
|     def _get_build_file(self) -> Path: | ||||
|         """Get build file. | ||||
|  | ||||
|         Must be run in executor. | ||||
|         """ | ||||
|         try: | ||||
|             build_file = find_one_filetype( | ||||
|             return find_one_filetype( | ||||
|                 self.addon.path_location, "build", FILE_SUFFIX_CONFIGURATION | ||||
|             ) | ||||
|         except ConfigurationFileError: | ||||
|             build_file = self.addon.path_location / "build.json" | ||||
|             return self.addon.path_location / "build.json" | ||||
|  | ||||
|         super().__init__(build_file, SCHEMA_BUILD_CONFIG) | ||||
|     async def read_data(self) -> None: | ||||
|         """Load data from file.""" | ||||
|         if not self._file: | ||||
|             self._file = await self.sys_run_in_executor(self._get_build_file) | ||||
|  | ||||
|     def save_data(self): | ||||
|         await super().read_data() | ||||
|  | ||||
|     async def save_data(self): | ||||
|         """Ignore save function.""" | ||||
|         raise RuntimeError() | ||||
|  | ||||
|     @cached_property | ||||
|     def arch(self) -> str: | ||||
|         """Return arch of the add-on.""" | ||||
|         return self.sys_arch.match(self.addon.arch) | ||||
|         return self.sys_arch.match([self.addon.arch]) | ||||
|  | ||||
|     @property | ||||
|     def base_image(self) -> str: | ||||
| @@ -67,13 +82,6 @@ class AddonBuild(FileConfiguration, CoreSysAttributes): | ||||
|             ) | ||||
|         return self._data[ATTR_BUILD_FROM][self.arch] | ||||
|  | ||||
|     @property | ||||
|     def dockerfile(self) -> Path: | ||||
|         """Return Dockerfile path.""" | ||||
|         if self.addon.path_location.joinpath(f"Dockerfile.{self.arch}").exists(): | ||||
|             return self.addon.path_location.joinpath(f"Dockerfile.{self.arch}") | ||||
|         return self.addon.path_location.joinpath("Dockerfile") | ||||
|  | ||||
|     @property | ||||
|     def squash(self) -> bool: | ||||
|         """Return True or False if squash is active.""" | ||||
| @@ -89,49 +97,89 @@ class AddonBuild(FileConfiguration, CoreSysAttributes): | ||||
|         """Return additional Docker labels.""" | ||||
|         return self._data[ATTR_LABELS] | ||||
|  | ||||
|     @property | ||||
|     def is_valid(self) -> bool: | ||||
|     def get_dockerfile(self) -> Path: | ||||
|         """Return Dockerfile path. | ||||
|  | ||||
|         Must be run in executor. | ||||
|         """ | ||||
|         if self.addon.path_location.joinpath(f"Dockerfile.{self.arch}").exists(): | ||||
|             return self.addon.path_location.joinpath(f"Dockerfile.{self.arch}") | ||||
|         return self.addon.path_location.joinpath("Dockerfile") | ||||
|  | ||||
|     async def is_valid(self) -> bool: | ||||
|         """Return true if the build env is valid.""" | ||||
|         try: | ||||
|  | ||||
|         def build_is_valid() -> bool: | ||||
|             return all( | ||||
|                 [ | ||||
|                     self.addon.path_location.is_dir(), | ||||
|                     self.dockerfile.is_file(), | ||||
|                     self.get_dockerfile().is_file(), | ||||
|                 ] | ||||
|             ) | ||||
|  | ||||
|         try: | ||||
|             return await self.sys_run_in_executor(build_is_valid) | ||||
|         except HassioArchNotFound: | ||||
|             return False | ||||
|  | ||||
|     def get_docker_args(self, version: AwesomeVersion): | ||||
|         """Create a dict with Docker build arguments.""" | ||||
|         args = { | ||||
|             "path": str(self.addon.path_location), | ||||
|             "tag": f"{self.addon.image}:{version!s}", | ||||
|             "dockerfile": str(self.dockerfile), | ||||
|             "pull": True, | ||||
|             "forcerm": not self.sys_dev, | ||||
|             "squash": self.squash, | ||||
|             "platform": MAP_ARCH[self.arch], | ||||
|             "labels": { | ||||
|                 "io.hass.version": version, | ||||
|                 "io.hass.arch": self.arch, | ||||
|                 "io.hass.type": META_ADDON, | ||||
|                 "io.hass.name": self._fix_label("name"), | ||||
|                 "io.hass.description": self._fix_label("description"), | ||||
|                 **self.additional_labels, | ||||
|             }, | ||||
|             "buildargs": { | ||||
|                 "BUILD_FROM": self.base_image, | ||||
|                 "BUILD_VERSION": version, | ||||
|                 "BUILD_ARCH": self.sys_arch.default, | ||||
|                 **self.additional_args, | ||||
|             }, | ||||
|     def get_docker_args( | ||||
|         self, version: AwesomeVersion, image_tag: str | ||||
|     ) -> dict[str, Any]: | ||||
|         """Create a dict with Docker run args.""" | ||||
|         dockerfile_path = self.get_dockerfile().relative_to(self.addon.path_location) | ||||
|  | ||||
|         build_cmd = [ | ||||
|             "docker", | ||||
|             "buildx", | ||||
|             "build", | ||||
|             ".", | ||||
|             "--tag", | ||||
|             image_tag, | ||||
|             "--file", | ||||
|             str(dockerfile_path), | ||||
|             "--platform", | ||||
|             MAP_ARCH[self.arch], | ||||
|             "--pull", | ||||
|         ] | ||||
|  | ||||
|         labels = { | ||||
|             "io.hass.version": version, | ||||
|             "io.hass.arch": self.arch, | ||||
|             "io.hass.type": META_ADDON, | ||||
|             "io.hass.name": self._fix_label("name"), | ||||
|             "io.hass.description": self._fix_label("description"), | ||||
|             **self.additional_labels, | ||||
|         } | ||||
|  | ||||
|         if self.addon.url: | ||||
|             args["labels"]["io.hass.url"] = self.addon.url | ||||
|             labels["io.hass.url"] = self.addon.url | ||||
|  | ||||
|         return args | ||||
|         for key, value in labels.items(): | ||||
|             build_cmd.extend(["--label", f"{key}={value}"]) | ||||
|  | ||||
|         build_args = { | ||||
|             "BUILD_FROM": self.base_image, | ||||
|             "BUILD_VERSION": version, | ||||
|             "BUILD_ARCH": self.sys_arch.default, | ||||
|             **self.additional_args, | ||||
|         } | ||||
|  | ||||
|         for key, value in build_args.items(): | ||||
|             build_cmd.extend(["--build-arg", f"{key}={value}"]) | ||||
|  | ||||
|         # The addon path will be mounted from the host system | ||||
|         addon_extern_path = self.sys_config.local_to_extern_path( | ||||
|             self.addon.path_location | ||||
|         ) | ||||
|  | ||||
|         return { | ||||
|             "command": build_cmd, | ||||
|             "volumes": { | ||||
|                 SOCKET_DOCKER: {"bind": "/var/run/docker.sock", "mode": "rw"}, | ||||
|                 addon_extern_path: {"bind": "/addon", "mode": "ro"}, | ||||
|             }, | ||||
|             "working_dir": "/addon", | ||||
|         } | ||||
|  | ||||
|     def _fix_label(self, label_name: str) -> str: | ||||
|         """Remove characters they are not supported.""" | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Add-on static data.""" | ||||
|  | ||||
| from datetime import timedelta | ||||
| from enum import StrEnum | ||||
|  | ||||
| @@ -28,6 +29,7 @@ class MappingType(StrEnum): | ||||
|  | ||||
|  | ||||
| ATTR_BACKUP = "backup" | ||||
| ATTR_BREAKING_VERSIONS = "breaking_versions" | ||||
| ATTR_CODENOTARY = "codenotary" | ||||
| ATTR_READ_ONLY = "read_only" | ||||
| ATTR_PATH = "path" | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Init file for Supervisor add-on data.""" | ||||
|  | ||||
| from copy import deepcopy | ||||
| from typing import Any | ||||
|  | ||||
| @@ -37,7 +38,7 @@ class AddonsData(FileConfiguration, CoreSysAttributes): | ||||
|         """Return local add-on data.""" | ||||
|         return self._data[ATTR_SYSTEM] | ||||
|  | ||||
|     def install(self, addon: AddonStore) -> None: | ||||
|     async def install(self, addon: AddonStore) -> None: | ||||
|         """Set addon as installed.""" | ||||
|         self.system[addon.slug] = deepcopy(addon.data) | ||||
|         self.user[addon.slug] = { | ||||
| @@ -45,26 +46,28 @@ class AddonsData(FileConfiguration, CoreSysAttributes): | ||||
|             ATTR_VERSION: addon.version, | ||||
|             ATTR_IMAGE: addon.image, | ||||
|         } | ||||
|         self.save_data() | ||||
|         await self.save_data() | ||||
|  | ||||
|     def uninstall(self, addon: Addon) -> None: | ||||
|     async def uninstall(self, addon: Addon) -> None: | ||||
|         """Set add-on as uninstalled.""" | ||||
|         self.system.pop(addon.slug, None) | ||||
|         self.user.pop(addon.slug, None) | ||||
|         self.save_data() | ||||
|         await self.save_data() | ||||
|  | ||||
|     def update(self, addon: AddonStore) -> None: | ||||
|     async def update(self, addon: AddonStore) -> None: | ||||
|         """Update version of add-on.""" | ||||
|         self.system[addon.slug] = deepcopy(addon.data) | ||||
|         self.user[addon.slug].update( | ||||
|             {ATTR_VERSION: addon.version, ATTR_IMAGE: addon.image} | ||||
|         ) | ||||
|         self.save_data() | ||||
|         await self.save_data() | ||||
|  | ||||
|     def restore(self, slug: str, user: Config, system: Config, image: str) -> None: | ||||
|     async def restore( | ||||
|         self, slug: str, user: Config, system: Config, image: str | ||||
|     ) -> None: | ||||
|         """Restore data to add-on.""" | ||||
|         self.user[slug] = deepcopy(user) | ||||
|         self.system[slug] = deepcopy(system) | ||||
|  | ||||
|         self.user[slug][ATTR_IMAGE] = image | ||||
|         self.save_data() | ||||
|         await self.save_data() | ||||
|   | ||||
| @@ -1,30 +1,31 @@ | ||||
| """Supervisor add-on manager.""" | ||||
|  | ||||
| import asyncio | ||||
| from collections.abc import Awaitable | ||||
| from contextlib import suppress | ||||
| import logging | ||||
| import tarfile | ||||
| from typing import Union | ||||
| from typing import Self, Union | ||||
|  | ||||
| from attr import evolve | ||||
|  | ||||
| from supervisor.jobs.const import JobConcurrency | ||||
|  | ||||
| from ..const import AddonBoot, AddonStartup, AddonState | ||||
| from ..coresys import CoreSys, CoreSysAttributes | ||||
| from ..exceptions import ( | ||||
|     AddonConfigurationError, | ||||
|     AddonNotSupportedError, | ||||
|     AddonsError, | ||||
|     AddonsJobError, | ||||
|     AddonsNotSupportedError, | ||||
|     CoreDNSError, | ||||
|     DockerAPIError, | ||||
|     DockerError, | ||||
|     DockerNotFound, | ||||
|     HassioError, | ||||
|     HomeAssistantAPIError, | ||||
| ) | ||||
| from ..jobs.decorator import Job, JobCondition | ||||
| from ..resolution.const import ContextType, IssueType, SuggestionType | ||||
| from ..store.addon import AddonStore | ||||
| from ..utils import check_exception_chain | ||||
| from ..utils.sentry import capture_exception | ||||
| from ..utils.sentry import async_capture_exception | ||||
| from .addon import Addon | ||||
| from .const import ADDON_UPDATE_CONDITIONS | ||||
| from .data import AddonsData | ||||
| @@ -68,6 +69,10 @@ class AddonManager(CoreSysAttributes): | ||||
|             return self.store.get(addon_slug) | ||||
|         return None | ||||
|  | ||||
|     def get_local_only(self, addon_slug: str) -> Addon | None: | ||||
|         """Return an installed add-on from slug.""" | ||||
|         return self.local.get(addon_slug) | ||||
|  | ||||
|     def from_token(self, token: str) -> Addon | None: | ||||
|         """Return an add-on from Supervisor token.""" | ||||
|         for addon in self.installed: | ||||
| @@ -75,17 +80,27 @@ class AddonManager(CoreSysAttributes): | ||||
|                 return addon | ||||
|         return None | ||||
|  | ||||
|     async def load_config(self) -> Self: | ||||
|         """Load config in executor.""" | ||||
|         await self.data.read_data() | ||||
|         return self | ||||
|  | ||||
|     async def load(self) -> None: | ||||
|         """Start up add-on management.""" | ||||
|         tasks = [] | ||||
|         # Refresh cache for all store addons | ||||
|         tasks: list[Awaitable[None]] = [ | ||||
|             store.refresh_path_cache() for store in self.store.values() | ||||
|         ] | ||||
|  | ||||
|         # Load all installed addons | ||||
|         for slug in self.data.system: | ||||
|             addon = self.local[slug] = Addon(self.coresys, slug) | ||||
|             tasks.append(self.sys_create_task(addon.load())) | ||||
|             tasks.append(addon.load()) | ||||
|  | ||||
|         # Run initial tasks | ||||
|         _LOGGER.info("Found %d installed add-ons", len(tasks)) | ||||
|         _LOGGER.info("Found %d installed add-ons", len(self.data.system)) | ||||
|         if tasks: | ||||
|             await asyncio.wait(tasks) | ||||
|             await asyncio.gather(*tasks) | ||||
|  | ||||
|         # Sync DNS | ||||
|         await self.sync_dns() | ||||
| @@ -112,15 +127,14 @@ class AddonManager(CoreSysAttributes): | ||||
|             try: | ||||
|                 if start_task := await addon.start(): | ||||
|                     wait_boot.append(start_task) | ||||
|             except AddonsError as err: | ||||
|                 # Check if there is an system/user issue | ||||
|                 if check_exception_chain( | ||||
|                     err, (DockerAPIError, DockerNotFound, AddonConfigurationError) | ||||
|                 ): | ||||
|                     addon.boot = AddonBoot.MANUAL | ||||
|                     addon.save_persist() | ||||
|             except HassioError: | ||||
|                 pass  # These are already handled | ||||
|                 self.sys_resolution.add_issue( | ||||
|                     evolve(addon.boot_failed_issue), | ||||
|                     suggestions=[ | ||||
|                         SuggestionType.EXECUTE_START, | ||||
|                         SuggestionType.DISABLE_BOOT, | ||||
|                     ], | ||||
|                 ) | ||||
|             else: | ||||
|                 continue | ||||
|  | ||||
| @@ -129,6 +143,19 @@ class AddonManager(CoreSysAttributes): | ||||
|         # Ignore exceptions from waiting for addon startup, addon errors handled elsewhere | ||||
|         await asyncio.gather(*wait_boot, return_exceptions=True) | ||||
|  | ||||
|         # After waiting for startup, create an issue for boot addons that are error or unknown state | ||||
|         # Ignore stopped as single shot addons can be run at boot and this is successful exit | ||||
|         # Timeout waiting for startup is not a failure, addon is probably just slow | ||||
|         for addon in tasks: | ||||
|             if addon.state in {AddonState.ERROR, AddonState.UNKNOWN}: | ||||
|                 self.sys_resolution.add_issue( | ||||
|                     evolve(addon.boot_failed_issue), | ||||
|                     suggestions=[ | ||||
|                         SuggestionType.EXECUTE_START, | ||||
|                         SuggestionType.DISABLE_BOOT, | ||||
|                     ], | ||||
|                 ) | ||||
|  | ||||
|     async def shutdown(self, stage: AddonStartup) -> None: | ||||
|         """Shutdown addons.""" | ||||
|         tasks: list[Addon] = [] | ||||
| @@ -149,14 +176,17 @@ class AddonManager(CoreSysAttributes): | ||||
|                 await addon.stop() | ||||
|             except Exception as err:  # pylint: disable=broad-except | ||||
|                 _LOGGER.warning("Can't stop Add-on %s: %s", addon.slug, err) | ||||
|                 capture_exception(err) | ||||
|                 await async_capture_exception(err) | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_manager_install", | ||||
|         conditions=ADDON_UPDATE_CONDITIONS, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.QUEUE, | ||||
|     ) | ||||
|     async def install(self, slug: str) -> None: | ||||
|     async def install( | ||||
|         self, slug: str, *, validation_complete: asyncio.Event | None = None | ||||
|     ) -> None: | ||||
|         """Install an add-on.""" | ||||
|         self.sys_jobs.current.reference = slug | ||||
|  | ||||
| @@ -169,17 +199,30 @@ class AddonManager(CoreSysAttributes): | ||||
|  | ||||
|         store.validate_availability() | ||||
|  | ||||
|         # If being run in the background, notify caller that validation has completed | ||||
|         if validation_complete: | ||||
|             validation_complete.set() | ||||
|  | ||||
|         await Addon(self.coresys, slug).install() | ||||
|  | ||||
|         _LOGGER.info("Add-on '%s' successfully installed", slug) | ||||
|  | ||||
|     async def uninstall(self, slug: str) -> None: | ||||
|     @Job(name="addon_manager_uninstall") | ||||
|     async def uninstall(self, slug: str, *, remove_config: bool = False) -> None: | ||||
|         """Remove an add-on.""" | ||||
|         if slug not in self.local: | ||||
|             _LOGGER.warning("Add-on %s is not installed", slug) | ||||
|             return | ||||
|  | ||||
|         await self.local[slug].uninstall() | ||||
|         shared_image = any( | ||||
|             self.local[slug].image == addon.image | ||||
|             and self.local[slug].version == addon.version | ||||
|             for addon in self.installed | ||||
|             if addon.slug != slug | ||||
|         ) | ||||
|         await self.local[slug].uninstall( | ||||
|             remove_config=remove_config, remove_image=not shared_image | ||||
|         ) | ||||
|  | ||||
|         _LOGGER.info("Add-on '%s' successfully removed", slug) | ||||
|  | ||||
| @@ -189,7 +232,11 @@ class AddonManager(CoreSysAttributes): | ||||
|         on_condition=AddonsJobError, | ||||
|     ) | ||||
|     async def update( | ||||
|         self, slug: str, backup: bool | None = False | ||||
|         self, | ||||
|         slug: str, | ||||
|         backup: bool | None = False, | ||||
|         *, | ||||
|         validation_complete: asyncio.Event | None = None, | ||||
|     ) -> asyncio.Task | None: | ||||
|         """Update add-on. | ||||
|  | ||||
| @@ -214,6 +261,10 @@ class AddonManager(CoreSysAttributes): | ||||
|         # Check if available, Maybe something have changed | ||||
|         store.validate_availability() | ||||
|  | ||||
|         # If being run in the background, notify caller that validation has completed | ||||
|         if validation_complete: | ||||
|             validation_complete.set() | ||||
|  | ||||
|         if backup: | ||||
|             await self.sys_backups.do_backup_partial( | ||||
|                 name=f"addon_{addon.slug}_{addon.version}", | ||||
| @@ -232,7 +283,7 @@ class AddonManager(CoreSysAttributes): | ||||
|         ], | ||||
|         on_condition=AddonsJobError, | ||||
|     ) | ||||
|     async def rebuild(self, slug: str) -> asyncio.Task | None: | ||||
|     async def rebuild(self, slug: str, *, force: bool = False) -> asyncio.Task | None: | ||||
|         """Perform a rebuild of local build add-on. | ||||
|  | ||||
|         Returns a Task that completes when addon has state 'started' (see addon.start) | ||||
| @@ -255,8 +306,8 @@ class AddonManager(CoreSysAttributes): | ||||
|             raise AddonsError( | ||||
|                 "Version changed, use Update instead Rebuild", _LOGGER.error | ||||
|             ) | ||||
|         if not addon.need_build: | ||||
|             raise AddonsNotSupportedError( | ||||
|         if not force and not addon.need_build: | ||||
|             raise AddonNotSupportedError( | ||||
|                 "Can't rebuild a image based add-on", _LOGGER.error | ||||
|             ) | ||||
|  | ||||
| @@ -284,7 +335,7 @@ class AddonManager(CoreSysAttributes): | ||||
|         if slug not in self.local: | ||||
|             _LOGGER.debug("Add-on %s is not local available for restore", slug) | ||||
|             addon = Addon(self.coresys, slug) | ||||
|             had_ingress = False | ||||
|             had_ingress: bool | None = False | ||||
|         else: | ||||
|             _LOGGER.debug("Add-on %s is local available for restore", slug) | ||||
|             addon = self.local[slug] | ||||
| @@ -359,7 +410,7 @@ class AddonManager(CoreSysAttributes): | ||||
|                     reference=addon.slug, | ||||
|                     suggestions=[SuggestionType.EXECUTE_REPAIR], | ||||
|                 ) | ||||
|                 capture_exception(err) | ||||
|                 await async_capture_exception(err) | ||||
|             else: | ||||
|                 add_host_coros.append( | ||||
|                     self.sys_plugins.dns.add_host( | ||||
|   | ||||
| @@ -1,14 +1,18 @@ | ||||
| """Init file for Supervisor add-ons.""" | ||||
|  | ||||
| from abc import ABC, abstractmethod | ||||
| from collections import defaultdict | ||||
| from collections.abc import Callable | ||||
| from collections.abc import Awaitable, Callable | ||||
| from contextlib import suppress | ||||
| from datetime import datetime | ||||
| import logging | ||||
| from pathlib import Path | ||||
| from typing import Any | ||||
|  | ||||
| from awesomeversion import AwesomeVersion, AwesomeVersionException | ||||
|  | ||||
| from supervisor.utils.dt import utc_from_timestamp | ||||
|  | ||||
| from ..const import ( | ||||
|     ATTR_ADVANCED, | ||||
|     ATTR_APPARMOR, | ||||
| @@ -43,7 +47,7 @@ from ..const import ( | ||||
|     ATTR_JOURNALD, | ||||
|     ATTR_KERNEL_MODULES, | ||||
|     ATTR_LEGACY, | ||||
|     ATTR_LOCATON, | ||||
|     ATTR_LOCATION, | ||||
|     ATTR_MACHINE, | ||||
|     ATTR_MAP, | ||||
|     ATTR_NAME, | ||||
| @@ -71,6 +75,7 @@ from ..const import ( | ||||
|     ATTR_URL, | ||||
|     ATTR_USB, | ||||
|     ATTR_VERSION, | ||||
|     ATTR_VERSION_TIMESTAMP, | ||||
|     ATTR_VIDEO, | ||||
|     ATTR_WATCHDOG, | ||||
|     ATTR_WEBUI, | ||||
| @@ -78,18 +83,25 @@ from ..const import ( | ||||
|     SECURITY_DISABLE, | ||||
|     SECURITY_PROFILE, | ||||
|     AddonBoot, | ||||
|     AddonBootConfig, | ||||
|     AddonStage, | ||||
|     AddonStartup, | ||||
| ) | ||||
| from ..coresys import CoreSys | ||||
| from ..docker.const import Capabilities | ||||
| from ..exceptions import AddonsNotSupportedError | ||||
| from ..exceptions import ( | ||||
|     AddonNotSupportedArchitectureError, | ||||
|     AddonNotSupportedError, | ||||
|     AddonNotSupportedHomeAssistantVersionError, | ||||
|     AddonNotSupportedMachineTypeError, | ||||
| ) | ||||
| from ..jobs.const import JOB_GROUP_ADDON | ||||
| from ..jobs.job_group import JobGroup | ||||
| from ..utils import version_is_new_enough | ||||
| from .configuration import FolderMapping | ||||
| from .const import ( | ||||
|     ATTR_BACKUP, | ||||
|     ATTR_BREAKING_VERSIONS, | ||||
|     ATTR_CODENOTARY, | ||||
|     ATTR_PATH, | ||||
|     ATTR_READ_ONLY, | ||||
| @@ -113,6 +125,10 @@ class AddonModel(JobGroup, ABC): | ||||
|             coresys, JOB_GROUP_ADDON.format_map(defaultdict(str, slug=slug)), slug | ||||
|         ) | ||||
|         self.slug: str = slug | ||||
|         self._path_icon_exists: bool = False | ||||
|         self._path_logo_exists: bool = False | ||||
|         self._path_changelog_exists: bool = False | ||||
|         self._path_documentation_exists: bool = False | ||||
|  | ||||
|     @property | ||||
|     @abstractmethod | ||||
| @@ -140,10 +156,15 @@ class AddonModel(JobGroup, ABC): | ||||
|         return self.data[ATTR_OPTIONS] | ||||
|  | ||||
|     @property | ||||
|     def boot(self) -> AddonBoot: | ||||
|         """Return boot config with prio local settings.""" | ||||
|     def boot_config(self) -> AddonBootConfig: | ||||
|         """Return boot config.""" | ||||
|         return self.data[ATTR_BOOT] | ||||
|  | ||||
|     @property | ||||
|     def boot(self) -> AddonBoot: | ||||
|         """Return boot config with prio local settings unless config is forced.""" | ||||
|         return AddonBoot(self.data[ATTR_BOOT]) | ||||
|  | ||||
|     @property | ||||
|     def auto_update(self) -> bool | None: | ||||
|         """Return if auto update is enable.""" | ||||
| @@ -194,18 +215,6 @@ class AddonModel(JobGroup, ABC): | ||||
|         """Return description of add-on.""" | ||||
|         return self.data[ATTR_DESCRIPTON] | ||||
|  | ||||
|     @property | ||||
|     def long_description(self) -> str | None: | ||||
|         """Return README.md as long_description.""" | ||||
|         readme = Path(self.path_location, "README.md") | ||||
|  | ||||
|         # If readme not exists | ||||
|         if not readme.exists(): | ||||
|             return None | ||||
|  | ||||
|         # Return data | ||||
|         return readme.read_text(encoding="utf-8") | ||||
|  | ||||
|     @property | ||||
|     def repository(self) -> str: | ||||
|         """Return repository of add-on.""" | ||||
| @@ -221,6 +230,11 @@ class AddonModel(JobGroup, ABC): | ||||
|         """Return latest version of add-on.""" | ||||
|         return self.data[ATTR_VERSION] | ||||
|  | ||||
|     @property | ||||
|     def latest_version_timestamp(self) -> datetime: | ||||
|         """Return when latest version was first seen.""" | ||||
|         return utc_from_timestamp(self.data[ATTR_VERSION_TIMESTAMP]) | ||||
|  | ||||
|     @property | ||||
|     def version(self) -> AwesomeVersion: | ||||
|         """Return version of add-on.""" | ||||
| @@ -285,7 +299,7 @@ class AddonModel(JobGroup, ABC): | ||||
|         return self.data.get(ATTR_WEBUI) | ||||
|  | ||||
|     @property | ||||
|     def watchdog(self) -> str | None: | ||||
|     def watchdog_url(self) -> str | None: | ||||
|         """Return URL to for watchdog or None.""" | ||||
|         return self.data.get(ATTR_WATCHDOG) | ||||
|  | ||||
| @@ -501,22 +515,22 @@ class AddonModel(JobGroup, ABC): | ||||
|     @property | ||||
|     def with_icon(self) -> bool: | ||||
|         """Return True if an icon exists.""" | ||||
|         return self.path_icon.exists() | ||||
|         return self._path_icon_exists | ||||
|  | ||||
|     @property | ||||
|     def with_logo(self) -> bool: | ||||
|         """Return True if a logo exists.""" | ||||
|         return self.path_logo.exists() | ||||
|         return self._path_logo_exists | ||||
|  | ||||
|     @property | ||||
|     def with_changelog(self) -> bool: | ||||
|         """Return True if a changelog exists.""" | ||||
|         return self.path_changelog.exists() | ||||
|         return self._path_changelog_exists | ||||
|  | ||||
|     @property | ||||
|     def with_documentation(self) -> bool: | ||||
|         """Return True if a documentation exists.""" | ||||
|         return self.path_documentation.exists() | ||||
|         return self._path_documentation_exists | ||||
|  | ||||
|     @property | ||||
|     def supported_arch(self) -> list[str]: | ||||
| @@ -560,7 +574,7 @@ class AddonModel(JobGroup, ABC): | ||||
|     @property | ||||
|     def path_location(self) -> Path: | ||||
|         """Return path to this add-on.""" | ||||
|         return Path(self.data[ATTR_LOCATON]) | ||||
|         return Path(self.data[ATTR_LOCATION]) | ||||
|  | ||||
|     @property | ||||
|     def path_icon(self) -> Path: | ||||
| @@ -597,7 +611,7 @@ class AddonModel(JobGroup, ABC): | ||||
|         return AddonOptions(self.coresys, raw_schema, self.name, self.slug) | ||||
|  | ||||
|     @property | ||||
|     def schema_ui(self) -> list[dict[any, any]] | None: | ||||
|     def schema_ui(self) -> list[dict[Any, Any]] | None: | ||||
|         """Create a UI schema for add-on options.""" | ||||
|         raw_schema = self.data[ATTR_SCHEMA] | ||||
|  | ||||
| @@ -620,25 +634,59 @@ class AddonModel(JobGroup, ABC): | ||||
|         """Return Signer email address for CAS.""" | ||||
|         return self.data.get(ATTR_CODENOTARY) | ||||
|  | ||||
|     @property | ||||
|     def breaking_versions(self) -> list[AwesomeVersion]: | ||||
|         """Return breaking versions of addon.""" | ||||
|         return self.data[ATTR_BREAKING_VERSIONS] | ||||
|  | ||||
|     async def long_description(self) -> str | None: | ||||
|         """Return README.md as long_description.""" | ||||
|  | ||||
|         def read_readme() -> str | None: | ||||
|             readme = Path(self.path_location, "README.md") | ||||
|  | ||||
|             # If readme not exists | ||||
|             if not readme.exists(): | ||||
|                 return None | ||||
|  | ||||
|             # Return data | ||||
|             return readme.read_text(encoding="utf-8", errors="replace") | ||||
|  | ||||
|         return await self.sys_run_in_executor(read_readme) | ||||
|  | ||||
|     def refresh_path_cache(self) -> Awaitable[None]: | ||||
|         """Refresh cache of existing paths.""" | ||||
|  | ||||
|         def check_paths(): | ||||
|             self._path_icon_exists = self.path_icon.exists() | ||||
|             self._path_logo_exists = self.path_logo.exists() | ||||
|             self._path_changelog_exists = self.path_changelog.exists() | ||||
|             self._path_documentation_exists = self.path_documentation.exists() | ||||
|  | ||||
|         return self.sys_run_in_executor(check_paths) | ||||
|  | ||||
|     def validate_availability(self) -> None: | ||||
|         """Validate if addon is available for current system.""" | ||||
|         return self._validate_availability(self.data, logger=_LOGGER.error) | ||||
|  | ||||
|     def __eq__(self, other): | ||||
|         """Compaired add-on objects.""" | ||||
|     def __eq__(self, other: Any) -> bool: | ||||
|         """Compare add-on objects.""" | ||||
|         if not isinstance(other, AddonModel): | ||||
|             return False | ||||
|         return self.slug == other.slug | ||||
|  | ||||
|     def __hash__(self) -> int: | ||||
|         """Hash for add-on objects.""" | ||||
|         return hash(self.slug) | ||||
|  | ||||
|     def _validate_availability( | ||||
|         self, config, *, logger: Callable[..., None] | None = None | ||||
|     ) -> None: | ||||
|         """Validate if addon is available for current system.""" | ||||
|         # Architecture | ||||
|         if not self.sys_arch.is_supported(config[ATTR_ARCH]): | ||||
|             raise AddonsNotSupportedError( | ||||
|                 f"Add-on {self.slug} not supported on this platform, supported architectures: {', '.join(config[ATTR_ARCH])}", | ||||
|                 logger, | ||||
|             raise AddonNotSupportedArchitectureError( | ||||
|                 logger, slug=self.slug, architectures=config[ATTR_ARCH] | ||||
|             ) | ||||
|  | ||||
|         # Machine / Hardware | ||||
| @@ -646,9 +694,8 @@ class AddonModel(JobGroup, ABC): | ||||
|         if machine and ( | ||||
|             f"!{self.sys_machine}" in machine or self.sys_machine not in machine | ||||
|         ): | ||||
|             raise AddonsNotSupportedError( | ||||
|                 f"Add-on {self.slug} not supported on this machine, supported machine types: {', '.join(machine)}", | ||||
|                 logger, | ||||
|             raise AddonNotSupportedMachineTypeError( | ||||
|                 logger, slug=self.slug, machine_types=machine | ||||
|             ) | ||||
|  | ||||
|         # Home Assistant | ||||
| @@ -657,16 +704,15 @@ class AddonModel(JobGroup, ABC): | ||||
|             if version and not version_is_new_enough( | ||||
|                 self.sys_homeassistant.version, version | ||||
|             ): | ||||
|                 raise AddonsNotSupportedError( | ||||
|                     f"Add-on {self.slug} not supported on this system, requires Home Assistant version {version} or greater", | ||||
|                     logger, | ||||
|                 raise AddonNotSupportedHomeAssistantVersionError( | ||||
|                     logger, slug=self.slug, version=str(version) | ||||
|                 ) | ||||
|  | ||||
|     def _available(self, config) -> bool: | ||||
|         """Return True if this add-on is available on this platform.""" | ||||
|         try: | ||||
|             self._validate_availability(config) | ||||
|         except AddonsNotSupportedError: | ||||
|         except AddonNotSupportedError: | ||||
|             return False | ||||
|  | ||||
|         return True | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Add-on Options / UI rendering.""" | ||||
|  | ||||
| import hashlib | ||||
| import logging | ||||
| from pathlib import Path | ||||
| @@ -92,15 +93,7 @@ class AddonOptions(CoreSysAttributes): | ||||
|  | ||||
|             typ = self.raw_schema[key] | ||||
|             try: | ||||
|                 if isinstance(typ, list): | ||||
|                     # nested value list | ||||
|                     options[key] = self._nested_validate_list(typ[0], value, key) | ||||
|                 elif isinstance(typ, dict): | ||||
|                     # nested value dict | ||||
|                     options[key] = self._nested_validate_dict(typ, value, key) | ||||
|                 else: | ||||
|                     # normal value | ||||
|                     options[key] = self._single_validate(typ, value, key) | ||||
|                 options[key] = self._validate_element(typ, value, key) | ||||
|             except (IndexError, KeyError): | ||||
|                 raise vol.Invalid( | ||||
|                     f"Type error for option '{key}' in {self._name} ({self._slug})" | ||||
| @@ -110,7 +103,20 @@ class AddonOptions(CoreSysAttributes): | ||||
|         return options | ||||
|  | ||||
|     # pylint: disable=no-value-for-parameter | ||||
|     def _single_validate(self, typ: str, value: Any, key: str): | ||||
|     def _validate_element(self, typ: Any, value: Any, key: str) -> Any: | ||||
|         """Validate a value against a type specification.""" | ||||
|         if isinstance(typ, list): | ||||
|             # nested value list | ||||
|             return self._nested_validate_list(typ[0], value, key) | ||||
|         elif isinstance(typ, dict): | ||||
|             # nested value dict | ||||
|             return self._nested_validate_dict(typ, value, key) | ||||
|         else: | ||||
|             # normal value | ||||
|             return self._single_validate(typ, value, key) | ||||
|  | ||||
|     # pylint: disable=no-value-for-parameter | ||||
|     def _single_validate(self, typ: str, value: Any, key: str) -> Any: | ||||
|         """Validate a single element.""" | ||||
|         # if required argument | ||||
|         if value is None: | ||||
| @@ -136,7 +142,7 @@ class AddonOptions(CoreSysAttributes): | ||||
|             ) from None | ||||
|  | ||||
|         # prepare range | ||||
|         range_args = {} | ||||
|         range_args: dict[str, Any] = {} | ||||
|         for group_name in _SCHEMA_LENGTH_PARTS: | ||||
|             group_value = match.group(group_name) | ||||
|             if group_value: | ||||
| @@ -181,13 +187,15 @@ class AddonOptions(CoreSysAttributes): | ||||
|  | ||||
|             # Device valid | ||||
|             self.devices.add(device) | ||||
|             return str(device.path) | ||||
|             return str(value) | ||||
|  | ||||
|         raise vol.Invalid( | ||||
|             f"Fatal error for option '{key}' with type '{typ}' in {self._name} ({self._slug})" | ||||
|         ) from None | ||||
|  | ||||
|     def _nested_validate_list(self, typ: Any, data_list: list[Any], key: str): | ||||
|     def _nested_validate_list( | ||||
|         self, typ: Any, data_list: list[Any], key: str | ||||
|     ) -> list[Any]: | ||||
|         """Validate nested items.""" | ||||
|         options = [] | ||||
|  | ||||
| @@ -200,17 +208,13 @@ class AddonOptions(CoreSysAttributes): | ||||
|         # Process list | ||||
|         for element in data_list: | ||||
|             # Nested? | ||||
|             if isinstance(typ, dict): | ||||
|                 c_options = self._nested_validate_dict(typ, element, key) | ||||
|                 options.append(c_options) | ||||
|             else: | ||||
|                 options.append(self._single_validate(typ, element, key)) | ||||
|             options.append(self._validate_element(typ, element, key)) | ||||
|  | ||||
|         return options | ||||
|  | ||||
|     def _nested_validate_dict( | ||||
|         self, typ: dict[Any, Any], data_dict: dict[Any, Any], key: str | ||||
|     ): | ||||
|     ) -> dict[Any, Any]: | ||||
|         """Validate nested items.""" | ||||
|         options = {} | ||||
|  | ||||
| @@ -230,12 +234,7 @@ class AddonOptions(CoreSysAttributes): | ||||
|                 continue | ||||
|  | ||||
|             # Nested? | ||||
|             if isinstance(typ[c_key], list): | ||||
|                 options[c_key] = self._nested_validate_list( | ||||
|                     typ[c_key][0], c_value, c_key | ||||
|                 ) | ||||
|             else: | ||||
|                 options[c_key] = self._single_validate(typ[c_key], c_value, c_key) | ||||
|             options[c_key] = self._validate_element(typ[c_key], c_value, c_key) | ||||
|  | ||||
|         self._check_missing_options(typ, options, key) | ||||
|         return options | ||||
| @@ -273,18 +272,28 @@ class UiOptions(CoreSysAttributes): | ||||
|  | ||||
|         # read options | ||||
|         for key, value in raw_schema.items(): | ||||
|             if isinstance(value, list): | ||||
|                 # nested value list | ||||
|                 self._nested_ui_list(ui_schema, value, key) | ||||
|             elif isinstance(value, dict): | ||||
|                 # nested value dict | ||||
|                 self._nested_ui_dict(ui_schema, value, key) | ||||
|             else: | ||||
|                 # normal value | ||||
|                 self._single_ui_option(ui_schema, value, key) | ||||
|             self._ui_schema_element(ui_schema, value, key) | ||||
|  | ||||
|         return ui_schema | ||||
|  | ||||
|     def _ui_schema_element( | ||||
|         self, | ||||
|         ui_schema: list[dict[str, Any]], | ||||
|         value: str, | ||||
|         key: str, | ||||
|         multiple: bool = False, | ||||
|     ): | ||||
|         if isinstance(value, list): | ||||
|             # nested value list | ||||
|             assert not multiple | ||||
|             self._nested_ui_list(ui_schema, value, key) | ||||
|         elif isinstance(value, dict): | ||||
|             # nested value dict | ||||
|             self._nested_ui_dict(ui_schema, value, key, multiple) | ||||
|         else: | ||||
|             # normal value | ||||
|             self._single_ui_option(ui_schema, value, key, multiple) | ||||
|  | ||||
|     def _single_ui_option( | ||||
|         self, | ||||
|         ui_schema: list[dict[str, Any]], | ||||
| @@ -376,10 +385,7 @@ class UiOptions(CoreSysAttributes): | ||||
|             _LOGGER.error("Invalid schema %s", key) | ||||
|             return | ||||
|  | ||||
|         if isinstance(element, dict): | ||||
|             self._nested_ui_dict(ui_schema, element, key, multiple=True) | ||||
|         else: | ||||
|             self._single_ui_option(ui_schema, element, key, multiple=True) | ||||
|         self._ui_schema_element(ui_schema, element, key, multiple=True) | ||||
|  | ||||
|     def _nested_ui_dict( | ||||
|         self, | ||||
| @@ -389,20 +395,16 @@ class UiOptions(CoreSysAttributes): | ||||
|         multiple: bool = False, | ||||
|     ) -> None: | ||||
|         """UI nested dict items.""" | ||||
|         ui_node = { | ||||
|         ui_node: dict[str, Any] = { | ||||
|             "name": key, | ||||
|             "type": "schema", | ||||
|             "optional": True, | ||||
|             "multiple": multiple, | ||||
|         } | ||||
|  | ||||
|         nested_schema = [] | ||||
|         nested_schema: list[dict[str, Any]] = [] | ||||
|         for c_key, c_value in option_dict.items(): | ||||
|             # Nested? | ||||
|             if isinstance(c_value, list): | ||||
|                 self._nested_ui_list(nested_schema, c_value, c_key) | ||||
|             else: | ||||
|                 self._single_ui_option(nested_schema, c_value, c_key) | ||||
|             self._ui_schema_element(nested_schema, c_value, c_key) | ||||
|  | ||||
|         ui_node["schema"] = nested_schema | ||||
|         ui_schema.append(ui_node) | ||||
| @@ -412,7 +414,7 @@ def _create_device_filter(str_filter: str) -> dict[str, Any]: | ||||
|     """Generate device Filter.""" | ||||
|     raw_filter = dict(value.split("=") for value in str_filter.split(";")) | ||||
|  | ||||
|     clean_filter = {} | ||||
|     clean_filter: dict[str, Any] = {} | ||||
|     for key, value in raw_filter.items(): | ||||
|         if key == "subsystem": | ||||
|             clean_filter[key] = UdevSubsystem(value) | ||||
|   | ||||
| @@ -1,9 +1,10 @@ | ||||
| """Util add-ons functions.""" | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| import asyncio | ||||
| import logging | ||||
| from pathlib import Path | ||||
| import subprocess | ||||
| from typing import TYPE_CHECKING | ||||
|  | ||||
| from ..const import ROLE_ADMIN, ROLE_MANAGER, SECURITY_DISABLE, SECURITY_PROFILE | ||||
| @@ -45,6 +46,7 @@ def rating_security(addon: AddonModel) -> int: | ||||
|             privilege in addon.privileged | ||||
|             for privilege in ( | ||||
|                 Capabilities.BPF, | ||||
|                 Capabilities.CHECKPOINT_RESTORE, | ||||
|                 Capabilities.DAC_READ_SEARCH, | ||||
|                 Capabilities.NET_ADMIN, | ||||
|                 Capabilities.NET_RAW, | ||||
| @@ -84,18 +86,20 @@ def rating_security(addon: AddonModel) -> int: | ||||
|     return max(min(8, rating), 1) | ||||
|  | ||||
|  | ||||
| async def remove_data(folder: Path) -> None: | ||||
|     """Remove folder and reset privileged.""" | ||||
|     try: | ||||
|         proc = await asyncio.create_subprocess_exec( | ||||
|             "rm", "-rf", str(folder), stdout=asyncio.subprocess.DEVNULL | ||||
|         ) | ||||
| def remove_data(folder: Path) -> None: | ||||
|     """Remove folder and reset privileged. | ||||
|  | ||||
|         _, error_msg = await proc.communicate() | ||||
|     Must be run in executor. | ||||
|     """ | ||||
|     try: | ||||
|         subprocess.run( | ||||
|             ["rm", "-rf", str(folder)], stdout=subprocess.DEVNULL, text=True, check=True | ||||
|         ) | ||||
|     except OSError as err: | ||||
|         error_msg = str(err) | ||||
|     except subprocess.CalledProcessError as procerr: | ||||
|         error_msg = procerr.stderr.strip() | ||||
|     else: | ||||
|         if proc.returncode == 0: | ||||
|             return | ||||
|         return | ||||
|  | ||||
|     _LOGGER.error("Can't remove Add-on Data: %s", error_msg) | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Validate add-ons options schema.""" | ||||
|  | ||||
| import logging | ||||
| import re | ||||
| import secrets | ||||
| @@ -31,6 +32,7 @@ from ..const import ( | ||||
|     ATTR_DISCOVERY, | ||||
|     ATTR_DOCKER_API, | ||||
|     ATTR_ENVIRONMENT, | ||||
|     ATTR_FIELDS, | ||||
|     ATTR_FULL_ACCESS, | ||||
|     ATTR_GPIO, | ||||
|     ATTR_HASSIO_API, | ||||
| @@ -54,7 +56,7 @@ from ..const import ( | ||||
|     ATTR_KERNEL_MODULES, | ||||
|     ATTR_LABELS, | ||||
|     ATTR_LEGACY, | ||||
|     ATTR_LOCATON, | ||||
|     ATTR_LOCATION, | ||||
|     ATTR_MACHINE, | ||||
|     ATTR_MAP, | ||||
|     ATTR_NAME, | ||||
| @@ -78,6 +80,8 @@ from ..const import ( | ||||
|     ATTR_STATE, | ||||
|     ATTR_STDIN, | ||||
|     ATTR_SYSTEM, | ||||
|     ATTR_SYSTEM_MANAGED, | ||||
|     ATTR_SYSTEM_MANAGED_CONFIG_ENTRY, | ||||
|     ATTR_TIMEOUT, | ||||
|     ATTR_TMPFS, | ||||
|     ATTR_TRANSLATIONS, | ||||
| @@ -95,11 +99,11 @@ from ..const import ( | ||||
|     ROLE_ALL, | ||||
|     ROLE_DEFAULT, | ||||
|     AddonBoot, | ||||
|     AddonBootConfig, | ||||
|     AddonStage, | ||||
|     AddonStartup, | ||||
|     AddonState, | ||||
| ) | ||||
| from ..discovery.validate import valid_discovery_service | ||||
| from ..docker.const import Capabilities | ||||
| from ..validate import ( | ||||
|     docker_image, | ||||
| @@ -112,6 +116,7 @@ from ..validate import ( | ||||
| ) | ||||
| from .const import ( | ||||
|     ATTR_BACKUP, | ||||
|     ATTR_BREAKING_VERSIONS, | ||||
|     ATTR_CODENOTARY, | ||||
|     ATTR_PATH, | ||||
|     ATTR_READ_ONLY, | ||||
| @@ -133,7 +138,19 @@ RE_DOCKER_IMAGE_BUILD = re.compile( | ||||
|     r"^([a-zA-Z\-\.:\d{}]+/)*?([\-\w{}]+)/([\-\w{}]+)(:[\.\-\w{}]+)?$" | ||||
| ) | ||||
|  | ||||
| SCHEMA_ELEMENT = vol.Match(RE_SCHEMA_ELEMENT) | ||||
| SCHEMA_ELEMENT = vol.Schema( | ||||
|     vol.Any( | ||||
|         vol.Match(RE_SCHEMA_ELEMENT), | ||||
|         [ | ||||
|             # A list may not directly contain another list | ||||
|             vol.Any( | ||||
|                 vol.Match(RE_SCHEMA_ELEMENT), | ||||
|                 {str: vol.Self}, | ||||
|             ) | ||||
|         ], | ||||
|         {str: vol.Self}, | ||||
|     ) | ||||
| ) | ||||
|  | ||||
| RE_MACHINE = re.compile( | ||||
|     r"^!?(?:" | ||||
| @@ -189,20 +206,6 @@ def _warn_addon_config(config: dict[str, Any]): | ||||
|             name, | ||||
|         ) | ||||
|  | ||||
|     invalid_services: list[str] = [] | ||||
|     for service in config.get(ATTR_DISCOVERY, []): | ||||
|         try: | ||||
|             valid_discovery_service(service) | ||||
|         except vol.Invalid: | ||||
|             invalid_services.append(service) | ||||
|  | ||||
|     if invalid_services: | ||||
|         _LOGGER.warning( | ||||
|             "Add-on lists the following unknown services for discovery: %s. Please report this to the maintainer of %s", | ||||
|             ", ".join(invalid_services), | ||||
|             name, | ||||
|         ) | ||||
|  | ||||
|     return config | ||||
|  | ||||
|  | ||||
| @@ -276,10 +279,23 @@ def _migrate_addon_config(protocol=False): | ||||
|         volumes = [] | ||||
|         for entry in config.get(ATTR_MAP, []): | ||||
|             if isinstance(entry, dict): | ||||
|                 # Validate that dict entries have required 'type' field | ||||
|                 if ATTR_TYPE not in entry: | ||||
|                     _LOGGER.warning( | ||||
|                         "Add-on config has invalid map entry missing 'type' field: %s. Skipping invalid entry for %s", | ||||
|                         entry, | ||||
|                         name, | ||||
|                     ) | ||||
|                     continue | ||||
|                 volumes.append(entry) | ||||
|             if isinstance(entry, str): | ||||
|                 result = RE_VOLUME.match(entry) | ||||
|                 if not result: | ||||
|                     _LOGGER.warning( | ||||
|                         "Add-on config has invalid map entry: %s. Skipping invalid entry for %s", | ||||
|                         entry, | ||||
|                         name, | ||||
|                     ) | ||||
|                     continue | ||||
|                 volumes.append( | ||||
|                     { | ||||
| @@ -288,8 +304,8 @@ def _migrate_addon_config(protocol=False): | ||||
|                     } | ||||
|                 ) | ||||
|  | ||||
|         if volumes: | ||||
|             config[ATTR_MAP] = volumes | ||||
|         # Always update config to clear potentially malformed ones | ||||
|         config[ATTR_MAP] = volumes | ||||
|  | ||||
|         # 2023-10 "config" became "homeassistant" so /config can be used for addon's public config | ||||
|         if any(volume[ATTR_TYPE] == MappingType.CONFIG for volume in volumes): | ||||
| @@ -332,7 +348,9 @@ _SCHEMA_ADDON_CONFIG = vol.Schema( | ||||
|         vol.Optional(ATTR_STARTUP, default=AddonStartup.APPLICATION): vol.Coerce( | ||||
|             AddonStartup | ||||
|         ), | ||||
|         vol.Optional(ATTR_BOOT, default=AddonBoot.AUTO): vol.Coerce(AddonBoot), | ||||
|         vol.Optional(ATTR_BOOT, default=AddonBootConfig.AUTO): vol.Coerce( | ||||
|             AddonBootConfig | ||||
|         ), | ||||
|         vol.Optional(ATTR_INIT, default=True): vol.Boolean(), | ||||
|         vol.Optional(ATTR_ADVANCED, default=False): vol.Boolean(), | ||||
|         vol.Optional(ATTR_STAGE, default=AddonStage.STABLE): vol.Coerce(AddonStage), | ||||
| @@ -401,20 +419,7 @@ _SCHEMA_ADDON_CONFIG = vol.Schema( | ||||
|         vol.Optional(ATTR_CODENOTARY): vol.Email(), | ||||
|         vol.Optional(ATTR_OPTIONS, default={}): dict, | ||||
|         vol.Optional(ATTR_SCHEMA, default={}): vol.Any( | ||||
|             vol.Schema( | ||||
|                 { | ||||
|                     str: vol.Any( | ||||
|                         SCHEMA_ELEMENT, | ||||
|                         [ | ||||
|                             vol.Any( | ||||
|                                 SCHEMA_ELEMENT, | ||||
|                                 {str: vol.Any(SCHEMA_ELEMENT, [SCHEMA_ELEMENT])}, | ||||
|                             ) | ||||
|                         ], | ||||
|                         vol.Schema({str: vol.Any(SCHEMA_ELEMENT, [SCHEMA_ELEMENT])}), | ||||
|                     ) | ||||
|                 } | ||||
|             ), | ||||
|             vol.Schema({str: SCHEMA_ELEMENT}), | ||||
|             False, | ||||
|         ), | ||||
|         vol.Optional(ATTR_IMAGE): docker_image, | ||||
| @@ -422,6 +427,7 @@ _SCHEMA_ADDON_CONFIG = vol.Schema( | ||||
|             vol.Coerce(int), vol.Range(min=10, max=300) | ||||
|         ), | ||||
|         vol.Optional(ATTR_JOURNALD, default=False): vol.Boolean(), | ||||
|         vol.Optional(ATTR_BREAKING_VERSIONS, default=list): [version_tag], | ||||
|     }, | ||||
|     extra=vol.REMOVE_EXTRA, | ||||
| ) | ||||
| @@ -449,6 +455,7 @@ SCHEMA_TRANSLATION_CONFIGURATION = vol.Schema( | ||||
|     { | ||||
|         vol.Required(ATTR_NAME): str, | ||||
|         vol.Optional(ATTR_DESCRIPTON): vol.Maybe(str), | ||||
|         vol.Optional(ATTR_FIELDS): {str: vol.Self}, | ||||
|     }, | ||||
|     extra=vol.REMOVE_EXTRA, | ||||
| ) | ||||
| @@ -480,6 +487,8 @@ SCHEMA_ADDON_USER = vol.Schema( | ||||
|         vol.Optional(ATTR_PROTECTED, default=True): vol.Boolean(), | ||||
|         vol.Optional(ATTR_INGRESS_PANEL, default=False): vol.Boolean(), | ||||
|         vol.Optional(ATTR_WATCHDOG, default=False): vol.Boolean(), | ||||
|         vol.Optional(ATTR_SYSTEM_MANAGED, default=False): vol.Boolean(), | ||||
|         vol.Optional(ATTR_SYSTEM_MANAGED_CONFIG_ENTRY, default=None): vol.Maybe(str), | ||||
|     }, | ||||
|     extra=vol.REMOVE_EXTRA, | ||||
| ) | ||||
| @@ -488,7 +497,7 @@ SCHEMA_ADDON_SYSTEM = vol.All( | ||||
|     _migrate_addon_config(), | ||||
|     _SCHEMA_ADDON_CONFIG.extend( | ||||
|         { | ||||
|             vol.Required(ATTR_LOCATON): str, | ||||
|             vol.Required(ATTR_LOCATION): str, | ||||
|             vol.Required(ATTR_REPOSITORY): str, | ||||
|             vol.Required(ATTR_TRANSLATIONS, default=dict): { | ||||
|                 str: SCHEMA_ADDON_TRANSLATIONS | ||||
|   | ||||
| @@ -1,20 +1,23 @@ | ||||
| """Init file for Supervisor RESTful API.""" | ||||
|  | ||||
| from dataclasses import dataclass | ||||
| from functools import partial | ||||
| import logging | ||||
| from pathlib import Path | ||||
| from typing import Any | ||||
|  | ||||
| from aiohttp import web | ||||
| from aiohttp_fast_url_dispatcher import FastUrlDispatcher, attach_fast_url_dispatcher | ||||
| from aiohttp import hdrs, web | ||||
|  | ||||
| from ..const import AddonState | ||||
| from ..const import SUPERVISOR_DOCKER_NAME, AddonState | ||||
| from ..coresys import CoreSys, CoreSysAttributes | ||||
| from ..exceptions import APIAddonNotInstalled | ||||
| from ..exceptions import APIAddonNotInstalled, HostNotSupportedError | ||||
| from ..utils.sentry import async_capture_exception | ||||
| from .addons import APIAddons | ||||
| from .audio import APIAudio | ||||
| from .auth import APIAuth | ||||
| from .backups import APIBackups | ||||
| from .cli import APICli | ||||
| from .const import CONTENT_TYPE_TEXT | ||||
| from .discovery import APIDiscovery | ||||
| from .dns import APICoreDNS | ||||
| from .docker import APIDocker | ||||
| @@ -36,7 +39,7 @@ from .security import APISecurity | ||||
| from .services import APIServices | ||||
| from .store import APIStore | ||||
| from .supervisor import APISupervisor | ||||
| from .utils import api_process | ||||
| from .utils import api_process, api_process_raw | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
|  | ||||
| @@ -45,6 +48,14 @@ MAX_CLIENT_SIZE: int = 1024**2 * 16 | ||||
| MAX_LINE_SIZE: int = 24570 | ||||
|  | ||||
|  | ||||
| @dataclass(slots=True, frozen=True) | ||||
| class StaticResourceConfig: | ||||
|     """Configuration for a static resource.""" | ||||
|  | ||||
|     prefix: str | ||||
|     path: Path | ||||
|  | ||||
|  | ||||
| class RestAPI(CoreSysAttributes): | ||||
|     """Handle RESTful API for Supervisor.""" | ||||
|  | ||||
| @@ -65,14 +76,19 @@ class RestAPI(CoreSysAttributes): | ||||
|                 "max_field_size": MAX_LINE_SIZE, | ||||
|             }, | ||||
|         ) | ||||
|         attach_fast_url_dispatcher(self.webapp, FastUrlDispatcher()) | ||||
|  | ||||
|         # service stuff | ||||
|         self._runner: web.AppRunner = web.AppRunner(self.webapp, shutdown_timeout=5) | ||||
|         self._site: web.TCPSite | None = None | ||||
|  | ||||
|         # share single host API handler for reuse in logging endpoints | ||||
|         self._api_host: APIHost = APIHost() | ||||
|         self._api_host.coresys = coresys | ||||
|  | ||||
|     async def load(self) -> None: | ||||
|         """Register REST API Calls.""" | ||||
|         static_resource_configs: list[StaticResourceConfig] = [] | ||||
|  | ||||
|         self._register_addons() | ||||
|         self._register_audio() | ||||
|         self._register_auth() | ||||
| @@ -91,7 +107,7 @@ class RestAPI(CoreSysAttributes): | ||||
|         self._register_network() | ||||
|         self._register_observer() | ||||
|         self._register_os() | ||||
|         self._register_panel() | ||||
|         static_resource_configs.extend(self._register_panel()) | ||||
|         self._register_proxy() | ||||
|         self._register_resolution() | ||||
|         self._register_root() | ||||
| @@ -100,12 +116,62 @@ class RestAPI(CoreSysAttributes): | ||||
|         self._register_store() | ||||
|         self._register_supervisor() | ||||
|  | ||||
|         if static_resource_configs: | ||||
|  | ||||
|             def process_configs() -> list[web.StaticResource]: | ||||
|                 return [ | ||||
|                     web.StaticResource(config.prefix, config.path) | ||||
|                     for config in static_resource_configs | ||||
|                 ] | ||||
|  | ||||
|             for resource in await self.sys_run_in_executor(process_configs): | ||||
|                 self.webapp.router.register_resource(resource) | ||||
|  | ||||
|         await self.start() | ||||
|  | ||||
|     def _register_advanced_logs(self, path: str, syslog_identifier: str): | ||||
|         """Register logs endpoint for a given path, returning logs for single syslog identifier.""" | ||||
|  | ||||
|         self.webapp.add_routes( | ||||
|             [ | ||||
|                 web.get( | ||||
|                     f"{path}/logs", | ||||
|                     partial(self._api_host.advanced_logs, identifier=syslog_identifier), | ||||
|                 ), | ||||
|                 web.get( | ||||
|                     f"{path}/logs/follow", | ||||
|                     partial( | ||||
|                         self._api_host.advanced_logs, | ||||
|                         identifier=syslog_identifier, | ||||
|                         follow=True, | ||||
|                     ), | ||||
|                 ), | ||||
|                 web.get( | ||||
|                     f"{path}/logs/latest", | ||||
|                     partial( | ||||
|                         self._api_host.advanced_logs, | ||||
|                         identifier=syslog_identifier, | ||||
|                         latest=True, | ||||
|                     ), | ||||
|                 ), | ||||
|                 web.get( | ||||
|                     f"{path}/logs/boots/{{bootid}}", | ||||
|                     partial(self._api_host.advanced_logs, identifier=syslog_identifier), | ||||
|                 ), | ||||
|                 web.get( | ||||
|                     f"{path}/logs/boots/{{bootid}}/follow", | ||||
|                     partial( | ||||
|                         self._api_host.advanced_logs, | ||||
|                         identifier=syslog_identifier, | ||||
|                         follow=True, | ||||
|                     ), | ||||
|                 ), | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|     def _register_host(self) -> None: | ||||
|         """Register hostcontrol functions.""" | ||||
|         api_host = APIHost() | ||||
|         api_host.coresys = self.coresys | ||||
|         api_host = self._api_host | ||||
|  | ||||
|         self.webapp.add_routes( | ||||
|             [ | ||||
| @@ -140,6 +206,7 @@ class RestAPI(CoreSysAttributes): | ||||
|                 web.post("/host/reload", api_host.reload), | ||||
|                 web.post("/host/options", api_host.options), | ||||
|                 web.get("/host/services", api_host.services), | ||||
|                 web.get("/host/disks/default/usage", api_host.disk_usage), | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
| @@ -179,9 +246,13 @@ class RestAPI(CoreSysAttributes): | ||||
|             [ | ||||
|                 web.get("/os/info", api_os.info), | ||||
|                 web.post("/os/update", api_os.update), | ||||
|                 web.get("/os/config/swap", api_os.config_swap_info), | ||||
|                 web.post("/os/config/swap", api_os.config_swap_options), | ||||
|                 web.post("/os/config/sync", api_os.config_sync), | ||||
|                 web.post("/os/datadisk/move", api_os.migrate_data), | ||||
|                 web.get("/os/datadisk/list", api_os.list_data), | ||||
|                 web.post("/os/datadisk/wipe", api_os.wipe_data), | ||||
|                 web.post("/os/boot-slot", api_os.set_boot_slot), | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
| @@ -219,6 +290,8 @@ class RestAPI(CoreSysAttributes): | ||||
|                 web.get("/jobs/info", api_jobs.info), | ||||
|                 web.post("/jobs/options", api_jobs.options), | ||||
|                 web.post("/jobs/reset", api_jobs.reset), | ||||
|                 web.get("/jobs/{uuid}", api_jobs.job_info), | ||||
|                 web.delete("/jobs/{uuid}", api_jobs.remove_job), | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
| @@ -257,11 +330,11 @@ class RestAPI(CoreSysAttributes): | ||||
|             [ | ||||
|                 web.get("/multicast/info", api_multicast.info), | ||||
|                 web.get("/multicast/stats", api_multicast.stats), | ||||
|                 web.get("/multicast/logs", api_multicast.logs), | ||||
|                 web.post("/multicast/update", api_multicast.update), | ||||
|                 web.post("/multicast/restart", api_multicast.restart), | ||||
|             ] | ||||
|         ) | ||||
|         self._register_advanced_logs("/multicast", "hassio_multicast") | ||||
|  | ||||
|     def _register_hardware(self) -> None: | ||||
|         """Register hardware functions.""" | ||||
| @@ -281,6 +354,9 @@ class RestAPI(CoreSysAttributes): | ||||
|         api_root.coresys = self.coresys | ||||
|  | ||||
|         self.webapp.add_routes([web.get("/info", api_root.info)]) | ||||
|         self.webapp.add_routes([web.post("/reload_updates", api_root.reload_updates)]) | ||||
|  | ||||
|         # Discouraged | ||||
|         self.webapp.add_routes([web.post("/refresh_updates", api_root.refresh_updates)]) | ||||
|         self.webapp.add_routes( | ||||
|             [web.get("/available_updates", api_root.available_updates)] | ||||
| @@ -334,6 +410,7 @@ class RestAPI(CoreSysAttributes): | ||||
|                 web.post("/auth", api_auth.auth), | ||||
|                 web.post("/auth/reset", api_auth.reset), | ||||
|                 web.delete("/auth/cache", api_auth.cache), | ||||
|                 web.get("/auth/list", api_auth.list_users), | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
| @@ -347,7 +424,6 @@ class RestAPI(CoreSysAttributes): | ||||
|                 web.get("/supervisor/ping", api_supervisor.ping), | ||||
|                 web.get("/supervisor/info", api_supervisor.info), | ||||
|                 web.get("/supervisor/stats", api_supervisor.stats), | ||||
|                 web.get("/supervisor/logs", api_supervisor.logs), | ||||
|                 web.post("/supervisor/update", api_supervisor.update), | ||||
|                 web.post("/supervisor/reload", api_supervisor.reload), | ||||
|                 web.post("/supervisor/restart", api_supervisor.restart), | ||||
| @@ -356,6 +432,44 @@ class RestAPI(CoreSysAttributes): | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|         async def get_supervisor_logs(*args, **kwargs): | ||||
|             try: | ||||
|                 return await self._api_host.advanced_logs_handler( | ||||
|                     *args, identifier=SUPERVISOR_DOCKER_NAME, **kwargs | ||||
|                 ) | ||||
|             except Exception as err:  # pylint: disable=broad-exception-caught | ||||
|                 # Supervisor logs are critical, so catch everything, log the exception | ||||
|                 # and try to return Docker container logs as the fallback | ||||
|                 _LOGGER.exception( | ||||
|                     "Failed to get supervisor logs using advanced_logs API" | ||||
|                 ) | ||||
|                 if not isinstance(err, HostNotSupportedError): | ||||
|                     # No need to capture HostNotSupportedError to Sentry, the cause | ||||
|                     # is known and reported to the user using the resolution center. | ||||
|                     await async_capture_exception(err) | ||||
|                 kwargs.pop("follow", None)  # Follow is not supported for Docker logs | ||||
|                 kwargs.pop("latest", None)  # Latest is not supported for Docker logs | ||||
|                 return await api_supervisor.logs(*args, **kwargs) | ||||
|  | ||||
|         self.webapp.add_routes( | ||||
|             [ | ||||
|                 web.get("/supervisor/logs", get_supervisor_logs), | ||||
|                 web.get( | ||||
|                     "/supervisor/logs/follow", | ||||
|                     partial(get_supervisor_logs, follow=True), | ||||
|                 ), | ||||
|                 web.get( | ||||
|                     "/supervisor/logs/latest", | ||||
|                     partial(get_supervisor_logs, latest=True), | ||||
|                 ), | ||||
|                 web.get("/supervisor/logs/boots/{bootid}", get_supervisor_logs), | ||||
|                 web.get( | ||||
|                     "/supervisor/logs/boots/{bootid}/follow", | ||||
|                     partial(get_supervisor_logs, follow=True), | ||||
|                 ), | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|     def _register_homeassistant(self) -> None: | ||||
|         """Register Home Assistant functions.""" | ||||
|         api_hass = APIHomeAssistant() | ||||
| @@ -364,7 +478,6 @@ class RestAPI(CoreSysAttributes): | ||||
|         self.webapp.add_routes( | ||||
|             [ | ||||
|                 web.get("/core/info", api_hass.info), | ||||
|                 web.get("/core/logs", api_hass.logs), | ||||
|                 web.get("/core/stats", api_hass.stats), | ||||
|                 web.post("/core/options", api_hass.options), | ||||
|                 web.post("/core/update", api_hass.update), | ||||
| @@ -376,11 +489,12 @@ class RestAPI(CoreSysAttributes): | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|         self._register_advanced_logs("/core", "homeassistant") | ||||
|  | ||||
|         # Reroute from legacy | ||||
|         self.webapp.add_routes( | ||||
|             [ | ||||
|                 web.get("/homeassistant/info", api_hass.info), | ||||
|                 web.get("/homeassistant/logs", api_hass.logs), | ||||
|                 web.get("/homeassistant/stats", api_hass.stats), | ||||
|                 web.post("/homeassistant/options", api_hass.options), | ||||
|                 web.post("/homeassistant/restart", api_hass.restart), | ||||
| @@ -392,6 +506,8 @@ class RestAPI(CoreSysAttributes): | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|         self._register_advanced_logs("/homeassistant", "homeassistant") | ||||
|  | ||||
|     def _register_proxy(self) -> None: | ||||
|         """Register Home Assistant API Proxy.""" | ||||
|         api_proxy = APIProxy() | ||||
| @@ -427,24 +543,49 @@ class RestAPI(CoreSysAttributes): | ||||
|  | ||||
|         self.webapp.add_routes( | ||||
|             [ | ||||
|                 web.get("/addons", api_addons.list), | ||||
|                 web.get("/addons", api_addons.list_addons), | ||||
|                 web.post("/addons/{addon}/uninstall", api_addons.uninstall), | ||||
|                 web.post("/addons/{addon}/start", api_addons.start), | ||||
|                 web.post("/addons/{addon}/stop", api_addons.stop), | ||||
|                 web.post("/addons/{addon}/restart", api_addons.restart), | ||||
|                 web.post("/addons/{addon}/options", api_addons.options), | ||||
|                 web.post("/addons/{addon}/sys_options", api_addons.sys_options), | ||||
|                 web.post( | ||||
|                     "/addons/{addon}/options/validate", api_addons.options_validate | ||||
|                 ), | ||||
|                 web.get("/addons/{addon}/options/config", api_addons.options_config), | ||||
|                 web.post("/addons/{addon}/rebuild", api_addons.rebuild), | ||||
|                 web.get("/addons/{addon}/logs", api_addons.logs), | ||||
|                 web.post("/addons/{addon}/stdin", api_addons.stdin), | ||||
|                 web.post("/addons/{addon}/security", api_addons.security), | ||||
|                 web.get("/addons/{addon}/stats", api_addons.stats), | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|         @api_process_raw(CONTENT_TYPE_TEXT, error_type=CONTENT_TYPE_TEXT) | ||||
|         async def get_addon_logs(request, *args, **kwargs): | ||||
|             addon = api_addons.get_addon_for_request(request) | ||||
|             kwargs["identifier"] = f"addon_{addon.slug}" | ||||
|             return await self._api_host.advanced_logs(request, *args, **kwargs) | ||||
|  | ||||
|         self.webapp.add_routes( | ||||
|             [ | ||||
|                 web.get("/addons/{addon}/logs", get_addon_logs), | ||||
|                 web.get( | ||||
|                     "/addons/{addon}/logs/follow", | ||||
|                     partial(get_addon_logs, follow=True), | ||||
|                 ), | ||||
|                 web.get( | ||||
|                     "/addons/{addon}/logs/latest", | ||||
|                     partial(get_addon_logs, latest=True), | ||||
|                 ), | ||||
|                 web.get("/addons/{addon}/logs/boots/{bootid}", get_addon_logs), | ||||
|                 web.get( | ||||
|                     "/addons/{addon}/logs/boots/{bootid}/follow", | ||||
|                     partial(get_addon_logs, follow=True), | ||||
|                 ), | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|         # Legacy routing to support requests for not installed addons | ||||
|         api_store = APIStore() | ||||
|         api_store.coresys = self.coresys | ||||
| @@ -474,7 +615,9 @@ class RestAPI(CoreSysAttributes): | ||||
|                 web.post("/ingress/session", api_ingress.create_session), | ||||
|                 web.post("/ingress/validate_session", api_ingress.validate_session), | ||||
|                 web.get("/ingress/panels", api_ingress.panels), | ||||
|                 web.view("/ingress/{token}/{path:.*}", api_ingress.handler), | ||||
|                 web.route( | ||||
|                     hdrs.METH_ANY, "/ingress/{token}/{path:.*}", api_ingress.handler | ||||
|                 ), | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
| @@ -485,7 +628,7 @@ class RestAPI(CoreSysAttributes): | ||||
|  | ||||
|         self.webapp.add_routes( | ||||
|             [ | ||||
|                 web.get("/backups", api_backups.list), | ||||
|                 web.get("/backups", api_backups.list_backups), | ||||
|                 web.get("/backups/info", api_backups.info), | ||||
|                 web.post("/backups/options", api_backups.options), | ||||
|                 web.post("/backups/reload", api_backups.reload), | ||||
| @@ -512,7 +655,7 @@ class RestAPI(CoreSysAttributes): | ||||
|  | ||||
|         self.webapp.add_routes( | ||||
|             [ | ||||
|                 web.get("/services", api_services.list), | ||||
|                 web.get("/services", api_services.list_services), | ||||
|                 web.get("/services/{service}", api_services.get_service), | ||||
|                 web.post("/services/{service}", api_services.set_service), | ||||
|                 web.delete("/services/{service}", api_services.del_service), | ||||
| @@ -526,7 +669,7 @@ class RestAPI(CoreSysAttributes): | ||||
|  | ||||
|         self.webapp.add_routes( | ||||
|             [ | ||||
|                 web.get("/discovery", api_discovery.list), | ||||
|                 web.get("/discovery", api_discovery.list_discovery), | ||||
|                 web.get("/discovery/{uuid}", api_discovery.get_discovery), | ||||
|                 web.delete("/discovery/{uuid}", api_discovery.del_discovery), | ||||
|                 web.post("/discovery", api_discovery.set_discovery), | ||||
| @@ -542,7 +685,6 @@ class RestAPI(CoreSysAttributes): | ||||
|             [ | ||||
|                 web.get("/dns/info", api_dns.info), | ||||
|                 web.get("/dns/stats", api_dns.stats), | ||||
|                 web.get("/dns/logs", api_dns.logs), | ||||
|                 web.post("/dns/update", api_dns.update), | ||||
|                 web.post("/dns/options", api_dns.options), | ||||
|                 web.post("/dns/restart", api_dns.restart), | ||||
| @@ -550,18 +692,17 @@ class RestAPI(CoreSysAttributes): | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|         self._register_advanced_logs("/dns", "hassio_dns") | ||||
|  | ||||
|     def _register_audio(self) -> None: | ||||
|         """Register Audio functions.""" | ||||
|         api_audio = APIAudio() | ||||
|         api_audio.coresys = self.coresys | ||||
|         api_host = APIHost() | ||||
|         api_host.coresys = self.coresys | ||||
|  | ||||
|         self.webapp.add_routes( | ||||
|             [ | ||||
|                 web.get("/audio/info", api_audio.info), | ||||
|                 web.get("/audio/stats", api_audio.stats), | ||||
|                 web.get("/audio/logs", api_audio.logs), | ||||
|                 web.post("/audio/update", api_audio.update), | ||||
|                 web.post("/audio/restart", api_audio.restart), | ||||
|                 web.post("/audio/reload", api_audio.reload), | ||||
| @@ -574,6 +715,8 @@ class RestAPI(CoreSysAttributes): | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|         self._register_advanced_logs("/audio", "hassio_audio") | ||||
|  | ||||
|     def _register_mounts(self) -> None: | ||||
|         """Register mounts endpoints.""" | ||||
|         api_mounts = APIMounts() | ||||
| @@ -600,7 +743,6 @@ class RestAPI(CoreSysAttributes): | ||||
|                 web.get("/store", api_store.store_info), | ||||
|                 web.get("/store/addons", api_store.addons_list), | ||||
|                 web.get("/store/addons/{addon}", api_store.addons_addon_info), | ||||
|                 web.get("/store/addons/{addon}/{version}", api_store.addons_addon_info), | ||||
|                 web.get("/store/addons/{addon}/icon", api_store.addons_addon_icon), | ||||
|                 web.get("/store/addons/{addon}/logo", api_store.addons_addon_logo), | ||||
|                 web.get( | ||||
| @@ -610,6 +752,10 @@ class RestAPI(CoreSysAttributes): | ||||
|                     "/store/addons/{addon}/documentation", | ||||
|                     api_store.addons_addon_documentation, | ||||
|                 ), | ||||
|                 web.get( | ||||
|                     "/store/addons/{addon}/availability", | ||||
|                     api_store.addons_addon_availability, | ||||
|                 ), | ||||
|                 web.post( | ||||
|                     "/store/addons/{addon}/install", api_store.addons_addon_install | ||||
|                 ), | ||||
| @@ -622,6 +768,8 @@ class RestAPI(CoreSysAttributes): | ||||
|                     "/store/addons/{addon}/update/{version}", | ||||
|                     api_store.addons_addon_update, | ||||
|                 ), | ||||
|                 # Must be below others since it has a wildcard in resource path | ||||
|                 web.get("/store/addons/{addon}/{version}", api_store.addons_addon_info), | ||||
|                 web.post("/store/reload", api_store.reload), | ||||
|                 web.get("/store/repositories", api_store.repositories_list), | ||||
|                 web.get( | ||||
| @@ -651,10 +799,9 @@ class RestAPI(CoreSysAttributes): | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|     def _register_panel(self) -> None: | ||||
|     def _register_panel(self) -> list[StaticResourceConfig]: | ||||
|         """Register panel for Home Assistant.""" | ||||
|         panel_dir = Path(__file__).parent.joinpath("panel") | ||||
|         self.webapp.add_routes([web.static("/app", panel_dir)]) | ||||
|         return [StaticResourceConfig("/app", Path(__file__).parent.joinpath("panel"))] | ||||
|  | ||||
|     def _register_docker(self) -> None: | ||||
|         """Register docker configuration functions.""" | ||||
| @@ -664,6 +811,7 @@ class RestAPI(CoreSysAttributes): | ||||
|         self.webapp.add_routes( | ||||
|             [ | ||||
|                 web.get("/docker/info", api_docker.info), | ||||
|                 web.post("/docker/options", api_docker.options), | ||||
|                 web.get("/docker/registries", api_docker.registries), | ||||
|                 web.post("/docker/registries", api_docker.create_registry), | ||||
|                 web.delete("/docker/registries/{hostname}", api_docker.remove_registry), | ||||
|   | ||||
| @@ -1,15 +1,15 @@ | ||||
| """Init file for Supervisor Home Assistant RESTful API.""" | ||||
|  | ||||
| import asyncio | ||||
| from collections.abc import Awaitable | ||||
| import logging | ||||
| from typing import Any | ||||
| from typing import Any, TypedDict | ||||
|  | ||||
| from aiohttp import web | ||||
| import voluptuous as vol | ||||
| from voluptuous.humanize import humanize_error | ||||
|  | ||||
| from ..addons.addon import Addon | ||||
| from ..addons.manager import AnyAddon | ||||
| from ..addons.utils import rating_security | ||||
| from ..const import ( | ||||
|     ATTR_ADDONS, | ||||
| @@ -36,6 +36,7 @@ from ..const import ( | ||||
|     ATTR_DNS, | ||||
|     ATTR_DOCKER_API, | ||||
|     ATTR_DOCUMENTATION, | ||||
|     ATTR_FORCE, | ||||
|     ATTR_FULL_ACCESS, | ||||
|     ATTR_GPIO, | ||||
|     ATTR_HASSIO_API, | ||||
| @@ -62,7 +63,6 @@ from ..const import ( | ||||
|     ATTR_MEMORY_LIMIT, | ||||
|     ATTR_MEMORY_PERCENT, | ||||
|     ATTR_MEMORY_USAGE, | ||||
|     ATTR_MESSAGE, | ||||
|     ATTR_NAME, | ||||
|     ATTR_NETWORK, | ||||
|     ATTR_NETWORK_DESCRIPTION, | ||||
| @@ -71,7 +71,6 @@ from ..const import ( | ||||
|     ATTR_OPTIONS, | ||||
|     ATTR_PRIVILEGED, | ||||
|     ATTR_PROTECTED, | ||||
|     ATTR_PWNED, | ||||
|     ATTR_RATING, | ||||
|     ATTR_REPOSITORY, | ||||
|     ATTR_SCHEMA, | ||||
| @@ -81,13 +80,14 @@ from ..const import ( | ||||
|     ATTR_STARTUP, | ||||
|     ATTR_STATE, | ||||
|     ATTR_STDIN, | ||||
|     ATTR_SYSTEM_MANAGED, | ||||
|     ATTR_SYSTEM_MANAGED_CONFIG_ENTRY, | ||||
|     ATTR_TRANSLATIONS, | ||||
|     ATTR_UART, | ||||
|     ATTR_UDEV, | ||||
|     ATTR_UPDATE_AVAILABLE, | ||||
|     ATTR_URL, | ||||
|     ATTR_USB, | ||||
|     ATTR_VALID, | ||||
|     ATTR_VERSION, | ||||
|     ATTR_VERSION_LATEST, | ||||
|     ATTR_VIDEO, | ||||
| @@ -95,6 +95,7 @@ from ..const import ( | ||||
|     ATTR_WEBUI, | ||||
|     REQUEST_FROM, | ||||
|     AddonBoot, | ||||
|     AddonBootConfig, | ||||
| ) | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..docker.stats import DockerStats | ||||
| @@ -102,12 +103,13 @@ from ..exceptions import ( | ||||
|     APIAddonNotInstalled, | ||||
|     APIError, | ||||
|     APIForbidden, | ||||
|     APINotFound, | ||||
|     PwnedError, | ||||
|     PwnedSecret, | ||||
| ) | ||||
| from ..validate import docker_ports | ||||
| from .const import ATTR_SIGNED, CONTENT_TYPE_BINARY | ||||
| from .utils import api_process, api_process_raw, api_validate, json_loads | ||||
| from .const import ATTR_BOOT_CONFIG, ATTR_REMOVE_CONFIG, ATTR_SIGNED | ||||
| from .utils import api_process, api_validate, json_loads | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
|  | ||||
| @@ -126,16 +128,37 @@ SCHEMA_OPTIONS = vol.Schema( | ||||
|     } | ||||
| ) | ||||
|  | ||||
| # pylint: disable=no-value-for-parameter | ||||
| SCHEMA_SYS_OPTIONS = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_SYSTEM_MANAGED): vol.Boolean(), | ||||
|         vol.Optional(ATTR_SYSTEM_MANAGED_CONFIG_ENTRY): vol.Maybe(str), | ||||
|     } | ||||
| ) | ||||
|  | ||||
| SCHEMA_SECURITY = vol.Schema({vol.Optional(ATTR_PROTECTED): vol.Boolean()}) | ||||
|  | ||||
| SCHEMA_UNINSTALL = vol.Schema( | ||||
|     {vol.Optional(ATTR_REMOVE_CONFIG, default=False): vol.Boolean()} | ||||
| ) | ||||
|  | ||||
| SCHEMA_REBUILD = vol.Schema({vol.Optional(ATTR_FORCE, default=False): vol.Boolean()}) | ||||
| # pylint: enable=no-value-for-parameter | ||||
|  | ||||
|  | ||||
| class OptionsValidateResponse(TypedDict): | ||||
|     """Response object for options validate.""" | ||||
|  | ||||
|     message: str | ||||
|     valid: bool | ||||
|     pwned: bool | None | ||||
|  | ||||
|  | ||||
| class APIAddons(CoreSysAttributes): | ||||
|     """Handle RESTful API for add-on functions.""" | ||||
|  | ||||
|     def _extract_addon(self, request: web.Request) -> Addon: | ||||
|         """Return addon, throw an exception it it doesn't exist.""" | ||||
|         addon_slug: str = request.match_info.get("addon") | ||||
|     def get_addon_for_request(self, request: web.Request) -> Addon: | ||||
|         """Return addon, throw an exception if it doesn't exist.""" | ||||
|         addon_slug: str = request.match_info["addon"] | ||||
|  | ||||
|         # Lookup itself | ||||
|         if addon_slug == "self": | ||||
| @@ -146,14 +169,14 @@ class APIAddons(CoreSysAttributes): | ||||
|  | ||||
|         addon = self.sys_addons.get(addon_slug) | ||||
|         if not addon: | ||||
|             raise APIError(f"Addon {addon_slug} does not exist") | ||||
|             raise APINotFound(f"Addon {addon_slug} does not exist") | ||||
|         if not isinstance(addon, Addon) or not addon.is_installed: | ||||
|             raise APIAddonNotInstalled("Addon is not installed") | ||||
|  | ||||
|         return addon | ||||
|  | ||||
|     @api_process | ||||
|     async def list(self, request: web.Request) -> dict[str, Any]: | ||||
|     async def list_addons(self, request: web.Request) -> dict[str, Any]: | ||||
|         """Return all add-ons or repositories.""" | ||||
|         data_addons = [ | ||||
|             { | ||||
| @@ -174,6 +197,7 @@ class APIAddons(CoreSysAttributes): | ||||
|                 ATTR_URL: addon.url, | ||||
|                 ATTR_ICON: addon.with_icon, | ||||
|                 ATTR_LOGO: addon.with_logo, | ||||
|                 ATTR_SYSTEM_MANAGED: addon.system_managed, | ||||
|             } | ||||
|             for addon in self.sys_addons.installed | ||||
|         ] | ||||
| @@ -187,7 +211,7 @@ class APIAddons(CoreSysAttributes): | ||||
|  | ||||
|     async def info(self, request: web.Request) -> dict[str, Any]: | ||||
|         """Return add-on information.""" | ||||
|         addon: AnyAddon = self._extract_addon(request) | ||||
|         addon: Addon = self.get_addon_for_request(request) | ||||
|  | ||||
|         data = { | ||||
|             ATTR_NAME: addon.name, | ||||
| @@ -195,13 +219,14 @@ class APIAddons(CoreSysAttributes): | ||||
|             ATTR_HOSTNAME: addon.hostname, | ||||
|             ATTR_DNS: addon.dns, | ||||
|             ATTR_DESCRIPTON: addon.description, | ||||
|             ATTR_LONG_DESCRIPTION: addon.long_description, | ||||
|             ATTR_LONG_DESCRIPTION: await addon.long_description(), | ||||
|             ATTR_ADVANCED: addon.advanced, | ||||
|             ATTR_STAGE: addon.stage, | ||||
|             ATTR_REPOSITORY: addon.repository, | ||||
|             ATTR_VERSION_LATEST: addon.latest_version, | ||||
|             ATTR_PROTECTED: addon.protected, | ||||
|             ATTR_RATING: rating_security(addon), | ||||
|             ATTR_BOOT_CONFIG: addon.boot_config, | ||||
|             ATTR_BOOT: addon.boot, | ||||
|             ATTR_OPTIONS: addon.options, | ||||
|             ATTR_SCHEMA: addon.schema_ui, | ||||
| @@ -261,6 +286,8 @@ class APIAddons(CoreSysAttributes): | ||||
|             ATTR_WATCHDOG: addon.watchdog, | ||||
|             ATTR_DEVICES: addon.static_devices | ||||
|             + [device.path for device in addon.devices], | ||||
|             ATTR_SYSTEM_MANAGED: addon.system_managed, | ||||
|             ATTR_SYSTEM_MANAGED_CONFIG_ENTRY: addon.system_managed_config_entry, | ||||
|         } | ||||
|  | ||||
|         return data | ||||
| @@ -268,7 +295,7 @@ class APIAddons(CoreSysAttributes): | ||||
|     @api_process | ||||
|     async def options(self, request: web.Request) -> None: | ||||
|         """Store user options for add-on.""" | ||||
|         addon = self._extract_addon(request) | ||||
|         addon = self.get_addon_for_request(request) | ||||
|  | ||||
|         # Update secrets for validation | ||||
|         await self.sys_homeassistant.secrets.reload() | ||||
| @@ -279,10 +306,14 @@ class APIAddons(CoreSysAttributes): | ||||
|         ) | ||||
|  | ||||
|         # Validate/Process Body | ||||
|         body = await api_validate(addon_schema, request, origin=[ATTR_OPTIONS]) | ||||
|         body = await api_validate(addon_schema, request) | ||||
|         if ATTR_OPTIONS in body: | ||||
|             addon.options = body[ATTR_OPTIONS] | ||||
|         if ATTR_BOOT in body: | ||||
|             if addon.boot_config == AddonBootConfig.MANUAL_ONLY: | ||||
|                 raise APIError( | ||||
|                     f"Addon {addon.slug} boot option is set to {addon.boot_config} so it cannot be changed" | ||||
|                 ) | ||||
|             addon.boot = body[ATTR_BOOT] | ||||
|         if ATTR_AUTO_UPDATE in body: | ||||
|             addon.auto_update = body[ATTR_AUTO_UPDATE] | ||||
| @@ -298,13 +329,27 @@ class APIAddons(CoreSysAttributes): | ||||
|         if ATTR_WATCHDOG in body: | ||||
|             addon.watchdog = body[ATTR_WATCHDOG] | ||||
|  | ||||
|         addon.save_persist() | ||||
|         await addon.save_persist() | ||||
|  | ||||
|     @api_process | ||||
|     async def options_validate(self, request: web.Request) -> None: | ||||
|     async def sys_options(self, request: web.Request) -> None: | ||||
|         """Store system options for an add-on.""" | ||||
|         addon = self.get_addon_for_request(request) | ||||
|  | ||||
|         # Validate/Process Body | ||||
|         body = await api_validate(SCHEMA_SYS_OPTIONS, request) | ||||
|         if ATTR_SYSTEM_MANAGED in body: | ||||
|             addon.system_managed = body[ATTR_SYSTEM_MANAGED] | ||||
|         if ATTR_SYSTEM_MANAGED_CONFIG_ENTRY in body: | ||||
|             addon.system_managed_config_entry = body[ATTR_SYSTEM_MANAGED_CONFIG_ENTRY] | ||||
|  | ||||
|         await addon.save_persist() | ||||
|  | ||||
|     @api_process | ||||
|     async def options_validate(self, request: web.Request) -> OptionsValidateResponse: | ||||
|         """Validate user options for add-on.""" | ||||
|         addon = self._extract_addon(request) | ||||
|         data = {ATTR_MESSAGE: "", ATTR_VALID: True, ATTR_PWNED: False} | ||||
|         addon = self.get_addon_for_request(request) | ||||
|         data = OptionsValidateResponse(message="", valid=True, pwned=False) | ||||
|  | ||||
|         options = await request.json(loads=json_loads) or addon.options | ||||
|  | ||||
| @@ -313,8 +358,8 @@ class APIAddons(CoreSysAttributes): | ||||
|         try: | ||||
|             options_schema.validate(options) | ||||
|         except vol.Invalid as ex: | ||||
|             data[ATTR_MESSAGE] = humanize_error(options, ex) | ||||
|             data[ATTR_VALID] = False | ||||
|             data["message"] = humanize_error(options, ex) | ||||
|             data["valid"] = False | ||||
|  | ||||
|         if not self.sys_security.pwned: | ||||
|             return data | ||||
| @@ -325,27 +370,27 @@ class APIAddons(CoreSysAttributes): | ||||
|                 await self.sys_security.verify_secret(secret) | ||||
|                 continue | ||||
|             except PwnedSecret: | ||||
|                 data[ATTR_PWNED] = True | ||||
|                 data["pwned"] = True | ||||
|             except PwnedError: | ||||
|                 data[ATTR_PWNED] = None | ||||
|                 data["pwned"] = None | ||||
|             break | ||||
|  | ||||
|         if self.sys_security.force and data[ATTR_PWNED] in (None, True): | ||||
|             data[ATTR_VALID] = False | ||||
|             if data[ATTR_PWNED] is None: | ||||
|                 data[ATTR_MESSAGE] = "Error happening on pwned secrets check!" | ||||
|         if self.sys_security.force and data["pwned"] in (None, True): | ||||
|             data["valid"] = False | ||||
|             if data["pwned"] is None: | ||||
|                 data["message"] = "Error happening on pwned secrets check!" | ||||
|             else: | ||||
|                 data[ATTR_MESSAGE] = "Add-on uses pwned secrets!" | ||||
|                 data["message"] = "Add-on uses pwned secrets!" | ||||
|  | ||||
|         return data | ||||
|  | ||||
|     @api_process | ||||
|     async def options_config(self, request: web.Request) -> None: | ||||
|         """Validate user options for add-on.""" | ||||
|         slug: str = request.match_info.get("addon") | ||||
|         slug: str = request.match_info["addon"] | ||||
|         if slug != "self": | ||||
|             raise APIForbidden("This can be only read by the Add-on itself!") | ||||
|         addon = self._extract_addon(request) | ||||
|         addon = self.get_addon_for_request(request) | ||||
|  | ||||
|         # Lookup/reload secrets | ||||
|         await self.sys_homeassistant.secrets.reload() | ||||
| @@ -357,19 +402,19 @@ class APIAddons(CoreSysAttributes): | ||||
|     @api_process | ||||
|     async def security(self, request: web.Request) -> None: | ||||
|         """Store security options for add-on.""" | ||||
|         addon = self._extract_addon(request) | ||||
|         addon = self.get_addon_for_request(request) | ||||
|         body: dict[str, Any] = await api_validate(SCHEMA_SECURITY, request) | ||||
|  | ||||
|         if ATTR_PROTECTED in body: | ||||
|             _LOGGER.warning("Changing protected flag for %s!", addon.slug) | ||||
|             addon.protected = body[ATTR_PROTECTED] | ||||
|  | ||||
|         addon.save_persist() | ||||
|         await addon.save_persist() | ||||
|  | ||||
|     @api_process | ||||
|     async def stats(self, request: web.Request) -> dict[str, Any]: | ||||
|         """Return resource information.""" | ||||
|         addon = self._extract_addon(request) | ||||
|         addon = self.get_addon_for_request(request) | ||||
|  | ||||
|         stats: DockerStats = await addon.stats() | ||||
|  | ||||
| @@ -385,48 +430,51 @@ class APIAddons(CoreSysAttributes): | ||||
|         } | ||||
|  | ||||
|     @api_process | ||||
|     def uninstall(self, request: web.Request) -> Awaitable[None]: | ||||
|     async def uninstall(self, request: web.Request) -> Awaitable[None]: | ||||
|         """Uninstall add-on.""" | ||||
|         addon = self._extract_addon(request) | ||||
|         return asyncio.shield(self.sys_addons.uninstall(addon.slug)) | ||||
|         addon = self.get_addon_for_request(request) | ||||
|         body: dict[str, Any] = await api_validate(SCHEMA_UNINSTALL, request) | ||||
|         return await asyncio.shield( | ||||
|             self.sys_addons.uninstall( | ||||
|                 addon.slug, remove_config=body[ATTR_REMOVE_CONFIG] | ||||
|             ) | ||||
|         ) | ||||
|  | ||||
|     @api_process | ||||
|     async def start(self, request: web.Request) -> None: | ||||
|         """Start add-on.""" | ||||
|         addon = self._extract_addon(request) | ||||
|         addon = self.get_addon_for_request(request) | ||||
|         if start_task := await asyncio.shield(addon.start()): | ||||
|             await start_task | ||||
|  | ||||
|     @api_process | ||||
|     def stop(self, request: web.Request) -> Awaitable[None]: | ||||
|         """Stop add-on.""" | ||||
|         addon = self._extract_addon(request) | ||||
|         addon = self.get_addon_for_request(request) | ||||
|         return asyncio.shield(addon.stop()) | ||||
|  | ||||
|     @api_process | ||||
|     async def restart(self, request: web.Request) -> None: | ||||
|         """Restart add-on.""" | ||||
|         addon: Addon = self._extract_addon(request) | ||||
|         addon: Addon = self.get_addon_for_request(request) | ||||
|         if start_task := await asyncio.shield(addon.restart()): | ||||
|             await start_task | ||||
|  | ||||
|     @api_process | ||||
|     async def rebuild(self, request: web.Request) -> None: | ||||
|         """Rebuild local build add-on.""" | ||||
|         addon = self._extract_addon(request) | ||||
|         if start_task := await asyncio.shield(self.sys_addons.rebuild(addon.slug)): | ||||
|             await start_task | ||||
|         addon = self.get_addon_for_request(request) | ||||
|         body: dict[str, Any] = await api_validate(SCHEMA_REBUILD, request) | ||||
|  | ||||
|     @api_process_raw(CONTENT_TYPE_BINARY) | ||||
|     def logs(self, request: web.Request) -> Awaitable[bytes]: | ||||
|         """Return logs from add-on.""" | ||||
|         addon = self._extract_addon(request) | ||||
|         return addon.logs() | ||||
|         if start_task := await asyncio.shield( | ||||
|             self.sys_addons.rebuild(addon.slug, force=body[ATTR_FORCE]) | ||||
|         ): | ||||
|             await start_task | ||||
|  | ||||
|     @api_process | ||||
|     async def stdin(self, request: web.Request) -> None: | ||||
|         """Write to stdin of add-on.""" | ||||
|         addon = self._extract_addon(request) | ||||
|         addon = self.get_addon_for_request(request) | ||||
|         if not addon.with_stdin: | ||||
|             raise APIError(f"STDIN not supported the {addon.slug} add-on") | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Init file for Supervisor Audio RESTful API.""" | ||||
|  | ||||
| import asyncio | ||||
| from collections.abc import Awaitable | ||||
| from dataclasses import asdict | ||||
| @@ -35,8 +36,7 @@ from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APIError | ||||
| from ..host.sound import StreamType | ||||
| from ..validate import version_tag | ||||
| from .const import CONTENT_TYPE_BINARY | ||||
| from .utils import api_process, api_process_raw, api_validate | ||||
| from .utils import api_process, api_validate | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
|  | ||||
| @@ -111,11 +111,6 @@ class APIAudio(CoreSysAttributes): | ||||
|             raise APIError(f"Version {version} is already in use") | ||||
|         await asyncio.shield(self.sys_plugins.audio.update(version)) | ||||
|  | ||||
|     @api_process_raw(CONTENT_TYPE_BINARY) | ||||
|     def logs(self, request: web.Request) -> Awaitable[bytes]: | ||||
|         """Return Audio Docker logs.""" | ||||
|         return self.sys_plugins.audio.logs() | ||||
|  | ||||
|     @api_process | ||||
|     def restart(self, request: web.Request) -> Awaitable[None]: | ||||
|         """Restart Audio plugin.""" | ||||
| @@ -129,7 +124,7 @@ class APIAudio(CoreSysAttributes): | ||||
|     @api_process | ||||
|     async def set_volume(self, request: web.Request) -> None: | ||||
|         """Set audio volume on stream.""" | ||||
|         source: StreamType = StreamType(request.match_info.get("source")) | ||||
|         source: StreamType = StreamType(request.match_info["source"]) | ||||
|         application: bool = request.path.endswith("application") | ||||
|         body = await api_validate(SCHEMA_VOLUME, request) | ||||
|  | ||||
| @@ -142,7 +137,7 @@ class APIAudio(CoreSysAttributes): | ||||
|     @api_process | ||||
|     async def set_mute(self, request: web.Request) -> None: | ||||
|         """Mute audio volume on stream.""" | ||||
|         source: StreamType = StreamType(request.match_info.get("source")) | ||||
|         source: StreamType = StreamType(request.match_info["source"]) | ||||
|         application: bool = request.path.endswith("application") | ||||
|         body = await api_validate(SCHEMA_MUTE, request) | ||||
|  | ||||
| @@ -155,7 +150,7 @@ class APIAudio(CoreSysAttributes): | ||||
|     @api_process | ||||
|     async def set_default(self, request: web.Request) -> None: | ||||
|         """Set audio default stream.""" | ||||
|         source: StreamType = StreamType(request.match_info.get("source")) | ||||
|         source: StreamType = StreamType(request.match_info["source"]) | ||||
|         body = await api_validate(SCHEMA_DEFAULT, request) | ||||
|  | ||||
|         await asyncio.shield(self.sys_host.sound.set_default(source, body[ATTR_NAME])) | ||||
|   | ||||
| @@ -1,19 +1,31 @@ | ||||
| """Init file for Supervisor auth/SSO RESTful API.""" | ||||
|  | ||||
| import asyncio | ||||
| from collections.abc import Awaitable | ||||
| import logging | ||||
| from typing import Any, cast | ||||
|  | ||||
| from aiohttp import BasicAuth, web | ||||
| from aiohttp.hdrs import AUTHORIZATION, CONTENT_TYPE, WWW_AUTHENTICATE | ||||
| from aiohttp.web import FileField | ||||
| from aiohttp.web_exceptions import HTTPUnauthorized | ||||
| from multidict import MultiDictProxy | ||||
| import voluptuous as vol | ||||
|  | ||||
| from ..addons.addon import Addon | ||||
| from ..const import ATTR_PASSWORD, ATTR_USERNAME, REQUEST_FROM | ||||
| from ..const import ATTR_NAME, ATTR_PASSWORD, ATTR_USERNAME, REQUEST_FROM | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APIForbidden | ||||
| from ..utils.json import json_loads | ||||
| from .const import CONTENT_TYPE_JSON, CONTENT_TYPE_URL | ||||
| from .utils import api_process, api_validate | ||||
| from .const import ( | ||||
|     ATTR_GROUP_IDS, | ||||
|     ATTR_IS_ACTIVE, | ||||
|     ATTR_IS_OWNER, | ||||
|     ATTR_LOCAL_ONLY, | ||||
|     ATTR_USERS, | ||||
|     CONTENT_TYPE_JSON, | ||||
|     CONTENT_TYPE_URL, | ||||
| ) | ||||
| from .utils import api_process, api_validate, json_loads | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
|  | ||||
| @@ -32,7 +44,7 @@ REALM_HEADER: dict[str, str] = { | ||||
| class APIAuth(CoreSysAttributes): | ||||
|     """Handle RESTful API for auth functions.""" | ||||
|  | ||||
|     def _process_basic(self, request: web.Request, addon: Addon) -> bool: | ||||
|     def _process_basic(self, request: web.Request, addon: Addon) -> Awaitable[bool]: | ||||
|         """Process login request with basic auth. | ||||
|  | ||||
|         Return a coroutine. | ||||
| @@ -41,8 +53,11 @@ class APIAuth(CoreSysAttributes): | ||||
|         return self.sys_auth.check_login(addon, auth.login, auth.password) | ||||
|  | ||||
|     def _process_dict( | ||||
|         self, request: web.Request, addon: Addon, data: dict[str, str] | ||||
|     ) -> bool: | ||||
|         self, | ||||
|         request: web.Request, | ||||
|         addon: Addon, | ||||
|         data: dict[str, Any] | MultiDictProxy[str | bytes | FileField], | ||||
|     ) -> Awaitable[bool]: | ||||
|         """Process login with dict data. | ||||
|  | ||||
|         Return a coroutine. | ||||
| @@ -50,14 +65,22 @@ class APIAuth(CoreSysAttributes): | ||||
|         username = data.get("username") or data.get("user") | ||||
|         password = data.get("password") | ||||
|  | ||||
|         return self.sys_auth.check_login(addon, username, password) | ||||
|         # Test that we did receive strings and not something else, raise if so | ||||
|         try: | ||||
|             _ = username.encode and password.encode  # type: ignore | ||||
|         except AttributeError: | ||||
|             raise HTTPUnauthorized(headers=REALM_HEADER) from None | ||||
|  | ||||
|         return self.sys_auth.check_login( | ||||
|             addon, cast(str, username), cast(str, password) | ||||
|         ) | ||||
|  | ||||
|     @api_process | ||||
|     async def auth(self, request: web.Request) -> bool: | ||||
|         """Process login request.""" | ||||
|         addon = request[REQUEST_FROM] | ||||
|  | ||||
|         if not addon.access_auth_api: | ||||
|         if not isinstance(addon, Addon) or not addon.access_auth_api: | ||||
|             raise APIForbidden("Can't use Home Assistant auth!") | ||||
|  | ||||
|         # BasicAuth | ||||
| @@ -69,13 +92,18 @@ class APIAuth(CoreSysAttributes): | ||||
|         # Json | ||||
|         if request.headers.get(CONTENT_TYPE) == CONTENT_TYPE_JSON: | ||||
|             data = await request.json(loads=json_loads) | ||||
|             return await self._process_dict(request, addon, data) | ||||
|             if not await self._process_dict(request, addon, data): | ||||
|                 raise HTTPUnauthorized() | ||||
|             return True | ||||
|  | ||||
|         # URL encoded | ||||
|         if request.headers.get(CONTENT_TYPE) == CONTENT_TYPE_URL: | ||||
|             data = await request.post() | ||||
|             return await self._process_dict(request, addon, data) | ||||
|             if not await self._process_dict(request, addon, data): | ||||
|                 raise HTTPUnauthorized() | ||||
|             return True | ||||
|  | ||||
|         # Advertise Basic authentication by default | ||||
|         raise HTTPUnauthorized(headers=REALM_HEADER) | ||||
|  | ||||
|     @api_process | ||||
| @@ -89,4 +117,22 @@ class APIAuth(CoreSysAttributes): | ||||
|     @api_process | ||||
|     async def cache(self, request: web.Request) -> None: | ||||
|         """Process cache reset request.""" | ||||
|         self.sys_auth.reset_data() | ||||
|         await self.sys_auth.reset_data() | ||||
|  | ||||
|     @api_process | ||||
|     async def list_users(self, request: web.Request) -> dict[str, list[dict[str, Any]]]: | ||||
|         """List users on the Home Assistant instance.""" | ||||
|         return { | ||||
|             ATTR_USERS: [ | ||||
|                 { | ||||
|                     ATTR_USERNAME: user[ATTR_USERNAME], | ||||
|                     ATTR_NAME: user[ATTR_NAME], | ||||
|                     ATTR_IS_OWNER: user[ATTR_IS_OWNER], | ||||
|                     ATTR_IS_ACTIVE: user[ATTR_IS_ACTIVE], | ||||
|                     ATTR_LOCAL_ONLY: user[ATTR_LOCAL_ONLY], | ||||
|                     ATTR_GROUP_IDS: user[ATTR_GROUP_IDS], | ||||
|                 } | ||||
|                 for user in await self.sys_auth.list_users() | ||||
|                 if user[ATTR_USERNAME] | ||||
|             ] | ||||
|         } | ||||
|   | ||||
| @@ -1,16 +1,23 @@ | ||||
| """Backups RESTful API.""" | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| import asyncio | ||||
| import errno | ||||
| from io import IOBase | ||||
| import logging | ||||
| from pathlib import Path | ||||
| import re | ||||
| from tempfile import TemporaryDirectory | ||||
| from typing import Any | ||||
| from typing import Any, cast | ||||
|  | ||||
| from aiohttp import web | ||||
| from aiohttp import BodyPartReader, web | ||||
| from aiohttp.hdrs import CONTENT_DISPOSITION | ||||
| import voluptuous as vol | ||||
| from voluptuous.humanize import humanize_error | ||||
|  | ||||
| from ..backups.backup import Backup | ||||
| from ..backups.const import LOCATION_CLOUD_BACKUP, LOCATION_TYPE | ||||
| from ..backups.validate import ALL_FOLDERS, FOLDER_HOMEASSISTANT, days_until_stale | ||||
| from ..const import ( | ||||
|     ATTR_ADDONS, | ||||
| @@ -19,77 +26,114 @@ from ..const import ( | ||||
|     ATTR_CONTENT, | ||||
|     ATTR_DATE, | ||||
|     ATTR_DAYS_UNTIL_STALE, | ||||
|     ATTR_EXTRA, | ||||
|     ATTR_FILENAME, | ||||
|     ATTR_FOLDERS, | ||||
|     ATTR_HOMEASSISTANT, | ||||
|     ATTR_HOMEASSISTANT_EXCLUDE_DATABASE, | ||||
|     ATTR_LOCATON, | ||||
|     ATTR_JOB_ID, | ||||
|     ATTR_LOCATION, | ||||
|     ATTR_NAME, | ||||
|     ATTR_PASSWORD, | ||||
|     ATTR_PROTECTED, | ||||
|     ATTR_REPOSITORIES, | ||||
|     ATTR_SIZE, | ||||
|     ATTR_SIZE_BYTES, | ||||
|     ATTR_SLUG, | ||||
|     ATTR_SUPERVISOR_VERSION, | ||||
|     ATTR_TIMEOUT, | ||||
|     ATTR_TYPE, | ||||
|     ATTR_VERSION, | ||||
|     REQUEST_FROM, | ||||
| ) | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APIError | ||||
| from ..exceptions import APIError, APIForbidden, APINotFound | ||||
| from ..mounts.const import MountUsage | ||||
| from ..resolution.const import UnhealthyReason | ||||
| from .const import CONTENT_TYPE_TAR | ||||
| from .utils import api_process, api_validate | ||||
| from .const import ( | ||||
|     ATTR_ADDITIONAL_LOCATIONS, | ||||
|     ATTR_BACKGROUND, | ||||
|     ATTR_LOCATION_ATTRIBUTES, | ||||
|     ATTR_LOCATIONS, | ||||
|     CONTENT_TYPE_TAR, | ||||
| ) | ||||
| from .utils import api_process, api_validate, background_task | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
|  | ||||
| ALL_ADDONS_FLAG = "ALL" | ||||
|  | ||||
| LOCATION_LOCAL = ".local" | ||||
|  | ||||
| RE_SLUGIFY_NAME = re.compile(r"[^A-Za-z0-9]+") | ||||
| RE_BACKUP_FILENAME = re.compile(r"^[^\\\/]+\.tar$") | ||||
|  | ||||
| # Backwards compatible | ||||
| # Remove: 2022.08 | ||||
| _ALL_FOLDERS = ALL_FOLDERS + [FOLDER_HOMEASSISTANT] | ||||
|  | ||||
|  | ||||
| def _ensure_list(item: Any) -> list: | ||||
|     """Ensure value is a list.""" | ||||
|     if not isinstance(item, list): | ||||
|         return [item] | ||||
|     return item | ||||
|  | ||||
|  | ||||
| def _convert_local_location(item: str | None) -> str | None: | ||||
|     """Convert local location value.""" | ||||
|     if item in {LOCATION_LOCAL, ""}: | ||||
|         return None | ||||
|     return item | ||||
|  | ||||
|  | ||||
| # pylint: disable=no-value-for-parameter | ||||
| SCHEMA_RESTORE_PARTIAL = vol.Schema( | ||||
| SCHEMA_FOLDERS = vol.All([vol.In(_ALL_FOLDERS)], vol.Unique()) | ||||
| SCHEMA_LOCATION = vol.All(vol.Maybe(str), _convert_local_location) | ||||
| SCHEMA_LOCATION_LIST = vol.All(_ensure_list, [SCHEMA_LOCATION], vol.Unique()) | ||||
|  | ||||
| SCHEMA_RESTORE_FULL = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_PASSWORD): vol.Maybe(str), | ||||
|         vol.Optional(ATTR_HOMEASSISTANT): vol.Boolean(), | ||||
|         vol.Optional(ATTR_ADDONS): vol.All([str], vol.Unique()), | ||||
|         vol.Optional(ATTR_FOLDERS): vol.All([vol.In(_ALL_FOLDERS)], vol.Unique()), | ||||
|         vol.Optional(ATTR_BACKGROUND, default=False): vol.Boolean(), | ||||
|         vol.Optional(ATTR_LOCATION): SCHEMA_LOCATION, | ||||
|     } | ||||
| ) | ||||
|  | ||||
| SCHEMA_RESTORE_FULL = vol.Schema({vol.Optional(ATTR_PASSWORD): vol.Maybe(str)}) | ||||
| SCHEMA_RESTORE_PARTIAL = SCHEMA_RESTORE_FULL.extend( | ||||
|     { | ||||
|         vol.Optional(ATTR_HOMEASSISTANT): vol.Boolean(), | ||||
|         vol.Optional(ATTR_ADDONS): vol.All([str], vol.Unique()), | ||||
|         vol.Optional(ATTR_FOLDERS): SCHEMA_FOLDERS, | ||||
|     } | ||||
| ) | ||||
|  | ||||
| SCHEMA_BACKUP_FULL = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_NAME): str, | ||||
|         vol.Optional(ATTR_FILENAME): vol.Match(RE_BACKUP_FILENAME), | ||||
|         vol.Optional(ATTR_PASSWORD): vol.Maybe(str), | ||||
|         vol.Optional(ATTR_COMPRESSED): vol.Maybe(vol.Boolean()), | ||||
|         vol.Optional(ATTR_LOCATON): vol.Maybe(str), | ||||
|         vol.Optional(ATTR_LOCATION): SCHEMA_LOCATION_LIST, | ||||
|         vol.Optional(ATTR_HOMEASSISTANT_EXCLUDE_DATABASE): vol.Boolean(), | ||||
|         vol.Optional(ATTR_BACKGROUND, default=False): vol.Boolean(), | ||||
|         vol.Optional(ATTR_EXTRA): dict, | ||||
|     } | ||||
| ) | ||||
|  | ||||
| SCHEMA_BACKUP_PARTIAL = SCHEMA_BACKUP_FULL.extend( | ||||
|     { | ||||
|         vol.Optional(ATTR_ADDONS): vol.All([str], vol.Unique()), | ||||
|         vol.Optional(ATTR_FOLDERS): vol.All([vol.In(_ALL_FOLDERS)], vol.Unique()), | ||||
|         vol.Optional(ATTR_ADDONS): vol.Or( | ||||
|             ALL_ADDONS_FLAG, vol.All([str], vol.Unique()) | ||||
|         ), | ||||
|         vol.Optional(ATTR_FOLDERS): SCHEMA_FOLDERS, | ||||
|         vol.Optional(ATTR_HOMEASSISTANT): vol.Boolean(), | ||||
|     } | ||||
| ) | ||||
|  | ||||
| SCHEMA_OPTIONS = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_DAYS_UNTIL_STALE): days_until_stale, | ||||
|     } | ||||
| ) | ||||
|  | ||||
| SCHEMA_FREEZE = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_TIMEOUT): vol.All(int, vol.Range(min=1)), | ||||
|     } | ||||
| ) | ||||
| SCHEMA_OPTIONS = vol.Schema({vol.Optional(ATTR_DAYS_UNTIL_STALE): days_until_stale}) | ||||
| SCHEMA_FREEZE = vol.Schema({vol.Optional(ATTR_TIMEOUT): vol.All(int, vol.Range(min=1))}) | ||||
| SCHEMA_REMOVE = vol.Schema({vol.Optional(ATTR_LOCATION): SCHEMA_LOCATION_LIST}) | ||||
|  | ||||
|  | ||||
| class APIBackups(CoreSysAttributes): | ||||
| @@ -99,9 +143,19 @@ class APIBackups(CoreSysAttributes): | ||||
|         """Return backup, throw an exception if it doesn't exist.""" | ||||
|         backup = self.sys_backups.get(request.match_info.get("slug")) | ||||
|         if not backup: | ||||
|             raise APIError("Backup does not exist") | ||||
|             raise APINotFound("Backup does not exist") | ||||
|         return backup | ||||
|  | ||||
|     def _make_location_attributes(self, backup: Backup) -> dict[str, dict[str, Any]]: | ||||
|         """Make location attributes dictionary.""" | ||||
|         return { | ||||
|             loc if loc else LOCATION_LOCAL: { | ||||
|                 ATTR_PROTECTED: backup.all_locations[loc].protected, | ||||
|                 ATTR_SIZE_BYTES: backup.all_locations[loc].size_bytes, | ||||
|             } | ||||
|             for loc in backup.locations | ||||
|         } | ||||
|  | ||||
|     def _list_backups(self): | ||||
|         """Return list of backups.""" | ||||
|         return [ | ||||
| @@ -111,8 +165,11 @@ class APIBackups(CoreSysAttributes): | ||||
|                 ATTR_DATE: backup.date, | ||||
|                 ATTR_TYPE: backup.sys_type, | ||||
|                 ATTR_SIZE: backup.size, | ||||
|                 ATTR_LOCATON: backup.location, | ||||
|                 ATTR_SIZE_BYTES: backup.size_bytes, | ||||
|                 ATTR_LOCATION: backup.location, | ||||
|                 ATTR_LOCATIONS: backup.locations, | ||||
|                 ATTR_PROTECTED: backup.protected, | ||||
|                 ATTR_LOCATION_ATTRIBUTES: self._make_location_attributes(backup), | ||||
|                 ATTR_COMPRESSED: backup.compressed, | ||||
|                 ATTR_CONTENT: { | ||||
|                     ATTR_HOMEASSISTANT: backup.homeassistant_version is not None, | ||||
| @@ -121,10 +178,11 @@ class APIBackups(CoreSysAttributes): | ||||
|                 }, | ||||
|             } | ||||
|             for backup in self.sys_backups.list_backups | ||||
|             if backup.location != LOCATION_CLOUD_BACKUP | ||||
|         ] | ||||
|  | ||||
|     @api_process | ||||
|     async def list(self, request): | ||||
|     async def list_backups(self, request): | ||||
|         """Return backup list.""" | ||||
|         data_backups = self._list_backups() | ||||
|  | ||||
| @@ -150,7 +208,7 @@ class APIBackups(CoreSysAttributes): | ||||
|         if ATTR_DAYS_UNTIL_STALE in body: | ||||
|             self.sys_backups.days_until_stale = body[ATTR_DAYS_UNTIL_STALE] | ||||
|  | ||||
|         self.sys_backups.save_data() | ||||
|         await self.sys_backups.save_data() | ||||
|  | ||||
|     @api_process | ||||
|     async def reload(self, _): | ||||
| @@ -180,125 +238,302 @@ class APIBackups(CoreSysAttributes): | ||||
|             ATTR_NAME: backup.name, | ||||
|             ATTR_DATE: backup.date, | ||||
|             ATTR_SIZE: backup.size, | ||||
|             ATTR_SIZE_BYTES: backup.size_bytes, | ||||
|             ATTR_COMPRESSED: backup.compressed, | ||||
|             ATTR_PROTECTED: backup.protected, | ||||
|             ATTR_LOCATION_ATTRIBUTES: self._make_location_attributes(backup), | ||||
|             ATTR_SUPERVISOR_VERSION: backup.supervisor_version, | ||||
|             ATTR_HOMEASSISTANT: backup.homeassistant_version, | ||||
|             ATTR_LOCATON: backup.location, | ||||
|             ATTR_LOCATION: backup.location, | ||||
|             ATTR_LOCATIONS: backup.locations, | ||||
|             ATTR_ADDONS: data_addons, | ||||
|             ATTR_REPOSITORIES: backup.repositories, | ||||
|             ATTR_FOLDERS: backup.folders, | ||||
|             ATTR_HOMEASSISTANT_EXCLUDE_DATABASE: backup.homeassistant_exclude_database, | ||||
|             ATTR_EXTRA: backup.extra, | ||||
|         } | ||||
|  | ||||
|     def _location_to_mount(self, body: dict[str, Any]) -> dict[str, Any]: | ||||
|         """Change location field to mount if necessary.""" | ||||
|         if not body.get(ATTR_LOCATON): | ||||
|             return body | ||||
|     def _location_to_mount(self, location: str | None) -> LOCATION_TYPE: | ||||
|         """Convert a single location to a mount if possible.""" | ||||
|         if not location or location == LOCATION_CLOUD_BACKUP: | ||||
|             return cast(LOCATION_TYPE, location) | ||||
|  | ||||
|         body[ATTR_LOCATON] = self.sys_mounts.get(body[ATTR_LOCATON]) | ||||
|         if body[ATTR_LOCATON].usage != MountUsage.BACKUP: | ||||
|         mount = self.sys_mounts.get(location) | ||||
|         if mount.usage != MountUsage.BACKUP: | ||||
|             raise APIError( | ||||
|                 f"Mount {body[ATTR_LOCATON].name} is not used for backups, cannot backup to there" | ||||
|                 f"Mount {mount.name} is not used for backups, cannot backup to there" | ||||
|             ) | ||||
|  | ||||
|         return mount | ||||
|  | ||||
|     def _location_field_to_mount(self, body: dict[str, Any]) -> dict[str, Any]: | ||||
|         """Change location field to mount if necessary.""" | ||||
|         body[ATTR_LOCATION] = self._location_to_mount(body.get(ATTR_LOCATION)) | ||||
|         return body | ||||
|  | ||||
|     def _validate_cloud_backup_location( | ||||
|         self, request: web.Request, location: list[str | None] | str | None | ||||
|     ) -> None: | ||||
|         """Cloud backup location is only available to Home Assistant.""" | ||||
|         if not isinstance(location, list): | ||||
|             location = [location] | ||||
|         if ( | ||||
|             LOCATION_CLOUD_BACKUP in location | ||||
|             and request.get(REQUEST_FROM) != self.sys_homeassistant | ||||
|         ): | ||||
|             raise APIForbidden( | ||||
|                 f"Location {LOCATION_CLOUD_BACKUP} is only available for Home Assistant" | ||||
|             ) | ||||
|  | ||||
|     @api_process | ||||
|     async def backup_full(self, request): | ||||
|     async def backup_full(self, request: web.Request): | ||||
|         """Create full backup.""" | ||||
|         body = await api_validate(SCHEMA_BACKUP_FULL, request) | ||||
|         locations: list[LOCATION_TYPE] | None = None | ||||
|  | ||||
|         backup = await asyncio.shield( | ||||
|             self.sys_backups.do_backup_full(**self._location_to_mount(body)) | ||||
|         if ATTR_LOCATION in body: | ||||
|             location_names: list[str | None] = body.pop(ATTR_LOCATION) | ||||
|             self._validate_cloud_backup_location(request, location_names) | ||||
|  | ||||
|             locations = [ | ||||
|                 self._location_to_mount(location) for location in location_names | ||||
|             ] | ||||
|             body[ATTR_LOCATION] = locations.pop(0) | ||||
|             if locations: | ||||
|                 body[ATTR_ADDITIONAL_LOCATIONS] = locations | ||||
|  | ||||
|         background = body.pop(ATTR_BACKGROUND) | ||||
|         backup_task, job_id = await background_task( | ||||
|             self, self.sys_backups.do_backup_full, **body | ||||
|         ) | ||||
|  | ||||
|         if background and not backup_task.done(): | ||||
|             return {ATTR_JOB_ID: job_id} | ||||
|  | ||||
|         backup: Backup = await backup_task | ||||
|         if backup: | ||||
|             return {ATTR_SLUG: backup.slug} | ||||
|         return False | ||||
|             return {ATTR_JOB_ID: job_id, ATTR_SLUG: backup.slug} | ||||
|         raise APIError( | ||||
|             f"An error occurred while making backup, check job '{job_id}' or supervisor logs for details", | ||||
|             job_id=job_id, | ||||
|         ) | ||||
|  | ||||
|     @api_process | ||||
|     async def backup_partial(self, request): | ||||
|     async def backup_partial(self, request: web.Request): | ||||
|         """Create a partial backup.""" | ||||
|         body = await api_validate(SCHEMA_BACKUP_PARTIAL, request) | ||||
|         backup = await asyncio.shield( | ||||
|             self.sys_backups.do_backup_partial(**self._location_to_mount(body)) | ||||
|         locations: list[LOCATION_TYPE] | None = None | ||||
|  | ||||
|         if ATTR_LOCATION in body: | ||||
|             location_names: list[str | None] = body.pop(ATTR_LOCATION) | ||||
|             self._validate_cloud_backup_location(request, location_names) | ||||
|  | ||||
|             locations = [ | ||||
|                 self._location_to_mount(location) for location in location_names | ||||
|             ] | ||||
|             body[ATTR_LOCATION] = locations.pop(0) | ||||
|             if locations: | ||||
|                 body[ATTR_ADDITIONAL_LOCATIONS] = locations | ||||
|  | ||||
|         if body.get(ATTR_ADDONS) == ALL_ADDONS_FLAG: | ||||
|             body[ATTR_ADDONS] = list(self.sys_addons.local) | ||||
|  | ||||
|         background = body.pop(ATTR_BACKGROUND) | ||||
|         backup_task, job_id = await background_task( | ||||
|             self, self.sys_backups.do_backup_partial, **body | ||||
|         ) | ||||
|  | ||||
|         if background and not backup_task.done(): | ||||
|             return {ATTR_JOB_ID: job_id} | ||||
|  | ||||
|         backup: Backup = await backup_task | ||||
|         if backup: | ||||
|             return {ATTR_SLUG: backup.slug} | ||||
|         return False | ||||
|             return {ATTR_JOB_ID: job_id, ATTR_SLUG: backup.slug} | ||||
|         raise APIError( | ||||
|             f"An error occurred while making backup, check job '{job_id}' or supervisor logs for details", | ||||
|             job_id=job_id, | ||||
|         ) | ||||
|  | ||||
|     @api_process | ||||
|     async def restore_full(self, request): | ||||
|     async def restore_full(self, request: web.Request): | ||||
|         """Full restore of a backup.""" | ||||
|         backup = self._extract_slug(request) | ||||
|         body = await api_validate(SCHEMA_RESTORE_FULL, request) | ||||
|         self._validate_cloud_backup_location( | ||||
|             request, body.get(ATTR_LOCATION, backup.location) | ||||
|         ) | ||||
|         background = body.pop(ATTR_BACKGROUND) | ||||
|         restore_task, job_id = await background_task( | ||||
|             self, self.sys_backups.do_restore_full, backup, **body | ||||
|         ) | ||||
|  | ||||
|         return await asyncio.shield(self.sys_backups.do_restore_full(backup, **body)) | ||||
|         if background and not restore_task.done() or await restore_task: | ||||
|             return {ATTR_JOB_ID: job_id} | ||||
|         raise APIError( | ||||
|             f"An error occurred during restore of {backup.slug}, check job '{job_id}' or supervisor logs for details", | ||||
|             job_id=job_id, | ||||
|         ) | ||||
|  | ||||
|     @api_process | ||||
|     async def restore_partial(self, request): | ||||
|     async def restore_partial(self, request: web.Request): | ||||
|         """Partial restore a backup.""" | ||||
|         backup = self._extract_slug(request) | ||||
|         body = await api_validate(SCHEMA_RESTORE_PARTIAL, request) | ||||
|         self._validate_cloud_backup_location( | ||||
|             request, body.get(ATTR_LOCATION, backup.location) | ||||
|         ) | ||||
|         background = body.pop(ATTR_BACKGROUND) | ||||
|         restore_task, job_id = await background_task( | ||||
|             self, self.sys_backups.do_restore_partial, backup, **body | ||||
|         ) | ||||
|  | ||||
|         return await asyncio.shield(self.sys_backups.do_restore_partial(backup, **body)) | ||||
|         if background and not restore_task.done() or await restore_task: | ||||
|             return {ATTR_JOB_ID: job_id} | ||||
|         raise APIError( | ||||
|             f"An error occurred during restore of {backup.slug}, check job '{job_id}' or supervisor logs for details", | ||||
|             job_id=job_id, | ||||
|         ) | ||||
|  | ||||
|     @api_process | ||||
|     async def freeze(self, request): | ||||
|     async def freeze(self, request: web.Request): | ||||
|         """Initiate manual freeze for external backup.""" | ||||
|         body = await api_validate(SCHEMA_FREEZE, request) | ||||
|         await asyncio.shield(self.sys_backups.freeze_all(**body)) | ||||
|  | ||||
|     @api_process | ||||
|     async def thaw(self, request): | ||||
|     async def thaw(self, request: web.Request): | ||||
|         """Begin thaw after manual freeze.""" | ||||
|         await self.sys_backups.thaw_all() | ||||
|  | ||||
|     @api_process | ||||
|     async def remove(self, request): | ||||
|     async def remove(self, request: web.Request): | ||||
|         """Remove a backup.""" | ||||
|         backup = self._extract_slug(request) | ||||
|         return self.sys_backups.remove(backup) | ||||
|         body = await api_validate(SCHEMA_REMOVE, request) | ||||
|         locations: list[LOCATION_TYPE] | None = None | ||||
|  | ||||
|     async def download(self, request): | ||||
|         if ATTR_LOCATION in body: | ||||
|             self._validate_cloud_backup_location(request, body[ATTR_LOCATION]) | ||||
|             locations = [self._location_to_mount(name) for name in body[ATTR_LOCATION]] | ||||
|         else: | ||||
|             self._validate_cloud_backup_location(request, backup.location) | ||||
|  | ||||
|         await self.sys_backups.remove(backup, locations=locations) | ||||
|  | ||||
|     @api_process | ||||
|     async def download(self, request: web.Request): | ||||
|         """Download a backup file.""" | ||||
|         backup = self._extract_slug(request) | ||||
|         # Query will give us '' for /backups, convert value to None | ||||
|         location = _convert_local_location( | ||||
|             request.query.get(ATTR_LOCATION, backup.location) | ||||
|         ) | ||||
|         self._validate_cloud_backup_location(request, location) | ||||
|         if location not in backup.all_locations: | ||||
|             raise APIError(f"Backup {backup.slug} is not in location {location}") | ||||
|  | ||||
|         _LOGGER.info("Downloading backup %s", backup.slug) | ||||
|         response = web.FileResponse(backup.tarfile) | ||||
|         filename = backup.all_locations[location].path | ||||
|         # If the file is missing, return 404 and trigger reload of location | ||||
|         if not await self.sys_run_in_executor(filename.is_file): | ||||
|             self.sys_create_task(self.sys_backups.reload(location)) | ||||
|             return web.Response(status=404) | ||||
|  | ||||
|         response = web.FileResponse(filename) | ||||
|         response.content_type = CONTENT_TYPE_TAR | ||||
|         response.headers[ | ||||
|             CONTENT_DISPOSITION | ||||
|         ] = f"attachment; filename={RE_SLUGIFY_NAME.sub('_', backup.name)}.tar" | ||||
|  | ||||
|         download_filename = filename.name | ||||
|         if download_filename == f"{backup.slug}.tar": | ||||
|             download_filename = f"{RE_SLUGIFY_NAME.sub('_', backup.name)}.tar" | ||||
|         response.headers[CONTENT_DISPOSITION] = ( | ||||
|             f"attachment; filename={download_filename}" | ||||
|         ) | ||||
|         return response | ||||
|  | ||||
|     @api_process | ||||
|     async def upload(self, request): | ||||
|     async def upload(self, request: web.Request): | ||||
|         """Upload a backup file.""" | ||||
|         with TemporaryDirectory(dir=str(self.sys_config.path_tmp)) as temp_dir: | ||||
|             tar_file = Path(temp_dir, "backup.tar") | ||||
|         location: LOCATION_TYPE = None | ||||
|         locations: list[LOCATION_TYPE] | None = None | ||||
|  | ||||
|         if ATTR_LOCATION in request.query: | ||||
|             location_names: list[str] = request.query.getall(ATTR_LOCATION, []) | ||||
|             self._validate_cloud_backup_location( | ||||
|                 request, cast(list[str | None], location_names) | ||||
|             ) | ||||
|             # Convert empty string to None if necessary | ||||
|             locations = [ | ||||
|                 self._location_to_mount(location) | ||||
|                 if _convert_local_location(location) | ||||
|                 else None | ||||
|                 for location in location_names | ||||
|             ] | ||||
|             location = locations.pop(0) | ||||
|  | ||||
|         filename: str | None = None | ||||
|         if ATTR_FILENAME in request.query: | ||||
|             filename = request.query.get(ATTR_FILENAME) | ||||
|             try: | ||||
|                 vol.Match(RE_BACKUP_FILENAME)(filename) | ||||
|             except vol.Invalid as ex: | ||||
|                 raise APIError(humanize_error(filename, ex)) from None | ||||
|  | ||||
|         tmp_path = await self.sys_backups.get_upload_path_for_location(location) | ||||
|         temp_dir: TemporaryDirectory | None = None | ||||
|         backup_file_stream: IOBase | None = None | ||||
|  | ||||
|         def open_backup_file() -> Path: | ||||
|             nonlocal temp_dir, backup_file_stream | ||||
|             temp_dir = TemporaryDirectory(dir=tmp_path.as_posix()) | ||||
|             tar_file = Path(temp_dir.name, "upload.tar") | ||||
|             backup_file_stream = tar_file.open("wb") | ||||
|             return tar_file | ||||
|  | ||||
|         def close_backup_file() -> None: | ||||
|             if backup_file_stream: | ||||
|                 # Make sure it got closed, in case of exception. It is safe to | ||||
|                 # close the file stream twice. | ||||
|                 backup_file_stream.close() | ||||
|             if temp_dir: | ||||
|                 temp_dir.cleanup() | ||||
|  | ||||
|         try: | ||||
|             reader = await request.multipart() | ||||
|             contents = await reader.next() | ||||
|             try: | ||||
|                 with tar_file.open("wb") as backup: | ||||
|                     while True: | ||||
|                         chunk = await contents.read_chunk() | ||||
|                         if not chunk: | ||||
|                             break | ||||
|                         backup.write(chunk) | ||||
|             if not isinstance(contents, BodyPartReader): | ||||
|                 raise APIError("Improperly formatted upload, could not read backup") | ||||
|  | ||||
|             except OSError as err: | ||||
|                 if err.errno == errno.EBADMSG: | ||||
|                     self.sys_resolution.unhealthy = UnhealthyReason.OSERROR_BAD_MESSAGE | ||||
|                 _LOGGER.error("Can't write new backup file: %s", err) | ||||
|                 return False | ||||
|             tar_file = await self.sys_run_in_executor(open_backup_file) | ||||
|             while chunk := await contents.read_chunk(size=2**16): | ||||
|                 await self.sys_run_in_executor( | ||||
|                     cast(IOBase, backup_file_stream).write, chunk | ||||
|                 ) | ||||
|             await self.sys_run_in_executor(cast(IOBase, backup_file_stream).close) | ||||
|  | ||||
|             except asyncio.CancelledError: | ||||
|                 return False | ||||
|             backup = await asyncio.shield( | ||||
|                 self.sys_backups.import_backup( | ||||
|                     tar_file, | ||||
|                     filename, | ||||
|                     location=location, | ||||
|                     additional_locations=locations, | ||||
|                 ) | ||||
|             ) | ||||
|         except OSError as err: | ||||
|             if err.errno == errno.EBADMSG and location in { | ||||
|                 LOCATION_CLOUD_BACKUP, | ||||
|                 None, | ||||
|             }: | ||||
|                 self.sys_resolution.add_unhealthy_reason( | ||||
|                     UnhealthyReason.OSERROR_BAD_MESSAGE | ||||
|                 ) | ||||
|             _LOGGER.error("Can't write new backup file: %s", err) | ||||
|             return False | ||||
|  | ||||
|             backup = await asyncio.shield(self.sys_backups.import_backup(tar_file)) | ||||
|         except asyncio.CancelledError: | ||||
|             return False | ||||
|  | ||||
|         finally: | ||||
|             await self.sys_run_in_executor(close_backup_file) | ||||
|  | ||||
|         if backup: | ||||
|             return {ATTR_SLUG: backup.slug} | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Init file for Supervisor HA cli RESTful API.""" | ||||
|  | ||||
| import asyncio | ||||
| import logging | ||||
| from typing import Any | ||||
|   | ||||
| @@ -1,18 +1,26 @@ | ||||
| """Const for API.""" | ||||
|  | ||||
| from enum import StrEnum | ||||
|  | ||||
| CONTENT_TYPE_BINARY = "application/octet-stream" | ||||
| CONTENT_TYPE_JSON = "application/json" | ||||
| CONTENT_TYPE_PNG = "image/png" | ||||
| CONTENT_TYPE_TAR = "application/tar" | ||||
| CONTENT_TYPE_TEXT = "text/plain" | ||||
| CONTENT_TYPE_URL = "application/x-www-form-urlencoded" | ||||
| CONTENT_TYPE_X_LOG = "text/x-log" | ||||
|  | ||||
| COOKIE_INGRESS = "ingress_session" | ||||
|  | ||||
| ATTR_ADDITIONAL_LOCATIONS = "additional_locations" | ||||
| ATTR_AGENT_VERSION = "agent_version" | ||||
| ATTR_APPARMOR_VERSION = "apparmor_version" | ||||
| ATTR_ATTRIBUTES = "attributes" | ||||
| ATTR_AVAILABLE_UPDATES = "available_updates" | ||||
| ATTR_BACKGROUND = "background" | ||||
| ATTR_BOOT_CONFIG = "boot_config" | ||||
| ATTR_BOOT_SLOT = "boot_slot" | ||||
| ATTR_BOOT_SLOTS = "boot_slots" | ||||
| ATTR_BOOT_TIMESTAMP = "boot_timestamp" | ||||
| ATTR_BOOTS = "boots" | ||||
| ATTR_BROADCAST_LLMNR = "broadcast_llmnr" | ||||
| @@ -30,25 +38,54 @@ ATTR_DT_UTC = "dt_utc" | ||||
| ATTR_EJECTABLE = "ejectable" | ||||
| ATTR_FALLBACK = "fallback" | ||||
| ATTR_FILESYSTEMS = "filesystems" | ||||
| ATTR_FORCE = "force" | ||||
| ATTR_GROUP_IDS = "group_ids" | ||||
| ATTR_IDENTIFIERS = "identifiers" | ||||
| ATTR_IS_ACTIVE = "is_active" | ||||
| ATTR_IS_OWNER = "is_owner" | ||||
| ATTR_JOBS = "jobs" | ||||
| ATTR_LLMNR = "llmnr" | ||||
| ATTR_LLMNR_HOSTNAME = "llmnr_hostname" | ||||
| ATTR_LOCAL_ONLY = "local_only" | ||||
| ATTR_LOCATION_ATTRIBUTES = "location_attributes" | ||||
| ATTR_LOCATIONS = "locations" | ||||
| ATTR_MAX_DEPTH = "max_depth" | ||||
| ATTR_MDNS = "mdns" | ||||
| ATTR_MODEL = "model" | ||||
| ATTR_MOUNTS = "mounts" | ||||
| ATTR_MOUNT_POINTS = "mount_points" | ||||
| ATTR_PANEL_PATH = "panel_path" | ||||
| ATTR_REMOVABLE = "removable" | ||||
| ATTR_REMOVE_CONFIG = "remove_config" | ||||
| ATTR_REVISION = "revision" | ||||
| ATTR_SAFE_MODE = "safe_mode" | ||||
| ATTR_SEAT = "seat" | ||||
| ATTR_SIGNED = "signed" | ||||
| ATTR_STARTUP_TIME = "startup_time" | ||||
| ATTR_STATUS = "status" | ||||
| ATTR_SUBSYSTEM = "subsystem" | ||||
| ATTR_SYSFS = "sysfs" | ||||
| ATTR_SYSTEM_HEALTH_LED = "system_health_led" | ||||
| ATTR_TIME_DETECTED = "time_detected" | ||||
| ATTR_UPDATE_TYPE = "update_type" | ||||
| ATTR_USE_NTP = "use_ntp" | ||||
| ATTR_USAGE = "usage" | ||||
| ATTR_USE_NTP = "use_ntp" | ||||
| ATTR_USERS = "users" | ||||
| ATTR_USER_PATH = "user_path" | ||||
| ATTR_VENDOR = "vendor" | ||||
| ATTR_VIRTUALIZATION = "virtualization" | ||||
|  | ||||
|  | ||||
| class BootSlot(StrEnum): | ||||
|     """Boot slots used by HAOS.""" | ||||
|  | ||||
|     A = "A" | ||||
|     B = "B" | ||||
|  | ||||
|  | ||||
| class DetectBlockingIO(StrEnum): | ||||
|     """Enable/Disable detection for blocking I/O in event loop.""" | ||||
|  | ||||
|     OFF = "off" | ||||
|     ON = "on" | ||||
|     ON_AT_STARTUP = "on-at-startup" | ||||
|   | ||||
| @@ -1,6 +1,9 @@ | ||||
| """Init file for Supervisor network RESTful API.""" | ||||
| import logging | ||||
|  | ||||
| import logging | ||||
| from typing import Any | ||||
|  | ||||
| from aiohttp import web | ||||
| import voluptuous as vol | ||||
|  | ||||
| from ..addons.addon import Addon | ||||
| @@ -15,8 +18,8 @@ from ..const import ( | ||||
|     AddonState, | ||||
| ) | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..discovery.validate import valid_discovery_service | ||||
| from ..exceptions import APIError, APIForbidden | ||||
| from ..discovery import Message | ||||
| from ..exceptions import APIForbidden, APINotFound | ||||
| from .utils import api_process, api_validate, require_home_assistant | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
| @@ -24,7 +27,7 @@ _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
| SCHEMA_DISCOVERY = vol.Schema( | ||||
|     { | ||||
|         vol.Required(ATTR_SERVICE): str, | ||||
|         vol.Optional(ATTR_CONFIG): vol.Maybe(dict), | ||||
|         vol.Required(ATTR_CONFIG): dict, | ||||
|     } | ||||
| ) | ||||
|  | ||||
| @@ -32,16 +35,16 @@ SCHEMA_DISCOVERY = vol.Schema( | ||||
| class APIDiscovery(CoreSysAttributes): | ||||
|     """Handle RESTful API for discovery functions.""" | ||||
|  | ||||
|     def _extract_message(self, request): | ||||
|     def _extract_message(self, request: web.Request) -> Message: | ||||
|         """Extract discovery message from URL.""" | ||||
|         message = self.sys_discovery.get(request.match_info.get("uuid")) | ||||
|         message = self.sys_discovery.get(request.match_info["uuid"]) | ||||
|         if not message: | ||||
|             raise APIError("Discovery message not found") | ||||
|             raise APINotFound("Discovery message not found") | ||||
|         return message | ||||
|  | ||||
|     @api_process | ||||
|     @require_home_assistant | ||||
|     async def list(self, request): | ||||
|     async def list_discovery(self, request: web.Request) -> dict[str, Any]: | ||||
|         """Show registered and available services.""" | ||||
|         # Get available discovery | ||||
|         discovery = [ | ||||
| @@ -52,12 +55,16 @@ class APIDiscovery(CoreSysAttributes): | ||||
|                 ATTR_CONFIG: message.config, | ||||
|             } | ||||
|             for message in self.sys_discovery.list_messages | ||||
|             if (addon := self.sys_addons.get(message.addon, local_only=True)) | ||||
|             and addon.state == AddonState.STARTED | ||||
|             if ( | ||||
|                 discovered := self.sys_addons.get_local_only( | ||||
|                     message.addon, | ||||
|                 ) | ||||
|             ) | ||||
|             and discovered.state == AddonState.STARTED | ||||
|         ] | ||||
|  | ||||
|         # Get available services/add-ons | ||||
|         services = {} | ||||
|         services: dict[str, list[str]] = {} | ||||
|         for addon in self.sys_addons.all: | ||||
|             for name in addon.discovery: | ||||
|                 services.setdefault(name, []).append(addon.slug) | ||||
| @@ -65,21 +72,12 @@ class APIDiscovery(CoreSysAttributes): | ||||
|         return {ATTR_DISCOVERY: discovery, ATTR_SERVICES: services} | ||||
|  | ||||
|     @api_process | ||||
|     async def set_discovery(self, request): | ||||
|     async def set_discovery(self, request: web.Request) -> dict[str, str]: | ||||
|         """Write data into a discovery pipeline.""" | ||||
|         body = await api_validate(SCHEMA_DISCOVERY, request) | ||||
|         addon: Addon = request[REQUEST_FROM] | ||||
|         service = body[ATTR_SERVICE] | ||||
|  | ||||
|         try: | ||||
|             valid_discovery_service(service) | ||||
|         except vol.Invalid: | ||||
|             _LOGGER.warning( | ||||
|                 "Received discovery message for unknown service %s from addon %s. Please report this to the maintainer of the add-on", | ||||
|                 service, | ||||
|                 addon.name, | ||||
|             ) | ||||
|  | ||||
|         # Access? | ||||
|         if body[ATTR_SERVICE] not in addon.discovery: | ||||
|             _LOGGER.error( | ||||
| @@ -92,13 +90,13 @@ class APIDiscovery(CoreSysAttributes): | ||||
|             ) | ||||
|  | ||||
|         # Process discovery message | ||||
|         message = self.sys_discovery.send(addon, **body) | ||||
|         message = await self.sys_discovery.send(addon, **body) | ||||
|  | ||||
|         return {ATTR_UUID: message.uuid} | ||||
|  | ||||
|     @api_process | ||||
|     @require_home_assistant | ||||
|     async def get_discovery(self, request): | ||||
|     async def get_discovery(self, request: web.Request) -> dict[str, Any]: | ||||
|         """Read data into a discovery message.""" | ||||
|         message = self._extract_message(request) | ||||
|  | ||||
| @@ -110,7 +108,7 @@ class APIDiscovery(CoreSysAttributes): | ||||
|         } | ||||
|  | ||||
|     @api_process | ||||
|     async def del_discovery(self, request): | ||||
|     async def del_discovery(self, request: web.Request) -> None: | ||||
|         """Delete data into a discovery message.""" | ||||
|         message = self._extract_message(request) | ||||
|         addon = request[REQUEST_FROM] | ||||
| @@ -119,5 +117,4 @@ class APIDiscovery(CoreSysAttributes): | ||||
|         if message.addon != addon.slug: | ||||
|             raise APIForbidden("Can't remove discovery message") | ||||
|  | ||||
|         self.sys_discovery.remove(message) | ||||
|         return True | ||||
|         await self.sys_discovery.remove(message) | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Init file for Supervisor DNS RESTful API.""" | ||||
|  | ||||
| import asyncio | ||||
| from collections.abc import Awaitable | ||||
| import logging | ||||
| @@ -26,8 +27,8 @@ from ..const import ( | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APIError | ||||
| from ..validate import dns_server_list, version_tag | ||||
| from .const import ATTR_FALLBACK, ATTR_LLMNR, ATTR_MDNS, CONTENT_TYPE_BINARY | ||||
| from .utils import api_process, api_process_raw, api_validate | ||||
| from .const import ATTR_FALLBACK, ATTR_LLMNR, ATTR_MDNS | ||||
| from .utils import api_process, api_validate | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
|  | ||||
| @@ -77,7 +78,7 @@ class APICoreDNS(CoreSysAttributes): | ||||
|         if restart_required: | ||||
|             self.sys_create_task(self.sys_plugins.dns.restart()) | ||||
|  | ||||
|         self.sys_plugins.dns.save_data() | ||||
|         await self.sys_plugins.dns.save_data() | ||||
|  | ||||
|     @api_process | ||||
|     async def stats(self, request: web.Request) -> dict[str, Any]: | ||||
| @@ -105,11 +106,6 @@ class APICoreDNS(CoreSysAttributes): | ||||
|             raise APIError(f"Version {version} is already in use") | ||||
|         await asyncio.shield(self.sys_plugins.dns.update(version)) | ||||
|  | ||||
|     @api_process_raw(CONTENT_TYPE_BINARY) | ||||
|     def logs(self, request: web.Request) -> Awaitable[bytes]: | ||||
|         """Return DNS Docker logs.""" | ||||
|         return self.sys_plugins.dns.logs() | ||||
|  | ||||
|     @api_process | ||||
|     def restart(self, request: web.Request) -> Awaitable[None]: | ||||
|         """Restart CoreDNS plugin.""" | ||||
|   | ||||
| @@ -1,13 +1,18 @@ | ||||
| """Init file for Supervisor Home Assistant RESTful API.""" | ||||
|  | ||||
| import logging | ||||
| from typing import Any | ||||
|  | ||||
| from aiohttp import web | ||||
| import voluptuous as vol | ||||
|  | ||||
| from supervisor.resolution.const import ContextType, IssueType, SuggestionType | ||||
|  | ||||
| from ..const import ( | ||||
|     ATTR_ENABLE_IPV6, | ||||
|     ATTR_HOSTNAME, | ||||
|     ATTR_LOGGING, | ||||
|     ATTR_MTU, | ||||
|     ATTR_PASSWORD, | ||||
|     ATTR_REGISTRIES, | ||||
|     ATTR_STORAGE, | ||||
| @@ -15,6 +20,7 @@ from ..const import ( | ||||
|     ATTR_VERSION, | ||||
| ) | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APINotFound | ||||
| from .utils import api_process, api_validate | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
| @@ -28,10 +34,65 @@ SCHEMA_DOCKER_REGISTRY = vol.Schema( | ||||
|     } | ||||
| ) | ||||
|  | ||||
| # pylint: disable=no-value-for-parameter | ||||
| SCHEMA_OPTIONS = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_ENABLE_IPV6): vol.Maybe(vol.Boolean()), | ||||
|         vol.Optional(ATTR_MTU): vol.Maybe(vol.All(int, vol.Range(min=68, max=65535))), | ||||
|     } | ||||
| ) | ||||
|  | ||||
|  | ||||
| class APIDocker(CoreSysAttributes): | ||||
|     """Handle RESTful API for Docker configuration.""" | ||||
|  | ||||
|     @api_process | ||||
|     async def info(self, request: web.Request): | ||||
|         """Get docker info.""" | ||||
|         data_registries = {} | ||||
|         for hostname, registry in self.sys_docker.config.registries.items(): | ||||
|             data_registries[hostname] = { | ||||
|                 ATTR_USERNAME: registry[ATTR_USERNAME], | ||||
|             } | ||||
|         return { | ||||
|             ATTR_VERSION: self.sys_docker.info.version, | ||||
|             ATTR_ENABLE_IPV6: self.sys_docker.config.enable_ipv6, | ||||
|             ATTR_MTU: self.sys_docker.config.mtu, | ||||
|             ATTR_STORAGE: self.sys_docker.info.storage, | ||||
|             ATTR_LOGGING: self.sys_docker.info.logging, | ||||
|             ATTR_REGISTRIES: data_registries, | ||||
|         } | ||||
|  | ||||
|     @api_process | ||||
|     async def options(self, request: web.Request) -> None: | ||||
|         """Set docker options.""" | ||||
|         body = await api_validate(SCHEMA_OPTIONS, request) | ||||
|  | ||||
|         reboot_required = False | ||||
|  | ||||
|         if ( | ||||
|             ATTR_ENABLE_IPV6 in body | ||||
|             and self.sys_docker.config.enable_ipv6 != body[ATTR_ENABLE_IPV6] | ||||
|         ): | ||||
|             self.sys_docker.config.enable_ipv6 = body[ATTR_ENABLE_IPV6] | ||||
|             reboot_required = True | ||||
|  | ||||
|         if ATTR_MTU in body and self.sys_docker.config.mtu != body[ATTR_MTU]: | ||||
|             self.sys_docker.config.mtu = body[ATTR_MTU] | ||||
|             reboot_required = True | ||||
|  | ||||
|         if reboot_required: | ||||
|             _LOGGER.info( | ||||
|                 "Host system reboot required to apply Docker configuration changes" | ||||
|             ) | ||||
|             self.sys_resolution.create_issue( | ||||
|                 IssueType.REBOOT_REQUIRED, | ||||
|                 ContextType.SYSTEM, | ||||
|                 suggestions=[SuggestionType.EXECUTE_REBOOT], | ||||
|             ) | ||||
|  | ||||
|         await self.sys_docker.config.save_data() | ||||
|  | ||||
|     @api_process | ||||
|     async def registries(self, request) -> dict[str, Any]: | ||||
|         """Return the list of registries.""" | ||||
| @@ -51,26 +112,14 @@ class APIDocker(CoreSysAttributes): | ||||
|         for hostname, registry in body.items(): | ||||
|             self.sys_docker.config.registries[hostname] = registry | ||||
|  | ||||
|         self.sys_docker.config.save_data() | ||||
|         await self.sys_docker.config.save_data() | ||||
|  | ||||
|     @api_process | ||||
|     async def remove_registry(self, request: web.Request): | ||||
|         """Delete a docker registry.""" | ||||
|         hostname = request.match_info.get(ATTR_HOSTNAME) | ||||
|         del self.sys_docker.config.registries[hostname] | ||||
|         self.sys_docker.config.save_data() | ||||
|         if hostname not in self.sys_docker.config.registries: | ||||
|             raise APINotFound(f"Hostname {hostname} does not exist in registries") | ||||
|  | ||||
|     @api_process | ||||
|     async def info(self, request: web.Request): | ||||
|         """Get docker info.""" | ||||
|         data_registries = {} | ||||
|         for hostname, registry in self.sys_docker.config.registries.items(): | ||||
|             data_registries[hostname] = { | ||||
|                 ATTR_USERNAME: registry[ATTR_USERNAME], | ||||
|             } | ||||
|         return { | ||||
|             ATTR_VERSION: self.sys_docker.info.version, | ||||
|             ATTR_STORAGE: self.sys_docker.info.storage, | ||||
|             ATTR_LOGGING: self.sys_docker.info.logging, | ||||
|             ATTR_REGISTRIES: data_registries, | ||||
|         } | ||||
|         del self.sys_docker.config.registries[hostname] | ||||
|         await self.sys_docker.config.save_data() | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Init file for Supervisor hardware RESTful API.""" | ||||
|  | ||||
| import logging | ||||
| from typing import Any | ||||
|  | ||||
| @@ -16,7 +17,7 @@ from ..const import ( | ||||
|     ATTR_SYSTEM, | ||||
| ) | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..dbus.udisks2 import UDisks2 | ||||
| from ..dbus.udisks2 import UDisks2Manager | ||||
| from ..dbus.udisks2.block import UDisks2Block | ||||
| from ..dbus.udisks2.drive import UDisks2Drive | ||||
| from ..hardware.data import Device | ||||
| @@ -67,12 +68,15 @@ def filesystem_struct(fs_block: UDisks2Block) -> dict[str, Any]: | ||||
|         ATTR_NAME: fs_block.id_label, | ||||
|         ATTR_SYSTEM: fs_block.hint_system, | ||||
|         ATTR_MOUNT_POINTS: [ | ||||
|             str(mount_point) for mount_point in fs_block.filesystem.mount_points | ||||
|             str(mount_point) | ||||
|             for mount_point in ( | ||||
|                 fs_block.filesystem.mount_points if fs_block.filesystem else [] | ||||
|             ) | ||||
|         ], | ||||
|     } | ||||
|  | ||||
|  | ||||
| def drive_struct(udisks2: UDisks2, drive: UDisks2Drive) -> dict[str, Any]: | ||||
| def drive_struct(udisks2: UDisks2Manager, drive: UDisks2Drive) -> dict[str, Any]: | ||||
|     """Return a dict with information of a disk to be used in the API.""" | ||||
|     return { | ||||
|         ATTR_VENDOR: drive.vendor, | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Init file for Supervisor Home Assistant RESTful API.""" | ||||
|  | ||||
| import asyncio | ||||
| from collections.abc import Awaitable | ||||
| import logging | ||||
| @@ -19,6 +20,7 @@ from ..const import ( | ||||
|     ATTR_CPU_PERCENT, | ||||
|     ATTR_IMAGE, | ||||
|     ATTR_IP_ADDRESS, | ||||
|     ATTR_JOB_ID, | ||||
|     ATTR_MACHINE, | ||||
|     ATTR_MEMORY_LIMIT, | ||||
|     ATTR_MEMORY_PERCENT, | ||||
| @@ -34,10 +36,10 @@ from ..const import ( | ||||
|     ATTR_WATCHDOG, | ||||
| ) | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APIError | ||||
| from ..exceptions import APIDBMigrationInProgress, APIError | ||||
| from ..validate import docker_image, network_port, version_tag | ||||
| from .const import CONTENT_TYPE_BINARY | ||||
| from .utils import api_process, api_process_raw, api_validate | ||||
| from .const import ATTR_BACKGROUND, ATTR_FORCE, ATTR_SAFE_MODE | ||||
| from .utils import api_process, api_validate, background_task | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
|  | ||||
| @@ -60,6 +62,20 @@ SCHEMA_UPDATE = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_VERSION): version_tag, | ||||
|         vol.Optional(ATTR_BACKUP): bool, | ||||
|         vol.Optional(ATTR_BACKGROUND, default=False): bool, | ||||
|     } | ||||
| ) | ||||
|  | ||||
| SCHEMA_RESTART = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_SAFE_MODE, default=False): vol.Boolean(), | ||||
|         vol.Optional(ATTR_FORCE, default=False): vol.Boolean(), | ||||
|     } | ||||
| ) | ||||
|  | ||||
| SCHEMA_STOP = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_FORCE, default=False): vol.Boolean(), | ||||
|     } | ||||
| ) | ||||
|  | ||||
| @@ -67,6 +83,17 @@ SCHEMA_UPDATE = vol.Schema( | ||||
| class APIHomeAssistant(CoreSysAttributes): | ||||
|     """Handle RESTful API for Home Assistant functions.""" | ||||
|  | ||||
|     async def _check_offline_migration(self, force: bool = False) -> None: | ||||
|         """Check and raise if there's an offline DB migration in progress.""" | ||||
|         if ( | ||||
|             not force | ||||
|             and (state := await self.sys_homeassistant.api.get_api_state()) | ||||
|             and state.offline_db_migration | ||||
|         ): | ||||
|             raise APIDBMigrationInProgress( | ||||
|                 "Offline database migration in progress, try again after it has completed" | ||||
|             ) | ||||
|  | ||||
|     @api_process | ||||
|     async def info(self, request: web.Request) -> dict[str, Any]: | ||||
|         """Return host information.""" | ||||
| @@ -93,7 +120,10 @@ class APIHomeAssistant(CoreSysAttributes): | ||||
|         body = await api_validate(SCHEMA_OPTIONS, request) | ||||
|  | ||||
|         if ATTR_IMAGE in body: | ||||
|             self.sys_homeassistant.image = body[ATTR_IMAGE] | ||||
|             self.sys_homeassistant.set_image(body[ATTR_IMAGE]) | ||||
|             self.sys_homeassistant.override_image = ( | ||||
|                 self.sys_homeassistant.image != self.sys_homeassistant.default_image | ||||
|             ) | ||||
|  | ||||
|         if ATTR_BOOT in body: | ||||
|             self.sys_homeassistant.boot = body[ATTR_BOOT] | ||||
| @@ -121,7 +151,7 @@ class APIHomeAssistant(CoreSysAttributes): | ||||
|                 ATTR_BACKUPS_EXCLUDE_DATABASE | ||||
|             ] | ||||
|  | ||||
|         self.sys_homeassistant.save_data() | ||||
|         await self.sys_homeassistant.save_data() | ||||
|  | ||||
|     @api_process | ||||
|     async def stats(self, request: web.Request) -> dict[Any, str]: | ||||
| @@ -142,21 +172,31 @@ class APIHomeAssistant(CoreSysAttributes): | ||||
|         } | ||||
|  | ||||
|     @api_process | ||||
|     async def update(self, request: web.Request) -> None: | ||||
|     async def update(self, request: web.Request) -> dict[str, str] | None: | ||||
|         """Update Home Assistant.""" | ||||
|         body = await api_validate(SCHEMA_UPDATE, request) | ||||
|         await self._check_offline_migration() | ||||
|  | ||||
|         await asyncio.shield( | ||||
|             self.sys_homeassistant.core.update( | ||||
|                 version=body.get(ATTR_VERSION, self.sys_homeassistant.latest_version), | ||||
|                 backup=body.get(ATTR_BACKUP), | ||||
|             ) | ||||
|         background = body[ATTR_BACKGROUND] | ||||
|         update_task, job_id = await background_task( | ||||
|             self, | ||||
|             self.sys_homeassistant.core.update, | ||||
|             version=body.get(ATTR_VERSION, self.sys_homeassistant.latest_version), | ||||
|             backup=body.get(ATTR_BACKUP), | ||||
|         ) | ||||
|  | ||||
|         if background and not update_task.done(): | ||||
|             return {ATTR_JOB_ID: job_id} | ||||
|  | ||||
|         return await update_task | ||||
|  | ||||
|     @api_process | ||||
|     def stop(self, request: web.Request) -> Awaitable[None]: | ||||
|     async def stop(self, request: web.Request) -> Awaitable[None]: | ||||
|         """Stop Home Assistant.""" | ||||
|         return asyncio.shield(self.sys_homeassistant.core.stop()) | ||||
|         body = await api_validate(SCHEMA_STOP, request) | ||||
|         await self._check_offline_migration(force=body[ATTR_FORCE]) | ||||
|  | ||||
|         return await asyncio.shield(self.sys_homeassistant.core.stop()) | ||||
|  | ||||
|     @api_process | ||||
|     def start(self, request: web.Request) -> Awaitable[None]: | ||||
| @@ -164,19 +204,24 @@ class APIHomeAssistant(CoreSysAttributes): | ||||
|         return asyncio.shield(self.sys_homeassistant.core.start()) | ||||
|  | ||||
|     @api_process | ||||
|     def restart(self, request: web.Request) -> Awaitable[None]: | ||||
|     async def restart(self, request: web.Request) -> None: | ||||
|         """Restart Home Assistant.""" | ||||
|         return asyncio.shield(self.sys_homeassistant.core.restart()) | ||||
|         body = await api_validate(SCHEMA_RESTART, request) | ||||
|         await self._check_offline_migration(force=body[ATTR_FORCE]) | ||||
|  | ||||
|         await asyncio.shield( | ||||
|             self.sys_homeassistant.core.restart(safe_mode=body[ATTR_SAFE_MODE]) | ||||
|         ) | ||||
|  | ||||
|     @api_process | ||||
|     def rebuild(self, request: web.Request) -> Awaitable[None]: | ||||
|     async def rebuild(self, request: web.Request) -> None: | ||||
|         """Rebuild Home Assistant.""" | ||||
|         return asyncio.shield(self.sys_homeassistant.core.rebuild()) | ||||
|         body = await api_validate(SCHEMA_RESTART, request) | ||||
|         await self._check_offline_migration(force=body[ATTR_FORCE]) | ||||
|  | ||||
|     @api_process_raw(CONTENT_TYPE_BINARY) | ||||
|     def logs(self, request: web.Request) -> Awaitable[bytes]: | ||||
|         """Return Home Assistant Docker logs.""" | ||||
|         return self.sys_homeassistant.core.logs() | ||||
|         await asyncio.shield( | ||||
|             self.sys_homeassistant.core.rebuild(safe_mode=body[ATTR_SAFE_MODE]) | ||||
|         ) | ||||
|  | ||||
|     @api_process | ||||
|     async def check(self, request: web.Request) -> None: | ||||
|   | ||||
| @@ -1,9 +1,18 @@ | ||||
| """Init file for Supervisor host RESTful API.""" | ||||
|  | ||||
| import asyncio | ||||
| from contextlib import suppress | ||||
| import json | ||||
| import logging | ||||
| from typing import Any | ||||
|  | ||||
| from aiohttp import web | ||||
| from aiohttp import ( | ||||
|     ClientConnectionResetError, | ||||
|     ClientError, | ||||
|     ClientPayloadError, | ||||
|     ClientTimeout, | ||||
|     web, | ||||
| ) | ||||
| from aiohttp.hdrs import ACCEPT, RANGE | ||||
| import voluptuous as vol | ||||
| from voluptuous.error import CoerceInvalid | ||||
| @@ -27,8 +36,16 @@ from ..const import ( | ||||
|     ATTR_TIMEZONE, | ||||
| ) | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APIError, HostLogError | ||||
| from ..host.const import PARAM_BOOT_ID, PARAM_FOLLOW, PARAM_SYSLOG_IDENTIFIER | ||||
| from ..exceptions import APIDBMigrationInProgress, APIError, HostLogError | ||||
| from ..host.const import ( | ||||
|     PARAM_BOOT_ID, | ||||
|     PARAM_FOLLOW, | ||||
|     PARAM_SYSLOG_IDENTIFIER, | ||||
|     LogFormat, | ||||
|     LogFormatter, | ||||
| ) | ||||
| from ..host.logs import SYSTEMD_JOURNAL_GATEWAYD_LINES_MAX | ||||
| from ..utils.systemd_journal import journal_logs_reader | ||||
| from .const import ( | ||||
|     ATTR_AGENT_VERSION, | ||||
|     ATTR_APPARMOR_VERSION, | ||||
| @@ -38,26 +55,49 @@ from .const import ( | ||||
|     ATTR_BROADCAST_MDNS, | ||||
|     ATTR_DT_SYNCHRONIZED, | ||||
|     ATTR_DT_UTC, | ||||
|     ATTR_FORCE, | ||||
|     ATTR_IDENTIFIERS, | ||||
|     ATTR_LLMNR_HOSTNAME, | ||||
|     ATTR_MAX_DEPTH, | ||||
|     ATTR_STARTUP_TIME, | ||||
|     ATTR_USE_NTP, | ||||
|     ATTR_VIRTUALIZATION, | ||||
|     CONTENT_TYPE_TEXT, | ||||
|     CONTENT_TYPE_X_LOG, | ||||
| ) | ||||
| from .utils import api_process, api_validate | ||||
| from .utils import api_process, api_process_raw, api_validate | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
|  | ||||
| IDENTIFIER = "identifier" | ||||
| BOOTID = "bootid" | ||||
| DEFAULT_RANGE = 100 | ||||
| DEFAULT_LINES = 100 | ||||
|  | ||||
| SCHEMA_OPTIONS = vol.Schema({vol.Optional(ATTR_HOSTNAME): str}) | ||||
|  | ||||
| # pylint: disable=no-value-for-parameter | ||||
| SCHEMA_SHUTDOWN = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_FORCE, default=False): vol.Boolean(), | ||||
|     } | ||||
| ) | ||||
| # pylint: enable=no-value-for-parameter | ||||
|  | ||||
|  | ||||
| class APIHost(CoreSysAttributes): | ||||
|     """Handle RESTful API for host functions.""" | ||||
|  | ||||
|     async def _check_ha_offline_migration(self, force: bool) -> None: | ||||
|         """Check if HA has an offline migration in progress and raise if not forced.""" | ||||
|         if ( | ||||
|             not force | ||||
|             and (state := await self.sys_homeassistant.api.get_api_state()) | ||||
|             and state.offline_db_migration | ||||
|         ): | ||||
|             raise APIDBMigrationInProgress( | ||||
|                 "Home Assistant offline database migration in progress, please wait until complete before shutting down host" | ||||
|             ) | ||||
|  | ||||
|     @api_process | ||||
|     async def info(self, request): | ||||
|         """Return host information.""" | ||||
| @@ -65,12 +105,13 @@ class APIHost(CoreSysAttributes): | ||||
|             ATTR_AGENT_VERSION: self.sys_dbus.agent.version, | ||||
|             ATTR_APPARMOR_VERSION: self.sys_host.apparmor.version, | ||||
|             ATTR_CHASSIS: self.sys_host.info.chassis, | ||||
|             ATTR_VIRTUALIZATION: self.sys_host.info.virtualization, | ||||
|             ATTR_CPE: self.sys_host.info.cpe, | ||||
|             ATTR_DEPLOYMENT: self.sys_host.info.deployment, | ||||
|             ATTR_DISK_FREE: self.sys_host.info.free_space, | ||||
|             ATTR_DISK_TOTAL: self.sys_host.info.total_space, | ||||
|             ATTR_DISK_USED: self.sys_host.info.used_space, | ||||
|             ATTR_DISK_LIFE_TIME: self.sys_host.info.disk_life_time, | ||||
|             ATTR_DISK_FREE: await self.sys_host.info.free_space(), | ||||
|             ATTR_DISK_TOTAL: await self.sys_host.info.total_space(), | ||||
|             ATTR_DISK_USED: await self.sys_host.info.used_space(), | ||||
|             ATTR_DISK_LIFE_TIME: await self.sys_host.info.disk_life_time(), | ||||
|             ATTR_FEATURES: self.sys_host.features, | ||||
|             ATTR_HOSTNAME: self.sys_host.info.hostname, | ||||
|             ATTR_LLMNR_HOSTNAME: self.sys_host.info.llmnr_hostname, | ||||
| @@ -98,14 +139,20 @@ class APIHost(CoreSysAttributes): | ||||
|             ) | ||||
|  | ||||
|     @api_process | ||||
|     def reboot(self, request): | ||||
|     async def reboot(self, request): | ||||
|         """Reboot host.""" | ||||
|         return asyncio.shield(self.sys_host.control.reboot()) | ||||
|         body = await api_validate(SCHEMA_SHUTDOWN, request) | ||||
|         await self._check_ha_offline_migration(force=body[ATTR_FORCE]) | ||||
|  | ||||
|         return await asyncio.shield(self.sys_host.control.reboot()) | ||||
|  | ||||
|     @api_process | ||||
|     def shutdown(self, request): | ||||
|     async def shutdown(self, request): | ||||
|         """Poweroff host.""" | ||||
|         return asyncio.shield(self.sys_host.control.shutdown()) | ||||
|         body = await api_validate(SCHEMA_SHUTDOWN, request) | ||||
|         await self._check_ha_offline_migration(force=body[ATTR_FORCE]) | ||||
|  | ||||
|         return await asyncio.shield(self.sys_host.control.shutdown()) | ||||
|  | ||||
|     @api_process | ||||
|     def reload(self, request): | ||||
| @@ -153,50 +200,194 @@ class APIHost(CoreSysAttributes): | ||||
|                 raise APIError() from err | ||||
|         return possible_offset | ||||
|  | ||||
|     @api_process | ||||
|     async def advanced_logs( | ||||
|         self, request: web.Request, identifier: str | None = None, follow: bool = False | ||||
|     async def advanced_logs_handler( | ||||
|         self, | ||||
|         request: web.Request, | ||||
|         identifier: str | None = None, | ||||
|         follow: bool = False, | ||||
|         latest: bool = False, | ||||
|     ) -> web.StreamResponse: | ||||
|         """Return systemd-journald logs.""" | ||||
|         params = {} | ||||
|         log_formatter = LogFormatter.PLAIN | ||||
|         params: dict[str, Any] = {} | ||||
|         if identifier: | ||||
|             params[PARAM_SYSLOG_IDENTIFIER] = identifier | ||||
|         elif IDENTIFIER in request.match_info: | ||||
|             params[PARAM_SYSLOG_IDENTIFIER] = request.match_info.get(IDENTIFIER) | ||||
|             params[PARAM_SYSLOG_IDENTIFIER] = request.match_info[IDENTIFIER] | ||||
|         else: | ||||
|             params[PARAM_SYSLOG_IDENTIFIER] = self.sys_host.logs.default_identifiers | ||||
|             # host logs should be always verbose, no matter what Accept header is used | ||||
|             log_formatter = LogFormatter.VERBOSE | ||||
|  | ||||
|         if BOOTID in request.match_info: | ||||
|             params[PARAM_BOOT_ID] = await self._get_boot_id( | ||||
|                 request.match_info.get(BOOTID) | ||||
|             ) | ||||
|             params[PARAM_BOOT_ID] = await self._get_boot_id(request.match_info[BOOTID]) | ||||
|         if follow: | ||||
|             params[PARAM_FOLLOW] = "" | ||||
|  | ||||
|         if latest: | ||||
|             if not identifier: | ||||
|                 raise APIError( | ||||
|                     "Latest logs can only be fetched for a specific identifier." | ||||
|                 ) | ||||
|  | ||||
|             try: | ||||
|                 epoch = await self._get_container_last_epoch(identifier) | ||||
|                 params["CONTAINER_LOG_EPOCH"] = epoch | ||||
|             except HostLogError as err: | ||||
|                 raise APIError( | ||||
|                     f"Cannot determine CONTAINER_LOG_EPOCH of {identifier}, latest logs not available." | ||||
|                 ) from err | ||||
|  | ||||
|         if ACCEPT in request.headers and request.headers[ACCEPT] not in [ | ||||
|             CONTENT_TYPE_TEXT, | ||||
|             CONTENT_TYPE_X_LOG, | ||||
|             "*/*", | ||||
|         ]: | ||||
|             raise APIError( | ||||
|                 "Invalid content type requested. Only text/plain supported for now." | ||||
|                 "Invalid content type requested. Only text/plain and text/x-log " | ||||
|                 "supported for now." | ||||
|             ) | ||||
|  | ||||
|         if RANGE in request.headers: | ||||
|             range_header = request.headers.get(RANGE) | ||||
|         if "verbose" in request.query or request.headers[ACCEPT] == CONTENT_TYPE_X_LOG: | ||||
|             log_formatter = LogFormatter.VERBOSE | ||||
|  | ||||
|         if "lines" in request.query: | ||||
|             lines = request.query.get("lines", DEFAULT_LINES) | ||||
|             try: | ||||
|                 lines = int(lines) | ||||
|             except ValueError: | ||||
|                 # If the user passed a non-integer value, just use the default instead of error. | ||||
|                 lines = DEFAULT_LINES | ||||
|             finally: | ||||
|                 # We can't use the entries= Range header syntax to refer to the last 1 line, | ||||
|                 # and passing 1 to the calculation below would return the 1st line of the logs | ||||
|                 # instead. Since this is really an edge case that doesn't matter much, we'll just | ||||
|                 # return 2 lines at minimum. | ||||
|                 lines = max(2, lines) | ||||
|             # entries=cursor[[:num_skip]:num_entries] | ||||
|             range_header = f"entries=:-{lines - 1}:{SYSTEMD_JOURNAL_GATEWAYD_LINES_MAX if follow else lines}" | ||||
|         elif latest: | ||||
|             range_header = f"entries=:0:{SYSTEMD_JOURNAL_GATEWAYD_LINES_MAX}" | ||||
|         elif RANGE in request.headers: | ||||
|             range_header = request.headers[RANGE] | ||||
|         else: | ||||
|             range_header = f"entries=:-{DEFAULT_RANGE}:" | ||||
|             range_header = f"entries=:-{DEFAULT_LINES - 1}:{SYSTEMD_JOURNAL_GATEWAYD_LINES_MAX if follow else DEFAULT_LINES}" | ||||
|  | ||||
|         async with self.sys_host.logs.journald_logs( | ||||
|             params=params, range_header=range_header | ||||
|             params=params, range_header=range_header, accept=LogFormat.JOURNAL | ||||
|         ) as resp: | ||||
|             try: | ||||
|                 response = web.StreamResponse() | ||||
|                 response.content_type = CONTENT_TYPE_TEXT | ||||
|                 await response.prepare(request) | ||||
|                 async for data in resp.content: | ||||
|                     await response.write(data) | ||||
|             except ConnectionResetError as ex: | ||||
|                 headers_returned = False | ||||
|                 async for cursor, line in journal_logs_reader(resp, log_formatter): | ||||
|                     try: | ||||
|                         if not headers_returned: | ||||
|                             if cursor: | ||||
|                                 response.headers["X-First-Cursor"] = cursor | ||||
|                             response.headers["X-Accel-Buffering"] = "no" | ||||
|                             await response.prepare(request) | ||||
|                             headers_returned = True | ||||
|                         await response.write(line.encode("utf-8") + b"\n") | ||||
|                     except ClientConnectionResetError as err: | ||||
|                         # When client closes the connection while reading busy logs, we | ||||
|                         # sometimes get this exception. It should be safe to ignore it. | ||||
|                         _LOGGER.debug( | ||||
|                             "ClientConnectionResetError raised when returning journal logs: %s", | ||||
|                             err, | ||||
|                         ) | ||||
|                         break | ||||
|                     except ConnectionError as err: | ||||
|                         _LOGGER.warning( | ||||
|                             "%s raised when returning journal logs: %s", | ||||
|                             type(err).__name__, | ||||
|                             err, | ||||
|                         ) | ||||
|                         break | ||||
|             except (ConnectionResetError, ClientPayloadError) as ex: | ||||
|                 # ClientPayloadError is most likely caused by the closing the connection | ||||
|                 raise APIError( | ||||
|                     "Connection reset when trying to fetch data from systemd-journald." | ||||
|                 ) from ex | ||||
|             return response | ||||
|  | ||||
|     @api_process_raw(CONTENT_TYPE_TEXT, error_type=CONTENT_TYPE_TEXT) | ||||
|     async def advanced_logs( | ||||
|         self, | ||||
|         request: web.Request, | ||||
|         identifier: str | None = None, | ||||
|         follow: bool = False, | ||||
|         latest: bool = False, | ||||
|     ) -> web.StreamResponse: | ||||
|         """Return systemd-journald logs. Wrapped as standard API handler.""" | ||||
|         return await self.advanced_logs_handler(request, identifier, follow, latest) | ||||
|  | ||||
|     @api_process | ||||
|     async def disk_usage(self, request: web.Request) -> dict: | ||||
|         """Return a breakdown of storage usage for the system.""" | ||||
|  | ||||
|         max_depth = request.query.get(ATTR_MAX_DEPTH, 1) | ||||
|         try: | ||||
|             max_depth = int(max_depth) | ||||
|         except ValueError: | ||||
|             max_depth = 1 | ||||
|  | ||||
|         disk = self.sys_hardware.disk | ||||
|  | ||||
|         total, used, _ = await self.sys_run_in_executor( | ||||
|             disk.disk_usage, self.sys_config.path_supervisor | ||||
|         ) | ||||
|  | ||||
|         known_paths = await self.sys_run_in_executor( | ||||
|             disk.get_dir_sizes, | ||||
|             { | ||||
|                 "addons_data": self.sys_config.path_addons_data, | ||||
|                 "addons_config": self.sys_config.path_addon_configs, | ||||
|                 "media": self.sys_config.path_media, | ||||
|                 "share": self.sys_config.path_share, | ||||
|                 "backup": self.sys_config.path_backup, | ||||
|                 "ssl": self.sys_config.path_ssl, | ||||
|                 "homeassistant": self.sys_config.path_homeassistant, | ||||
|             }, | ||||
|             max_depth, | ||||
|         ) | ||||
|         return { | ||||
|             # this can be the disk/partition ID in the future | ||||
|             "id": "root", | ||||
|             "label": "Root", | ||||
|             "total_bytes": total, | ||||
|             "used_bytes": used, | ||||
|             "children": [ | ||||
|                 { | ||||
|                     "id": "system", | ||||
|                     "label": "System", | ||||
|                     "used_bytes": used | ||||
|                     - sum(path["used_bytes"] for path in known_paths), | ||||
|                 }, | ||||
|                 *known_paths, | ||||
|             ], | ||||
|         } | ||||
|  | ||||
|     async def _get_container_last_epoch(self, identifier: str) -> str | None: | ||||
|         """Get Docker's internal log epoch of the latest log entry for the given identifier.""" | ||||
|         try: | ||||
|             async with self.sys_host.logs.journald_logs( | ||||
|                 params={"CONTAINER_NAME": identifier}, | ||||
|                 range_header="entries=:-1:2",  # -1 = next to the last entry | ||||
|                 accept=LogFormat.JSON, | ||||
|                 timeout=ClientTimeout(total=10), | ||||
|             ) as resp: | ||||
|                 text = await resp.text() | ||||
|         except (ClientError, TimeoutError) as err: | ||||
|             raise HostLogError( | ||||
|                 "Could not get last container epoch from systemd-journal-gatewayd", | ||||
|                 _LOGGER.error, | ||||
|             ) from err | ||||
|  | ||||
|         try: | ||||
|             return json.loads(text.strip().split("\n")[-1])["CONTAINER_LOG_EPOCH"] | ||||
|         except (json.JSONDecodeError, KeyError, IndexError) as err: | ||||
|             raise HostLogError( | ||||
|                 f"Failed to parse CONTAINER_LOG_EPOCH of {identifier} container, got: {text}", | ||||
|                 _LOGGER.error, | ||||
|             ) from err | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Supervisor Add-on ingress service.""" | ||||
|  | ||||
| import asyncio | ||||
| from ipaddress import ip_address | ||||
| import logging | ||||
| @@ -82,7 +83,7 @@ class APIIngress(CoreSysAttributes): | ||||
|  | ||||
|     def _extract_addon(self, request: web.Request) -> Addon: | ||||
|         """Return addon, throw an exception it it doesn't exist.""" | ||||
|         token = request.match_info.get("token") | ||||
|         token = request.match_info["token"] | ||||
|  | ||||
|         # Find correct add-on | ||||
|         addon = self.sys_ingress.get(token) | ||||
| @@ -131,7 +132,7 @@ class APIIngress(CoreSysAttributes): | ||||
|  | ||||
|     @api_process | ||||
|     @require_home_assistant | ||||
|     async def validate_session(self, request: web.Request) -> dict[str, Any]: | ||||
|     async def validate_session(self, request: web.Request) -> None: | ||||
|         """Validate session and extending how long it's valid for.""" | ||||
|         data = await api_validate(VALIDATE_SESSION_DATA, request) | ||||
|  | ||||
| @@ -146,14 +147,14 @@ class APIIngress(CoreSysAttributes): | ||||
|         """Route data to Supervisor ingress service.""" | ||||
|  | ||||
|         # Check Ingress Session | ||||
|         session = request.cookies.get(COOKIE_INGRESS) | ||||
|         session = request.cookies.get(COOKIE_INGRESS, "") | ||||
|         if not self.sys_ingress.validate_session(session): | ||||
|             _LOGGER.warning("No valid ingress session %s", session) | ||||
|             raise HTTPUnauthorized() | ||||
|  | ||||
|         # Process requests | ||||
|         addon = self._extract_addon(request) | ||||
|         path = request.match_info.get("path") | ||||
|         path = request.match_info.get("path", "") | ||||
|         session_data = self.sys_ingress.get_session_data(session) | ||||
|         try: | ||||
|             # Websocket | ||||
| @@ -182,7 +183,7 @@ class APIIngress(CoreSysAttributes): | ||||
|                 for proto in request.headers[hdrs.SEC_WEBSOCKET_PROTOCOL].split(",") | ||||
|             ] | ||||
|         else: | ||||
|             req_protocols = () | ||||
|             req_protocols = [] | ||||
|  | ||||
|         ws_server = web.WebSocketResponse( | ||||
|             protocols=req_protocols, autoclose=False, autoping=False | ||||
| @@ -198,21 +199,25 @@ class APIIngress(CoreSysAttributes): | ||||
|             url = f"{url}?{request.query_string}" | ||||
|  | ||||
|         # Start proxy | ||||
|         async with self.sys_websession.ws_connect( | ||||
|             url, | ||||
|             headers=source_header, | ||||
|             protocols=req_protocols, | ||||
|             autoclose=False, | ||||
|             autoping=False, | ||||
|         ) as ws_client: | ||||
|             # Proxy requests | ||||
|             await asyncio.wait( | ||||
|                 [ | ||||
|                     self.sys_create_task(_websocket_forward(ws_server, ws_client)), | ||||
|                     self.sys_create_task(_websocket_forward(ws_client, ws_server)), | ||||
|                 ], | ||||
|                 return_when=asyncio.FIRST_COMPLETED, | ||||
|             ) | ||||
|         try: | ||||
|             _LOGGER.debug("Proxing WebSocket to %s, upstream url: %s", addon.slug, url) | ||||
|             async with self.sys_websession.ws_connect( | ||||
|                 url, | ||||
|                 headers=source_header, | ||||
|                 protocols=req_protocols, | ||||
|                 autoclose=False, | ||||
|                 autoping=False, | ||||
|             ) as ws_client: | ||||
|                 # Proxy requests | ||||
|                 await asyncio.wait( | ||||
|                     [ | ||||
|                         self.sys_create_task(_websocket_forward(ws_server, ws_client)), | ||||
|                         self.sys_create_task(_websocket_forward(ws_client, ws_server)), | ||||
|                     ], | ||||
|                     return_when=asyncio.FIRST_COMPLETED, | ||||
|                 ) | ||||
|         except TimeoutError: | ||||
|             _LOGGER.warning("WebSocket proxy to %s timed out", addon.slug) | ||||
|  | ||||
|         return ws_server | ||||
|  | ||||
| @@ -276,14 +281,16 @@ class APIIngress(CoreSysAttributes): | ||||
|             response.content_type = content_type | ||||
|  | ||||
|             try: | ||||
|                 response.headers["X-Accel-Buffering"] = "no" | ||||
|                 await response.prepare(request) | ||||
|                 async for data in result.content.iter_chunked(4096): | ||||
|                 async for data, _ in result.content.iter_chunks(): | ||||
|                     await response.write(data) | ||||
|  | ||||
|             except ( | ||||
|                 aiohttp.ClientError, | ||||
|                 aiohttp.ClientPayloadError, | ||||
|                 ConnectionResetError, | ||||
|                 ConnectionError, | ||||
|             ) as err: | ||||
|                 _LOGGER.error("Stream error with %s: %s", url, err) | ||||
|  | ||||
| @@ -307,9 +314,9 @@ class APIIngress(CoreSysAttributes): | ||||
|  | ||||
| def _init_header( | ||||
|     request: web.Request, addon: Addon, session_data: IngressSessionData | None | ||||
| ) -> CIMultiDict | dict[str, str]: | ||||
| ) -> CIMultiDict[str]: | ||||
|     """Create initial header.""" | ||||
|     headers = {} | ||||
|     headers = CIMultiDict[str]() | ||||
|  | ||||
|     if session_data is not None: | ||||
|         headers[HEADER_REMOTE_USER_ID] = session_data.user.id | ||||
| @@ -335,19 +342,20 @@ def _init_header( | ||||
|             istr(HEADER_REMOTE_USER_DISPLAY_NAME), | ||||
|         ): | ||||
|             continue | ||||
|         headers[name] = value | ||||
|         headers.add(name, value) | ||||
|  | ||||
|     # Update X-Forwarded-For | ||||
|     forward_for = request.headers.get(hdrs.X_FORWARDED_FOR) | ||||
|     connected_ip = ip_address(request.transport.get_extra_info("peername")[0]) | ||||
|     headers[hdrs.X_FORWARDED_FOR] = f"{forward_for}, {connected_ip!s}" | ||||
|     if request.transport: | ||||
|         forward_for = request.headers.get(hdrs.X_FORWARDED_FOR) | ||||
|         connected_ip = ip_address(request.transport.get_extra_info("peername")[0]) | ||||
|         headers[hdrs.X_FORWARDED_FOR] = f"{forward_for}, {connected_ip!s}" | ||||
|  | ||||
|     return headers | ||||
|  | ||||
|  | ||||
| def _response_header(response: aiohttp.ClientResponse) -> dict[str, str]: | ||||
| def _response_header(response: aiohttp.ClientResponse) -> CIMultiDict[str]: | ||||
|     """Create response header.""" | ||||
|     headers = {} | ||||
|     headers = CIMultiDict[str]() | ||||
|  | ||||
|     for name, value in response.headers.items(): | ||||
|         if name in ( | ||||
| @@ -357,7 +365,7 @@ def _response_header(response: aiohttp.ClientResponse) -> dict[str, str]: | ||||
|             hdrs.CONTENT_ENCODING, | ||||
|         ): | ||||
|             continue | ||||
|         headers[name] = value | ||||
|         headers.add(name, value) | ||||
|  | ||||
|     return headers | ||||
|  | ||||
| @@ -383,9 +391,9 @@ async def _websocket_forward(ws_from, ws_to): | ||||
|             elif msg.type == aiohttp.WSMsgType.BINARY: | ||||
|                 await ws_to.send_bytes(msg.data) | ||||
|             elif msg.type == aiohttp.WSMsgType.PING: | ||||
|                 await ws_to.ping() | ||||
|                 await ws_to.ping(msg.data) | ||||
|             elif msg.type == aiohttp.WSMsgType.PONG: | ||||
|                 await ws_to.pong() | ||||
|                 await ws_to.pong(msg.data) | ||||
|             elif ws_to.closed: | ||||
|                 await ws_to.close(code=ws_to.close_code, message=msg.extra) | ||||
|     except RuntimeError: | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Init file for Supervisor Jobs RESTful API.""" | ||||
|  | ||||
| import logging | ||||
| from typing import Any | ||||
|  | ||||
| @@ -6,6 +7,7 @@ from aiohttp import web | ||||
| import voluptuous as vol | ||||
|  | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APIError, APINotFound, JobNotFound | ||||
| from ..jobs import SupervisorJob | ||||
| from ..jobs.const import ATTR_IGNORE_CONDITIONS, JobCondition | ||||
| from .const import ATTR_JOBS | ||||
| @@ -21,10 +23,24 @@ SCHEMA_OPTIONS = vol.Schema( | ||||
| class APIJobs(CoreSysAttributes): | ||||
|     """Handle RESTful API for OS functions.""" | ||||
|  | ||||
|     def _list_jobs(self) -> list[dict[str, Any]]: | ||||
|         """Return current job tree.""" | ||||
|     def _extract_job(self, request: web.Request) -> SupervisorJob: | ||||
|         """Extract job from request or raise.""" | ||||
|         try: | ||||
|             return self.sys_jobs.get_job(request.match_info["uuid"]) | ||||
|         except JobNotFound: | ||||
|             raise APINotFound("Job does not exist") from None | ||||
|  | ||||
|     def _list_jobs(self, start: SupervisorJob | None = None) -> list[dict[str, Any]]: | ||||
|         """Return current job tree. | ||||
|  | ||||
|         Jobs are added to cache as they are created so by default they are in oldest to newest. | ||||
|         This is correct ordering for child jobs as it makes logical sense to present those in | ||||
|         the order they occurred within the parent. For the list as a whole, sort from newest | ||||
|         to oldest as its likely any client is most interested in the newer ones. | ||||
|         """ | ||||
|         # Initially sort oldest to newest so all child lists end up in correct order | ||||
|         jobs_by_parent: dict[str | None, list[SupervisorJob]] = {} | ||||
|         for job in self.sys_jobs.jobs: | ||||
|         for job in sorted(self.sys_jobs.jobs): | ||||
|             if job.internal: | ||||
|                 continue | ||||
|  | ||||
| @@ -33,10 +49,16 @@ class APIJobs(CoreSysAttributes): | ||||
|             else: | ||||
|                 jobs_by_parent[job.parent_id].append(job) | ||||
|  | ||||
|         # After parent-child organization, sort the root jobs only from newest to oldest | ||||
|         job_list: list[dict[str, Any]] = [] | ||||
|         queue: list[tuple[list[dict[str, Any]], SupervisorJob]] = [ | ||||
|             (job_list, job) for job in jobs_by_parent.get(None, []) | ||||
|         ] | ||||
|         queue: list[tuple[list[dict[str, Any]], SupervisorJob]] = ( | ||||
|             [(job_list, start)] | ||||
|             if start | ||||
|             else [ | ||||
|                 (job_list, job) | ||||
|                 for job in sorted(jobs_by_parent.get(None, []), reverse=True) | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
|         while queue: | ||||
|             (current_list, current_job) = queue.pop(0) | ||||
| @@ -49,7 +71,10 @@ class APIJobs(CoreSysAttributes): | ||||
|  | ||||
|             if current_job.uuid in jobs_by_parent: | ||||
|                 queue.extend( | ||||
|                     [(child_jobs, job) for job in jobs_by_parent.get(current_job.uuid)] | ||||
|                     [ | ||||
|                         (child_jobs, job) | ||||
|                         for job in jobs_by_parent.get(current_job.uuid, []) | ||||
|                     ] | ||||
|                 ) | ||||
|  | ||||
|         return job_list | ||||
| @@ -70,11 +95,27 @@ class APIJobs(CoreSysAttributes): | ||||
|         if ATTR_IGNORE_CONDITIONS in body: | ||||
|             self.sys_jobs.ignore_conditions = body[ATTR_IGNORE_CONDITIONS] | ||||
|  | ||||
|         self.sys_jobs.save_data() | ||||
|         await self.sys_jobs.save_data() | ||||
|  | ||||
|         await self.sys_resolution.evaluate.evaluate_system() | ||||
|  | ||||
|     @api_process | ||||
|     async def reset(self, request: web.Request) -> None: | ||||
|         """Reset options for JobManager.""" | ||||
|         self.sys_jobs.reset_data() | ||||
|         await self.sys_jobs.reset_data() | ||||
|  | ||||
|     @api_process | ||||
|     async def job_info(self, request: web.Request) -> dict[str, Any]: | ||||
|         """Get details of a job by ID.""" | ||||
|         job = self._extract_job(request) | ||||
|         return self._list_jobs(job)[0] | ||||
|  | ||||
|     @api_process | ||||
|     async def remove_job(self, request: web.Request) -> None: | ||||
|         """Remove a completed job.""" | ||||
|         job = self._extract_job(request) | ||||
|  | ||||
|         if not job.done: | ||||
|             raise APIError(f"Job {job.uuid} is not done!") | ||||
|  | ||||
|         self.sys_jobs.remove_job(job) | ||||
|   | ||||
| @@ -1,13 +1,17 @@ | ||||
| """Handle security part of this API.""" | ||||
|  | ||||
| from collections.abc import Callable | ||||
| import logging | ||||
| import re | ||||
| from typing import Final | ||||
| from urllib.parse import unquote | ||||
|  | ||||
| from aiohttp.web import Request, RequestHandler, Response, middleware | ||||
| from aiohttp.web import Request, Response, middleware | ||||
| from aiohttp.web_exceptions import HTTPBadRequest, HTTPForbidden, HTTPUnauthorized | ||||
| from awesomeversion import AwesomeVersion | ||||
|  | ||||
| from supervisor.homeassistant.const import LANDINGPAGE | ||||
|  | ||||
| from ...addons.const import RE_SLUG | ||||
| from ...const import ( | ||||
|     REQUEST_FROM, | ||||
| @@ -16,11 +20,11 @@ from ...const import ( | ||||
|     ROLE_DEFAULT, | ||||
|     ROLE_HOMEASSISTANT, | ||||
|     ROLE_MANAGER, | ||||
|     CoreState, | ||||
|     VALID_API_STATES, | ||||
| ) | ||||
| from ...coresys import CoreSys, CoreSysAttributes | ||||
| from ...utils import version_is_new_enough | ||||
| from ..utils import api_return_error, excract_supervisor_token | ||||
| from ..utils import api_return_error, extract_supervisor_token | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
| _CORE_VERSION: Final = AwesomeVersion("2023.3.4") | ||||
| @@ -77,6 +81,13 @@ ADDONS_API_BYPASS: Final = re.compile( | ||||
|     r")$" | ||||
| ) | ||||
|  | ||||
| # Home Assistant only | ||||
| CORE_ONLY_PATHS: Final = re.compile( | ||||
|     r"^(?:" | ||||
|     r"/addons/" + RE_SLUG + "/sys_options" | ||||
|     r")$" | ||||
| ) | ||||
|  | ||||
| # Policy role add-on API access | ||||
| ADDONS_ROLE_ACCESS: dict[str, re.Pattern] = { | ||||
|     ROLE_DEFAULT: re.compile( | ||||
| @@ -103,6 +114,8 @@ ADDONS_ROLE_ACCESS: dict[str, re.Pattern] = { | ||||
|         r"|/addons(?:/" + RE_SLUG + r"/(?!security).+|/reload)?" | ||||
|         r"|/audio/.+" | ||||
|         r"|/auth/cache" | ||||
|         r"|/available_updates" | ||||
|         r"|/backups.*" | ||||
|         r"|/cli/.+" | ||||
|         r"|/core/.+" | ||||
|         r"|/dns/.+" | ||||
| @@ -112,16 +125,17 @@ ADDONS_ROLE_ACCESS: dict[str, re.Pattern] = { | ||||
|         r"|/hassos/.+" | ||||
|         r"|/homeassistant/.+" | ||||
|         r"|/host/.+" | ||||
|         r"|/mounts.*" | ||||
|         r"|/multicast/.+" | ||||
|         r"|/network/.+" | ||||
|         r"|/observer/.+" | ||||
|         r"|/os/.+" | ||||
|         r"|/os/(?!datadisk/wipe).+" | ||||
|         r"|/refresh_updates" | ||||
|         r"|/resolution/.+" | ||||
|         r"|/backups.*" | ||||
|         r"|/security/.+" | ||||
|         r"|/snapshots.*" | ||||
|         r"|/store.*" | ||||
|         r"|/supervisor/.+" | ||||
|         r"|/security/.+" | ||||
|         r")$" | ||||
|     ), | ||||
|     ROLE_ADMIN: re.compile( | ||||
| @@ -166,9 +180,7 @@ class SecurityMiddleware(CoreSysAttributes): | ||||
|         return unquoted | ||||
|  | ||||
|     @middleware | ||||
|     async def block_bad_requests( | ||||
|         self, request: Request, handler: RequestHandler | ||||
|     ) -> Response: | ||||
|     async def block_bad_requests(self, request: Request, handler: Callable) -> Response: | ||||
|         """Process request and tblock commonly known exploit attempts.""" | ||||
|         if FILTERS.search(self._recursive_unquote(request.path)): | ||||
|             _LOGGER.warning( | ||||
| @@ -186,15 +198,9 @@ class SecurityMiddleware(CoreSysAttributes): | ||||
|         return await handler(request) | ||||
|  | ||||
|     @middleware | ||||
|     async def system_validation( | ||||
|         self, request: Request, handler: RequestHandler | ||||
|     ) -> Response: | ||||
|     async def system_validation(self, request: Request, handler: Callable) -> Response: | ||||
|         """Check if core is ready to response.""" | ||||
|         if self.sys_core.state not in ( | ||||
|             CoreState.STARTUP, | ||||
|             CoreState.RUNNING, | ||||
|             CoreState.FREEZE, | ||||
|         ): | ||||
|         if self.sys_core.state not in VALID_API_STATES: | ||||
|             return api_return_error( | ||||
|                 message=f"System is not ready with state: {self.sys_core.state}" | ||||
|             ) | ||||
| @@ -202,12 +208,10 @@ class SecurityMiddleware(CoreSysAttributes): | ||||
|         return await handler(request) | ||||
|  | ||||
|     @middleware | ||||
|     async def token_validation( | ||||
|         self, request: Request, handler: RequestHandler | ||||
|     ) -> Response: | ||||
|     async def token_validation(self, request: Request, handler: Callable) -> Response: | ||||
|         """Check security access of this layer.""" | ||||
|         request_from = None | ||||
|         supervisor_token = excract_supervisor_token(request) | ||||
|         request_from: CoreSysAttributes | None = None | ||||
|         supervisor_token = extract_supervisor_token(request) | ||||
|  | ||||
|         # Blacklist | ||||
|         if BLACKLIST.match(request.path): | ||||
| @@ -229,6 +233,9 @@ class SecurityMiddleware(CoreSysAttributes): | ||||
|         if supervisor_token == self.sys_homeassistant.supervisor_token: | ||||
|             _LOGGER.debug("%s access from Home Assistant", request.path) | ||||
|             request_from = self.sys_homeassistant | ||||
|         elif CORE_ONLY_PATHS.match(request.path): | ||||
|             _LOGGER.warning("Attempted access to %s from client besides Home Assistant") | ||||
|             raise HTTPForbidden() | ||||
|  | ||||
|         # Host | ||||
|         if supervisor_token == self.sys_plugins.cli.supervisor_token: | ||||
| @@ -272,10 +279,12 @@ class SecurityMiddleware(CoreSysAttributes): | ||||
|         raise HTTPForbidden() | ||||
|  | ||||
|     @middleware | ||||
|     async def core_proxy(self, request: Request, handler: RequestHandler) -> Response: | ||||
|     async def core_proxy(self, request: Request, handler: Callable) -> Response: | ||||
|         """Validate user from Core API proxy.""" | ||||
|         if request[REQUEST_FROM] != self.sys_homeassistant or version_is_new_enough( | ||||
|             self.sys_homeassistant.version, _CORE_VERSION | ||||
|         if ( | ||||
|             request[REQUEST_FROM] != self.sys_homeassistant | ||||
|             or self.sys_homeassistant.version == LANDINGPAGE | ||||
|             or version_is_new_enough(self.sys_homeassistant.version, _CORE_VERSION) | ||||
|         ): | ||||
|             return await handler(request) | ||||
|  | ||||
|   | ||||
| @@ -1,17 +1,17 @@ | ||||
| """Inits file for supervisor mounts REST API.""" | ||||
|  | ||||
| from typing import Any | ||||
| from typing import Any, cast | ||||
|  | ||||
| from aiohttp import web | ||||
| import voluptuous as vol | ||||
|  | ||||
| from ..const import ATTR_NAME, ATTR_STATE | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APIError | ||||
| from ..exceptions import APIError, APINotFound | ||||
| from ..mounts.const import ATTR_DEFAULT_BACKUP_MOUNT, MountUsage | ||||
| from ..mounts.mount import Mount | ||||
| from ..mounts.validate import SCHEMA_MOUNT_CONFIG | ||||
| from .const import ATTR_MOUNTS | ||||
| from ..mounts.validate import SCHEMA_MOUNT_CONFIG, MountData | ||||
| from .const import ATTR_MOUNTS, ATTR_USER_PATH | ||||
| from .utils import api_process, api_validate | ||||
|  | ||||
| SCHEMA_OPTIONS = vol.Schema( | ||||
| @@ -24,6 +24,13 @@ SCHEMA_OPTIONS = vol.Schema( | ||||
| class APIMounts(CoreSysAttributes): | ||||
|     """Handle REST API for mounting options.""" | ||||
|  | ||||
|     def _extract_mount(self, request: web.Request) -> Mount: | ||||
|         """Extract mount from request or raise.""" | ||||
|         name = request.match_info["mount"] | ||||
|         if name not in self.sys_mounts: | ||||
|             raise APINotFound(f"No mount exists with name {name}") | ||||
|         return self.sys_mounts.get(name) | ||||
|  | ||||
|     @api_process | ||||
|     async def info(self, request: web.Request) -> dict[str, Any]: | ||||
|         """Return MountManager info.""" | ||||
| @@ -32,7 +39,13 @@ class APIMounts(CoreSysAttributes): | ||||
|             if self.sys_mounts.default_backup_mount | ||||
|             else None, | ||||
|             ATTR_MOUNTS: [ | ||||
|                 mount.to_dict() | {ATTR_STATE: mount.state} | ||||
|                 mount.to_dict() | ||||
|                 | { | ||||
|                     ATTR_STATE: mount.state, | ||||
|                     ATTR_USER_PATH: mount.container_where.as_posix() | ||||
|                     if mount.container_where | ||||
|                     else None, | ||||
|                 } | ||||
|                 for mount in self.sys_mounts.mounts | ||||
|             ], | ||||
|         } | ||||
| @@ -53,15 +66,15 @@ class APIMounts(CoreSysAttributes): | ||||
|             else: | ||||
|                 self.sys_mounts.default_backup_mount = mount | ||||
|  | ||||
|         self.sys_mounts.save_data() | ||||
|         await self.sys_mounts.save_data() | ||||
|  | ||||
|     @api_process | ||||
|     async def create_mount(self, request: web.Request) -> None: | ||||
|         """Create a new mount in supervisor.""" | ||||
|         body = await api_validate(SCHEMA_MOUNT_CONFIG, request) | ||||
|         body = cast(MountData, await api_validate(SCHEMA_MOUNT_CONFIG, request)) | ||||
|  | ||||
|         if body[ATTR_NAME] in self.sys_mounts: | ||||
|             raise APIError(f"A mount already exists with name {body[ATTR_NAME]}") | ||||
|         if body["name"] in self.sys_mounts: | ||||
|             raise APIError(f"A mount already exists with name {body['name']}") | ||||
|  | ||||
|         mount = Mount.from_dict(self.coresys, body) | ||||
|         await self.sys_mounts.create_mount(mount) | ||||
| @@ -74,19 +87,20 @@ class APIMounts(CoreSysAttributes): | ||||
|             if not self.sys_mounts.default_backup_mount: | ||||
|                 self.sys_mounts.default_backup_mount = mount | ||||
|  | ||||
|         self.sys_mounts.save_data() | ||||
|         await self.sys_mounts.save_data() | ||||
|  | ||||
|     @api_process | ||||
|     async def update_mount(self, request: web.Request) -> None: | ||||
|         """Update an existing mount in supervisor.""" | ||||
|         name = request.match_info.get("mount") | ||||
|         current = self._extract_mount(request) | ||||
|         name_schema = vol.Schema( | ||||
|             {vol.Optional(ATTR_NAME, default=name): name}, extra=vol.ALLOW_EXTRA | ||||
|             {vol.Optional(ATTR_NAME, default=current.name): current.name}, | ||||
|             extra=vol.ALLOW_EXTRA, | ||||
|         ) | ||||
|         body = cast( | ||||
|             MountData, | ||||
|             await api_validate(vol.All(name_schema, SCHEMA_MOUNT_CONFIG), request), | ||||
|         ) | ||||
|         body = await api_validate(vol.All(name_schema, SCHEMA_MOUNT_CONFIG), request) | ||||
|  | ||||
|         if name not in self.sys_mounts: | ||||
|             raise APIError(f"No mount exists with name {name}") | ||||
|  | ||||
|         mount = Mount.from_dict(self.coresys, body) | ||||
|         await self.sys_mounts.create_mount(mount) | ||||
| @@ -99,26 +113,26 @@ class APIMounts(CoreSysAttributes): | ||||
|         elif self.sys_mounts.default_backup_mount == mount: | ||||
|             self.sys_mounts.default_backup_mount = None | ||||
|  | ||||
|         self.sys_mounts.save_data() | ||||
|         await self.sys_mounts.save_data() | ||||
|  | ||||
|     @api_process | ||||
|     async def delete_mount(self, request: web.Request) -> None: | ||||
|         """Delete an existing mount in supervisor.""" | ||||
|         name = request.match_info.get("mount") | ||||
|         mount = await self.sys_mounts.remove_mount(name) | ||||
|         current = self._extract_mount(request) | ||||
|         mount = await self.sys_mounts.remove_mount(current.name) | ||||
|  | ||||
|         # If it was a backup mount, reload backups | ||||
|         if mount.usage == MountUsage.BACKUP: | ||||
|             self.sys_create_task(self.sys_backups.reload()) | ||||
|  | ||||
|         self.sys_mounts.save_data() | ||||
|         await self.sys_mounts.save_data() | ||||
|  | ||||
|     @api_process | ||||
|     async def reload_mount(self, request: web.Request) -> None: | ||||
|         """Reload an existing mount in supervisor.""" | ||||
|         name = request.match_info.get("mount") | ||||
|         await self.sys_mounts.reload_mount(name) | ||||
|         mount = self._extract_mount(request) | ||||
|         await self.sys_mounts.reload_mount(mount.name) | ||||
|  | ||||
|         # If it's a backup mount, reload backups | ||||
|         if self.sys_mounts.get(name).usage == MountUsage.BACKUP: | ||||
|         if mount.usage == MountUsage.BACKUP: | ||||
|             self.sys_create_task(self.sys_backups.reload()) | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Init file for Supervisor Multicast RESTful API.""" | ||||
|  | ||||
| import asyncio | ||||
| from collections.abc import Awaitable | ||||
| import logging | ||||
| @@ -23,8 +24,7 @@ from ..const import ( | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APIError | ||||
| from ..validate import version_tag | ||||
| from .const import CONTENT_TYPE_BINARY | ||||
| from .utils import api_process, api_process_raw, api_validate | ||||
| from .utils import api_process, api_validate | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
|  | ||||
| @@ -69,11 +69,6 @@ class APIMulticast(CoreSysAttributes): | ||||
|             raise APIError(f"Version {version} is already in use") | ||||
|         await asyncio.shield(self.sys_plugins.multicast.update(version)) | ||||
|  | ||||
|     @api_process_raw(CONTENT_TYPE_BINARY) | ||||
|     def logs(self, request: web.Request) -> Awaitable[bytes]: | ||||
|         """Return Multicast Docker logs.""" | ||||
|         return self.sys_plugins.multicast.logs() | ||||
|  | ||||
|     @api_process | ||||
|     def restart(self, request: web.Request) -> Awaitable[None]: | ||||
|         """Restart Multicast plugin.""" | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| """REST API for network.""" | ||||
|  | ||||
| import asyncio | ||||
| from collections.abc import Awaitable | ||||
| from dataclasses import replace | ||||
| from ipaddress import ip_address, ip_interface | ||||
| from ipaddress import IPv4Address, IPv4Interface, IPv6Address, IPv6Interface | ||||
| from typing import Any | ||||
|  | ||||
| from aiohttp import web | ||||
| @@ -10,6 +10,7 @@ import voluptuous as vol | ||||
|  | ||||
| from ..const import ( | ||||
|     ATTR_ACCESSPOINTS, | ||||
|     ATTR_ADDR_GEN_MODE, | ||||
|     ATTR_ADDRESS, | ||||
|     ATTR_AUTH, | ||||
|     ATTR_CONNECTED, | ||||
| @@ -22,9 +23,12 @@ from ..const import ( | ||||
|     ATTR_ID, | ||||
|     ATTR_INTERFACE, | ||||
|     ATTR_INTERFACES, | ||||
|     ATTR_IP6_PRIVACY, | ||||
|     ATTR_IPV4, | ||||
|     ATTR_IPV6, | ||||
|     ATTR_LLMNR, | ||||
|     ATTR_MAC, | ||||
|     ATTR_MDNS, | ||||
|     ATTR_METHOD, | ||||
|     ATTR_MODE, | ||||
|     ATTR_NAMESERVERS, | ||||
| @@ -38,28 +42,44 @@ from ..const import ( | ||||
|     ATTR_TYPE, | ||||
|     ATTR_VLAN, | ||||
|     ATTR_WIFI, | ||||
|     DOCKER_IPV4_NETWORK_MASK, | ||||
|     DOCKER_NETWORK, | ||||
|     DOCKER_NETWORK_MASK, | ||||
| ) | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APIError, HostNetworkNotFound | ||||
| from ..exceptions import APIError, APINotFound, HostNetworkNotFound | ||||
| from ..host.configuration import ( | ||||
|     AccessPoint, | ||||
|     Interface, | ||||
|     InterfaceAddrGenMode, | ||||
|     InterfaceIp6Privacy, | ||||
|     InterfaceMethod, | ||||
|     Ip6Setting, | ||||
|     IpConfig, | ||||
|     IpSetting, | ||||
|     MulticastDnsMode, | ||||
|     VlanConfig, | ||||
|     WifiConfig, | ||||
| ) | ||||
| from ..host.const import AuthMethod, InterfaceType, WifiMode | ||||
| from .utils import api_process, api_validate | ||||
|  | ||||
| _SCHEMA_IP_CONFIG = vol.Schema( | ||||
| _SCHEMA_IPV4_CONFIG = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_ADDRESS): [vol.Coerce(ip_interface)], | ||||
|         vol.Optional(ATTR_ADDRESS): [vol.Coerce(IPv4Interface)], | ||||
|         vol.Optional(ATTR_METHOD): vol.Coerce(InterfaceMethod), | ||||
|         vol.Optional(ATTR_GATEWAY): vol.Coerce(ip_address), | ||||
|         vol.Optional(ATTR_NAMESERVERS): [vol.Coerce(ip_address)], | ||||
|         vol.Optional(ATTR_GATEWAY): vol.Coerce(IPv4Address), | ||||
|         vol.Optional(ATTR_NAMESERVERS): [vol.Coerce(IPv4Address)], | ||||
|     } | ||||
| ) | ||||
|  | ||||
| _SCHEMA_IPV6_CONFIG = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_ADDRESS): [vol.Coerce(IPv6Interface)], | ||||
|         vol.Optional(ATTR_METHOD): vol.Coerce(InterfaceMethod), | ||||
|         vol.Optional(ATTR_ADDR_GEN_MODE): vol.Coerce(InterfaceAddrGenMode), | ||||
|         vol.Optional(ATTR_IP6_PRIVACY): vol.Coerce(InterfaceIp6Privacy), | ||||
|         vol.Optional(ATTR_GATEWAY): vol.Coerce(IPv6Address), | ||||
|         vol.Optional(ATTR_NAMESERVERS): [vol.Coerce(IPv6Address)], | ||||
|     } | ||||
| ) | ||||
|  | ||||
| @@ -76,18 +96,33 @@ _SCHEMA_WIFI_CONFIG = vol.Schema( | ||||
| # pylint: disable=no-value-for-parameter | ||||
| SCHEMA_UPDATE = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_IPV4): _SCHEMA_IP_CONFIG, | ||||
|         vol.Optional(ATTR_IPV6): _SCHEMA_IP_CONFIG, | ||||
|         vol.Optional(ATTR_IPV4): _SCHEMA_IPV4_CONFIG, | ||||
|         vol.Optional(ATTR_IPV6): _SCHEMA_IPV6_CONFIG, | ||||
|         vol.Optional(ATTR_WIFI): _SCHEMA_WIFI_CONFIG, | ||||
|         vol.Optional(ATTR_ENABLED): vol.Boolean(), | ||||
|         vol.Optional(ATTR_MDNS): vol.Coerce(MulticastDnsMode), | ||||
|         vol.Optional(ATTR_LLMNR): vol.Coerce(MulticastDnsMode), | ||||
|     } | ||||
| ) | ||||
|  | ||||
|  | ||||
| def ipconfig_struct(config: IpConfig) -> dict[str, Any]: | ||||
|     """Return a dict with information about ip configuration.""" | ||||
| def ip4config_struct(config: IpConfig, setting: IpSetting) -> dict[str, Any]: | ||||
|     """Return a dict with information about IPv4 configuration.""" | ||||
|     return { | ||||
|         ATTR_METHOD: config.method, | ||||
|         ATTR_METHOD: setting.method, | ||||
|         ATTR_ADDRESS: [address.with_prefixlen for address in config.address], | ||||
|         ATTR_NAMESERVERS: [str(address) for address in config.nameservers], | ||||
|         ATTR_GATEWAY: str(config.gateway) if config.gateway else None, | ||||
|         ATTR_READY: config.ready, | ||||
|     } | ||||
|  | ||||
|  | ||||
| def ip6config_struct(config: IpConfig, setting: Ip6Setting) -> dict[str, Any]: | ||||
|     """Return a dict with information about IPv6 configuration.""" | ||||
|     return { | ||||
|         ATTR_METHOD: setting.method, | ||||
|         ATTR_ADDR_GEN_MODE: setting.addr_gen_mode, | ||||
|         ATTR_IP6_PRIVACY: setting.ip6_privacy, | ||||
|         ATTR_ADDRESS: [address.with_prefixlen for address in config.address], | ||||
|         ATTR_NAMESERVERS: [str(address) for address in config.nameservers], | ||||
|         ATTR_GATEWAY: str(config.gateway) if config.gateway else None, | ||||
| @@ -122,10 +157,16 @@ def interface_struct(interface: Interface) -> dict[str, Any]: | ||||
|         ATTR_CONNECTED: interface.connected, | ||||
|         ATTR_PRIMARY: interface.primary, | ||||
|         ATTR_MAC: interface.mac, | ||||
|         ATTR_IPV4: ipconfig_struct(interface.ipv4) if interface.ipv4 else None, | ||||
|         ATTR_IPV6: ipconfig_struct(interface.ipv6) if interface.ipv6 else None, | ||||
|         ATTR_IPV4: ip4config_struct(interface.ipv4, interface.ipv4setting) | ||||
|         if interface.ipv4 and interface.ipv4setting | ||||
|         else None, | ||||
|         ATTR_IPV6: ip6config_struct(interface.ipv6, interface.ipv6setting) | ||||
|         if interface.ipv6 and interface.ipv6setting | ||||
|         else None, | ||||
|         ATTR_WIFI: wifi_struct(interface.wifi) if interface.wifi else None, | ||||
|         ATTR_VLAN: vlan_struct(interface.vlan) if interface.vlan else None, | ||||
|         ATTR_MDNS: interface.mdns, | ||||
|         ATTR_LLMNR: interface.llmnr, | ||||
|     } | ||||
|  | ||||
|  | ||||
| @@ -157,7 +198,7 @@ class APINetwork(CoreSysAttributes): | ||||
|             except HostNetworkNotFound: | ||||
|                 pass | ||||
|  | ||||
|         raise APIError(f"Interface {name} does not exist") from None | ||||
|         raise APINotFound(f"Interface {name} does not exist") from None | ||||
|  | ||||
|     @api_process | ||||
|     async def info(self, request: web.Request) -> dict[str, Any]: | ||||
| @@ -169,7 +210,7 @@ class APINetwork(CoreSysAttributes): | ||||
|             ], | ||||
|             ATTR_DOCKER: { | ||||
|                 ATTR_INTERFACE: DOCKER_NETWORK, | ||||
|                 ATTR_ADDRESS: str(DOCKER_NETWORK_MASK), | ||||
|                 ATTR_ADDRESS: str(DOCKER_IPV4_NETWORK_MASK), | ||||
|                 ATTR_GATEWAY: str(self.sys_docker.network.gateway), | ||||
|                 ATTR_DNS: str(self.sys_docker.network.dns), | ||||
|             }, | ||||
| @@ -180,14 +221,14 @@ class APINetwork(CoreSysAttributes): | ||||
|     @api_process | ||||
|     async def interface_info(self, request: web.Request) -> dict[str, Any]: | ||||
|         """Return network information for a interface.""" | ||||
|         interface = self._get_interface(request.match_info.get(ATTR_INTERFACE)) | ||||
|         interface = self._get_interface(request.match_info[ATTR_INTERFACE]) | ||||
|  | ||||
|         return interface_struct(interface) | ||||
|  | ||||
|     @api_process | ||||
|     async def interface_update(self, request: web.Request) -> None: | ||||
|         """Update the configuration of an interface.""" | ||||
|         interface = self._get_interface(request.match_info.get(ATTR_INTERFACE)) | ||||
|         interface = self._get_interface(request.match_info[ATTR_INTERFACE]) | ||||
|  | ||||
|         # Validate data | ||||
|         body = await api_validate(SCHEMA_UPDATE, request) | ||||
| @@ -197,27 +238,39 @@ class APINetwork(CoreSysAttributes): | ||||
|         # Apply config | ||||
|         for key, config in body.items(): | ||||
|             if key == ATTR_IPV4: | ||||
|                 interface.ipv4 = replace( | ||||
|                     interface.ipv4 | ||||
|                     or IpConfig(InterfaceMethod.STATIC, [], None, [], None), | ||||
|                     **config, | ||||
|                 interface.ipv4setting = IpSetting( | ||||
|                     method=config.get(ATTR_METHOD, InterfaceMethod.STATIC), | ||||
|                     address=config.get(ATTR_ADDRESS, []), | ||||
|                     gateway=config.get(ATTR_GATEWAY), | ||||
|                     nameservers=config.get(ATTR_NAMESERVERS, []), | ||||
|                 ) | ||||
|             elif key == ATTR_IPV6: | ||||
|                 interface.ipv6 = replace( | ||||
|                     interface.ipv6 | ||||
|                     or IpConfig(InterfaceMethod.STATIC, [], None, [], None), | ||||
|                     **config, | ||||
|                 interface.ipv6setting = Ip6Setting( | ||||
|                     method=config.get(ATTR_METHOD, InterfaceMethod.STATIC), | ||||
|                     addr_gen_mode=config.get( | ||||
|                         ATTR_ADDR_GEN_MODE, InterfaceAddrGenMode.DEFAULT | ||||
|                     ), | ||||
|                     ip6_privacy=config.get( | ||||
|                         ATTR_IP6_PRIVACY, InterfaceIp6Privacy.DEFAULT | ||||
|                     ), | ||||
|                     address=config.get(ATTR_ADDRESS, []), | ||||
|                     gateway=config.get(ATTR_GATEWAY), | ||||
|                     nameservers=config.get(ATTR_NAMESERVERS, []), | ||||
|                 ) | ||||
|             elif key == ATTR_WIFI: | ||||
|                 interface.wifi = replace( | ||||
|                     interface.wifi | ||||
|                     or WifiConfig( | ||||
|                         WifiMode.INFRASTRUCTURE, "", AuthMethod.OPEN, None, None | ||||
|                     ), | ||||
|                     **config, | ||||
|                 interface.wifi = WifiConfig( | ||||
|                     mode=config.get(ATTR_MODE, WifiMode.INFRASTRUCTURE), | ||||
|                     ssid=config.get(ATTR_SSID, ""), | ||||
|                     auth=config.get(ATTR_AUTH, AuthMethod.OPEN), | ||||
|                     psk=config.get(ATTR_PSK, None), | ||||
|                     signal=None, | ||||
|                 ) | ||||
|             elif key == ATTR_ENABLED: | ||||
|                 interface.enabled = config | ||||
|             elif key == ATTR_MDNS: | ||||
|                 interface.mdns = config | ||||
|             elif key == ATTR_LLMNR: | ||||
|                 interface.llmnr = config | ||||
|  | ||||
|         await asyncio.shield(self.sys_host.network.apply_changes(interface)) | ||||
|  | ||||
| @@ -231,7 +284,7 @@ class APINetwork(CoreSysAttributes): | ||||
|     @api_process | ||||
|     async def scan_accesspoints(self, request: web.Request) -> dict[str, Any]: | ||||
|         """Scan and return a list of available networks.""" | ||||
|         interface = self._get_interface(request.match_info.get(ATTR_INTERFACE)) | ||||
|         interface = self._get_interface(request.match_info[ATTR_INTERFACE]) | ||||
|  | ||||
|         # Only wlan is supported | ||||
|         if interface.type != InterfaceType.WIRELESS: | ||||
| @@ -244,8 +297,10 @@ class APINetwork(CoreSysAttributes): | ||||
|     @api_process | ||||
|     async def create_vlan(self, request: web.Request) -> None: | ||||
|         """Create a new vlan.""" | ||||
|         interface = self._get_interface(request.match_info.get(ATTR_INTERFACE)) | ||||
|         vlan = int(request.match_info.get(ATTR_VLAN)) | ||||
|         interface = self._get_interface(request.match_info[ATTR_INTERFACE]) | ||||
|         vlan = int(request.match_info.get(ATTR_VLAN, -1)) | ||||
|         if vlan < 0: | ||||
|             raise APIError(f"Invalid vlan specified: {vlan}") | ||||
|  | ||||
|         # Only ethernet is supported | ||||
|         if interface.type != InterfaceType.ETHERNET: | ||||
| @@ -256,37 +311,54 @@ class APINetwork(CoreSysAttributes): | ||||
|  | ||||
|         vlan_config = VlanConfig(vlan, interface.name) | ||||
|  | ||||
|         ipv4_config = None | ||||
|         mdns_mode = MulticastDnsMode.DEFAULT | ||||
|         llmnr_mode = MulticastDnsMode.DEFAULT | ||||
|  | ||||
|         if ATTR_MDNS in body: | ||||
|             mdns_mode = body[ATTR_MDNS] | ||||
|  | ||||
|         if ATTR_LLMNR in body: | ||||
|             llmnr_mode = body[ATTR_LLMNR] | ||||
|  | ||||
|         ipv4_setting = None | ||||
|         if ATTR_IPV4 in body: | ||||
|             ipv4_config = IpConfig( | ||||
|                 body[ATTR_IPV4].get(ATTR_METHOD, InterfaceMethod.AUTO), | ||||
|                 body[ATTR_IPV4].get(ATTR_ADDRESS, []), | ||||
|                 body[ATTR_IPV4].get(ATTR_GATEWAY, None), | ||||
|                 body[ATTR_IPV4].get(ATTR_NAMESERVERS, []), | ||||
|                 None, | ||||
|             ipv4_setting = IpSetting( | ||||
|                 method=body[ATTR_IPV4].get(ATTR_METHOD, InterfaceMethod.AUTO), | ||||
|                 address=body[ATTR_IPV4].get(ATTR_ADDRESS, []), | ||||
|                 gateway=body[ATTR_IPV4].get(ATTR_GATEWAY, None), | ||||
|                 nameservers=body[ATTR_IPV4].get(ATTR_NAMESERVERS, []), | ||||
|             ) | ||||
|  | ||||
|         ipv6_config = None | ||||
|         ipv6_setting = None | ||||
|         if ATTR_IPV6 in body: | ||||
|             ipv6_config = IpConfig( | ||||
|                 body[ATTR_IPV6].get(ATTR_METHOD, InterfaceMethod.AUTO), | ||||
|                 body[ATTR_IPV6].get(ATTR_ADDRESS, []), | ||||
|                 body[ATTR_IPV6].get(ATTR_GATEWAY, None), | ||||
|                 body[ATTR_IPV6].get(ATTR_NAMESERVERS, []), | ||||
|                 None, | ||||
|             ipv6_setting = Ip6Setting( | ||||
|                 method=body[ATTR_IPV6].get(ATTR_METHOD, InterfaceMethod.AUTO), | ||||
|                 addr_gen_mode=body[ATTR_IPV6].get( | ||||
|                     ATTR_ADDR_GEN_MODE, InterfaceAddrGenMode.DEFAULT | ||||
|                 ), | ||||
|                 ip6_privacy=body[ATTR_IPV6].get( | ||||
|                     ATTR_IP6_PRIVACY, InterfaceIp6Privacy.DEFAULT | ||||
|                 ), | ||||
|                 address=body[ATTR_IPV6].get(ATTR_ADDRESS, []), | ||||
|                 gateway=body[ATTR_IPV6].get(ATTR_GATEWAY, None), | ||||
|                 nameservers=body[ATTR_IPV6].get(ATTR_NAMESERVERS, []), | ||||
|             ) | ||||
|  | ||||
|         vlan_interface = Interface( | ||||
|             "", | ||||
|             f"{interface.name}.{vlan}", | ||||
|             "", | ||||
|             "", | ||||
|             True, | ||||
|             True, | ||||
|             False, | ||||
|             InterfaceType.VLAN, | ||||
|             ipv4_config, | ||||
|             ipv6_config, | ||||
|             None, | ||||
|             ipv4_setting, | ||||
|             None, | ||||
|             ipv6_setting, | ||||
|             None, | ||||
|             vlan_config, | ||||
|             mdns=mdns_mode, | ||||
|             llmnr=llmnr_mode, | ||||
|         ) | ||||
|         await asyncio.shield(self.sys_host.network.apply_changes(vlan_interface)) | ||||
|         await asyncio.shield(self.sys_host.network.create_vlan(vlan_interface)) | ||||
|   | ||||
| @@ -1,4 +1,5 @@ | ||||
| """Init file for Supervisor Observer RESTful API.""" | ||||
|  | ||||
| import asyncio | ||||
| import logging | ||||
| from typing import Any | ||||
|   | ||||
| @@ -1,7 +1,9 @@ | ||||
| """Init file for Supervisor HassOS RESTful API.""" | ||||
|  | ||||
| import asyncio | ||||
| from collections.abc import Awaitable | ||||
| import logging | ||||
| import re | ||||
| from typing import Any | ||||
|  | ||||
| from aiohttp import web | ||||
| @@ -19,22 +21,29 @@ from ..const import ( | ||||
|     ATTR_POWER_LED, | ||||
|     ATTR_SERIAL, | ||||
|     ATTR_SIZE, | ||||
|     ATTR_STATE, | ||||
|     ATTR_SWAP_SIZE, | ||||
|     ATTR_SWAPPINESS, | ||||
|     ATTR_UPDATE_AVAILABLE, | ||||
|     ATTR_VERSION, | ||||
|     ATTR_VERSION_LATEST, | ||||
| ) | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import BoardInvalidError | ||||
| from ..exceptions import APINotFound, BoardInvalidError | ||||
| from ..resolution.const import ContextType, IssueType, SuggestionType | ||||
| from ..validate import version_tag | ||||
| from .const import ( | ||||
|     ATTR_BOOT_SLOT, | ||||
|     ATTR_BOOT_SLOTS, | ||||
|     ATTR_DATA_DISK, | ||||
|     ATTR_DEV_PATH, | ||||
|     ATTR_DEVICE, | ||||
|     ATTR_DISKS, | ||||
|     ATTR_MODEL, | ||||
|     ATTR_STATUS, | ||||
|     ATTR_SYSTEM_HEALTH_LED, | ||||
|     ATTR_VENDOR, | ||||
|     BootSlot, | ||||
| ) | ||||
| from .utils import api_process, api_validate | ||||
|  | ||||
| @@ -42,6 +51,7 @@ _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
|  | ||||
| # pylint: disable=no-value-for-parameter | ||||
| SCHEMA_VERSION = vol.Schema({vol.Optional(ATTR_VERSION): version_tag}) | ||||
| SCHEMA_SET_BOOT_SLOT = vol.Schema({vol.Required(ATTR_BOOT_SLOT): vol.Coerce(BootSlot)}) | ||||
| SCHEMA_DISK = vol.Schema({vol.Required(ATTR_DEVICE): str}) | ||||
|  | ||||
| SCHEMA_YELLOW_OPTIONS = vol.Schema( | ||||
| @@ -58,6 +68,15 @@ SCHEMA_GREEN_OPTIONS = vol.Schema( | ||||
|         vol.Optional(ATTR_SYSTEM_HEALTH_LED): vol.Boolean(), | ||||
|     } | ||||
| ) | ||||
|  | ||||
| RE_SWAP_SIZE = re.compile(r"^\d+([KMG](i?B)?|B)?$", re.IGNORECASE) | ||||
|  | ||||
| SCHEMA_SWAP_OPTIONS = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_SWAP_SIZE): vol.Match(RE_SWAP_SIZE), | ||||
|         vol.Optional(ATTR_SWAPPINESS): vol.All(int, vol.Range(min=0, max=200)), | ||||
|     } | ||||
| ) | ||||
| # pylint: enable=no-value-for-parameter | ||||
|  | ||||
|  | ||||
| @@ -74,6 +93,15 @@ class APIOS(CoreSysAttributes): | ||||
|             ATTR_BOARD: self.sys_os.board, | ||||
|             ATTR_BOOT: self.sys_dbus.rauc.boot_slot, | ||||
|             ATTR_DATA_DISK: self.sys_os.datadisk.disk_used_id, | ||||
|             ATTR_BOOT_SLOTS: { | ||||
|                 slot.bootname: { | ||||
|                     ATTR_STATE: slot.state, | ||||
|                     ATTR_STATUS: slot.boot_status, | ||||
|                     ATTR_VERSION: slot.bundle_version, | ||||
|                 } | ||||
|                 for slot in self.sys_os.slots | ||||
|                 if slot.bootname | ||||
|             }, | ||||
|         } | ||||
|  | ||||
|     @api_process | ||||
| @@ -96,6 +124,17 @@ class APIOS(CoreSysAttributes): | ||||
|  | ||||
|         await asyncio.shield(self.sys_os.datadisk.migrate_disk(body[ATTR_DEVICE])) | ||||
|  | ||||
|     @api_process | ||||
|     def wipe_data(self, request: web.Request) -> Awaitable[None]: | ||||
|         """Trigger data disk wipe on Host.""" | ||||
|         return asyncio.shield(self.sys_os.datadisk.wipe_disk()) | ||||
|  | ||||
|     @api_process | ||||
|     async def set_boot_slot(self, request: web.Request) -> None: | ||||
|         """Change the active boot slot and reboot into it.""" | ||||
|         body = await api_validate(SCHEMA_SET_BOOT_SLOT, request) | ||||
|         await asyncio.shield(self.sys_os.set_boot_slot(body[ATTR_BOOT_SLOT])) | ||||
|  | ||||
|     @api_process | ||||
|     async def list_data(self, request: web.Request) -> dict[str, Any]: | ||||
|         """Return possible data targets.""" | ||||
| @@ -130,15 +169,19 @@ class APIOS(CoreSysAttributes): | ||||
|         body = await api_validate(SCHEMA_GREEN_OPTIONS, request) | ||||
|  | ||||
|         if ATTR_ACTIVITY_LED in body: | ||||
|             self.sys_dbus.agent.board.green.activity_led = body[ATTR_ACTIVITY_LED] | ||||
|             await self.sys_dbus.agent.board.green.set_activity_led( | ||||
|                 body[ATTR_ACTIVITY_LED] | ||||
|             ) | ||||
|  | ||||
|         if ATTR_POWER_LED in body: | ||||
|             self.sys_dbus.agent.board.green.power_led = body[ATTR_POWER_LED] | ||||
|             await self.sys_dbus.agent.board.green.set_power_led(body[ATTR_POWER_LED]) | ||||
|  | ||||
|         if ATTR_SYSTEM_HEALTH_LED in body: | ||||
|             self.sys_dbus.agent.board.green.user_led = body[ATTR_SYSTEM_HEALTH_LED] | ||||
|             await self.sys_dbus.agent.board.green.set_user_led( | ||||
|                 body[ATTR_SYSTEM_HEALTH_LED] | ||||
|             ) | ||||
|  | ||||
|         self.sys_dbus.agent.board.green.save_data() | ||||
|         await self.sys_dbus.agent.board.green.save_data() | ||||
|  | ||||
|     @api_process | ||||
|     async def boards_yellow_info(self, request: web.Request) -> dict[str, Any]: | ||||
| @@ -155,15 +198,17 @@ class APIOS(CoreSysAttributes): | ||||
|         body = await api_validate(SCHEMA_YELLOW_OPTIONS, request) | ||||
|  | ||||
|         if ATTR_DISK_LED in body: | ||||
|             self.sys_dbus.agent.board.yellow.disk_led = body[ATTR_DISK_LED] | ||||
|             await self.sys_dbus.agent.board.yellow.set_disk_led(body[ATTR_DISK_LED]) | ||||
|  | ||||
|         if ATTR_HEARTBEAT_LED in body: | ||||
|             self.sys_dbus.agent.board.yellow.heartbeat_led = body[ATTR_HEARTBEAT_LED] | ||||
|             await self.sys_dbus.agent.board.yellow.set_heartbeat_led( | ||||
|                 body[ATTR_HEARTBEAT_LED] | ||||
|             ) | ||||
|  | ||||
|         if ATTR_POWER_LED in body: | ||||
|             self.sys_dbus.agent.board.yellow.power_led = body[ATTR_POWER_LED] | ||||
|             await self.sys_dbus.agent.board.yellow.set_power_led(body[ATTR_POWER_LED]) | ||||
|  | ||||
|         self.sys_dbus.agent.board.yellow.save_data() | ||||
|         await self.sys_dbus.agent.board.yellow.save_data() | ||||
|         self.sys_resolution.create_issue( | ||||
|             IssueType.REBOOT_REQUIRED, | ||||
|             ContextType.SYSTEM, | ||||
| @@ -179,3 +224,53 @@ class APIOS(CoreSysAttributes): | ||||
|             ) | ||||
|  | ||||
|         return {} | ||||
|  | ||||
|     @api_process | ||||
|     async def config_swap_info(self, request: web.Request) -> dict[str, Any]: | ||||
|         """Get swap settings.""" | ||||
|         if ( | ||||
|             not self.coresys.os.available | ||||
|             or not self.coresys.os.version | ||||
|             or self.coresys.os.version < "15.0" | ||||
|         ): | ||||
|             raise APINotFound( | ||||
|                 "Home Assistant OS 15.0 or newer required for swap settings" | ||||
|             ) | ||||
|  | ||||
|         return { | ||||
|             ATTR_SWAP_SIZE: self.sys_dbus.agent.swap.swap_size, | ||||
|             ATTR_SWAPPINESS: self.sys_dbus.agent.swap.swappiness, | ||||
|         } | ||||
|  | ||||
|     @api_process | ||||
|     async def config_swap_options(self, request: web.Request) -> None: | ||||
|         """Update swap settings.""" | ||||
|         if ( | ||||
|             not self.coresys.os.available | ||||
|             or not self.coresys.os.version | ||||
|             or self.coresys.os.version < "15.0" | ||||
|         ): | ||||
|             raise APINotFound( | ||||
|                 "Home Assistant OS 15.0 or newer required for swap settings" | ||||
|             ) | ||||
|  | ||||
|         body = await api_validate(SCHEMA_SWAP_OPTIONS, request) | ||||
|  | ||||
|         reboot_required = False | ||||
|  | ||||
|         if ATTR_SWAP_SIZE in body: | ||||
|             old_size = self.sys_dbus.agent.swap.swap_size | ||||
|             await self.sys_dbus.agent.swap.set_swap_size(body[ATTR_SWAP_SIZE]) | ||||
|             reboot_required = reboot_required or old_size != body[ATTR_SWAP_SIZE] | ||||
|  | ||||
|         if ATTR_SWAPPINESS in body: | ||||
|             old_swappiness = self.sys_dbus.agent.swap.swappiness | ||||
|             await self.sys_dbus.agent.swap.set_swappiness(body[ATTR_SWAPPINESS]) | ||||
|             reboot_required = reboot_required or old_swappiness != body[ATTR_SWAPPINESS] | ||||
|  | ||||
|         if reboot_required: | ||||
|             self.sys_resolution.create_issue( | ||||
|                 IssueType.REBOOT_REQUIRED, | ||||
|                 ContextType.SYSTEM, | ||||
|                 suggestions=[SuggestionType.EXECUTE_REBOOT], | ||||
|             ) | ||||
|   | ||||
| @@ -1 +1 @@ | ||||
| !function(){function n(n){var t=document.createElement("script");t.src=n,document.body.appendChild(t)}if(/.*Version\/(?:11|12)(?:\.\d+)*.*Safari\//.test(navigator.userAgent))n("/api/hassio/app/frontend_es5/entrypoint-5yRSddAJzJ4.js");else try{new Function("import('/api/hassio/app/frontend_latest/entrypoint-qzB1D0O4L9U.js')")()}catch(t){n("/api/hassio/app/frontend_es5/entrypoint-5yRSddAJzJ4.js")}}() | ||||
| !function(){function d(d){var e=document.createElement("script");e.src=d,document.body.appendChild(e)}if(/Edge?\/(13\d|1[4-9]\d|[2-9]\d{2}|\d{4,})\.\d+(\.\d+|)|Firefox\/(13[1-9]|1[4-9]\d|[2-9]\d{2}|\d{4,})\.\d+(\.\d+|)|Chrom(ium|e)\/(10[5-9]|1[1-9]\d|[2-9]\d{2}|\d{4,})\.\d+(\.\d+|)|(Maci|X1{2}).+ Version\/(18\.([1-9]|\d{2,})|(19|[2-9]\d|\d{3,})\.\d+)([,.]\d+|)( \(\w+\)|)( Mobile\/\w+|) Safari\/|Chrome.+OPR\/(1{2}[5-9]|1[2-9]\d|[2-9]\d{2}|\d{4,})\.\d+\.\d+|(CPU[ +]OS|iPhone[ +]OS|CPU[ +]iPhone|CPU IPhone OS|CPU iPad OS)[ +]+(18[._]([1-9]|\d{2,})|(19|[2-9]\d|\d{3,})[._]\d+)([._]\d+|)|Android:?[ /-](13\d|1[4-9]\d|[2-9]\d{2}|\d{4,})(\.\d+|)(\.\d+|)|Mobile Safari.+OPR\/([89]\d|\d{3,})\.\d+\.\d+|Android.+Firefox\/(13[1-9]|1[4-9]\d|[2-9]\d{2}|\d{4,})\.\d+(\.\d+|)|Android.+Chrom(ium|e)\/(13\d|1[4-9]\d|[2-9]\d{2}|\d{4,})\.\d+(\.\d+|)|SamsungBrowser\/(2[89]|[3-9]\d|\d{3,})\.\d+|Home As{2}istant\/[\d.]+ \(.+; macOS (1[3-9]|[2-9]\d|\d{3,})\.\d+(\.\d+)?\)/.test(navigator.userAgent))try{new Function("import('/api/hassio/app/frontend_latest/entrypoint.1e251476306cafd4.js')")()}catch(e){d("/api/hassio/app/frontend_es5/entrypoint.601ff5d4dddd11f9.js")}else d("/api/hassio/app/frontend_es5/entrypoint.601ff5d4dddd11f9.js")}() | ||||
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/entrypoint.js.br
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/entrypoint.js.br
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										2
									
								
								supervisor/api/panel/frontend_es5/10.02c74d8ffd9bf568.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								supervisor/api/panel/frontend_es5/10.02c74d8ffd9bf568.js
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/10.02c74d8ffd9bf568.js.br
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/10.02c74d8ffd9bf568.js.br
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/10.02c74d8ffd9bf568.js.gz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/10.02c74d8ffd9bf568.js.gz
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1008.c2e44b88f5829db4.js.br
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1008.c2e44b88f5829db4.js.br
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1008.c2e44b88f5829db4.js.gz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1008.c2e44b88f5829db4.js.gz
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,2 +0,0 @@ | ||||
| "use strict";(self.webpackChunkhome_assistant_frontend=self.webpackChunkhome_assistant_frontend||[]).push([[1047],{32594:function(e,t,r){r.d(t,{U:function(){return n}});var n=function(e){return e.stopPropagation()}},75054:function(e,t,r){r.r(t),r.d(t,{HaTimeDuration:function(){return f}});var n,a=r(88962),i=r(33368),o=r(71650),d=r(82390),u=r(69205),l=r(70906),s=r(91808),c=r(68144),v=r(79932),f=(r(47289),(0,s.Z)([(0,v.Mo)("ha-selector-duration")],(function(e,t){var r=function(t){(0,u.Z)(n,t);var r=(0,l.Z)(n);function n(){var t;(0,o.Z)(this,n);for(var a=arguments.length,i=new Array(a),u=0;u<a;u++)i[u]=arguments[u];return t=r.call.apply(r,[this].concat(i)),e((0,d.Z)(t)),t}return(0,i.Z)(n)}(t);return{F:r,d:[{kind:"field",decorators:[(0,v.Cb)({attribute:!1})],key:"hass",value:void 0},{kind:"field",decorators:[(0,v.Cb)({attribute:!1})],key:"selector",value:void 0},{kind:"field",decorators:[(0,v.Cb)({attribute:!1})],key:"value",value:void 0},{kind:"field",decorators:[(0,v.Cb)()],key:"label",value:void 0},{kind:"field",decorators:[(0,v.Cb)()],key:"helper",value:void 0},{kind:"field",decorators:[(0,v.Cb)({type:Boolean})],key:"disabled",value:function(){return!1}},{kind:"field",decorators:[(0,v.Cb)({type:Boolean})],key:"required",value:function(){return!0}},{kind:"method",key:"render",value:function(){var e;return(0,c.dy)(n||(n=(0,a.Z)([' <ha-duration-input .label="','" .helper="','" .data="','" .disabled="','" .required="','" ?enableDay="','"></ha-duration-input> '])),this.label,this.helper,this.value,this.disabled,this.required,null===(e=this.selector.duration)||void 0===e?void 0:e.enable_day)}}]}}),c.oi))}}]); | ||||
| //# sourceMappingURL=1047-g7fFLS9eP4I.js.map | ||||
										
											Binary file not shown.
										
									
								
							| @@ -1 +0,0 @@ | ||||
| {"version":3,"file":"1047-g7fFLS9eP4I.js","mappings":"yKAAO,IAAMA,EAAkB,SAACC,GAAE,OAAKA,EAAGD,iBAAiB,C,qLCQ9CE,G,UAAcC,EAAAA,EAAAA,GAAA,EAD1BC,EAAAA,EAAAA,IAAc,0BAAuB,SAAAC,EAAAC,GAAA,IACzBJ,EAAc,SAAAK,IAAAC,EAAAA,EAAAA,GAAAN,EAAAK,GAAA,IAAAE,GAAAC,EAAAA,EAAAA,GAAAR,GAAA,SAAAA,IAAA,IAAAS,GAAAC,EAAAA,EAAAA,GAAA,KAAAV,GAAA,QAAAW,EAAAC,UAAAC,OAAAC,EAAA,IAAAC,MAAAJ,GAAAK,EAAA,EAAAA,EAAAL,EAAAK,IAAAF,EAAAE,GAAAJ,UAAAI,GAAA,OAAAP,EAAAF,EAAAU,KAAAC,MAAAX,EAAA,OAAAY,OAAAL,IAAAX,GAAAiB,EAAAA,EAAAA,GAAAX,IAAAA,CAAA,QAAAY,EAAAA,EAAAA,GAAArB,EAAA,EAAAI,GAAA,OAAAkB,EAAdtB,EAAcuB,EAAA,EAAAC,KAAA,QAAAC,WAAA,EACxBC,EAAAA,EAAAA,IAAS,CAAEC,WAAW,KAAQC,IAAA,OAAAC,WAAA,IAAAL,KAAA,QAAAC,WAAA,EAE9BC,EAAAA,EAAAA,IAAS,CAAEC,WAAW,KAAQC,IAAA,WAAAC,WAAA,IAAAL,KAAA,QAAAC,WAAA,EAE9BC,EAAAA,EAAAA,IAAS,CAAEC,WAAW,KAAQC,IAAA,QAAAC,WAAA,IAAAL,KAAA,QAAAC,WAAA,EAE9BC,EAAAA,EAAAA,OAAUE,IAAA,QAAAC,WAAA,IAAAL,KAAA,QAAAC,WAAA,EAEVC,EAAAA,EAAAA,OAAUE,IAAA,SAAAC,WAAA,IAAAL,KAAA,QAAAC,WAAA,EAEVC,EAAAA,EAAAA,IAAS,CAAEI,KAAMC,WAAUH,IAAA,WAAAC,MAAA,kBAAmB,CAAK,IAAAL,KAAA,QAAAC,WAAA,EAEnDC,EAAAA,EAAAA,IAAS,CAAEI,KAAMC,WAAUH,IAAA,WAAAC,MAAA,kBAAmB,CAAI,IAAAL,KAAA,SAAAI,IAAA,SAAAC,MAEnD,WAAmB,IAAAG,EACjB,OAAOC,EAAAA,EAAAA,IAAIC,IAAAA,GAAAC,EAAAA,EAAAA,GAAA,wIAEEC,KAAKC,MACJD,KAAKE,OACPF,KAAKP,MACDO,KAAKG,SACLH,KAAKI,SACkB,QADVR,EACZI,KAAKK,SAASC,gBAAQ,IAAAV,OAAA,EAAtBA,EAAwBW,WAG3C,IAAC,GA1BiCC,EAAAA,I","sources":["https://raw.githubusercontent.com/home-assistant/frontend/20230703.0/src/common/dom/stop_propagation.ts","https://raw.githubusercontent.com/home-assistant/frontend/20230703.0/src/components/ha-selector/ha-selector-duration.ts"],"names":["stopPropagation","ev","HaTimeDuration","_decorate","customElement","_initialize","_LitElement","_LitElement2","_inherits","_super","_createSuper","_this","_classCallCheck","_len","arguments","length","args","Array","_key","call","apply","concat","_assertThisInitialized","_createClass","F","d","kind","decorators","property","attribute","key","value","type","Boolean","_this$selector$durati","html","_templateObject","_taggedTemplateLiteral","this","label","helper","disabled","required","selector","duration","enable_day","LitElement"],"sourceRoot":""} | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1057.d306824fd6aa0497.js.br
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1057.d306824fd6aa0497.js.br
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1057.d306824fd6aa0497.js.gz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1057.d306824fd6aa0497.js.gz
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -0,0 +1 @@ | ||||
| {"version":3,"file":"1057.d306824fd6aa0497.js","sources":["https://raw.githubusercontent.com/home-assistant/frontend/20250925.1/src/data/auth.ts","https://raw.githubusercontent.com/home-assistant/frontend/20250925.1/src/data/entity.ts","https://raw.githubusercontent.com/home-assistant/frontend/20250925.1/src/data/media-player.ts","https://raw.githubusercontent.com/home-assistant/frontend/20250925.1/src/data/tts.ts","https://raw.githubusercontent.com/home-assistant/frontend/20250925.1/src/util/brands-url.ts"],"names":["autocompleteLoginFields","schema","map","field","type","name","Object","assign","autocomplete","autofocus","getSignedPath","hass","path","callWS","UNAVAILABLE","UNKNOWN","ON","OFF","UNAVAILABLE_STATES","OFF_STATES","isUnavailableState","arrayLiteralIncludes","MediaPlayerEntityFeature","BROWSER_PLAYER","MediaClassBrowserSettings","album","icon","layout","app","show_list_images","artist","mdiAccountMusic","channel","mdiTelevisionClassic","thumbnail_ratio","composer","contributing_artist","directory","episode","game","genre","image","movie","music","playlist","podcast","season","track","tv_show","url","video","browseMediaPlayer","entityId","mediaContentId","mediaContentType","entity_id","media_content_id","media_content_type","convertTextToSpeech","data","callApi","TTS_MEDIA_SOURCE_PREFIX","isTTSMediaSource","startsWith","getProviderFromTTSMediaSource","substring","listTTSEngines","language","country","getTTSEngine","engine_id","listTTSVoices","brandsUrl","options","brand","useFallback","domain","darkOptimized","extractDomainFromBrandUrl","split","isBrandUrl","thumbnail"],"mappings":"2QAyBO,MAEMA,EAA2BC,GACtCA,EAAOC,IAAKC,IACV,GAAmB,WAAfA,EAAMC,KAAmB,OAAOD,EACpC,OAAQA,EAAME,MACZ,IAAK,WACH,OAAAC,OAAAC,OAAAD,OAAAC,OAAA,GAAYJ,GAAK,IAAEK,aAAc,WAAYC,WAAW,IAC1D,IAAK,WACH,OAAAH,OAAAC,OAAAD,OAAAC,OAAA,GAAYJ,GAAK,IAAEK,aAAc,qBACnC,IAAK,OACH,OAAAF,OAAAC,OAAAD,OAAAC,OAAA,GAAYJ,GAAK,IAAEK,aAAc,gBAAiBC,WAAW,IAC/D,QACE,OAAON,KAIFO,EAAgBA,CAC3BC,EACAC,IACwBD,EAAKE,OAAO,CAAET,KAAM,iBAAkBQ,Q,gMC3CzD,MAAME,EAAc,cACdC,EAAU,UACVC,EAAK,KACLC,EAAM,MAENC,EAAqB,CAACJ,EAAaC,GACnCI,EAAa,CAACL,EAAaC,EAASE,GAEpCG,GAAqBC,EAAAA,EAAAA,GAAqBH,IAC7BG,EAAAA,EAAAA,GAAqBF,E,+gCCuExC,IAAWG,EAAA,SAAAA,G,qnBAAAA,C,CAAA,C,IAyBX,MAAMC,EAAiB,UAWjBC,EAGT,CACFC,MAAO,CAAEC,K,mQAAgBC,OAAQ,QACjCC,IAAK,CAAEF,K,6GAAsBC,OAAQ,OAAQE,kBAAkB,GAC/DC,OAAQ,CAAEJ,KAAMK,EAAiBJ,OAAQ,OAAQE,kBAAkB,GACnEG,QAAS,CACPN,KAAMO,EACNC,gBAAiB,WACjBP,OAAQ,OACRE,kBAAkB,GAEpBM,SAAU,CACRT,K,4cACAC,OAAQ,OACRE,kBAAkB,GAEpBO,oBAAqB,CACnBV,KAAMK,EACNJ,OAAQ,OACRE,kBAAkB,GAEpBQ,UAAW,CAAEX,K,gGAAiBC,OAAQ,OAAQE,kBAAkB,GAChES,QAAS,CACPZ,KAAMO,EACNN,OAAQ,OACRO,gBAAiB,WACjBL,kBAAkB,GAEpBU,KAAM,CACJb,K,qWACAC,OAAQ,OACRO,gBAAiB,YAEnBM,MAAO,CAAEd,K,4hCAAqBC,OAAQ,OAAQE,kBAAkB,GAChEY,MAAO,CAAEf,K,sHAAgBC,OAAQ,OAAQE,kBAAkB,GAC3Da,MAAO,CACLhB,K,6GACAQ,gBAAiB,WACjBP,OAAQ,OACRE,kBAAkB,GAEpBc,MAAO,CAAEjB,K,+NAAgBG,kBAAkB,GAC3Ce,SAAU,CAAElB,K,mJAAwBC,OAAQ,OAAQE,kBAAkB,GACtEgB,QAAS,CAAEnB,K,qpBAAkBC,OAAQ,QACrCmB,OAAQ,CACNpB,KAAMO,EACNN,OAAQ,OACRO,gBAAiB,WACjBL,kBAAkB,GAEpBkB,MAAO,CAAErB,K,mLACTsB,QAAS,CACPtB,KAAMO,EACNN,OAAQ,OACRO,gBAAiB,YAEnBe,IAAK,CAAEvB,K,w5BACPwB,MAAO,CAAExB,K,2GAAgBC,OAAQ,OAAQE,kBAAkB,IAkChDsB,EAAoBA,CAC/BxC,EACAyC,EACAC,EACAC,IAEA3C,EAAKE,OAAwB,CAC3BT,KAAM,4BACNmD,UAAWH,EACXI,iBAAkBH,EAClBI,mBAAoBH,G,yLC/MjB,MAAMI,EAAsBA,CACjC/C,EACAgD,IAOGhD,EAAKiD,QAAuC,OAAQ,cAAeD,GAElEE,EAA0B,sBAEnBC,EAAoBT,GAC/BA,EAAeU,WAAWF,GAEfG,EAAiCX,GAC5CA,EAAeY,UAAUJ,IAEdK,EAAiBA,CAC5BvD,EACAwD,EACAC,IAEAzD,EAAKE,OAAO,CACVT,KAAM,kBACN+D,WACAC,YAGSC,EAAeA,CAC1B1D,EACA2D,IAEA3D,EAAKE,OAAO,CACVT,KAAM,iBACNkE,cAGSC,EAAgBA,CAC3B5D,EACA2D,EACAH,IAEAxD,EAAKE,OAAO,CACVT,KAAM,oBACNkE,YACAH,Y,kHC9CG,MAAMK,EAAaC,GACxB,oCAAoCA,EAAQC,MAAQ,UAAY,KAC9DD,EAAQE,YAAc,KAAO,KAC5BF,EAAQG,UAAUH,EAAQI,cAAgB,QAAU,KACrDJ,EAAQrE,WAQC0E,EAA6B7B,GAAgBA,EAAI8B,MAAM,KAAK,GAE5DC,EAAcC,GACzBA,EAAUlB,WAAW,oC"} | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -0,0 +1,23 @@ | ||||
| /** | ||||
|  * @license | ||||
|  * Copyright 2018 Google LLC | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  */ | ||||
|  | ||||
| /** | ||||
|  * @license | ||||
|  * Copyright 2018 Google LLC | ||||
|  * SPDX-License-Identifier: BSD-3-Clause | ||||
|  */ | ||||
|  | ||||
| /** | ||||
|  * @license | ||||
|  * Copyright 2020 Google LLC | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  */ | ||||
|  | ||||
| /** | ||||
|  * @license | ||||
|  * Copyright 2021 Google LLC | ||||
|  * SPDX-LIcense-Identifier: Apache-2.0 | ||||
|  */ | ||||
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1076.205340b2a7c5d559.js.br
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1076.205340b2a7c5d559.js.br
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1076.205340b2a7c5d559.js.gz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1076.205340b2a7c5d559.js.gz
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -0,0 +1,23 @@ | ||||
| /** | ||||
|  * @license | ||||
|  * Copyright 2018 Google LLC | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  */ | ||||
|  | ||||
| /** | ||||
|  * @license | ||||
|  * Copyright 2018 Google LLC | ||||
|  * SPDX-License-Identifier: BSD-3-Clause | ||||
|  */ | ||||
|  | ||||
| /** | ||||
|  * @license | ||||
|  * Copyright 2020 Google LLC | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  */ | ||||
|  | ||||
| /** | ||||
|  * @license | ||||
|  * Copyright 2021 Google LLC | ||||
|  * SPDX-LIcense-Identifier: Apache-2.0 | ||||
|  */ | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user