mirror of
				https://github.com/home-assistant/supervisor.git
				synced 2025-10-25 19:49:44 +00:00 
			
		
		
		
	Compare commits
	
		
			281 Commits
		
	
	
		
			2025.05.1
			...
			fix-websoc
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 3e307c5c8b | ||
|   | 1cd499b4a5 | ||
|   | 53a8044aff | ||
|   | c71553f37d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c1eb97d8ab | ||
|   | 190b734332 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 559b6982a3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 301362e9e5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fc928d294c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f42aeb4937 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fd21886de9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e4bb415e30 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 622dda5382 | ||
|   | 78a2e15ebb | ||
|   | f3e1e0f423 | ||
|   | 5779b567f1 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3c5f4920a0 | ||
|   | 64f94a159c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ab3b147876 | ||
| ![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 | 
							
								
								
									
										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> | ||||
|  | ||||
							
								
								
									
										9
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.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 | ||||
| @@ -76,7 +75,7 @@ body: | ||||
|       description: > | ||||
|         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: | ||||
| @@ -86,7 +85,7 @@ body: | ||||
|         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 | ||||
							
								
								
									
										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. | ||||
							
								
								
									
										27
									
								
								.github/workflows/builder.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										27
									
								
								.github/workflows/builder.yml
									
									
									
									
										vendored
									
									
								
							| @@ -53,7 +53,7 @@ jobs: | ||||
|       requirements: ${{ steps.requirements.outputs.changed }} | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         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.2.2 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
| @@ -104,9 +104,10 @@ 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@2025.03.0 | ||||
|         uses: home-assistant/wheels@2025.09.1 | ||||
|         with: | ||||
|           abi: cp313 | ||||
|           tag: musllinux_1_2 | ||||
| @@ -125,15 +126,15 @@ jobs: | ||||
|  | ||||
|       - name: Set up Python ${{ env.DEFAULT_PYTHON }} | ||||
|         if: needs.init.outputs.publish == 'true' | ||||
|         uses: actions/setup-python@v5.6.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.8.2 | ||||
|         uses: sigstore/cosign-installer@d7543c93d881b35a8faa02e8e3605f69b7a1ce62 # v3.10.0 | ||||
|         with: | ||||
|           cosign-release: "v2.4.0" | ||||
|           cosign-release: "v2.5.3" | ||||
|  | ||||
|       - name: Install dirhash and calc hash | ||||
|         if: needs.init.outputs.publish == 'true' | ||||
| @@ -149,7 +150,7 @@ jobs: | ||||
|  | ||||
|       - name: Login to GitHub Container Registry | ||||
|         if: needs.init.outputs.publish == 'true' | ||||
|         uses: docker/login-action@v3.4.0 | ||||
|         uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.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@2025.03.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.2.2 | ||||
|         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.2.2 | ||||
|         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@2025.03.0 | ||||
|         uses: home-assistant/builder@2025.09.0 | ||||
|         with: | ||||
|           args: | | ||||
|             --test \ | ||||
|   | ||||
							
								
								
									
										122
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										122
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -10,6 +10,7 @@ on: | ||||
| env: | ||||
|   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.2.2 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python | ||||
|         id: python | ||||
|         uses: actions/setup-python@v5.6.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@v4.2.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@v4.2.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: ${{ env.PRE_COMMIT_CACHE }} | ||||
|           lookup-only: true | ||||
| @@ -67,15 +68,15 @@ jobs: | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.6.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@v4.2.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -87,7 +88,7 @@ jobs: | ||||
|           exit 1 | ||||
|       - name: Restore pre-commit environment from cache | ||||
|         id: cache-precommit | ||||
|         uses: actions/cache@v4.2.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: ${{ env.PRE_COMMIT_CACHE }} | ||||
|           key: | | ||||
| @@ -110,15 +111,15 @@ jobs: | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.6.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@v4.2.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -130,7 +131,7 @@ jobs: | ||||
|           exit 1 | ||||
|       - name: Restore pre-commit environment from cache | ||||
|         id: cache-precommit | ||||
|         uses: actions/cache@v4.2.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: ${{ env.PRE_COMMIT_CACHE }} | ||||
|           key: | | ||||
| @@ -153,7 +154,7 @@ jobs: | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Register hadolint problem matcher | ||||
|         run: | | ||||
|           echo "::add-matcher::.github/workflows/matchers/hadolint.json" | ||||
| @@ -168,15 +169,15 @@ jobs: | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.6.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@v4.2.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -188,7 +189,7 @@ jobs: | ||||
|           exit 1 | ||||
|       - name: Restore pre-commit environment from cache | ||||
|         id: cache-precommit | ||||
|         uses: actions/cache@v4.2.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: ${{ env.PRE_COMMIT_CACHE }} | ||||
|           key: | | ||||
| @@ -212,15 +213,15 @@ jobs: | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.6.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@v4.2.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -232,7 +233,7 @@ jobs: | ||||
|           exit 1 | ||||
|       - name: Restore pre-commit environment from cache | ||||
|         id: cache-precommit | ||||
|         uses: actions/cache@v4.2.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: ${{ env.PRE_COMMIT_CACHE }} | ||||
|           key: | | ||||
| @@ -256,15 +257,15 @@ jobs: | ||||
|     needs: prepare | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.6.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@v4.2.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -286,25 +287,71 @@ jobs: | ||||
|           . 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@v4.2.2 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.6.0 | ||||
|         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@v3.8.2 | ||||
|         uses: sigstore/cosign-installer@d7543c93d881b35a8faa02e8e3605f69b7a1ce62 # v3.10.0 | ||||
|         with: | ||||
|           cosign-release: "v2.4.0" | ||||
|           cosign-release: "v2.5.3" | ||||
|       - name: Restore Python virtual environment | ||||
|         id: cache-venv | ||||
|         uses: actions/cache@v4.2.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -339,9 +386,9 @@ jobs: | ||||
|             -o console_output_style=count \ | ||||
|             tests | ||||
|       - name: Upload coverage artifact | ||||
|         uses: actions/upload-artifact@v4.6.2 | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         with: | ||||
|           name: coverage-${{ matrix.python-version }} | ||||
|           name: coverage | ||||
|           path: .coverage | ||||
|           include-hidden-files: true | ||||
|  | ||||
| @@ -351,15 +398,15 @@ jobs: | ||||
|     needs: ["pytest", "prepare"] | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Set up Python ${{ needs.prepare.outputs.python-version }} | ||||
|         uses: actions/setup-python@v5.6.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@v4.2.3 | ||||
|         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||
|         with: | ||||
|           path: venv | ||||
|           key: | | ||||
| @@ -370,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.3.0 | ||||
|         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||
|         with: | ||||
|           name: coverage | ||||
|           path: coverage/ | ||||
|       - name: Combine coverage results | ||||
|         run: | | ||||
|           . venv/bin/activate | ||||
| @@ -378,4 +428,4 @@ jobs: | ||||
|           coverage report | ||||
|           coverage xml | ||||
|       - name: Upload coverage to Codecov | ||||
|         uses: codecov/codecov-action@v5.4.2 | ||||
|         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" | ||||
|   | ||||
							
								
								
									
										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.2.2 | ||||
|         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@v6.1.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.2.2 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Sentry Release | ||||
|         uses: getsentry/action-release@v3.1.1 | ||||
|         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.1.0 | ||||
|       - uses: actions/stale@5f858e3efba33a5ca4407a664cc011ad407f2008 # v10.1.0 | ||||
|         with: | ||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           days-before-stale: 30 | ||||
|   | ||||
							
								
								
									
										10
									
								
								.github/workflows/update_frontend.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/update_frontend.yml
									
									
									
									
										vendored
									
									
								
							| @@ -14,10 +14,10 @@ jobs: | ||||
|       latest_version: ${{ steps.latest_frontend_version.outputs.latest_tag }} | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Get latest frontend release | ||||
|         id: latest_frontend_version | ||||
|         uses: abatilo/release-info-action@v1.3.3 | ||||
|         uses: abatilo/release-info-action@32cb932219f1cee3fc4f4a298fd65ead5d35b661 # v1.3.3 | ||||
|         with: | ||||
|           owner: home-assistant | ||||
|           repo: frontend | ||||
| @@ -49,7 +49,7 @@ jobs: | ||||
|     if: needs.check-version.outputs.skip != 'true' | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|       - name: Clear www folder | ||||
|         run: | | ||||
|           rm -rf supervisor/api/panel/* | ||||
| @@ -57,7 +57,7 @@ jobs: | ||||
|         run: | | ||||
|           echo "${{ needs.check-version.outputs.latest_version }}" > .ha-frontend-version | ||||
|       - name: Download release assets | ||||
|         uses: robinraju/release-downloader@v1 | ||||
|         uses: robinraju/release-downloader@daf26c55d821e836577a15f77d86ddc078948b05 # v1.12 | ||||
|         with: | ||||
|           repository: 'home-assistant/frontend' | ||||
|           tag: ${{ needs.check-version.outputs.latest_version }} | ||||
| @@ -68,7 +68,7 @@ jobs: | ||||
|         run: | | ||||
|           rm -f supervisor/api/panel/home_assistant_frontend_supervisor-*.tar.gz | ||||
|       - name: Create PR | ||||
|         uses: peter-evans/create-pull-request@v7 | ||||
|         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 | ||||
|   | ||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -100,3 +100,6 @@ ENV/ | ||||
| # mypy | ||||
| /.mypy_cache/* | ||||
| /.dmypy.json | ||||
|  | ||||
| # Mac | ||||
| .DS_Store | ||||
| @@ -1 +1 @@ | ||||
| 20250401.0 | ||||
| 20250925.1 | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| repos: | ||||
|   - repo: https://github.com/astral-sh/ruff-pre-commit | ||||
|     rev: v0.9.1 | ||||
|     rev: v0.11.10 | ||||
|     hooks: | ||||
|       - id: ruff | ||||
|         args: | ||||
| @@ -13,3 +13,15 @@ repos: | ||||
|       - id: check-executables-have-shebangs | ||||
|         stages: [manual] | ||||
|       - id: check-json | ||||
|   - repo: local | ||||
|     hooks: | ||||
|       # 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)$ | ||||
|   | ||||
| @@ -29,7 +29,7 @@ RUN \ | ||||
|     \ | ||||
|     && curl -Lso /usr/bin/cosign "https://github.com/home-assistant/cosign/releases/download/${COSIGN_VERSION}/cosign_${BUILD_ARCH}" \ | ||||
|     && chmod a+x /usr/bin/cosign \ | ||||
|     && pip3 install uv==0.6.17 | ||||
|     && pip3 install uv==0.8.9 | ||||
|  | ||||
| # Install requirements | ||||
| COPY requirements.txt . | ||||
|   | ||||
							
								
								
									
										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.13-alpine3.21 | ||||
|   armhf: ghcr.io/home-assistant/armhf-base-python:3.13-alpine3.21 | ||||
|   armv7: ghcr.io/home-assistant/armv7-base-python:3.13-alpine3.21 | ||||
|   amd64: ghcr.io/home-assistant/amd64-base-python:3.13-alpine3.21 | ||||
|   i386: ghcr.io/home-assistant/i386-base-python:3.13-alpine3.21 | ||||
|   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.4.0 | ||||
|   COSIGN_VERSION: 2.5.3 | ||||
| labels: | ||||
|   io.hass.type: supervisor | ||||
|   org.opencontainers.image.title: Home Assistant Supervisor | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| [build-system] | ||||
| requires = ["setuptools~=80.4.0", "wheel~=0.46.1"] | ||||
| requires = ["setuptools~=80.9.0", "wheel~=0.46.1"] | ||||
| build-backend = "setuptools.build_meta" | ||||
|  | ||||
| [project] | ||||
|   | ||||
| @@ -1,30 +1,30 @@ | ||||
| aiodns==3.4.0 | ||||
| aiohttp==3.11.18 | ||||
| aiodns==3.5.0 | ||||
| aiohttp==3.13.0 | ||||
| atomicwrites-homeassistant==1.4.1 | ||||
| attrs==25.3.0 | ||||
| awesomeversion==24.6.0 | ||||
| blockbuster==1.5.24 | ||||
| attrs==25.4.0 | ||||
| awesomeversion==25.8.0 | ||||
| blockbuster==1.5.25 | ||||
| brotli==1.1.0 | ||||
| ciso8601==2.3.2 | ||||
| ciso8601==2.3.3 | ||||
| colorlog==6.9.0 | ||||
| cpe==1.3.1 | ||||
| cryptography==44.0.3 | ||||
| debugpy==1.8.14 | ||||
| cryptography==46.0.2 | ||||
| debugpy==1.8.17 | ||||
| deepmerge==2.0 | ||||
| dirhash==0.5.0 | ||||
| docker==7.1.0 | ||||
| faust-cchardet==2.1.19 | ||||
| gitpython==3.1.44 | ||||
| gitpython==3.1.45 | ||||
| jinja2==3.1.6 | ||||
| log-rate-limit==1.4.2 | ||||
| orjson==3.10.18 | ||||
| orjson==3.11.3 | ||||
| pulsectl==24.12.0 | ||||
| pyudev==0.24.3 | ||||
| PyYAML==6.0.2 | ||||
| requests==2.32.3 | ||||
| PyYAML==6.0.3 | ||||
| requests==2.32.5 | ||||
| securetar==2025.2.1 | ||||
| sentry-sdk==2.28.0 | ||||
| setuptools==80.4.0 | ||||
| sentry-sdk==2.40.0 | ||||
| setuptools==80.9.0 | ||||
| voluptuous==0.15.2 | ||||
| dbus-fast==2.44.1 | ||||
| dbus-fast==2.44.5 | ||||
| zlib-fast==0.2.1 | ||||
|   | ||||
| @@ -1,12 +1,16 @@ | ||||
| astroid==3.3.10 | ||||
| coverage==7.8.0 | ||||
| pre-commit==4.2.0 | ||||
| pylint==3.3.7 | ||||
| astroid==3.3.11 | ||||
| coverage==7.10.7 | ||||
| mypy==1.18.2 | ||||
| pre-commit==4.3.0 | ||||
| pylint==3.3.9 | ||||
| pytest-aiohttp==1.1.0 | ||||
| pytest-asyncio==0.25.2 | ||||
| pytest-cov==6.1.1 | ||||
| pytest-cov==7.0.0 | ||||
| pytest-timeout==2.4.0 | ||||
| pytest==8.3.5 | ||||
| ruff==0.11.9 | ||||
| time-machine==2.16.0 | ||||
| urllib3==2.4.0 | ||||
| pytest==8.4.2 | ||||
| ruff==0.14.0 | ||||
| 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 "$@" | ||||
| @@ -13,7 +13,7 @@ zlib_fast.enable() | ||||
|  | ||||
| # pylint: disable=wrong-import-position | ||||
| from supervisor import bootstrap  # noqa: E402 | ||||
| from supervisor.utils.blockbuster import activate_blockbuster  # 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 | ||||
| @@ -55,7 +55,7 @@ if __name__ == "__main__": | ||||
|     coresys = loop.run_until_complete(bootstrap.initialize_coresys()) | ||||
|     loop.set_debug(coresys.config.debug) | ||||
|     if coresys.config.detect_blocking_io: | ||||
|         activate_blockbuster() | ||||
|         BlockBusterManager.activate() | ||||
|     loop.run_until_complete(coresys.core.connect()) | ||||
|  | ||||
|     loop.run_until_complete(bootstrap.supervisor_debugger(coresys)) | ||||
| @@ -66,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") | ||||
|   | ||||
| @@ -67,17 +67,16 @@ from ..docker.monitor import DockerContainerStateEvent | ||||
| from ..docker.stats import DockerStats | ||||
| from ..exceptions import ( | ||||
|     AddonConfigurationError, | ||||
|     AddonNotSupportedError, | ||||
|     AddonsError, | ||||
|     AddonsJobError, | ||||
|     AddonsNotSupportedError, | ||||
|     ConfigurationFileError, | ||||
|     DockerError, | ||||
|     HomeAssistantAPIError, | ||||
|     HostAppArmorError, | ||||
| ) | ||||
| from ..hardware.data import Device | ||||
| from ..homeassistant.const import WSEvent | ||||
| from ..jobs.const import JobExecutionLimit | ||||
| from ..jobs.const import JobConcurrency, JobThrottle | ||||
| from ..jobs.decorator import Job | ||||
| from ..resolution.const import ContextType, IssueType, UnhealthyReason | ||||
| from ..resolution.data import Issue | ||||
| @@ -227,6 +226,7 @@ class Addon(AddonModel): | ||||
|         ) | ||||
|  | ||||
|         await self._check_ingress_port() | ||||
|  | ||||
|         default_image = self._image(self.data) | ||||
|         try: | ||||
|             await self.instance.attach(version=self.version) | ||||
| @@ -360,7 +360,7 @@ class Addon(AddonModel): | ||||
|     @property | ||||
|     def auto_update(self) -> bool: | ||||
|         """Return if auto update is enable.""" | ||||
|         return self.persist.get(ATTR_AUTO_UPDATE, super().auto_update) | ||||
|         return self.persist.get(ATTR_AUTO_UPDATE, False) | ||||
|  | ||||
|     @auto_update.setter | ||||
|     def auto_update(self, value: bool) -> None: | ||||
| @@ -733,8 +733,8 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_unload", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def unload(self) -> None: | ||||
|         """Unload add-on and remove data.""" | ||||
| @@ -766,8 +766,8 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_install", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def install(self) -> None: | ||||
|         """Install and setup this addon.""" | ||||
| @@ -775,7 +775,6 @@ class Addon(AddonModel): | ||||
|             raise AddonsError("Missing from store, cannot install!") | ||||
|  | ||||
|         await self.sys_addons.data.install(self.addon_store) | ||||
|         await self.load() | ||||
|  | ||||
|         def setup_data(): | ||||
|             if not self.path_data.is_dir(): | ||||
| @@ -798,6 +797,9 @@ class Addon(AddonModel): | ||||
|             await self.sys_addons.data.uninstall(self) | ||||
|             raise AddonsError() from err | ||||
|  | ||||
|         # Finish initialization and set up listeners | ||||
|         await self.load() | ||||
|  | ||||
|         # Add to addon manager | ||||
|         self.sys_addons.local[self.slug] = self | ||||
|  | ||||
| @@ -807,8 +809,8 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_uninstall", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def uninstall( | ||||
|         self, *, remove_config: bool, remove_image: bool = True | ||||
| @@ -842,8 +844,7 @@ class Addon(AddonModel): | ||||
|         # Cleanup Ingress panel from sidebar | ||||
|         if self.ingress_panel: | ||||
|             self.ingress_panel = False | ||||
|             with suppress(HomeAssistantAPIError): | ||||
|                 await self.sys_ingress.update_hass_panel(self) | ||||
|             await self.sys_ingress.update_hass_panel(self) | ||||
|  | ||||
|         # Cleanup Ingress dynamic port assignment | ||||
|         need_ingress_token_cleanup = False | ||||
| @@ -873,8 +874,8 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_update", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def update(self) -> asyncio.Task | None: | ||||
|         """Update this addon to latest version. | ||||
| @@ -923,8 +924,8 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_rebuild", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def rebuild(self) -> asyncio.Task | None: | ||||
|         """Rebuild this addons container and image. | ||||
| @@ -1068,8 +1069,8 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_start", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def start(self) -> asyncio.Task: | ||||
|         """Set options and start add-on. | ||||
| @@ -1117,8 +1118,8 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_stop", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def stop(self) -> None: | ||||
|         """Stop add-on.""" | ||||
| @@ -1131,8 +1132,8 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_restart", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def restart(self) -> asyncio.Task: | ||||
|         """Restart add-on. | ||||
| @@ -1166,13 +1167,13 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_write_stdin", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def write_stdin(self, data) -> None: | ||||
|         """Write data to add-on stdin.""" | ||||
|         if not self.with_stdin: | ||||
|             raise AddonsNotSupportedError( | ||||
|             raise AddonNotSupportedError( | ||||
|                 f"Add-on {self.slug} does not support writing to stdin!", _LOGGER.error | ||||
|             ) | ||||
|  | ||||
| @@ -1200,8 +1201,8 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_begin_backup", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def begin_backup(self) -> bool: | ||||
|         """Execute pre commands or stop addon if necessary. | ||||
| @@ -1222,8 +1223,8 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_end_backup", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def end_backup(self) -> asyncio.Task | None: | ||||
|         """Execute post commands or restart addon if necessary. | ||||
| @@ -1260,8 +1261,8 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_backup", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def backup(self, tar_file: tarfile.TarFile) -> asyncio.Task | None: | ||||
|         """Backup state of an add-on. | ||||
| @@ -1359,9 +1360,7 @@ class Addon(AddonModel): | ||||
|             ) | ||||
|             _LOGGER.info("Finish backup for addon %s", self.slug) | ||||
|         except (tarfile.TarError, OSError, AddFileError) as err: | ||||
|             raise AddonsError( | ||||
|                 f"Can't write tarfile {tar_file}: {err}", _LOGGER.error | ||||
|             ) from err | ||||
|             raise AddonsError(f"Can't write tarfile: {err}", _LOGGER.error) from err | ||||
|         finally: | ||||
|             if was_running: | ||||
|                 wait_for_start = await self.end_backup() | ||||
| @@ -1370,8 +1369,8 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_restore", | ||||
|         limit=JobExecutionLimit.GROUP_ONCE, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.GROUP_REJECT, | ||||
|     ) | ||||
|     async def restore(self, tar_file: tarfile.TarFile) -> asyncio.Task | None: | ||||
|         """Restore state of an add-on. | ||||
| @@ -1421,7 +1420,7 @@ class Addon(AddonModel): | ||||
|  | ||||
|             # If available | ||||
|             if not self._available(data[ATTR_SYSTEM]): | ||||
|                 raise AddonsNotSupportedError( | ||||
|                 raise AddonNotSupportedError( | ||||
|                     f"Add-on {self.slug} is not available for this platform", | ||||
|                     _LOGGER.error, | ||||
|                 ) | ||||
| @@ -1523,10 +1522,10 @@ class Addon(AddonModel): | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_restart_after_problem", | ||||
|         limit=JobExecutionLimit.GROUP_THROTTLE_RATE_LIMIT, | ||||
|         throttle_period=WATCHDOG_THROTTLE_PERIOD, | ||||
|         throttle_max_calls=WATCHDOG_THROTTLE_MAX_CALLS, | ||||
|         on_condition=AddonsJobError, | ||||
|         throttle=JobThrottle.GROUP_RATE_LIMIT, | ||||
|     ) | ||||
|     async def _restart_after_problem(self, state: ContainerState): | ||||
|         """Restart unhealthy or failed addon.""" | ||||
|   | ||||
| @@ -15,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 | ||||
| @@ -121,39 +122,64 @@ class AddonBuild(FileConfiguration, CoreSysAttributes): | ||||
|         except HassioArchNotFound: | ||||
|             return False | ||||
|  | ||||
|     def get_docker_args(self, version: AwesomeVersion, image: str | None = None): | ||||
|         """Create a dict with Docker build arguments. | ||||
|     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) | ||||
|  | ||||
|         Must be run in executor. | ||||
|         """ | ||||
|         args: dict[str, Any] = { | ||||
|             "path": str(self.addon.path_location), | ||||
|             "tag": f"{image or self.addon.image}:{version!s}", | ||||
|             "dockerfile": str(self.get_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, | ||||
|             }, | ||||
|         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.""" | ||||
|   | ||||
| @@ -12,14 +12,15 @@ from attr import evolve | ||||
| from ..const import AddonBoot, AddonStartup, AddonState | ||||
| from ..coresys import CoreSys, CoreSysAttributes | ||||
| from ..exceptions import ( | ||||
|     AddonNotSupportedError, | ||||
|     AddonsError, | ||||
|     AddonsJobError, | ||||
|     AddonsNotSupportedError, | ||||
|     CoreDNSError, | ||||
|     DockerError, | ||||
|     HassioError, | ||||
|     HomeAssistantAPIError, | ||||
| ) | ||||
| from ..jobs import ChildJobSyncFilter | ||||
| from ..jobs.const import JobConcurrency | ||||
| from ..jobs.decorator import Job, JobCondition | ||||
| from ..resolution.const import ContextType, IssueType, SuggestionType | ||||
| from ..store.addon import AddonStore | ||||
| @@ -67,6 +68,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: | ||||
| @@ -176,8 +181,14 @@ class AddonManager(CoreSysAttributes): | ||||
|         name="addon_manager_install", | ||||
|         conditions=ADDON_UPDATE_CONDITIONS, | ||||
|         on_condition=AddonsJobError, | ||||
|         concurrency=JobConcurrency.QUEUE, | ||||
|         child_job_syncs=[ | ||||
|             ChildJobSyncFilter("docker_interface_install", progress_allocation=1.0) | ||||
|         ], | ||||
|     ) | ||||
|     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 | ||||
|  | ||||
| @@ -190,6 +201,10 @@ 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) | ||||
| @@ -217,9 +232,20 @@ class AddonManager(CoreSysAttributes): | ||||
|         name="addon_manager_update", | ||||
|         conditions=ADDON_UPDATE_CONDITIONS, | ||||
|         on_condition=AddonsJobError, | ||||
|         # We assume for now the docker image pull is 100% of this task for progress | ||||
|         # allocation. But from a user perspective that isn't true. Other steps | ||||
|         # that take time which is not accounted for in progress include: | ||||
|         # partial backup, image cleanup, apparmor update, and addon restart | ||||
|         child_job_syncs=[ | ||||
|             ChildJobSyncFilter("docker_interface_install", progress_allocation=1.0) | ||||
|         ], | ||||
|     ) | ||||
|     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. | ||||
|  | ||||
| @@ -244,6 +270,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}", | ||||
| @@ -251,7 +281,10 @@ class AddonManager(CoreSysAttributes): | ||||
|                 addons=[addon.slug], | ||||
|             ) | ||||
|  | ||||
|         return await addon.update() | ||||
|         task = await addon.update() | ||||
|  | ||||
|         _LOGGER.info("Add-on '%s' successfully updated", slug) | ||||
|         return task | ||||
|  | ||||
|     @Job( | ||||
|         name="addon_manager_rebuild", | ||||
| @@ -262,7 +295,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) | ||||
| @@ -285,8 +318,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 | ||||
|             ) | ||||
|  | ||||
| @@ -330,8 +363,7 @@ class AddonManager(CoreSysAttributes): | ||||
|         # Update ingress | ||||
|         if had_ingress != addon.ingress_panel: | ||||
|             await self.sys_ingress.reload() | ||||
|             with suppress(HomeAssistantAPIError): | ||||
|                 await self.sys_ingress.update_hass_panel(addon) | ||||
|             await self.sys_ingress.update_hass_panel(addon) | ||||
|  | ||||
|         return wait_for_start | ||||
|  | ||||
|   | ||||
| @@ -72,6 +72,7 @@ from ..const import ( | ||||
|     ATTR_TYPE, | ||||
|     ATTR_UART, | ||||
|     ATTR_UDEV, | ||||
|     ATTR_ULIMITS, | ||||
|     ATTR_URL, | ||||
|     ATTR_USB, | ||||
|     ATTR_VERSION, | ||||
| @@ -89,7 +90,12 @@ from ..const import ( | ||||
| ) | ||||
| 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 | ||||
| @@ -457,6 +463,11 @@ class AddonModel(JobGroup, ABC): | ||||
|         """Return True if the add-on have his own udev.""" | ||||
|         return self.data[ATTR_UDEV] | ||||
|  | ||||
|     @property | ||||
|     def ulimits(self) -> dict[str, Any]: | ||||
|         """Return ulimits configuration.""" | ||||
|         return self.data[ATTR_ULIMITS] | ||||
|  | ||||
|     @property | ||||
|     def with_kernel_modules(self) -> bool: | ||||
|         """Return True if the add-on access to kernel modules.""" | ||||
| @@ -645,7 +656,7 @@ class AddonModel(JobGroup, ABC): | ||||
|                 return None | ||||
|  | ||||
|             # Return data | ||||
|             return readme.read_text(encoding="utf-8") | ||||
|             return readme.read_text(encoding="utf-8", errors="replace") | ||||
|  | ||||
|         return await self.sys_run_in_executor(read_readme) | ||||
|  | ||||
| @@ -664,21 +675,24 @@ class AddonModel(JobGroup, ABC): | ||||
|         """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 | ||||
| @@ -686,9 +700,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 | ||||
| @@ -697,16 +710,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 | ||||
|   | ||||
| @@ -93,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})" | ||||
| @@ -111,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: | ||||
| @@ -182,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 = [] | ||||
|  | ||||
| @@ -201,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 = {} | ||||
|  | ||||
| @@ -231,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 | ||||
| @@ -274,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]], | ||||
| @@ -377,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, | ||||
| @@ -399,11 +404,7 @@ class UiOptions(CoreSysAttributes): | ||||
|  | ||||
|         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) | ||||
|   | ||||
| @@ -32,6 +32,7 @@ from ..const import ( | ||||
|     ATTR_DISCOVERY, | ||||
|     ATTR_DOCKER_API, | ||||
|     ATTR_ENVIRONMENT, | ||||
|     ATTR_FIELDS, | ||||
|     ATTR_FULL_ACCESS, | ||||
|     ATTR_GPIO, | ||||
|     ATTR_HASSIO_API, | ||||
| @@ -87,6 +88,7 @@ from ..const import ( | ||||
|     ATTR_TYPE, | ||||
|     ATTR_UART, | ||||
|     ATTR_UDEV, | ||||
|     ATTR_ULIMITS, | ||||
|     ATTR_URL, | ||||
|     ATTR_USB, | ||||
|     ATTR_USER, | ||||
| @@ -137,7 +139,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"^!?(?:" | ||||
| @@ -266,10 +280,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( | ||||
|                     { | ||||
| @@ -278,8 +305,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): | ||||
| @@ -393,23 +420,24 @@ _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, | ||||
|         vol.Optional(ATTR_ULIMITS, default=dict): vol.Any( | ||||
|             {str: vol.Coerce(int)},  # Simple format: {name: limit} | ||||
|             { | ||||
|                 str: vol.Any( | ||||
|                     vol.Coerce(int),  # Simple format for individual entries | ||||
|                     vol.Schema( | ||||
|                         {  # Detailed format for individual entries | ||||
|                             vol.Required("soft"): vol.Coerce(int), | ||||
|                             vol.Required("hard"): vol.Coerce(int), | ||||
|                         } | ||||
|                     ), | ||||
|                 ) | ||||
|             }, | ||||
|         ), | ||||
|         vol.Optional(ATTR_TIMEOUT, default=10): vol.All( | ||||
|             vol.Coerce(int), vol.Range(min=10, max=300) | ||||
|         ), | ||||
| @@ -442,6 +470,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, | ||||
| ) | ||||
|   | ||||
| @@ -8,7 +8,7 @@ from typing import Any | ||||
|  | ||||
| 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, HostNotSupportedError | ||||
| from ..utils.sentry import async_capture_exception | ||||
| @@ -146,6 +146,14 @@ class RestAPI(CoreSysAttributes): | ||||
|                         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), | ||||
| @@ -198,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), | ||||
|             ] | ||||
|         ) | ||||
|  | ||||
| @@ -426,7 +435,7 @@ class RestAPI(CoreSysAttributes): | ||||
|         async def get_supervisor_logs(*args, **kwargs): | ||||
|             try: | ||||
|                 return await self._api_host.advanced_logs_handler( | ||||
|                     *args, identifier="hassio_supervisor", **kwargs | ||||
|                     *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 | ||||
| @@ -439,6 +448,7 @@ class RestAPI(CoreSysAttributes): | ||||
|                     # 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( | ||||
| @@ -448,6 +458,10 @@ class RestAPI(CoreSysAttributes): | ||||
|                     "/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", | ||||
| @@ -560,6 +574,10 @@ class RestAPI(CoreSysAttributes): | ||||
|                     "/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", | ||||
| @@ -734,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 | ||||
|                 ), | ||||
| @@ -789,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), | ||||
|   | ||||
| @@ -36,6 +36,7 @@ from ..const import ( | ||||
|     ATTR_DNS, | ||||
|     ATTR_DOCKER_API, | ||||
|     ATTR_DOCUMENTATION, | ||||
|     ATTR_FORCE, | ||||
|     ATTR_FULL_ACCESS, | ||||
|     ATTR_GPIO, | ||||
|     ATTR_HASSIO_API, | ||||
| @@ -139,6 +140,8 @@ 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 | ||||
|  | ||||
|  | ||||
| @@ -303,7 +306,7 @@ 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: | ||||
| @@ -461,7 +464,11 @@ class APIAddons(CoreSysAttributes): | ||||
|     async def rebuild(self, request: web.Request) -> None: | ||||
|         """Rebuild local build add-on.""" | ||||
|         addon = self.get_addon_for_request(request) | ||||
|         if start_task := await asyncio.shield(self.sys_addons.rebuild(addon.slug)): | ||||
|         body: dict[str, Any] = await api_validate(SCHEMA_REBUILD, request) | ||||
|  | ||||
|         if start_task := await asyncio.shield( | ||||
|             self.sys_addons.rebuild(addon.slug, force=body[ATTR_FORCE]) | ||||
|         ): | ||||
|             await start_task | ||||
|  | ||||
|     @api_process | ||||
|   | ||||
| @@ -3,11 +3,13 @@ | ||||
| import asyncio | ||||
| from collections.abc import Awaitable | ||||
| import logging | ||||
| from typing import Any | ||||
| 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 | ||||
| @@ -51,7 +53,10 @@ 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] | ||||
|         self, | ||||
|         request: web.Request, | ||||
|         addon: Addon, | ||||
|         data: dict[str, Any] | MultiDictProxy[str | bytes | FileField], | ||||
|     ) -> Awaitable[bool]: | ||||
|         """Process login with dict data. | ||||
|  | ||||
| @@ -60,7 +65,15 @@ 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: | ||||
| @@ -79,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 | ||||
|   | ||||
| @@ -3,7 +3,6 @@ | ||||
| from __future__ import annotations | ||||
|  | ||||
| import asyncio | ||||
| from collections.abc import Callable | ||||
| import errno | ||||
| from io import IOBase | ||||
| import logging | ||||
| @@ -46,12 +45,9 @@ from ..const import ( | ||||
|     ATTR_TYPE, | ||||
|     ATTR_VERSION, | ||||
|     REQUEST_FROM, | ||||
|     BusEvent, | ||||
|     CoreState, | ||||
| ) | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APIError, APIForbidden, APINotFound | ||||
| from ..jobs import JobSchedulerOptions, SupervisorJob | ||||
| from ..mounts.const import MountUsage | ||||
| from ..resolution.const import UnhealthyReason | ||||
| from .const import ( | ||||
| @@ -61,7 +57,7 @@ from .const import ( | ||||
|     ATTR_LOCATIONS, | ||||
|     CONTENT_TYPE_TAR, | ||||
| ) | ||||
| from .utils import api_process, api_validate | ||||
| from .utils import api_process, api_validate, background_task | ||||
|  | ||||
| _LOGGER: logging.Logger = logging.getLogger(__name__) | ||||
|  | ||||
| @@ -289,41 +285,6 @@ class APIBackups(CoreSysAttributes): | ||||
|                 f"Location {LOCATION_CLOUD_BACKUP} is only available for Home Assistant" | ||||
|             ) | ||||
|  | ||||
|     async def _background_backup_task( | ||||
|         self, backup_method: Callable, *args, **kwargs | ||||
|     ) -> tuple[asyncio.Task, str]: | ||||
|         """Start backup task in  background and return task and job ID.""" | ||||
|         event = asyncio.Event() | ||||
|         job, backup_task = cast( | ||||
|             tuple[SupervisorJob, asyncio.Task], | ||||
|             self.sys_jobs.schedule_job( | ||||
|                 backup_method, JobSchedulerOptions(), *args, **kwargs | ||||
|             ), | ||||
|         ) | ||||
|  | ||||
|         async def release_on_freeze(new_state: CoreState): | ||||
|             if new_state == CoreState.FREEZE: | ||||
|                 event.set() | ||||
|  | ||||
|         # Wait for system to get into freeze state before returning | ||||
|         # If the backup fails validation it will raise before getting there | ||||
|         listener = self.sys_bus.register_event( | ||||
|             BusEvent.SUPERVISOR_STATE_CHANGE, release_on_freeze | ||||
|         ) | ||||
|         try: | ||||
|             event_task = self.sys_create_task(event.wait()) | ||||
|             _, pending = await asyncio.wait( | ||||
|                 (backup_task, event_task), | ||||
|                 return_when=asyncio.FIRST_COMPLETED, | ||||
|             ) | ||||
|             # It seems backup returned early (error or something), make sure to cancel | ||||
|             # the event task to avoid "Task was destroyed but it is pending!" errors. | ||||
|             if event_task in pending: | ||||
|                 event_task.cancel() | ||||
|             return (backup_task, job.uuid) | ||||
|         finally: | ||||
|             self.sys_bus.remove_listener(listener) | ||||
|  | ||||
|     @api_process | ||||
|     async def backup_full(self, request: web.Request): | ||||
|         """Create full backup.""" | ||||
| @@ -342,8 +303,8 @@ class APIBackups(CoreSysAttributes): | ||||
|                 body[ATTR_ADDITIONAL_LOCATIONS] = locations | ||||
|  | ||||
|         background = body.pop(ATTR_BACKGROUND) | ||||
|         backup_task, job_id = await self._background_backup_task( | ||||
|             self.sys_backups.do_backup_full, **body | ||||
|         backup_task, job_id = await background_task( | ||||
|             self, self.sys_backups.do_backup_full, **body | ||||
|         ) | ||||
|  | ||||
|         if background and not backup_task.done(): | ||||
| @@ -378,8 +339,8 @@ class APIBackups(CoreSysAttributes): | ||||
|             body[ATTR_ADDONS] = list(self.sys_addons.local) | ||||
|  | ||||
|         background = body.pop(ATTR_BACKGROUND) | ||||
|         backup_task, job_id = await self._background_backup_task( | ||||
|             self.sys_backups.do_backup_partial, **body | ||||
|         backup_task, job_id = await background_task( | ||||
|             self, self.sys_backups.do_backup_partial, **body | ||||
|         ) | ||||
|  | ||||
|         if background and not backup_task.done(): | ||||
| @@ -402,8 +363,8 @@ class APIBackups(CoreSysAttributes): | ||||
|             request, body.get(ATTR_LOCATION, backup.location) | ||||
|         ) | ||||
|         background = body.pop(ATTR_BACKGROUND) | ||||
|         restore_task, job_id = await self._background_backup_task( | ||||
|             self.sys_backups.do_restore_full, backup, **body | ||||
|         restore_task, job_id = await background_task( | ||||
|             self, self.sys_backups.do_restore_full, backup, **body | ||||
|         ) | ||||
|  | ||||
|         if background and not restore_task.done() or await restore_task: | ||||
| @@ -422,8 +383,8 @@ class APIBackups(CoreSysAttributes): | ||||
|             request, body.get(ATTR_LOCATION, backup.location) | ||||
|         ) | ||||
|         background = body.pop(ATTR_BACKGROUND) | ||||
|         restore_task, job_id = await self._background_backup_task( | ||||
|             self.sys_backups.do_restore_partial, backup, **body | ||||
|         restore_task, job_id = await background_task( | ||||
|             self, self.sys_backups.do_restore_partial, backup, **body | ||||
|         ) | ||||
|  | ||||
|         if background and not restore_task.done() or await restore_task: | ||||
|   | ||||
| @@ -49,6 +49,7 @@ 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" | ||||
| @@ -87,4 +88,4 @@ class DetectBlockingIO(StrEnum): | ||||
|  | ||||
|     OFF = "off" | ||||
|     ON = "on" | ||||
|     ON_AT_STARTUP = "on_at_startup" | ||||
|     ON_AT_STARTUP = "on-at-startup" | ||||
|   | ||||
| @@ -1,7 +1,7 @@ | ||||
| """Init file for Supervisor network RESTful API.""" | ||||
|  | ||||
| import logging | ||||
| from typing import Any, cast | ||||
| from typing import Any | ||||
|  | ||||
| from aiohttp import web | ||||
| import voluptuous as vol | ||||
| @@ -56,8 +56,8 @@ class APIDiscovery(CoreSysAttributes): | ||||
|             } | ||||
|             for message in self.sys_discovery.list_messages | ||||
|             if ( | ||||
|                 discovered := cast( | ||||
|                     Addon, self.sys_addons.get(message.addon, local_only=True) | ||||
|                 discovered := self.sys_addons.get_local_only( | ||||
|                     message.addon, | ||||
|                 ) | ||||
|             ) | ||||
|             and discovered.state == AddonState.STARTED | ||||
|   | ||||
| @@ -6,9 +6,13 @@ 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, | ||||
| @@ -30,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.""" | ||||
| @@ -64,18 +123,3 @@ class APIDocker(CoreSysAttributes): | ||||
|  | ||||
|         del self.sys_docker.config.registries[hostname] | ||||
|         await self.sys_docker.config.save_data() | ||||
|  | ||||
|     @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, | ||||
|         } | ||||
|   | ||||
| @@ -20,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, | ||||
| @@ -37,8 +38,8 @@ from ..const import ( | ||||
| from ..coresys import CoreSysAttributes | ||||
| from ..exceptions import APIDBMigrationInProgress, APIError | ||||
| from ..validate import docker_image, network_port, version_tag | ||||
| from .const import ATTR_FORCE, ATTR_SAFE_MODE | ||||
| from .utils import api_process, 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__) | ||||
|  | ||||
| @@ -61,6 +62,7 @@ SCHEMA_UPDATE = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(ATTR_VERSION): version_tag, | ||||
|         vol.Optional(ATTR_BACKUP): bool, | ||||
|         vol.Optional(ATTR_BACKGROUND, default=False): bool, | ||||
|     } | ||||
| ) | ||||
|  | ||||
| @@ -170,18 +172,24 @@ 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 | ||||
|     async def stop(self, request: web.Request) -> Awaitable[None]: | ||||
|         """Stop Home Assistant.""" | ||||
|   | ||||
| @@ -2,10 +2,17 @@ | ||||
|  | ||||
| import asyncio | ||||
| from contextlib import suppress | ||||
| import json | ||||
| import logging | ||||
| from typing import Any | ||||
|  | ||||
| from aiohttp import ClientConnectionResetError, ClientPayloadError, web | ||||
| from aiohttp import ( | ||||
|     ClientConnectionResetError, | ||||
|     ClientError, | ||||
|     ClientPayloadError, | ||||
|     ClientTimeout, | ||||
|     web, | ||||
| ) | ||||
| from aiohttp.hdrs import ACCEPT, RANGE | ||||
| import voluptuous as vol | ||||
| from voluptuous.error import CoerceInvalid | ||||
| @@ -51,6 +58,7 @@ from .const import ( | ||||
|     ATTR_FORCE, | ||||
|     ATTR_IDENTIFIERS, | ||||
|     ATTR_LLMNR_HOSTNAME, | ||||
|     ATTR_MAX_DEPTH, | ||||
|     ATTR_STARTUP_TIME, | ||||
|     ATTR_USE_NTP, | ||||
|     ATTR_VIRTUALIZATION, | ||||
| @@ -193,7 +201,11 @@ class APIHost(CoreSysAttributes): | ||||
|         return possible_offset | ||||
|  | ||||
|     async def advanced_logs_handler( | ||||
|         self, request: web.Request, identifier: str | None = None, follow: bool = False | ||||
|         self, | ||||
|         request: web.Request, | ||||
|         identifier: str | None = None, | ||||
|         follow: bool = False, | ||||
|         latest: bool = False, | ||||
|     ) -> web.StreamResponse: | ||||
|         """Return systemd-journald logs.""" | ||||
|         log_formatter = LogFormatter.PLAIN | ||||
| @@ -212,6 +224,20 @@ class APIHost(CoreSysAttributes): | ||||
|         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, | ||||
| @@ -240,6 +266,8 @@ class APIHost(CoreSysAttributes): | ||||
|                 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: | ||||
| @@ -269,6 +297,13 @@ class APIHost(CoreSysAttributes): | ||||
|                             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( | ||||
| @@ -278,7 +313,81 @@ class APIHost(CoreSysAttributes): | ||||
|  | ||||
|     @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 | ||||
|         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) | ||||
|         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 | ||||
|   | ||||
| @@ -199,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 | ||||
|  | ||||
| @@ -286,6 +290,7 @@ class APIIngress(CoreSysAttributes): | ||||
|                 aiohttp.ClientError, | ||||
|                 aiohttp.ClientPayloadError, | ||||
|                 ConnectionResetError, | ||||
|                 ConnectionError, | ||||
|             ) as err: | ||||
|                 _LOGGER.error("Stream error with %s: %s", url, err) | ||||
|  | ||||
| @@ -309,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 | ||||
| @@ -337,7 +342,7 @@ def _init_header( | ||||
|             istr(HEADER_REMOTE_USER_DISPLAY_NAME), | ||||
|         ): | ||||
|             continue | ||||
|         headers[name] = value | ||||
|         headers.add(name, value) | ||||
|  | ||||
|     # Update X-Forwarded-For | ||||
|     if request.transport: | ||||
| @@ -348,9 +353,9 @@ def _init_header( | ||||
|     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 ( | ||||
| @@ -360,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 | ||||
|  | ||||
| @@ -386,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: | ||||
|   | ||||
| @@ -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,17 +42,21 @@ 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, APINotFound, HostNetworkNotFound | ||||
| from ..host.configuration import ( | ||||
|     AccessPoint, | ||||
|     Interface, | ||||
|     InterfaceAddrGenMode, | ||||
|     InterfaceIp6Privacy, | ||||
|     InterfaceMethod, | ||||
|     Ip6Setting, | ||||
|     IpConfig, | ||||
|     IpSetting, | ||||
|     MulticastDnsMode, | ||||
|     VlanConfig, | ||||
|     WifiConfig, | ||||
| ) | ||||
| @@ -68,6 +76,8 @@ _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)], | ||||
|     } | ||||
| @@ -90,12 +100,14 @@ SCHEMA_UPDATE = vol.Schema( | ||||
|         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, setting: IpSetting) -> 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: setting.method, | ||||
|         ATTR_ADDRESS: [address.with_prefixlen for address in config.address], | ||||
| @@ -105,6 +117,19 @@ def ipconfig_struct(config: IpConfig, setting: IpSetting) -> dict[str, Any]: | ||||
|     } | ||||
|  | ||||
|  | ||||
| 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, | ||||
|         ATTR_READY: config.ready, | ||||
|     } | ||||
|  | ||||
|  | ||||
| def wifi_struct(config: WifiConfig) -> dict[str, Any]: | ||||
|     """Return a dict with information about wifi configuration.""" | ||||
|     return { | ||||
| @@ -132,14 +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, interface.ipv4setting) | ||||
|         ATTR_IPV4: ip4config_struct(interface.ipv4, interface.ipv4setting) | ||||
|         if interface.ipv4 and interface.ipv4setting | ||||
|         else None, | ||||
|         ATTR_IPV6: ipconfig_struct(interface.ipv6, interface.ipv6setting) | ||||
|         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, | ||||
|     } | ||||
|  | ||||
|  | ||||
| @@ -183,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), | ||||
|             }, | ||||
| @@ -212,28 +239,38 @@ class APINetwork(CoreSysAttributes): | ||||
|         for key, config in body.items(): | ||||
|             if key == ATTR_IPV4: | ||||
|                 interface.ipv4setting = IpSetting( | ||||
|                     config.get(ATTR_METHOD, InterfaceMethod.STATIC), | ||||
|                     config.get(ATTR_ADDRESS, []), | ||||
|                     config.get(ATTR_GATEWAY), | ||||
|                     config.get(ATTR_NAMESERVERS, []), | ||||
|                     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.ipv6setting = IpSetting( | ||||
|                     config.get(ATTR_METHOD, InterfaceMethod.STATIC), | ||||
|                     config.get(ATTR_ADDRESS, []), | ||||
|                     config.get(ATTR_GATEWAY), | ||||
|                     config.get(ATTR_NAMESERVERS, []), | ||||
|                 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 = WifiConfig( | ||||
|                     config.get(ATTR_MODE, WifiMode.INFRASTRUCTURE), | ||||
|                     config.get(ATTR_SSID, ""), | ||||
|                     config.get(ATTR_AUTH, AuthMethod.OPEN), | ||||
|                     config.get(ATTR_PSK, None), | ||||
|                     None, | ||||
|                     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)) | ||||
|  | ||||
| @@ -274,26 +311,41 @@ class APINetwork(CoreSysAttributes): | ||||
|  | ||||
|         vlan_config = VlanConfig(vlan, interface.name) | ||||
|  | ||||
|         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_setting = IpSetting( | ||||
|                 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, []), | ||||
|                 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_setting = None | ||||
|         if ATTR_IPV6 in body: | ||||
|             ipv6_setting = IpSetting( | ||||
|                 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, []), | ||||
|             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, | ||||
| @@ -306,5 +358,7 @@ class APINetwork(CoreSysAttributes): | ||||
|             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 +1 @@ | ||||
| !function(){function d(d){var e=document.createElement("script");e.src=d,document.body.appendChild(e)}if(/Edge?\/(12[4-9]|1[3-9]\d|[2-9]\d{2}|\d{4,})\.\d+(\.\d+|)|Firefox\/(12[5-9]|1[3-9]\d|[2-9]\d{2}|\d{4,})\.\d+(\.\d+|)|Chrom(ium|e)\/(109|1[1-9]\d|[2-9]\d{2}|\d{4,})\.\d+(\.\d+|)|(Maci|X1{2}).+ Version\/(17\.([5-9]|\d{2,})|(1[89]|[2-9]\d|\d{3,})\.\d+)([,.]\d+|)( \(\w+\)|)( Mobile\/\w+|) Safari\/|Chrome.+OPR\/(1{2}\d|1[2-9]\d|[2-9]\d{2}|\d{4,})\.\d+\.\d+|(CPU[ +]OS|iPhone[ +]OS|CPU[ +]iPhone|CPU IPhone OS|CPU iPad OS)[ +]+(15[._]([6-9]|\d{2,})|(1[6-9]|[2-9]\d|\d{3,})[._]\d+)([._]\d+|)|Android:?[ /-](12[4-9]|1[3-9]\d|[2-9]\d{2}|\d{4,})(\.\d+|)(\.\d+|)|Mobile Safari.+OPR\/([89]\d|\d{3,})\.\d+\.\d+|Android.+Firefox\/(12[5-9]|1[3-9]\d|[2-9]\d{2}|\d{4,})\.\d+(\.\d+|)|Android.+Chrom(ium|e)\/(12[4-9]|1[3-9]\d|[2-9]\d{2}|\d{4,})\.\d+(\.\d+|)|SamsungBrowser\/(2[5-9]|[3-9]\d|\d{3,})\.\d+|Home As{2}istant\/[\d.]+ \(.+; macOS (1[2-9]|[2-9]\d|\d{3,})\.\d+(\.\d+)?\)/.test(navigator.userAgent))try{new Function("import('/api/hassio/app/frontend_latest/entrypoint.35399ae87c70acf8.js')")()}catch(e){d("/api/hassio/app/frontend_es5/entrypoint.476bfed22da63267.js")}else d("/api/hassio/app/frontend_es5/entrypoint.476bfed22da63267.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")}() | ||||
										
											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
											
										
									
								
							
							
								
								
									
										
											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
											
										
									
								
							
							
								
								
									
										
											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.
										
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -1 +0,0 @@ | ||||
| {"version":3,"file":"1081.91949d686e61cc12.js","sources":["https://raw.githubusercontent.com/home-assistant/frontend/20250401.0/src/components/ha-button-toggle-group.ts","https://raw.githubusercontent.com/home-assistant/frontend/20250401.0/src/components/ha-selector/ha-selector-button-toggle.ts"],"names":["_decorate","customElement","_initialize","_LitElement","F","constructor","args","d","kind","decorators","property","attribute","key","value","type","Boolean","queryAll","html","_t","_","this","buttons","map","button","iconPath","_t2","label","active","_handleClick","_t3","styleMap","width","fullWidth","length","dense","_this$_buttons","_buttons","forEach","async","updateComplete","shadowRoot","querySelector","style","margin","ev","currentTarget","fireEvent","static","css","_t4","LitElement","HaButtonToggleSelector","_this$selector$button","_this$selector$button2","_this$selector$button3","options","selector","button_toggle","option","translationKey","translation_key","localizeValue","localizedLabel","sort","a","b","caseInsensitiveStringCompare","hass","locale","language","toggleButtons","item","_valueChanged","_ev$detail","_this$value","stopPropagation","detail","target","disabled","undefined"],"mappings":"qXAWgCA,EAAAA,EAAAA,GAAA,EAD/BC,EAAAA,EAAAA,IAAc,4BAAyB,SAAAC,EAAAC,GAkIvC,OAAAC,EAlID,cACgCD,EAAoBE,WAAAA,IAAAC,GAAA,SAAAA,GAAAJ,EAAA,QAApBK,EAAA,EAAAC,KAAA,QAAAC,WAAA,EAC7BC,EAAAA,EAAAA,IAAS,CAAEC,WAAW,KAAQC,IAAA,UAAAC,WAAA,IAAAL,KAAA,QAAAC,WAAA,EAE9BC,EAAAA,EAAAA,OAAUE,IAAA,SAAAC,WAAA,IAAAL,KAAA,QAAAC,WAAA,EAEVC,EAAAA,EAAAA,IAAS,CAAEC,UAAW,aAAcG,KAAMC,WAAUH,IAAA,YAAAC,KAAAA,GAAA,OAClC,CAAK,IAAAL,KAAA,QAAAC,WAAA,EAEvBC,EAAAA,EAAAA,IAAS,CAAEI,KAAMC,WAAUH,IAAA,QAAAC,KAAAA,GAAA,OAAgB,CAAK,IAAAL,KAAA,QAAAC,WAAA,EAEhDO,EAAAA,EAAAA,IAAS,eAAaJ,IAAA,WAAAC,WAAA,IAAAL,KAAA,SAAAI,IAAA,SAAAC,MAEvB,WACE,OAAOI,EAAAA,EAAAA,IAAIC,IAAAA,EAAAC,CAAA,uBAELC,KAAKC,QAAQC,KAAKC,GAClBA,EAAOC,UACHP,EAAAA,EAAAA,IAAIQ,IAAAA,EAAAN,CAAA,2GACOI,EAAOG,MACRH,EAAOC,SACND,EAAOV,MACNO,KAAKO,SAAWJ,EAAOV,MACxBO,KAAKQ,eAEhBX,EAAAA,EAAAA,IAAIY,IAAAA,EAAAV,CAAA,iHACMW,EAAAA,EAAAA,GAAS,CACfC,MAAOX,KAAKY,UACL,IAAMZ,KAAKC,QAAQY,OAAtB,IACA,YAGGb,KAAKc,MACLX,EAAOV,MACNO,KAAKO,SAAWJ,EAAOV,MACxBO,KAAKQ,aACXL,EAAOG,SAKxB,GAAC,CAAAlB,KAAA,SAAAI,IAAA,UAAAC,MAED,WAAoB,IAAAsB,EAEL,QAAbA,EAAAf,KAAKgB,gBAAQ,IAAAD,GAAbA,EAAeE,SAAQC,gBACff,EAAOgB,eAEXhB,EAAOiB,WAAYC,cAAc,UACjCC,MAAMC,OAAS,GAAG,GAExB,GAAC,CAAAnC,KAAA,SAAAI,IAAA,eAAAC,MAED,SAAqB+B,GACnBxB,KAAKO,OAASiB,EAAGC,cAAchC,OAC/BiC,EAAAA,EAAAA,GAAU1B,KAAM,gBAAiB,CAAEP,MAAOO,KAAKO,QACjD,GAAC,CAAAnB,KAAA,QAAAuC,QAAA,EAAAnC,IAAA,SAAAC,KAAAA,GAAA,OAEemC,EAAAA,EAAAA,IAAGC,IAAAA,EAAA9B,CAAA,u0CAzDoB+B,EAAAA,I,MCD5BC,GAAsBnD,EAAAA,EAAAA,GAAA,EADlCC,EAAAA,EAAAA,IAAc,+BAA4B,SAAAC,EAAAC,GA4F1C,OAAAC,EA5FD,cACmCD,EAAoBE,WAAAA,IAAAC,GAAA,SAAAA,GAAAJ,EAAA,QAApBK,EAAA,EAAAC,KAAA,QAAAC,WAAA,EAChCC,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,OAAUE,IAAA,QAAAC,WAAA,IAAAL,KAAA,QAAAC,WAAA,EAEVC,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,CAAEC,WAAW,KAAQC,IAAA,gBAAAC,WAAA,IAAAL,KAAA,QAAAC,WAAA,EAG9BC,EAAAA,EAAAA,IAAS,CAAEI,KAAMC,WAAUH,IAAA,WAAAC,KAAAA,GAAA,OAAmB,CAAK,IAAAL,KAAA,QAAAC,WAAA,EAEnDC,EAAAA,EAAAA,IAAS,CAAEI,KAAMC,WAAUH,IAAA,WAAAC,KAAAA,GAAA,OAAmB,CAAI,IAAAL,KAAA,SAAAI,IAAA,SAAAC,MAEnD,WAAmB,IAAAuC,EAAAC,EAAAC,EACjB,MAAMC,GACuB,QAA3BH,EAAAhC,KAAKoC,SAASC,qBAAa,IAAAL,GAAS,QAATA,EAA3BA,EAA6BG,eAAO,IAAAH,OAAA,EAApCA,EAAsC9B,KAAKoC,GACvB,iBAAXA,EACFA,EACA,CAAE7C,MAAO6C,EAAQhC,MAAOgC,OAC1B,GAEDC,EAA4C,QAA9BN,EAAGjC,KAAKoC,SAASC,qBAAa,IAAAJ,OAAA,EAA3BA,EAA6BO,gBAEhDxC,KAAKyC,eAAiBF,GACxBJ,EAAQlB,SAASqB,IACf,MAAMI,EAAiB1C,KAAKyC,cAC1B,GAAGF,aAA0BD,EAAO7C,SAElCiD,IACFJ,EAAOhC,MAAQoC,EACjB,IAI2B,QAA/BR,EAAIlC,KAAKoC,SAASC,qBAAa,IAAAH,GAA3BA,EAA6BS,MAC/BR,EAAQQ,MAAK,CAACC,EAAGC,KACfC,EAAAA,EAAAA,IACEF,EAAEtC,MACFuC,EAAEvC,MACFN,KAAK+C,KAAKC,OAAOC,YAKvB,MAAMC,EAAgCf,EAAQjC,KAAKiD,IAAkB,CACnE7C,MAAO6C,EAAK7C,MACZb,MAAO0D,EAAK1D,UAGd,OAAOI,EAAAA,EAAAA,IAAIC,IAAAA,EAAAC,CAAA,iHACPC,KAAKM,MAEM4C,EACDlD,KAAKP,MACEO,KAAKoD,cAG5B,GAAC,CAAAhE,KAAA,SAAAI,IAAA,gBAAAC,MAED,SAAsB+B,GAAI,IAAA6B,EAAAC,EACxB9B,EAAG+B,kBAEH,MAAM9D,GAAiB,QAAT4D,EAAA7B,EAAGgC,cAAM,IAAAH,OAAA,EAATA,EAAW5D,QAAS+B,EAAGiC,OAAOhE,MACxCO,KAAK0D,eAAsBC,IAAVlE,GAAuBA,KAAqB,QAAhB6D,EAAMtD,KAAKP,aAAK,IAAA6D,EAAAA,EAAI,MAGrE5B,EAAAA,EAAAA,GAAU1B,KAAM,gBAAiB,CAC/BP,MAAOA,GAEX,GAAC,CAAAL,KAAA,QAAAuC,QAAA,EAAAnC,IAAA,SAAAC,KAAAA,GAAA,OAEemC,EAAAA,EAAAA,IAAGvB,IAAAA,EAAAN,CAAA,wLA5EuB+B,EAAAA,G"} | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1180.89c3426e7a24fa5c.js.br
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1180.89c3426e7a24fa5c.js.br
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1180.89c3426e7a24fa5c.js.gz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/1180.89c3426e7a24fa5c.js.gz
									
									
									
									
									
										Normal file
									
								
							
										
											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([["12"],{5739:function(e,a,t){t.a(e,(async function(e,i){try{t.r(a),t.d(a,{HaNavigationSelector:()=>c});var d=t(73577),r=(t(71695),t(47021),t(57243)),n=t(50778),l=t(36522),o=t(63297),s=e([o]);o=(s.then?(await s)():s)[0];let u,h=e=>e,c=(0,d.Z)([(0,n.Mo)("ha-selector-navigation")],(function(e,a){return{F:class extends a{constructor(...a){super(...a),e(this)}},d:[{kind:"field",decorators:[(0,n.Cb)({attribute:!1})],key:"hass",value:void 0},{kind:"field",decorators:[(0,n.Cb)({attribute:!1})],key:"selector",value:void 0},{kind:"field",decorators:[(0,n.Cb)()],key:"value",value:void 0},{kind:"field",decorators:[(0,n.Cb)()],key:"label",value:void 0},{kind:"field",decorators:[(0,n.Cb)()],key:"helper",value:void 0},{kind:"field",decorators:[(0,n.Cb)({type:Boolean,reflect:!0})],key:"disabled",value(){return!1}},{kind:"field",decorators:[(0,n.Cb)({type:Boolean})],key:"required",value(){return!0}},{kind:"method",key:"render",value:function(){return(0,r.dy)(u||(u=h` <ha-navigation-picker .hass="${0}" .label="${0}" .value="${0}" .required="${0}" .disabled="${0}" .helper="${0}" @value-changed="${0}"></ha-navigation-picker> `),this.hass,this.label,this.value,this.required,this.disabled,this.helper,this._valueChanged)}},{kind:"method",key:"_valueChanged",value:function(e){(0,l.B)(this,"value-changed",{value:e.detail.value})}}]}}),r.oi);i()}catch(u){i(u)}}))}}]); | ||||
| //# sourceMappingURL=12.ffa1bdc0a98802fa.js.map | ||||
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -1 +0,0 @@ | ||||
| {"version":3,"file":"12.ffa1bdc0a98802fa.js","sources":["https://raw.githubusercontent.com/home-assistant/frontend/20250401.0/src/components/ha-selector/ha-selector-navigation.ts"],"names":["HaNavigationSelector","_decorate","customElement","_initialize","_LitElement","F","constructor","args","d","kind","decorators","property","attribute","key","value","type","Boolean","reflect","html","_t","_","this","hass","label","required","disabled","helper","_valueChanged","ev","fireEvent","detail","LitElement"],"mappings":"mVAQaA,GAAoBC,EAAAA,EAAAA,GAAA,EADhCC,EAAAA,EAAAA,IAAc,4BAAyB,SAAAC,EAAAC,GAiCvC,OAAAC,EAjCD,cACiCD,EAAoBE,WAAAA,IAAAC,GAAA,SAAAA,GAAAJ,EAAA,QAApBK,EAAA,EAAAC,KAAA,QAAAC,WAAA,EAC9BC,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,OAAUE,IAAA,QAAAC,WAAA,IAAAL,KAAA,QAAAC,WAAA,EAEVC,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,QAASC,SAAS,KAAOJ,IAAA,WAAAC,KAAAA,GAAA,OAAmB,CAAK,IAAAL,KAAA,QAAAC,WAAA,EAElEC,EAAAA,EAAAA,IAAS,CAAEI,KAAMC,WAAUH,IAAA,WAAAC,KAAAA,GAAA,OAAmB,CAAI,IAAAL,KAAA,SAAAI,IAAA,SAAAC,MAEnD,WACE,OAAOI,EAAAA,EAAAA,IAAIC,IAAAA,EAAAC,CAAA,mKAECC,KAAKC,KACJD,KAAKE,MACLF,KAAKP,MACFO,KAAKG,SACLH,KAAKI,SACPJ,KAAKK,OACEL,KAAKM,cAG5B,GAAC,CAAAlB,KAAA,SAAAI,IAAA,gBAAAC,MAED,SAAsBc,IACpBC,EAAAA,EAAAA,GAAUR,KAAM,gBAAiB,CAAEP,MAAOc,EAAGE,OAAOhB,OACtD,IAAC,GA/BuCiB,EAAAA,I"} | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/120.c5f670671b56cb1c.js.br
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/120.c5f670671b56cb1c.js.br
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/120.c5f670671b56cb1c.js.gz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/120.c5f670671b56cb1c.js.gz
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,2 +0,0 @@ | ||||
| (self.webpackChunkhome_assistant_frontend=self.webpackChunkhome_assistant_frontend||[]).push([["1236"],{4121:function(){Intl.PluralRules&&"function"==typeof Intl.PluralRules.__addLocaleData&&Intl.PluralRules.__addLocaleData({data:{categories:{cardinal:["one","other"],ordinal:["one","two","few","other"]},fn:function(e,n){var t=String(e).split("."),a=!t[1],l=Number(t[0])==e,o=l&&t[0].slice(-1),r=l&&t[0].slice(-2);return n?1==o&&11!=r?"one":2==o&&12!=r?"two":3==o&&13!=r?"few":"other":1==e&&a?"one":"other"}},locale:"en"})}}]); | ||||
| //# sourceMappingURL=1236.64ca65d0ea4d76d4.js.map | ||||
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -1 +0,0 @@ | ||||
| {"version":3,"file":"1236.64ca65d0ea4d76d4.js","sources":["/unknown/node_modules/@formatjs/intl-pluralrules/locale-data/en.js"],"names":["Intl","PluralRules","__addLocaleData","n","ord","s","String","split","v0","t0","Number","n10","slice","n100"],"mappings":"wHAEIA,KAAKC,aAA2D,mBAArCD,KAAKC,YAAYC,iBAC9CF,KAAKC,YAAYC,gBAAgB,CAAC,KAAO,CAAC,WAAa,CAAC,SAAW,CAAC,MAAM,SAAS,QAAU,CAAC,MAAM,MAAM,MAAM,UAAU,GAAK,SAASC,EAAGC,GAC3I,IAAIC,EAAIC,OAAOH,GAAGI,MAAM,KAAMC,GAAMH,EAAE,GAAII,EAAKC,OAAOL,EAAE,KAAOF,EAAGQ,EAAMF,GAAMJ,EAAE,GAAGO,OAAO,GAAIC,EAAOJ,GAAMJ,EAAE,GAAGO,OAAO,GACvH,OAAIR,EAAmB,GAAPO,GAAoB,IAARE,EAAa,MAC9B,GAAPF,GAAoB,IAARE,EAAa,MAClB,GAAPF,GAAoB,IAARE,EAAa,MACzB,QACQ,GAALV,GAAUK,EAAK,MAAQ,OAChC,GAAG,OAAS,M"} | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1 +0,0 @@ | ||||
| "use strict";(self.webpackChunkhome_assistant_frontend=self.webpackChunkhome_assistant_frontend||[]).push([["1295"],{21393:function(s,n,e){e.r(n)}}]); | ||||
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/131.566df1af9c07775a.js.br
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/131.566df1af9c07775a.js.br
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/131.566df1af9c07775a.js.gz
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								supervisor/api/panel/frontend_es5/131.566df1af9c07775a.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.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user