mirror of
				https://github.com/home-assistant/core.git
				synced 2025-10-22 10:09:33 +00:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			cursor/add
			...
			get_config
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 6d082a87a4 | 
| @@ -1,77 +0,0 @@ | ||||
| --- | ||||
| name: quality-scale-rule-verifier | ||||
| description: | | ||||
|   Use this agent when you need to verify that a Home Assistant integration follows a specific quality scale rule. This includes checking if the integration implements required patterns, configurations, or code structures defined by the quality scale system. | ||||
|  | ||||
|   <example> | ||||
|   Context: The user wants to verify if an integration follows a specific quality scale rule. | ||||
|   user: "Check if the peblar integration follows the config-flow rule" | ||||
|   assistant: "I'll use the quality scale rule verifier to check if the peblar integration properly implements the config-flow rule." | ||||
|   <commentary> | ||||
|   Since the user is asking to verify a quality scale rule implementation, use the quality-scale-rule-verifier agent. | ||||
|   </commentary> | ||||
|   </example> | ||||
|  | ||||
|   <example> | ||||
|   Context: The user is reviewing if an integration reaches a specific quality scale level. | ||||
|   user: "Verify that this integration reaches the bronze quality scale" | ||||
|   assistant: "Let me use the quality scale rule verifier to check the bronze quality scale implementation." | ||||
|   <commentary> | ||||
|   The user wants to verify the integration has reached a certain quality level, so use multiple quality-scale-rule-verifier agents to verify each bronze rule. | ||||
|   </commentary> | ||||
|   </example> | ||||
| model: inherit | ||||
| color: yellow | ||||
| tools: Read, Bash, Grep, Glob, WebFetch | ||||
| --- | ||||
|  | ||||
| You are an expert Home Assistant integration quality scale auditor specializing in verifying compliance with specific quality scale rules. You have deep knowledge of Home Assistant's architecture, best practices, and the quality scale system that ensures integration consistency and reliability. | ||||
|  | ||||
| You will verify if an integration follows a specific quality scale rule by: | ||||
|  | ||||
| 1. **Fetching Rule Documentation**: Retrieve the official rule documentation from: | ||||
|    `https://raw.githubusercontent.com/home-assistant/developers.home-assistant/refs/heads/master/docs/core/integration-quality-scale/rules/{rule_name}.md` | ||||
|    where `{rule_name}` is the rule identifier (e.g., 'config-flow', 'entity-unique-id', 'parallel-updates') | ||||
|  | ||||
| 2. **Understanding Rule Requirements**: Parse the rule documentation to identify: | ||||
|    - Core requirements and mandatory implementations | ||||
|    - Specific code patterns or configurations required | ||||
|    - Common violations and anti-patterns | ||||
|    - Exemption criteria (when a rule might not apply) | ||||
|    - The quality tier this rule belongs to (Bronze, Silver, Gold, Platinum) | ||||
|  | ||||
| 3. **Analyzing Integration Code**: Examine the integration's codebase at `homeassistant/components/<integration domain>` focusing on: | ||||
|    - `manifest.json` for quality scale declaration and configuration | ||||
|    - `quality_scale.yaml` for rule status (done, todo, exempt) | ||||
|    - Relevant Python modules based on the rule requirements | ||||
|    - Configuration files and service definitions as needed | ||||
|  | ||||
| 4. **Verification Process**: | ||||
|    - Check if the rule is marked as 'done', 'todo', or 'exempt' in quality_scale.yaml | ||||
|    - If marked 'exempt', verify the exemption reason is valid | ||||
|    - If marked 'done', verify the actual implementation matches requirements | ||||
|    - Identify specific files and code sections that demonstrate compliance or violations | ||||
|    - Consider the integration's declared quality tier when applying rules | ||||
|    - To fetch the integration docs, use WebFetch to fetch from `https://raw.githubusercontent.com/home-assistant/home-assistant.io/refs/heads/current/source/_integrations/<integration domain>.markdown` | ||||
|    - To fetch information about a PyPI package, use the URL `https://pypi.org/pypi/<package>/json` | ||||
|  | ||||
| 5. **Reporting Findings**: Provide a comprehensive verification report that includes: | ||||
|    - **Rule Summary**: Brief description of what the rule requires | ||||
|    - **Compliance Status**: Clear pass/fail/exempt determination | ||||
|    - **Evidence**: Specific code examples showing compliance or violations | ||||
|    - **Issues Found**: Detailed list of any non-compliance issues with file locations | ||||
|    - **Recommendations**: Actionable steps to achieve compliance if needed | ||||
|    - **Exemption Analysis**: If applicable, whether the exemption is justified | ||||
|  | ||||
| When examining code, you will: | ||||
| - Look for exact implementation patterns specified in the rule | ||||
| - Verify all required components are present and properly configured | ||||
| - Check for common mistakes and anti-patterns | ||||
| - Consider edge cases and error handling requirements | ||||
| - Validate that implementations follow Home Assistant conventions | ||||
|  | ||||
| You will be thorough but focused, examining only the aspects relevant to the specific rule being verified. You will provide clear, actionable feedback that helps developers understand both what needs to be fixed and why it matters for integration quality. | ||||
|  | ||||
| If you cannot access the rule documentation or find the integration code, clearly state what information is missing and what you would need to complete the verification. | ||||
|  | ||||
| Remember that quality scale rules are cumulative - Bronze rules apply to all integrations with a quality scale, Silver rules apply to Silver+ integrations, and so on. Always consider the integration's target quality level when determining which rules should be enforced. | ||||
| @@ -58,7 +58,6 @@ base_platforms: &base_platforms | ||||
| # Extra components that trigger the full suite | ||||
| components: &components | ||||
|   - homeassistant/components/alexa/** | ||||
|   - homeassistant/components/analytics/** | ||||
|   - homeassistant/components/application_credentials/** | ||||
|   - homeassistant/components/assist_pipeline/** | ||||
|   - homeassistant/components/auth/** | ||||
|   | ||||
| @@ -8,9 +8,6 @@ | ||||
|     "PYTHONASYNCIODEBUG": "1" | ||||
|   }, | ||||
|   "features": { | ||||
|     // Node feature required for Claude Code until fixed https://github.com/anthropics/devcontainer-features/issues/28 | ||||
|     "ghcr.io/devcontainers/features/node:1": {}, | ||||
|     "ghcr.io/anthropics/devcontainer-features/claude-code:1.0": {}, | ||||
|     "ghcr.io/devcontainers/features/github-cli:1": {} | ||||
|   }, | ||||
|   // Port 5683 udp is used by Shelly integration | ||||
|   | ||||
| @@ -14,8 +14,7 @@ tests | ||||
|  | ||||
| # Other virtualization methods | ||||
| venv | ||||
| .venv | ||||
| .vagrant | ||||
|  | ||||
| # Temporary files | ||||
| **/__pycache__ | ||||
| **/__pycache__ | ||||
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,9 +6,9 @@ body: | ||||
|       value: | | ||||
|         This issue form is for reporting bugs only! | ||||
|  | ||||
|         If you have a feature or enhancement request, please [request them here instead][fr]. | ||||
|         If you have a feature or enhancement request, please use the [feature request][fr] section of our [Community Forum][fr]. | ||||
|  | ||||
|         [fr]: https://github.com/orgs/home-assistant/discussions | ||||
|         [fr]: https://community.home-assistant.io/c/feature-requests | ||||
|   - type: textarea | ||||
|     validations: | ||||
|       required: true | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,8 +10,8 @@ contact_links: | ||||
|     url: https://www.home-assistant.io/help | ||||
|     about: We use GitHub for tracking bugs, check our website for resources on getting help. | ||||
|   - name: Feature Request | ||||
|     url: https://github.com/orgs/home-assistant/discussions | ||||
|     about: Please use this link to request new features or enhancements to existing features. | ||||
|     url: https://community.home-assistant.io/c/feature-requests | ||||
|     about: Please use our Community Forum for making feature requests. | ||||
|   - name: I'm unsure where to go | ||||
|     url: https://www.home-assistant.io/join-chat | ||||
|     about: If you are unsure where to go, then joining our chat is recommended; Just ask! | ||||
|   | ||||
							
								
								
									
										53
									
								
								.github/ISSUE_TEMPLATE/task.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										53
									
								
								.github/ISSUE_TEMPLATE/task.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,53 +0,0 @@ | ||||
| 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, authorized contributors, and integration code owners only.** | ||||
|  | ||||
|         If you are a community member wanting to contribute, please: | ||||
|         - For bug reports: Use the [bug report form](https://github.com/home-assistant/core/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 | ||||
							
								
								
									
										5
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							| @@ -55,12 +55,8 @@ | ||||
|   creating the PR. If you're unsure about any of them, don't hesitate to ask. | ||||
|   We're here to help! This is simply a reminder of what we are going to look | ||||
|   for before merging your code. | ||||
|  | ||||
|   AI tools are welcome, but contributors are responsible for *fully* | ||||
|   understanding the code before submitting a PR. | ||||
| --> | ||||
|  | ||||
| - [ ] I understand the code I am submitting and can explain how it works. | ||||
| - [ ] The code change is tested and works locally. | ||||
| - [ ] Local tests pass. **Your PR cannot be merged unless tests pass** | ||||
| - [ ] There is no commented out code in this PR. | ||||
| @@ -68,7 +64,6 @@ | ||||
| - [ ] I have followed the [perfect PR recommendations][perfect-pr] | ||||
| - [ ] The code has been formatted using Ruff (`ruff format homeassistant tests`) | ||||
| - [ ] Tests have been added to verify that the new code works. | ||||
| - [ ] Any generated code has been carefully reviewed for correctness and compliance with project standards. | ||||
|  | ||||
| If user exposed functionality or configuration variables are added/changed: | ||||
|  | ||||
|   | ||||
							
								
								
									
										1254
									
								
								.github/copilot-instructions.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1254
									
								
								.github/copilot-instructions.md
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										3
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -6,6 +6,3 @@ updates: | ||||
|       interval: daily | ||||
|       time: "06:00" | ||||
|     open-pull-requests-limit: 10 | ||||
|     labels: | ||||
|       - dependency | ||||
|       - github_actions | ||||
|   | ||||
							
								
								
									
										58
									
								
								.github/workflows/builder.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										58
									
								
								.github/workflows/builder.yml
									
									
									
									
										vendored
									
									
								
							| @@ -27,12 +27,12 @@ jobs: | ||||
|       publish: ${{ steps.version.outputs.publish }} | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|  | ||||
|       - name: Set up Python ${{ env.DEFAULT_PYTHON }} | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         uses: actions/setup-python@v5.5.0 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON }} | ||||
|  | ||||
| @@ -69,7 +69,7 @@ jobs: | ||||
|         run: find ./homeassistant/components/*/translations -name "*.json" | tar zcvf translations.tar.gz -T - | ||||
|  | ||||
|       - name: Upload translations | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         uses: actions/upload-artifact@v4.6.2 | ||||
|         with: | ||||
|           name: translations | ||||
|           path: translations.tar.gz | ||||
| @@ -90,11 +90,11 @@ jobs: | ||||
|         arch: ${{ fromJson(needs.init.outputs.architectures) }} | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Download nightly wheels of frontend | ||||
|         if: needs.init.outputs.channel == 'dev' | ||||
|         uses: dawidd6/action-download-artifact@ac66b43f0e6a346234dd65d4d0c8fbb31cb316e5 # v11 | ||||
|         uses: dawidd6/action-download-artifact@v9 | ||||
|         with: | ||||
|           github_token: ${{secrets.GITHUB_TOKEN}} | ||||
|           repo: home-assistant/frontend | ||||
| @@ -105,10 +105,10 @@ jobs: | ||||
|  | ||||
|       - name: Download nightly wheels of intents | ||||
|         if: needs.init.outputs.channel == 'dev' | ||||
|         uses: dawidd6/action-download-artifact@ac66b43f0e6a346234dd65d4d0c8fbb31cb316e5 # v11 | ||||
|         uses: dawidd6/action-download-artifact@v9 | ||||
|         with: | ||||
|           github_token: ${{secrets.GITHUB_TOKEN}} | ||||
|           repo: OHF-Voice/intents-package | ||||
|           repo: home-assistant/intents-package | ||||
|           branch: main | ||||
|           workflow: nightly.yaml | ||||
|           workflow_conclusion: success | ||||
| @@ -116,7 +116,7 @@ jobs: | ||||
|  | ||||
|       - name: Set up Python ${{ env.DEFAULT_PYTHON }} | ||||
|         if: needs.init.outputs.channel == 'dev' | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         uses: actions/setup-python@v5.5.0 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON }} | ||||
|  | ||||
| @@ -175,7 +175,7 @@ jobs: | ||||
|           sed -i "s|pykrakenapi|# pykrakenapi|g" requirements_all.txt | ||||
|  | ||||
|       - name: Download translations | ||||
|         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||
|         uses: actions/download-artifact@v4.2.1 | ||||
|         with: | ||||
|           name: translations | ||||
|  | ||||
| @@ -190,15 +190,14 @@ jobs: | ||||
|           echo "${{ github.sha }};${{ github.ref }};${{ github.event_name }};${{ github.actor }}" > rootfs/OFFICIAL_IMAGE | ||||
|  | ||||
|       - name: Login to GitHub Container Registry | ||||
|         uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0 | ||||
|         uses: docker/login-action@v3.4.0 | ||||
|         with: | ||||
|           registry: ghcr.io | ||||
|           username: ${{ github.repository_owner }} | ||||
|           password: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
|       # home-assistant/builder doesn't support sha pinning | ||||
|       - name: Build base image | ||||
|         uses: home-assistant/builder@2025.09.0 | ||||
|         uses: home-assistant/builder@2025.03.0 | ||||
|         with: | ||||
|           args: | | ||||
|             $BUILD_ARGS \ | ||||
| @@ -243,7 +242,7 @@ jobs: | ||||
|           - green | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Set build additional args | ||||
|         run: | | ||||
| @@ -257,15 +256,14 @@ jobs: | ||||
|           fi | ||||
|  | ||||
|       - name: Login to GitHub Container Registry | ||||
|         uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0 | ||||
|         uses: docker/login-action@v3.4.0 | ||||
|         with: | ||||
|           registry: ghcr.io | ||||
|           username: ${{ github.repository_owner }} | ||||
|           password: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
|       # home-assistant/builder doesn't support sha pinning | ||||
|       - name: Build base image | ||||
|         uses: home-assistant/builder@2025.09.0 | ||||
|         uses: home-assistant/builder@2025.03.0 | ||||
|         with: | ||||
|           args: | | ||||
|             $BUILD_ARGS \ | ||||
| @@ -281,7 +279,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Initialize git | ||||
|         uses: home-assistant/actions/helpers/git-init@master | ||||
| @@ -323,23 +321,23 @@ jobs: | ||||
|         registry: ["ghcr.io/home-assistant", "docker.io/homeassistant"] | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Install Cosign | ||||
|         uses: sigstore/cosign-installer@d7543c93d881b35a8faa02e8e3605f69b7a1ce62 # v3.10.0 | ||||
|         uses: sigstore/cosign-installer@v3.8.1 | ||||
|         with: | ||||
|           cosign-release: "v2.2.3" | ||||
|  | ||||
|       - name: Login to DockerHub | ||||
|         if: matrix.registry == 'docker.io/homeassistant' | ||||
|         uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0 | ||||
|         uses: docker/login-action@v3.4.0 | ||||
|         with: | ||||
|           username: ${{ secrets.DOCKERHUB_USERNAME }} | ||||
|           password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||
|  | ||||
|       - name: Login to GitHub Container Registry | ||||
|         if: matrix.registry == 'ghcr.io/home-assistant' | ||||
|         uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0 | ||||
|         uses: docker/login-action@v3.4.0 | ||||
|         with: | ||||
|           registry: ghcr.io | ||||
|           username: ${{ github.repository_owner }} | ||||
| @@ -456,15 +454,15 @@ jobs: | ||||
|     if: github.repository_owner == 'home-assistant' && needs.init.outputs.publish == 'true' | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Set up Python ${{ env.DEFAULT_PYTHON }} | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         uses: actions/setup-python@v5.5.0 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON }} | ||||
|  | ||||
|       - name: Download translations | ||||
|         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||
|         uses: actions/download-artifact@v4.2.1 | ||||
|         with: | ||||
|           name: translations | ||||
|  | ||||
| @@ -482,7 +480,7 @@ jobs: | ||||
|           python -m build | ||||
|  | ||||
|       - name: Upload package to PyPI | ||||
|         uses: pypa/gh-action-pypi-publish@ed0c53931b1dc9bd32cbe73a98c7f6766f8a527e # v1.13.0 | ||||
|         uses: pypa/gh-action-pypi-publish@v1.12.4 | ||||
|         with: | ||||
|           skip-existing: true | ||||
|  | ||||
| @@ -501,17 +499,17 @@ jobs: | ||||
|       HASSFEST_IMAGE_TAG: ghcr.io/home-assistant/hassfest:${{ needs.init.outputs.version }} | ||||
|     steps: | ||||
|       - name: Checkout repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2 | ||||
|  | ||||
|       - name: Login to GitHub Container Registry | ||||
|         uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0 | ||||
|         uses: docker/login-action@74a5d142397b4f367a81961eba4e8cd7edddf772 # v3.4.0 | ||||
|         with: | ||||
|           registry: ghcr.io | ||||
|           username: ${{ github.repository_owner }} | ||||
|           password: ${{ secrets.GITHUB_TOKEN }} | ||||
|  | ||||
|       - name: Build Docker image | ||||
|         uses: docker/build-push-action@263435318d21b8e681c14492fe198d362a7d2c83 # v6.18.0 | ||||
|         uses: docker/build-push-action@471d1dc4e07e5cdedd4c2171150001c434f0b7a4 # v6.15.0 | ||||
|         with: | ||||
|           context: . # So action will not pull the repository again | ||||
|           file: ./script/hassfest/docker/Dockerfile | ||||
| @@ -524,7 +522,7 @@ jobs: | ||||
|       - name: Push Docker image | ||||
|         if: needs.init.outputs.channel != 'dev' && needs.init.outputs.publish == 'true' | ||||
|         id: push | ||||
|         uses: docker/build-push-action@263435318d21b8e681c14492fe198d362a7d2c83 # v6.18.0 | ||||
|         uses: docker/build-push-action@471d1dc4e07e5cdedd4c2171150001c434f0b7a4 # v6.15.0 | ||||
|         with: | ||||
|           context: . # So action will not pull the repository again | ||||
|           file: ./script/hassfest/docker/Dockerfile | ||||
| @@ -533,7 +531,7 @@ jobs: | ||||
|  | ||||
|       - name: Generate artifact attestation | ||||
|         if: needs.init.outputs.channel != 'dev' && needs.init.outputs.publish == 'true' | ||||
|         uses: actions/attest-build-provenance@977bb373ede98d70efdf65b84cb5f73e068dcc2a # v3.0.0 | ||||
|         uses: actions/attest-build-provenance@c074443f1aee8d4aeeae555aebba3282517141b2 # v2.2.3 | ||||
|         with: | ||||
|           subject-name: ${{ env.HASSFEST_IMAGE_NAME }} | ||||
|           subject-digest: ${{ steps.push.outputs.digest }} | ||||
|   | ||||
							
								
								
									
										772
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										772
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										6
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/codeql.yml
									
									
									
									
										vendored
									
									
								
							| @@ -21,14 +21,14 @@ jobs: | ||||
|  | ||||
|     steps: | ||||
|       - name: Check out code from GitHub | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Initialize CodeQL | ||||
|         uses: github/codeql-action/init@e296a935590eb16afc0c0108289f68c87e2a89a5 # v4.30.7 | ||||
|         uses: github/codeql-action/init@v3.28.13 | ||||
|         with: | ||||
|           languages: python | ||||
|  | ||||
|       - name: Perform CodeQL Analysis | ||||
|         uses: github/codeql-action/analyze@e296a935590eb16afc0c0108289f68c87e2a89a5 # v4.30.7 | ||||
|         uses: github/codeql-action/analyze@v3.28.13 | ||||
|         with: | ||||
|           category: "/language:python" | ||||
|   | ||||
							
								
								
									
										385
									
								
								.github/workflows/detect-duplicate-issues.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										385
									
								
								.github/workflows/detect-duplicate-issues.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,385 +0,0 @@ | ||||
| name: Auto-detect duplicate issues | ||||
|  | ||||
| # yamllint disable-line rule:truthy | ||||
| on: | ||||
|   issues: | ||||
|     types: [labeled] | ||||
|  | ||||
| permissions: | ||||
|   issues: write | ||||
|   models: read | ||||
|  | ||||
| jobs: | ||||
|   detect-duplicates: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - name: Check if integration label was added and extract details | ||||
|         id: extract | ||||
|         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | ||||
|         with: | ||||
|           script: | | ||||
|             // Debug: Log the event payload | ||||
|             console.log('Event name:', context.eventName); | ||||
|             console.log('Event action:', context.payload.action); | ||||
|             console.log('Event payload keys:', Object.keys(context.payload)); | ||||
|  | ||||
|             // Check the specific label that was added | ||||
|             const addedLabel = context.payload.label; | ||||
|             if (!addedLabel) { | ||||
|               console.log('No label found in labeled event payload'); | ||||
|               core.setOutput('should_continue', 'false'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             console.log(`Label added: ${addedLabel.name}`); | ||||
|  | ||||
|             if (!addedLabel.name.startsWith('integration:')) { | ||||
|               console.log('Added label is not an integration label, skipping duplicate detection'); | ||||
|               core.setOutput('should_continue', 'false'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             console.log(`Integration label added: ${addedLabel.name}`); | ||||
|  | ||||
|             let currentIssue; | ||||
|             let integrationLabels = []; | ||||
|  | ||||
|             try { | ||||
|               const issue = await github.rest.issues.get({ | ||||
|                 owner: context.repo.owner, | ||||
|                 repo: context.repo.repo, | ||||
|                 issue_number: context.payload.issue.number | ||||
|               }); | ||||
|  | ||||
|               currentIssue = issue.data; | ||||
|  | ||||
|               // Check if potential-duplicate label already exists | ||||
|               const hasPotentialDuplicateLabel = currentIssue.labels | ||||
|                 .some(label => label.name === 'potential-duplicate'); | ||||
|  | ||||
|               if (hasPotentialDuplicateLabel) { | ||||
|                 console.log('Issue already has potential-duplicate label, skipping duplicate detection'); | ||||
|                 core.setOutput('should_continue', 'false'); | ||||
|                 return; | ||||
|               } | ||||
|  | ||||
|               integrationLabels = currentIssue.labels | ||||
|                 .filter(label => label.name.startsWith('integration:')) | ||||
|                 .map(label => label.name); | ||||
|             } catch (error) { | ||||
|               core.error(`Failed to fetch issue #${context.payload.issue.number}:`, error.message); | ||||
|               core.setOutput('should_continue', 'false'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             // Check if we've already posted a duplicate detection comment recently | ||||
|             let comments; | ||||
|             try { | ||||
|               comments = await github.rest.issues.listComments({ | ||||
|                 owner: context.repo.owner, | ||||
|                 repo: context.repo.repo, | ||||
|                 issue_number: context.payload.issue.number, | ||||
|                 per_page: 10 | ||||
|               }); | ||||
|             } catch (error) { | ||||
|               core.error('Failed to fetch comments:', error.message); | ||||
|               // Continue anyway, worst case we might post a duplicate comment | ||||
|               comments = { data: [] }; | ||||
|             } | ||||
|  | ||||
|             // Check if we've already posted a duplicate detection comment | ||||
|             const recentDuplicateComment = comments.data.find(comment => | ||||
|               comment.user && comment.user.login === 'github-actions[bot]' && | ||||
|               comment.body.includes('<!-- workflow: detect-duplicate-issues -->') | ||||
|             ); | ||||
|  | ||||
|             if (recentDuplicateComment) { | ||||
|               console.log('Already posted duplicate detection comment, skipping'); | ||||
|               core.setOutput('should_continue', 'false'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             core.setOutput('should_continue', 'true'); | ||||
|             core.setOutput('current_number', currentIssue.number); | ||||
|             core.setOutput('current_title', currentIssue.title); | ||||
|             core.setOutput('current_body', currentIssue.body); | ||||
|             core.setOutput('current_url', currentIssue.html_url); | ||||
|             core.setOutput('integration_labels', JSON.stringify(integrationLabels)); | ||||
|  | ||||
|             console.log(`Current issue: #${currentIssue.number}`); | ||||
|             console.log(`Integration labels: ${integrationLabels.join(', ')}`); | ||||
|  | ||||
|       - name: Fetch similar issues | ||||
|         id: fetch_similar | ||||
|         if: steps.extract.outputs.should_continue == 'true' | ||||
|         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | ||||
|         env: | ||||
|           INTEGRATION_LABELS: ${{ steps.extract.outputs.integration_labels }} | ||||
|           CURRENT_NUMBER: ${{ steps.extract.outputs.current_number }} | ||||
|         with: | ||||
|           script: | | ||||
|             const integrationLabels = JSON.parse(process.env.INTEGRATION_LABELS); | ||||
|             const currentNumber = parseInt(process.env.CURRENT_NUMBER); | ||||
|  | ||||
|             if (integrationLabels.length === 0) { | ||||
|               console.log('No integration labels found, skipping duplicate detection'); | ||||
|               core.setOutput('has_similar', 'false'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             // Use GitHub search API to find issues with matching integration labels | ||||
|             console.log(`Searching for issues with integration labels: ${integrationLabels.join(', ')}`); | ||||
|  | ||||
|             // Build search query for issues with any of the current integration labels | ||||
|             const labelQueries = integrationLabels.map(label => `label:"${label}"`); | ||||
|  | ||||
|             // Calculate date 6 months ago | ||||
|             const sixMonthsAgo = new Date(); | ||||
|             sixMonthsAgo.setMonth(sixMonthsAgo.getMonth() - 6); | ||||
|             const dateFilter = `created:>=${sixMonthsAgo.toISOString().split('T')[0]}`; | ||||
|  | ||||
|             let searchQuery; | ||||
|  | ||||
|             if (labelQueries.length === 1) { | ||||
|               searchQuery = `repo:${context.repo.owner}/${context.repo.repo} is:issue ${labelQueries[0]} ${dateFilter}`; | ||||
|             } else { | ||||
|               searchQuery = `repo:${context.repo.owner}/${context.repo.repo} is:issue (${labelQueries.join(' OR ')}) ${dateFilter}`; | ||||
|             } | ||||
|  | ||||
|             console.log(`Search query: ${searchQuery}`); | ||||
|  | ||||
|             let result; | ||||
|             try { | ||||
|               result = await github.rest.search.issuesAndPullRequests({ | ||||
|                 q: searchQuery, | ||||
|                 per_page: 15, | ||||
|                 sort: 'updated', | ||||
|                 order: 'desc' | ||||
|               }); | ||||
|             } catch (error) { | ||||
|               core.error('Failed to search for similar issues:', error.message); | ||||
|               if (error.status === 403 && error.message.includes('rate limit')) { | ||||
|                 core.error('GitHub API rate limit exceeded'); | ||||
|               } | ||||
|               core.setOutput('has_similar', 'false'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             // Filter out the current issue, pull requests, and newer issues (higher numbers) | ||||
|             const similarIssues = result.data.items | ||||
|               .filter(item => | ||||
|                 item.number !== currentNumber && | ||||
|                 !item.pull_request && | ||||
|                 item.number < currentNumber // Only include older issues (lower numbers) | ||||
|               ) | ||||
|               .map(item => ({ | ||||
|                 number: item.number, | ||||
|                 title: item.title, | ||||
|                 body: item.body, | ||||
|                 url: item.html_url, | ||||
|                 state: item.state, | ||||
|                 createdAt: item.created_at, | ||||
|                 updatedAt: item.updated_at, | ||||
|                 comments: item.comments, | ||||
|                 labels: item.labels.map(l => l.name) | ||||
|               })); | ||||
|  | ||||
|             console.log(`Found ${similarIssues.length} issues with matching integration labels`); | ||||
|             console.log('Raw similar issues:', JSON.stringify(similarIssues.slice(0, 3), null, 2)); | ||||
|  | ||||
|             if (similarIssues.length === 0) { | ||||
|               console.log('No similar issues found, setting has_similar to false'); | ||||
|               core.setOutput('has_similar', 'false'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             console.log('Similar issues found, setting has_similar to true'); | ||||
|             core.setOutput('has_similar', 'true'); | ||||
|  | ||||
|             // Clean the issue data to prevent JSON parsing issues | ||||
|             const cleanedIssues = similarIssues.slice(0, 15).map(item => { | ||||
|               // Handle body with improved truncation and null handling | ||||
|               let cleanBody = ''; | ||||
|               if (item.body && typeof item.body === 'string') { | ||||
|                 // Remove control characters | ||||
|                 const cleaned = item.body.replace(/[\u0000-\u001F\u007F-\u009F]/g, ''); | ||||
|                 // Truncate to 1000 characters and add ellipsis if needed | ||||
|                 cleanBody = cleaned.length > 1000 | ||||
|                   ? cleaned.substring(0, 1000) + '...' | ||||
|                   : cleaned; | ||||
|               } | ||||
|  | ||||
|               return { | ||||
|                 number: item.number, | ||||
|                 title: item.title.replace(/[\u0000-\u001F\u007F-\u009F]/g, ''), // Remove control characters | ||||
|                 body: cleanBody, | ||||
|                 url: item.url, | ||||
|                 state: item.state, | ||||
|                 createdAt: item.createdAt, | ||||
|                 updatedAt: item.updatedAt, | ||||
|                 comments: item.comments, | ||||
|                 labels: item.labels | ||||
|               }; | ||||
|             }); | ||||
|  | ||||
|             console.log(`Cleaned issues count: ${cleanedIssues.length}`); | ||||
|             console.log('First cleaned issue:', JSON.stringify(cleanedIssues[0], null, 2)); | ||||
|  | ||||
|             core.setOutput('similar_issues', JSON.stringify(cleanedIssues)); | ||||
|  | ||||
|       - name: Detect duplicates using AI | ||||
|         id: ai_detection | ||||
|         if: steps.extract.outputs.should_continue == 'true' && steps.fetch_similar.outputs.has_similar == 'true' | ||||
|         uses: actions/ai-inference@a1c11829223a786afe3b5663db904a3aa1eac3a2 # v2.0.1 | ||||
|         with: | ||||
|           model: openai/gpt-4o | ||||
|           system-prompt: | | ||||
|             You are a Home Assistant issue duplicate detector. Your task is to identify TRUE DUPLICATES - issues that report the EXACT SAME problem, not just similar or related issues. | ||||
|  | ||||
|             CRITICAL: An issue is ONLY a duplicate if: | ||||
|             - It describes the SAME problem with the SAME root cause | ||||
|             - Issues about the same integration but different problems are NOT duplicates | ||||
|             - Issues with similar symptoms but different causes are NOT duplicates | ||||
|  | ||||
|             Important considerations: | ||||
|             - Open issues are more relevant than closed ones for duplicate detection | ||||
|             - Recently updated issues may indicate ongoing work or discussion | ||||
|             - Issues with more comments are generally more relevant and active | ||||
|             - Older closed issues might be resolved differently than newer approaches | ||||
|             - Consider the time between issues - very old issues may have different contexts | ||||
|  | ||||
|             Rules: | ||||
|             1. ONLY mark as duplicate if the issues describe IDENTICAL problems | ||||
|             2. Look for issues that report the same problem or request the same functionality | ||||
|             3. Different error messages = NOT a duplicate (even if same integration) | ||||
|             4. For CLOSED issues, only mark as duplicate if they describe the EXACT same problem | ||||
|             5. For OPEN issues, use a lower threshold (90%+ similarity) | ||||
|             6. Prioritize issues with higher comment counts as they indicate more activity/relevance | ||||
|             7. When in doubt, do NOT mark as duplicate | ||||
|             8. Return ONLY a JSON array of issue numbers that are duplicates | ||||
|             9. If no duplicates are found, return an empty array: [] | ||||
|             10. Maximum 5 potential duplicates, prioritize open issues with comments | ||||
|             11. Consider the age of issues - prefer recent duplicates over very old ones | ||||
|  | ||||
|             Example response format: | ||||
|             [1234, 5678, 9012] | ||||
|  | ||||
|           prompt: | | ||||
|             Current issue (just created): | ||||
|             Title: ${{ steps.extract.outputs.current_title }} | ||||
|             Body: ${{ steps.extract.outputs.current_body }} | ||||
|  | ||||
|             Other issues to compare against (each includes state, creation date, last update, and comment count): | ||||
|             ${{ steps.fetch_similar.outputs.similar_issues }} | ||||
|  | ||||
|             Analyze these issues and identify which ones describe IDENTICAL problems and thus are duplicates of the current issue. When sorting them, consider their state (open/closed), how recently they were updated, and their comment count (higher = more relevant). | ||||
|  | ||||
|           max-tokens: 100 | ||||
|  | ||||
|       - name: Post duplicate detection results | ||||
|         id: post_results | ||||
|         if: steps.extract.outputs.should_continue == 'true' && steps.fetch_similar.outputs.has_similar == 'true' | ||||
|         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | ||||
|         env: | ||||
|           AI_RESPONSE: ${{ steps.ai_detection.outputs.response }} | ||||
|           SIMILAR_ISSUES: ${{ steps.fetch_similar.outputs.similar_issues }} | ||||
|         with: | ||||
|           script: | | ||||
|             const aiResponse = process.env.AI_RESPONSE; | ||||
|  | ||||
|             console.log('Raw AI response:', JSON.stringify(aiResponse)); | ||||
|  | ||||
|             let duplicateNumbers = []; | ||||
|             try { | ||||
|               // Clean the response of any potential control characters | ||||
|               const cleanResponse = aiResponse.trim().replace(/[\u0000-\u001F\u007F-\u009F]/g, ''); | ||||
|               console.log('Cleaned AI response:', cleanResponse); | ||||
|  | ||||
|               duplicateNumbers = JSON.parse(cleanResponse); | ||||
|  | ||||
|               // Ensure it's an array and contains only numbers | ||||
|               if (!Array.isArray(duplicateNumbers)) { | ||||
|                 console.log('AI response is not an array, trying to extract numbers'); | ||||
|                 const numberMatches = cleanResponse.match(/\d+/g); | ||||
|                 duplicateNumbers = numberMatches ? numberMatches.map(n => parseInt(n)) : []; | ||||
|               } | ||||
|  | ||||
|               // Filter to only valid numbers | ||||
|               duplicateNumbers = duplicateNumbers.filter(n => typeof n === 'number' && !isNaN(n)); | ||||
|  | ||||
|             } catch (error) { | ||||
|               console.log('Failed to parse AI response as JSON:', error.message); | ||||
|               console.log('Raw response:', aiResponse); | ||||
|  | ||||
|               // Fallback: try to extract numbers from the response | ||||
|               const numberMatches = aiResponse.match(/\d+/g); | ||||
|               duplicateNumbers = numberMatches ? numberMatches.map(n => parseInt(n)) : []; | ||||
|               console.log('Extracted numbers as fallback:', duplicateNumbers); | ||||
|             } | ||||
|  | ||||
|             if (!Array.isArray(duplicateNumbers) || duplicateNumbers.length === 0) { | ||||
|               console.log('No duplicates detected by AI'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             console.log(`AI detected ${duplicateNumbers.length} potential duplicates: ${duplicateNumbers.join(', ')}`); | ||||
|  | ||||
|             // Get details of detected duplicates | ||||
|             const similarIssues = JSON.parse(process.env.SIMILAR_ISSUES); | ||||
|             const duplicates = similarIssues.filter(issue => duplicateNumbers.includes(issue.number)); | ||||
|  | ||||
|             if (duplicates.length === 0) { | ||||
|               console.log('No matching issues found for detected numbers'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             // Create comment with duplicate detection results | ||||
|             const duplicateLinks = duplicates.map(issue => `- [#${issue.number}: ${issue.title}](${issue.url})`).join('\n'); | ||||
|  | ||||
|             const commentBody = [ | ||||
|               '<!-- workflow: detect-duplicate-issues -->', | ||||
|               '### 🔍 **Potential duplicate detection**', | ||||
|               '', | ||||
|               'I\'ve analyzed similar issues and found the following potential duplicates:', | ||||
|               '', | ||||
|               duplicateLinks, | ||||
|               '', | ||||
|               '**What to do next:**', | ||||
|               '1. Please review these issues to see if they match your issue', | ||||
|               '2. If you find an existing issue that covers your problem:', | ||||
|               '   - Consider closing this issue', | ||||
|               '   - Add your findings or 👍 on the existing issue instead', | ||||
|               '3. If your issue is different or adds new aspects, please clarify how it differs', | ||||
|               '', | ||||
|               'This helps keep our issues organized and ensures similar issues are consolidated for better visibility.', | ||||
|               '', | ||||
|               '*This message was generated automatically by our duplicate detection system.*' | ||||
|             ].join('\n'); | ||||
|  | ||||
|             try { | ||||
|               await github.rest.issues.createComment({ | ||||
|                 owner: context.repo.owner, | ||||
|                 repo: context.repo.repo, | ||||
|                 issue_number: context.payload.issue.number, | ||||
|                 body: commentBody | ||||
|               }); | ||||
|  | ||||
|               console.log(`Posted duplicate detection comment with ${duplicates.length} potential duplicates`); | ||||
|  | ||||
|               // Add the potential-duplicate label | ||||
|               await github.rest.issues.addLabels({ | ||||
|                 owner: context.repo.owner, | ||||
|                 repo: context.repo.repo, | ||||
|                 issue_number: context.payload.issue.number, | ||||
|                 labels: ['potential-duplicate'] | ||||
|               }); | ||||
|  | ||||
|               console.log('Added potential-duplicate label to the issue'); | ||||
|             } catch (error) { | ||||
|               core.error('Failed to post duplicate detection comment or add label:', error.message); | ||||
|               if (error.status === 403) { | ||||
|                 core.error('Permission denied or rate limit exceeded'); | ||||
|               } | ||||
|               // Don't throw - we've done the analysis, just couldn't post the result | ||||
|             } | ||||
							
								
								
									
										193
									
								
								.github/workflows/detect-non-english-issues.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										193
									
								
								.github/workflows/detect-non-english-issues.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,193 +0,0 @@ | ||||
| name: Auto-detect non-English issues | ||||
|  | ||||
| # yamllint disable-line rule:truthy | ||||
| on: | ||||
|   issues: | ||||
|     types: [opened] | ||||
|  | ||||
| permissions: | ||||
|   issues: write | ||||
|   models: read | ||||
|  | ||||
| jobs: | ||||
|   detect-language: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - name: Check issue language | ||||
|         id: detect_language | ||||
|         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | ||||
|         env: | ||||
|           ISSUE_NUMBER: ${{ github.event.issue.number }} | ||||
|           ISSUE_TITLE: ${{ github.event.issue.title }} | ||||
|           ISSUE_BODY: ${{ github.event.issue.body }} | ||||
|           ISSUE_USER_TYPE: ${{ github.event.issue.user.type }} | ||||
|         with: | ||||
|           script: | | ||||
|             // Get the issue details from environment variables | ||||
|             const issueNumber = process.env.ISSUE_NUMBER; | ||||
|             const issueTitle = process.env.ISSUE_TITLE || ''; | ||||
|             const issueBody = process.env.ISSUE_BODY || ''; | ||||
|             const userType = process.env.ISSUE_USER_TYPE; | ||||
|  | ||||
|             // Skip language detection for bot users | ||||
|             if (userType === 'Bot') { | ||||
|               console.log('Skipping language detection for bot user'); | ||||
|               core.setOutput('should_continue', 'false'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             console.log(`Checking language for issue #${issueNumber}`); | ||||
|             console.log(`Title: ${issueTitle}`); | ||||
|  | ||||
|             // Combine title and body for language detection | ||||
|             const fullText = `${issueTitle}\n\n${issueBody}`; | ||||
|  | ||||
|             // Check if the text is too short to reliably detect language | ||||
|             if (fullText.trim().length < 20) { | ||||
|               console.log('Text too short for reliable language detection'); | ||||
|               core.setOutput('should_continue', 'false'); // Skip processing for very short text | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             core.setOutput('issue_number', issueNumber); | ||||
|             core.setOutput('issue_text', fullText); | ||||
|             core.setOutput('should_continue', 'true'); | ||||
|  | ||||
|       - name: Detect language using AI | ||||
|         id: ai_language_detection | ||||
|         if: steps.detect_language.outputs.should_continue == 'true' | ||||
|         uses: actions/ai-inference@a1c11829223a786afe3b5663db904a3aa1eac3a2 # v2.0.1 | ||||
|         with: | ||||
|           model: openai/gpt-4o-mini | ||||
|           system-prompt: | | ||||
|             You are a language detection system. Your task is to determine if the provided text is written in English or another language. | ||||
|  | ||||
|             Rules: | ||||
|             1. Analyze the text and determine the primary language of the USER'S DESCRIPTION only | ||||
|             2. IGNORE markdown headers (lines starting with #, ##, ###, etc.) as these are from issue templates, not user input | ||||
|             3. IGNORE all code blocks (text between ``` or ` markers) as they may contain system-generated error messages in other languages | ||||
|             4. IGNORE error messages, logs, and system output even if not in code blocks - these often appear in the user's system language | ||||
|             5. Consider technical terms, code snippets, URLs, and file paths as neutral (they don't indicate non-English) | ||||
|             6. Focus ONLY on the actual sentences and descriptions written by the user explaining their issue | ||||
|             7. If the user's explanation/description is in English but includes non-English error messages or logs, consider it ENGLISH | ||||
|             8. Return ONLY a JSON object with two fields: | ||||
|                - "is_english": boolean (true if the user's description is primarily in English, false otherwise) | ||||
|                - "detected_language": string (the name of the detected language, e.g., "English", "Spanish", "Chinese", etc.) | ||||
|             9. Be lenient - if the user's explanation is in English with non-English system output, it's still English | ||||
|             10. Common programming terms, error messages, and technical jargon should not be considered as non-English | ||||
|             11. If you cannot reliably determine the language, set detected_language to "undefined" | ||||
|  | ||||
|             Example response: | ||||
|             {"is_english": false, "detected_language": "Spanish"} | ||||
|  | ||||
|           prompt: | | ||||
|             Please analyze the following issue text and determine if it is written in English: | ||||
|  | ||||
|             ${{ steps.detect_language.outputs.issue_text }} | ||||
|  | ||||
|           max-tokens: 50 | ||||
|  | ||||
|       - name: Process non-English issues | ||||
|         if: steps.detect_language.outputs.should_continue == 'true' | ||||
|         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8.0.0 | ||||
|         env: | ||||
|           AI_RESPONSE: ${{ steps.ai_language_detection.outputs.response }} | ||||
|           ISSUE_NUMBER: ${{ steps.detect_language.outputs.issue_number }} | ||||
|         with: | ||||
|           script: | | ||||
|             const issueNumber = parseInt(process.env.ISSUE_NUMBER); | ||||
|             const aiResponse = process.env.AI_RESPONSE; | ||||
|  | ||||
|             console.log('AI language detection response:', aiResponse); | ||||
|  | ||||
|             let languageResult; | ||||
|             try { | ||||
|               languageResult = JSON.parse(aiResponse.trim()); | ||||
|  | ||||
|               // Validate the response structure | ||||
|               if (!languageResult || typeof languageResult.is_english !== 'boolean') { | ||||
|                 throw new Error('Invalid response structure'); | ||||
|               } | ||||
|             } catch (error) { | ||||
|               core.error(`Failed to parse AI response: ${error.message}`); | ||||
|               console.log('Raw AI response:', aiResponse); | ||||
|  | ||||
|               // Log more details for debugging | ||||
|               core.warning('Defaulting to English due to parsing error'); | ||||
|  | ||||
|               // Default to English if we can't parse the response | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             if (languageResult.is_english) { | ||||
|               console.log('Issue is in English, no action needed'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             // If language is undefined or not detected, skip processing | ||||
|             if (!languageResult.detected_language || languageResult.detected_language === 'undefined') { | ||||
|               console.log('Language could not be determined, skipping processing'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             console.log(`Issue detected as non-English: ${languageResult.detected_language}`); | ||||
|  | ||||
|             // Post comment explaining the language requirement | ||||
|             const commentBody = [ | ||||
|               '<!-- workflow: detect-non-english-issues -->', | ||||
|               '### 🌐 Non-English issue detected', | ||||
|               '', | ||||
|               `This issue appears to be written in **${languageResult.detected_language}** rather than English.`, | ||||
|               '', | ||||
|               'The Home Assistant project uses English as the primary language for issues to ensure that everyone in our international community can participate and help resolve issues. This allows any of our thousands of contributors to jump in and provide assistance.', | ||||
|               '', | ||||
|               '**What to do:**', | ||||
|               '1. Re-create the issue using the English language', | ||||
|               '2. If you need help with translation, consider using:', | ||||
|               '   - Translation tools like Google Translate', | ||||
|               '   - AI assistants like ChatGPT or Claude', | ||||
|               '', | ||||
|               'This helps our community provide the best possible support and ensures your issue gets the attention it deserves from our global contributor base.', | ||||
|               '', | ||||
|               'Thank you for your understanding! 🙏' | ||||
|             ].join('\n'); | ||||
|  | ||||
|             try { | ||||
|               // Add comment | ||||
|               await github.rest.issues.createComment({ | ||||
|                 owner: context.repo.owner, | ||||
|                 repo: context.repo.repo, | ||||
|                 issue_number: issueNumber, | ||||
|                 body: commentBody | ||||
|               }); | ||||
|  | ||||
|               console.log('Posted language requirement comment'); | ||||
|  | ||||
|               // Add non-english label | ||||
|               await github.rest.issues.addLabels({ | ||||
|                 owner: context.repo.owner, | ||||
|                 repo: context.repo.repo, | ||||
|                 issue_number: issueNumber, | ||||
|                 labels: ['non-english'] | ||||
|               }); | ||||
|  | ||||
|               console.log('Added non-english label'); | ||||
|  | ||||
|               // Close the issue | ||||
|               await github.rest.issues.update({ | ||||
|                 owner: context.repo.owner, | ||||
|                 repo: context.repo.repo, | ||||
|                 issue_number: issueNumber, | ||||
|                 state: 'closed', | ||||
|                 state_reason: 'not_planned' | ||||
|               }); | ||||
|  | ||||
|               console.log('Closed the issue'); | ||||
|  | ||||
|             } catch (error) { | ||||
|               core.error('Failed to process non-English issue:', error.message); | ||||
|               if (error.status === 403) { | ||||
|                 core.error('Permission denied or rate limit exceeded'); | ||||
|               } | ||||
|             } | ||||
							
								
								
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,7 +10,7 @@ jobs: | ||||
|     if: github.repository_owner == 'home-assistant' | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: dessant/lock-threads@1bf7ec25051fe7c00bdd17e6a7cf3d7bfb7dc771 # v5.0.1 | ||||
|       - uses: dessant/lock-threads@v5.0.1 | ||||
|         with: | ||||
|           github-token: ${{ github.token }} | ||||
|           issue-inactive-days: "30" | ||||
|   | ||||
							
								
								
									
										84
									
								
								.github/workflows/restrict-task-creation.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										84
									
								
								.github/workflows/restrict-task-creation.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,84 +0,0 @@ | ||||
| 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; | ||||
|  | ||||
|             // First 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, no need to check further | ||||
|             } catch (error) { | ||||
|               console.log(`ℹ️ ${issueAuthor} is not an organization member, checking codeowners...`); | ||||
|             } | ||||
|  | ||||
|             // If not an org member, check if they're a codeowner | ||||
|             try { | ||||
|               // Fetch CODEOWNERS file from the repository | ||||
|               const { data: codeownersFile } = await github.rest.repos.getContent({ | ||||
|                 owner: context.repo.owner, | ||||
|                 repo: context.repo.repo, | ||||
|                 path: 'CODEOWNERS', | ||||
|                 ref: 'dev' | ||||
|               }); | ||||
|  | ||||
|               // Decode the content (it's base64 encoded) | ||||
|               const codeownersContent = Buffer.from(codeownersFile.content, 'base64').toString('utf-8'); | ||||
|  | ||||
|               // Check if the issue author is mentioned in CODEOWNERS | ||||
|               // GitHub usernames in CODEOWNERS are prefixed with @ | ||||
|               if (codeownersContent.includes(`@${issueAuthor}`)) { | ||||
|                 console.log(`✅ ${issueAuthor} is a integration code owner`); | ||||
|                 return; // Authorized | ||||
|               } | ||||
|             } catch (error) { | ||||
|               console.error('Error checking CODEOWNERS:', error); | ||||
|             } | ||||
|  | ||||
|             // If we reach here, user is not authorized | ||||
|             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, authorized contributors, and integration code owners.\n\n` + | ||||
|                     `If you would like to:\n` + | ||||
|                     `- Report a bug: Please use the [bug report form](https://github.com/home-assistant/core/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'] | ||||
|             }); | ||||
							
								
								
									
										6
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							| @@ -17,7 +17,7 @@ jobs: | ||||
|       # - No PRs marked as no-stale | ||||
|       # - No issues (-1) | ||||
|       - name: 60 days stale PRs policy | ||||
|         uses: actions/stale@5f858e3efba33a5ca4407a664cc011ad407f2008 # v10.1.0 | ||||
|         uses: actions/stale@v9.1.0 | ||||
|         with: | ||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           days-before-stale: 60 | ||||
| @@ -57,7 +57,7 @@ jobs: | ||||
|       # - No issues marked as no-stale or help-wanted | ||||
|       # - No PRs (-1) | ||||
|       - name: 90 days stale issues | ||||
|         uses: actions/stale@5f858e3efba33a5ca4407a664cc011ad407f2008 # v10.1.0 | ||||
|         uses: actions/stale@v9.1.0 | ||||
|         with: | ||||
|           repo-token: ${{ steps.token.outputs.token }} | ||||
|           days-before-stale: 90 | ||||
| @@ -87,7 +87,7 @@ jobs: | ||||
|       # - No Issues marked as no-stale or help-wanted | ||||
|       # - No PRs (-1) | ||||
|       - name: Needs more information stale issues policy | ||||
|         uses: actions/stale@5f858e3efba33a5ca4407a664cc011ad407f2008 # v10.1.0 | ||||
|         uses: actions/stale@v9.1.0 | ||||
|         with: | ||||
|           repo-token: ${{ steps.token.outputs.token }} | ||||
|           only-labels: "needs-more-information" | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/workflows/translations.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/translations.yml
									
									
									
									
										vendored
									
									
								
							| @@ -19,10 +19,10 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Set up Python ${{ env.DEFAULT_PYTHON }} | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         uses: actions/setup-python@v5.5.0 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON }} | ||||
|  | ||||
|   | ||||
							
								
								
									
										36
									
								
								.github/workflows/wheels.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										36
									
								
								.github/workflows/wheels.yml
									
									
									
									
										vendored
									
									
								
							| @@ -32,11 +32,11 @@ jobs: | ||||
|       architectures: ${{ steps.info.outputs.architectures }} | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Set up Python ${{ env.DEFAULT_PYTHON }} | ||||
|         id: python | ||||
|         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||
|         uses: actions/setup-python@v5.5.0 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON }} | ||||
|           check-latest: true | ||||
| @@ -91,7 +91,7 @@ jobs: | ||||
|           ) > build_constraints.txt | ||||
|  | ||||
|       - name: Upload env_file | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         uses: actions/upload-artifact@v4.6.2 | ||||
|         with: | ||||
|           name: env_file | ||||
|           path: ./.env_file | ||||
| @@ -99,14 +99,14 @@ jobs: | ||||
|           overwrite: true | ||||
|  | ||||
|       - name: Upload build_constraints | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         uses: actions/upload-artifact@v4.6.2 | ||||
|         with: | ||||
|           name: build_constraints | ||||
|           path: ./build_constraints.txt | ||||
|           overwrite: true | ||||
|  | ||||
|       - name: Upload requirements_diff | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         uses: actions/upload-artifact@v4.6.2 | ||||
|         with: | ||||
|           name: requirements_diff | ||||
|           path: ./requirements_diff.txt | ||||
| @@ -118,7 +118,7 @@ jobs: | ||||
|           python -m script.gen_requirements_all ci | ||||
|  | ||||
|       - name: Upload requirements_all_wheels | ||||
|         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||
|         uses: actions/upload-artifact@v4.6.2 | ||||
|         with: | ||||
|           name: requirements_all_wheels | ||||
|           path: ./requirements_all_wheels_*.txt | ||||
| @@ -135,20 +135,20 @@ jobs: | ||||
|         arch: ${{ fromJson(needs.init.outputs.architectures) }} | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Download env_file | ||||
|         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||
|         uses: actions/download-artifact@v4.2.1 | ||||
|         with: | ||||
|           name: env_file | ||||
|  | ||||
|       - name: Download build_constraints | ||||
|         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||
|         uses: actions/download-artifact@v4.2.1 | ||||
|         with: | ||||
|           name: build_constraints | ||||
|  | ||||
|       - name: Download requirements_diff | ||||
|         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||
|         uses: actions/download-artifact@v4.2.1 | ||||
|         with: | ||||
|           name: requirements_diff | ||||
|  | ||||
| @@ -158,9 +158,8 @@ jobs: | ||||
|           sed -i "/uv/d" requirements.txt | ||||
|           sed -i "/uv/d" requirements_diff.txt | ||||
|  | ||||
|       # home-assistant/wheels doesn't support sha pinning | ||||
|       - name: Build wheels | ||||
|         uses: home-assistant/wheels@2025.09.1 | ||||
|         uses: home-assistant/wheels@2025.03.0 | ||||
|         with: | ||||
|           abi: ${{ matrix.abi }} | ||||
|           tag: musllinux_1_2 | ||||
| @@ -185,25 +184,25 @@ jobs: | ||||
|         arch: ${{ fromJson(needs.init.outputs.architectures) }} | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||
|         uses: actions/checkout@v4.2.2 | ||||
|  | ||||
|       - name: Download env_file | ||||
|         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||
|         uses: actions/download-artifact@v4.2.1 | ||||
|         with: | ||||
|           name: env_file | ||||
|  | ||||
|       - name: Download build_constraints | ||||
|         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||
|         uses: actions/download-artifact@v4.2.1 | ||||
|         with: | ||||
|           name: build_constraints | ||||
|  | ||||
|       - name: Download requirements_diff | ||||
|         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||
|         uses: actions/download-artifact@v4.2.1 | ||||
|         with: | ||||
|           name: requirements_diff | ||||
|  | ||||
|       - name: Download requirements_all_wheels | ||||
|         uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0 | ||||
|         uses: actions/download-artifact@v4.2.1 | ||||
|         with: | ||||
|           name: requirements_all_wheels | ||||
|  | ||||
| @@ -219,9 +218,8 @@ jobs: | ||||
|           sed -i "/uv/d" requirements.txt | ||||
|           sed -i "/uv/d" requirements_diff.txt | ||||
|  | ||||
|       # home-assistant/wheels doesn't support sha pinning | ||||
|       - name: Build wheels | ||||
|         uses: home-assistant/wheels@2025.09.1 | ||||
|         uses: home-assistant/wheels@2025.03.0 | ||||
|         with: | ||||
|           abi: ${{ matrix.abi }} | ||||
|           tag: musllinux_1_2 | ||||
|   | ||||
							
								
								
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -137,8 +137,4 @@ tmp_cache | ||||
| .ropeproject | ||||
|  | ||||
| # Will be created from script/split_tests.py | ||||
| pytest_buckets.txt | ||||
|  | ||||
| # AI tooling | ||||
| .claude/settings.local.json | ||||
|  | ||||
| pytest_buckets.txt | ||||
| @@ -1,8 +1,8 @@ | ||||
| repos: | ||||
|   - repo: https://github.com/astral-sh/ruff-pre-commit | ||||
|     rev: v0.13.0 | ||||
|     rev: v0.11.0 | ||||
|     hooks: | ||||
|       - id: ruff-check | ||||
|       - id: ruff | ||||
|         args: | ||||
|           - --fix | ||||
|       - id: ruff-format | ||||
| @@ -18,7 +18,7 @@ repos: | ||||
|         exclude_types: [csv, json, html] | ||||
|         exclude: ^tests/fixtures/|homeassistant/generated/|tests/components/.*/snapshots/ | ||||
|   - repo: https://github.com/pre-commit/pre-commit-hooks | ||||
|     rev: v6.0.0 | ||||
|     rev: v5.0.0 | ||||
|     hooks: | ||||
|       - id: check-executables-have-shebangs | ||||
|         stages: [manual] | ||||
| @@ -30,7 +30,7 @@ repos: | ||||
|           - --branch=master | ||||
|           - --branch=rc | ||||
|   - repo: https://github.com/adrienverge/yamllint.git | ||||
|     rev: v1.37.1 | ||||
|     rev: v1.35.1 | ||||
|     hooks: | ||||
|       - id: yamllint | ||||
|   - repo: https://github.com/pre-commit/mirrors-prettier | ||||
|   | ||||
| @@ -53,7 +53,6 @@ homeassistant.components.air_quality.* | ||||
| homeassistant.components.airgradient.* | ||||
| homeassistant.components.airly.* | ||||
| homeassistant.components.airnow.* | ||||
| homeassistant.components.airos.* | ||||
| homeassistant.components.airq.* | ||||
| homeassistant.components.airthings.* | ||||
| homeassistant.components.airthings_ble.* | ||||
| @@ -66,9 +65,7 @@ homeassistant.components.aladdin_connect.* | ||||
| homeassistant.components.alarm_control_panel.* | ||||
| homeassistant.components.alert.* | ||||
| homeassistant.components.alexa.* | ||||
| homeassistant.components.alexa_devices.* | ||||
| homeassistant.components.alpha_vantage.* | ||||
| homeassistant.components.altruist.* | ||||
| homeassistant.components.amazon_polly.* | ||||
| homeassistant.components.amberelectric.* | ||||
| homeassistant.components.ambient_network.* | ||||
| @@ -142,7 +139,6 @@ homeassistant.components.cloud.* | ||||
| homeassistant.components.co2signal.* | ||||
| homeassistant.components.comelit.* | ||||
| homeassistant.components.command_line.* | ||||
| homeassistant.components.compit.* | ||||
| homeassistant.components.config.* | ||||
| homeassistant.components.configurator.* | ||||
| homeassistant.components.cookidoo.* | ||||
| @@ -170,7 +166,6 @@ homeassistant.components.dnsip.* | ||||
| homeassistant.components.doorbird.* | ||||
| homeassistant.components.dormakaba_dkey.* | ||||
| homeassistant.components.downloader.* | ||||
| homeassistant.components.droplet.* | ||||
| homeassistant.components.dsmr.* | ||||
| homeassistant.components.duckdns.* | ||||
| homeassistant.components.dunehd.* | ||||
| @@ -203,7 +198,6 @@ homeassistant.components.feedreader.* | ||||
| homeassistant.components.file_upload.* | ||||
| homeassistant.components.filesize.* | ||||
| homeassistant.components.filter.* | ||||
| homeassistant.components.firefly_iii.* | ||||
| homeassistant.components.fitbit.* | ||||
| homeassistant.components.flexit_bacnet.* | ||||
| homeassistant.components.flux_led.* | ||||
| @@ -221,7 +215,6 @@ homeassistant.components.generic_thermostat.* | ||||
| homeassistant.components.geo_location.* | ||||
| homeassistant.components.geocaching.* | ||||
| homeassistant.components.gios.* | ||||
| homeassistant.components.github.* | ||||
| homeassistant.components.glances.* | ||||
| homeassistant.components.go2rtc.* | ||||
| homeassistant.components.goalzero.* | ||||
| @@ -277,7 +270,6 @@ homeassistant.components.image_processing.* | ||||
| homeassistant.components.image_upload.* | ||||
| homeassistant.components.imap.* | ||||
| homeassistant.components.imgw_pib.* | ||||
| homeassistant.components.immich.* | ||||
| homeassistant.components.incomfort.* | ||||
| homeassistant.components.input_button.* | ||||
| homeassistant.components.input_select.* | ||||
| @@ -299,7 +291,6 @@ homeassistant.components.kaleidescape.* | ||||
| homeassistant.components.knocki.* | ||||
| homeassistant.components.knx.* | ||||
| homeassistant.components.kraken.* | ||||
| homeassistant.components.kulersky.* | ||||
| homeassistant.components.lacrosse.* | ||||
| homeassistant.components.lacrosse_view.* | ||||
| homeassistant.components.lamarzocco.* | ||||
| @@ -311,10 +302,10 @@ homeassistant.components.ld2410_ble.* | ||||
| homeassistant.components.led_ble.* | ||||
| homeassistant.components.lektrico.* | ||||
| homeassistant.components.letpot.* | ||||
| homeassistant.components.libre_hardware_monitor.* | ||||
| homeassistant.components.lidarr.* | ||||
| homeassistant.components.lifx.* | ||||
| homeassistant.components.light.* | ||||
| homeassistant.components.linear_garage_door.* | ||||
| homeassistant.components.linkplay.* | ||||
| homeassistant.components.litejet.* | ||||
| homeassistant.components.litterrobot.* | ||||
| @@ -327,7 +318,6 @@ homeassistant.components.london_underground.* | ||||
| homeassistant.components.lookin.* | ||||
| homeassistant.components.lovelace.* | ||||
| homeassistant.components.luftdaten.* | ||||
| homeassistant.components.lunatone.* | ||||
| homeassistant.components.madvr.* | ||||
| homeassistant.components.manual.* | ||||
| homeassistant.components.mastodon.* | ||||
| @@ -341,7 +331,6 @@ homeassistant.components.media_player.* | ||||
| homeassistant.components.media_source.* | ||||
| homeassistant.components.met_eireann.* | ||||
| homeassistant.components.metoffice.* | ||||
| homeassistant.components.miele.* | ||||
| homeassistant.components.mikrotik.* | ||||
| homeassistant.components.min_max.* | ||||
| homeassistant.components.minecraft_server.* | ||||
| @@ -373,23 +362,18 @@ homeassistant.components.no_ip.* | ||||
| homeassistant.components.nordpool.* | ||||
| homeassistant.components.notify.* | ||||
| homeassistant.components.notion.* | ||||
| homeassistant.components.ntfy.* | ||||
| homeassistant.components.number.* | ||||
| homeassistant.components.nut.* | ||||
| homeassistant.components.ohme.* | ||||
| homeassistant.components.onboarding.* | ||||
| homeassistant.components.oncue.* | ||||
| homeassistant.components.onedrive.* | ||||
| homeassistant.components.onewire.* | ||||
| homeassistant.components.onkyo.* | ||||
| homeassistant.components.open_meteo.* | ||||
| homeassistant.components.open_router.* | ||||
| homeassistant.components.openai_conversation.* | ||||
| homeassistant.components.openexchangerates.* | ||||
| homeassistant.components.opensky.* | ||||
| homeassistant.components.openuv.* | ||||
| homeassistant.components.opnsense.* | ||||
| homeassistant.components.opower.* | ||||
| homeassistant.components.oralb.* | ||||
| homeassistant.components.otbr.* | ||||
| homeassistant.components.overkiz.* | ||||
| @@ -397,16 +381,13 @@ homeassistant.components.overseerr.* | ||||
| homeassistant.components.p1_monitor.* | ||||
| homeassistant.components.pandora.* | ||||
| homeassistant.components.panel_custom.* | ||||
| homeassistant.components.paperless_ngx.* | ||||
| homeassistant.components.peblar.* | ||||
| homeassistant.components.peco.* | ||||
| homeassistant.components.pegel_online.* | ||||
| homeassistant.components.persistent_notification.* | ||||
| homeassistant.components.person.* | ||||
| homeassistant.components.pi_hole.* | ||||
| homeassistant.components.ping.* | ||||
| homeassistant.components.plugwise.* | ||||
| homeassistant.components.portainer.* | ||||
| homeassistant.components.powerfox.* | ||||
| homeassistant.components.powerwall.* | ||||
| homeassistant.components.private_ble_device.* | ||||
| @@ -446,9 +427,9 @@ homeassistant.components.rituals_perfume_genie.* | ||||
| homeassistant.components.roborock.* | ||||
| homeassistant.components.roku.* | ||||
| homeassistant.components.romy.* | ||||
| homeassistant.components.route_b_smart_meter.* | ||||
| homeassistant.components.rpi_power.* | ||||
| homeassistant.components.rss_feed_template.* | ||||
| homeassistant.components.rtsp_to_webrtc.* | ||||
| homeassistant.components.russound_rio.* | ||||
| homeassistant.components.ruuvi_gateway.* | ||||
| homeassistant.components.ruuvitag_ble.* | ||||
| @@ -467,7 +448,6 @@ homeassistant.components.sensorpush_cloud.* | ||||
| homeassistant.components.sensoterra.* | ||||
| homeassistant.components.senz.* | ||||
| homeassistant.components.sfr_box.* | ||||
| homeassistant.components.sftp_storage.* | ||||
| homeassistant.components.shell_command.* | ||||
| homeassistant.components.shelly.* | ||||
| homeassistant.components.shopping_list.* | ||||
| @@ -476,11 +456,9 @@ homeassistant.components.simplisafe.* | ||||
| homeassistant.components.siren.* | ||||
| homeassistant.components.skybell.* | ||||
| homeassistant.components.slack.* | ||||
| homeassistant.components.sleep_as_android.* | ||||
| homeassistant.components.sleepiq.* | ||||
| homeassistant.components.smhi.* | ||||
| homeassistant.components.smlight.* | ||||
| homeassistant.components.smtp.* | ||||
| homeassistant.components.snooz.* | ||||
| homeassistant.components.solarlog.* | ||||
| homeassistant.components.sonarr.* | ||||
| @@ -512,12 +490,10 @@ homeassistant.components.tag.* | ||||
| homeassistant.components.tailscale.* | ||||
| homeassistant.components.tailwind.* | ||||
| homeassistant.components.tami4.* | ||||
| homeassistant.components.tankerkoenig.* | ||||
| homeassistant.components.tautulli.* | ||||
| homeassistant.components.tcp.* | ||||
| homeassistant.components.technove.* | ||||
| homeassistant.components.tedee.* | ||||
| homeassistant.components.telegram_bot.* | ||||
| homeassistant.components.text.* | ||||
| homeassistant.components.thethingsnetwork.* | ||||
| homeassistant.components.threshold.* | ||||
| @@ -548,7 +524,6 @@ homeassistant.components.unifiprotect.* | ||||
| homeassistant.components.upcloud.* | ||||
| homeassistant.components.update.* | ||||
| homeassistant.components.uptime.* | ||||
| homeassistant.components.uptime_kuma.* | ||||
| homeassistant.components.uptimerobot.* | ||||
| homeassistant.components.usb.* | ||||
| homeassistant.components.uvc.* | ||||
| @@ -556,10 +531,8 @@ homeassistant.components.vacuum.* | ||||
| homeassistant.components.vallox.* | ||||
| homeassistant.components.valve.* | ||||
| homeassistant.components.velbus.* | ||||
| homeassistant.components.vivotek.* | ||||
| homeassistant.components.vlc_telnet.* | ||||
| homeassistant.components.vodafone_station.* | ||||
| homeassistant.components.volvo.* | ||||
| homeassistant.components.wake_on_lan.* | ||||
| homeassistant.components.wake_word.* | ||||
| homeassistant.components.wallbox.* | ||||
|   | ||||
							
								
								
									
										2
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							| @@ -45,7 +45,7 @@ | ||||
|     { | ||||
|       "label": "Ruff", | ||||
|       "type": "shell", | ||||
|       "command": "pre-commit run ruff-check --all-files", | ||||
|       "command": "pre-commit run ruff --all-files", | ||||
|       "group": { | ||||
|         "kind": "test", | ||||
|         "isDefault": true | ||||
|   | ||||
							
								
								
									
										255
									
								
								CODEOWNERS
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										255
									
								
								CODEOWNERS
									
									
									
										generated
									
									
									
								
							| @@ -46,8 +46,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/accuweather/ @bieniu | ||||
| /homeassistant/components/acmeda/ @atmurray | ||||
| /tests/components/acmeda/ @atmurray | ||||
| /homeassistant/components/adax/ @danielhiversen @lazytarget | ||||
| /tests/components/adax/ @danielhiversen @lazytarget | ||||
| /homeassistant/components/adax/ @danielhiversen | ||||
| /tests/components/adax/ @danielhiversen | ||||
| /homeassistant/components/adguard/ @frenck | ||||
| /tests/components/adguard/ @frenck | ||||
| /homeassistant/components/ads/ @mrpasztoradam | ||||
| @@ -57,8 +57,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/aemet/ @Noltari | ||||
| /homeassistant/components/agent_dvr/ @ispysoftware | ||||
| /tests/components/agent_dvr/ @ispysoftware | ||||
| /homeassistant/components/ai_task/ @home-assistant/core | ||||
| /tests/components/ai_task/ @home-assistant/core | ||||
| /homeassistant/components/air_quality/ @home-assistant/core | ||||
| /tests/components/air_quality/ @home-assistant/core | ||||
| /homeassistant/components/airgradient/ @airgradienthq @joostlek | ||||
| @@ -67,8 +65,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/airly/ @bieniu | ||||
| /homeassistant/components/airnow/ @asymworks | ||||
| /tests/components/airnow/ @asymworks | ||||
| /homeassistant/components/airos/ @CoMPaTech | ||||
| /tests/components/airos/ @CoMPaTech | ||||
| /homeassistant/components/airq/ @Sibgatulin @dl2080 | ||||
| /tests/components/airq/ @Sibgatulin @dl2080 | ||||
| /homeassistant/components/airthings/ @danielhiversen @LaStrada | ||||
| @@ -87,18 +83,12 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/airzone/ @Noltari | ||||
| /homeassistant/components/airzone_cloud/ @Noltari | ||||
| /tests/components/airzone_cloud/ @Noltari | ||||
| /homeassistant/components/aladdin_connect/ @swcloudgenie | ||||
| /tests/components/aladdin_connect/ @swcloudgenie | ||||
| /homeassistant/components/alarm_control_panel/ @home-assistant/core | ||||
| /tests/components/alarm_control_panel/ @home-assistant/core | ||||
| /homeassistant/components/alert/ @home-assistant/core @frenck | ||||
| /tests/components/alert/ @home-assistant/core @frenck | ||||
| /homeassistant/components/alexa/ @home-assistant/cloud @ochlocracy @jbouwh | ||||
| /tests/components/alexa/ @home-assistant/cloud @ochlocracy @jbouwh | ||||
| /homeassistant/components/alexa_devices/ @chemelli74 | ||||
| /tests/components/alexa_devices/ @chemelli74 | ||||
| /homeassistant/components/altruist/ @airalab @LoSk-p | ||||
| /tests/components/altruist/ @airalab @LoSk-p | ||||
| /homeassistant/components/amazon_polly/ @jschlyter | ||||
| /homeassistant/components/amberelectric/ @madpilot | ||||
| /tests/components/amberelectric/ @madpilot | ||||
| @@ -107,8 +97,8 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/ambient_station/ @bachya | ||||
| /tests/components/ambient_station/ @bachya | ||||
| /homeassistant/components/amcrest/ @flacjacket | ||||
| /homeassistant/components/analytics/ @home-assistant/core | ||||
| /tests/components/analytics/ @home-assistant/core | ||||
| /homeassistant/components/analytics/ @home-assistant/core @ludeeus | ||||
| /tests/components/analytics/ @home-assistant/core @ludeeus | ||||
| /homeassistant/components/analytics_insights/ @joostlek | ||||
| /tests/components/analytics_insights/ @joostlek | ||||
| /homeassistant/components/android_ip_webcam/ @engrbm87 | ||||
| @@ -154,12 +144,12 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/arve/ @ikalnyi | ||||
| /homeassistant/components/aseko_pool_live/ @milanmeu | ||||
| /tests/components/aseko_pool_live/ @milanmeu | ||||
| /homeassistant/components/assist_pipeline/ @synesthesiam @arturpragacz | ||||
| /tests/components/assist_pipeline/ @synesthesiam @arturpragacz | ||||
| /homeassistant/components/assist_satellite/ @home-assistant/core @synesthesiam @arturpragacz | ||||
| /tests/components/assist_satellite/ @home-assistant/core @synesthesiam @arturpragacz | ||||
| /homeassistant/components/asuswrt/ @kennedyshead @ollo69 @Vaskivskyi | ||||
| /tests/components/asuswrt/ @kennedyshead @ollo69 @Vaskivskyi | ||||
| /homeassistant/components/assist_pipeline/ @balloob @synesthesiam | ||||
| /tests/components/assist_pipeline/ @balloob @synesthesiam | ||||
| /homeassistant/components/assist_satellite/ @home-assistant/core @synesthesiam | ||||
| /tests/components/assist_satellite/ @home-assistant/core @synesthesiam | ||||
| /homeassistant/components/asuswrt/ @kennedyshead @ollo69 | ||||
| /tests/components/asuswrt/ @kennedyshead @ollo69 | ||||
| /homeassistant/components/atag/ @MatsNL | ||||
| /tests/components/atag/ @MatsNL | ||||
| /homeassistant/components/aten_pe/ @mtdcr | ||||
| @@ -181,8 +171,6 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/avea/ @pattyland | ||||
| /homeassistant/components/awair/ @ahayworth @danielsjf | ||||
| /tests/components/awair/ @ahayworth @danielsjf | ||||
| /homeassistant/components/aws_s3/ @tomasbedrich | ||||
| /tests/components/aws_s3/ @tomasbedrich | ||||
| /homeassistant/components/axis/ @Kane610 | ||||
| /tests/components/axis/ @Kane610 | ||||
| /homeassistant/components/azure_data_explorer/ @kaareseras | ||||
| @@ -212,8 +200,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/blebox/ @bbx-a @swistakm | ||||
| /homeassistant/components/blink/ @fronzbot @mkmer | ||||
| /tests/components/blink/ @fronzbot @mkmer | ||||
| /homeassistant/components/blue_current/ @gleeuwen @NickKoepr @jtodorova23 | ||||
| /tests/components/blue_current/ @gleeuwen @NickKoepr @jtodorova23 | ||||
| /homeassistant/components/blue_current/ @Floris272 @gleeuwen | ||||
| /tests/components/blue_current/ @Floris272 @gleeuwen | ||||
| /homeassistant/components/bluemaestro/ @bdraco | ||||
| /tests/components/bluemaestro/ @bdraco | ||||
| /homeassistant/components/blueprint/ @home-assistant/core | ||||
| @@ -292,16 +280,14 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/command_line/ @gjohansson-ST | ||||
| /homeassistant/components/compensation/ @Petro31 | ||||
| /tests/components/compensation/ @Petro31 | ||||
| /homeassistant/components/compit/ @Przemko92 | ||||
| /tests/components/compit/ @Przemko92 | ||||
| /homeassistant/components/config/ @home-assistant/core | ||||
| /tests/components/config/ @home-assistant/core | ||||
| /homeassistant/components/configurator/ @home-assistant/core | ||||
| /tests/components/configurator/ @home-assistant/core | ||||
| /homeassistant/components/control4/ @lawtancool | ||||
| /tests/components/control4/ @lawtancool | ||||
| /homeassistant/components/conversation/ @home-assistant/core @synesthesiam @arturpragacz | ||||
| /tests/components/conversation/ @home-assistant/core @synesthesiam @arturpragacz | ||||
| /homeassistant/components/conversation/ @home-assistant/core @synesthesiam | ||||
| /tests/components/conversation/ @home-assistant/core @synesthesiam | ||||
| /homeassistant/components/cookidoo/ @miaucl | ||||
| /tests/components/cookidoo/ @miaucl | ||||
| /homeassistant/components/coolmaster/ @OnFreund | ||||
| @@ -315,9 +301,6 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/crownstone/ @Crownstone @RicArch97 | ||||
| /tests/components/crownstone/ @Crownstone @RicArch97 | ||||
| /homeassistant/components/cups/ @fabaff | ||||
| /tests/components/cups/ @fabaff | ||||
| /homeassistant/components/cync/ @Kinachi249 | ||||
| /tests/components/cync/ @Kinachi249 | ||||
| /homeassistant/components/daikin/ @fredrike | ||||
| /tests/components/daikin/ @fredrike | ||||
| /homeassistant/components/date/ @home-assistant/core | ||||
| @@ -339,8 +322,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/demo/ @home-assistant/core | ||||
| /homeassistant/components/denonavr/ @ol-iver @starkillerOG | ||||
| /tests/components/denonavr/ @ol-iver @starkillerOG | ||||
| /homeassistant/components/derivative/ @afaucogney @karwosts | ||||
| /tests/components/derivative/ @afaucogney @karwosts | ||||
| /homeassistant/components/derivative/ @afaucogney | ||||
| /tests/components/derivative/ @afaucogney | ||||
| /homeassistant/components/devialet/ @fwestenberg | ||||
| /tests/components/devialet/ @fwestenberg | ||||
| /homeassistant/components/device_automation/ @home-assistant/core | ||||
| @@ -381,8 +364,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/dremel_3d_printer/ @tkdrob | ||||
| /homeassistant/components/drop_connect/ @ChandlerSystems @pfrazer | ||||
| /tests/components/drop_connect/ @ChandlerSystems @pfrazer | ||||
| /homeassistant/components/droplet/ @sarahseidman | ||||
| /tests/components/droplet/ @sarahseidman | ||||
| /homeassistant/components/dsmr/ @Robbie1221 | ||||
| /tests/components/dsmr/ @Robbie1221 | ||||
| /homeassistant/components/dsmr_reader/ @sorted-bits @glodenox @erwindouna | ||||
| @@ -412,8 +393,6 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/egardia/ @jeroenterheerdt | ||||
| /homeassistant/components/eheimdigital/ @autinerd | ||||
| /tests/components/eheimdigital/ @autinerd | ||||
| /homeassistant/components/ekeybionyx/ @richardpolzer | ||||
| /tests/components/ekeybionyx/ @richardpolzer | ||||
| /homeassistant/components/electrasmart/ @jafar-atili | ||||
| /tests/components/electrasmart/ @jafar-atili | ||||
| /homeassistant/components/electric_kiwi/ @mikey0000 | ||||
| @@ -432,8 +411,6 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/emby/ @mezz64 | ||||
| /homeassistant/components/emoncms/ @borpin @alexandrecuer | ||||
| /tests/components/emoncms/ @borpin @alexandrecuer | ||||
| /homeassistant/components/emoncms_history/ @alexandrecuer | ||||
| /tests/components/emoncms_history/ @alexandrecuer | ||||
| /homeassistant/components/emonitor/ @bdraco | ||||
| /tests/components/emonitor/ @bdraco | ||||
| /homeassistant/components/emulated_hue/ @bdraco @Tho85 | ||||
| @@ -448,12 +425,14 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/energyzero/ @klaasnicolaas | ||||
| /homeassistant/components/enigma2/ @autinerd | ||||
| /tests/components/enigma2/ @autinerd | ||||
| /homeassistant/components/enphase_envoy/ @bdraco @cgarwood @catsmanac | ||||
| /tests/components/enphase_envoy/ @bdraco @cgarwood @catsmanac | ||||
| /homeassistant/components/enocean/ @bdurrer | ||||
| /tests/components/enocean/ @bdurrer | ||||
| /homeassistant/components/enphase_envoy/ @bdraco @cgarwood @joostlek @catsmanac | ||||
| /tests/components/enphase_envoy/ @bdraco @cgarwood @joostlek @catsmanac | ||||
| /homeassistant/components/entur_public_transport/ @hfurubotten | ||||
| /homeassistant/components/environment_canada/ @gwww @michaeldavie | ||||
| /tests/components/environment_canada/ @gwww @michaeldavie | ||||
| /homeassistant/components/ephember/ @ttroy50 @roberty99 | ||||
| /homeassistant/components/ephember/ @ttroy50 | ||||
| /homeassistant/components/epic_games_store/ @hacf-fr @Quentame | ||||
| /tests/components/epic_games_store/ @hacf-fr @Quentame | ||||
| /homeassistant/components/epion/ @lhgravendeel | ||||
| @@ -464,16 +443,18 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/eq3btsmart/ @eulemitkeule @dbuezas | ||||
| /homeassistant/components/escea/ @lazdavila | ||||
| /tests/components/escea/ @lazdavila | ||||
| /homeassistant/components/esphome/ @jesserockz @kbx81 @bdraco | ||||
| /tests/components/esphome/ @jesserockz @kbx81 @bdraco | ||||
| /homeassistant/components/esphome/ @OttoWinter @jesserockz @kbx81 @bdraco | ||||
| /tests/components/esphome/ @OttoWinter @jesserockz @kbx81 @bdraco | ||||
| /homeassistant/components/eufylife_ble/ @bdr99 | ||||
| /tests/components/eufylife_ble/ @bdr99 | ||||
| /homeassistant/components/event/ @home-assistant/core | ||||
| /tests/components/event/ @home-assistant/core | ||||
| /homeassistant/components/evil_genius_labs/ @balloob | ||||
| /tests/components/evil_genius_labs/ @balloob | ||||
| /homeassistant/components/evohome/ @zxdavb | ||||
| /tests/components/evohome/ @zxdavb | ||||
| /homeassistant/components/ezviz/ @RenierM26 | ||||
| /tests/components/ezviz/ @RenierM26 | ||||
| /homeassistant/components/ezviz/ @RenierM26 @baqs | ||||
| /tests/components/ezviz/ @RenierM26 @baqs | ||||
| /homeassistant/components/faa_delays/ @ntilley905 | ||||
| /tests/components/faa_delays/ @ntilley905 | ||||
| /homeassistant/components/fan/ @home-assistant/core | ||||
| @@ -492,8 +473,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/filesize/ @gjohansson-ST | ||||
| /homeassistant/components/filter/ @dgomes | ||||
| /tests/components/filter/ @dgomes | ||||
| /homeassistant/components/firefly_iii/ @erwindouna | ||||
| /tests/components/firefly_iii/ @erwindouna | ||||
| /homeassistant/components/fireservicerota/ @cyberjunky | ||||
| /tests/components/fireservicerota/ @cyberjunky | ||||
| /homeassistant/components/firmata/ @DaAwesomeP | ||||
| @@ -521,8 +500,8 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/forked_daapd/ @uvjustin | ||||
| /tests/components/forked_daapd/ @uvjustin | ||||
| /homeassistant/components/fortios/ @kimfrellsen | ||||
| /homeassistant/components/foscam/ @Foscam-wangzhengyu | ||||
| /tests/components/foscam/ @Foscam-wangzhengyu | ||||
| /homeassistant/components/foscam/ @krmarien | ||||
| /tests/components/foscam/ @krmarien | ||||
| /homeassistant/components/freebox/ @hacf-fr @Quentame | ||||
| /tests/components/freebox/ @hacf-fr @Quentame | ||||
| /homeassistant/components/freedompro/ @stefano055415 | ||||
| @@ -656,8 +635,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/homeassistant/ @home-assistant/core | ||||
| /homeassistant/components/homeassistant_alerts/ @home-assistant/core | ||||
| /tests/components/homeassistant_alerts/ @home-assistant/core | ||||
| /homeassistant/components/homeassistant_connect_zbt2/ @home-assistant/core | ||||
| /tests/components/homeassistant_connect_zbt2/ @home-assistant/core | ||||
| /homeassistant/components/homeassistant_green/ @home-assistant/core | ||||
| /tests/components/homeassistant_green/ @home-assistant/core | ||||
| /homeassistant/components/homeassistant_hardware/ @home-assistant/core | ||||
| @@ -686,8 +663,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/http/ @home-assistant/core | ||||
| /homeassistant/components/huawei_lte/ @scop @fphammerle | ||||
| /tests/components/huawei_lte/ @scop @fphammerle | ||||
| /homeassistant/components/hue/ @marcelveldt | ||||
| /tests/components/hue/ @marcelveldt | ||||
| /homeassistant/components/hue/ @balloob @marcelveldt | ||||
| /tests/components/hue/ @balloob @marcelveldt | ||||
| /homeassistant/components/huisbaasje/ @dennisschroer | ||||
| /tests/components/huisbaasje/ @dennisschroer | ||||
| /homeassistant/components/humidifier/ @home-assistant/core @Shulyaka | ||||
| @@ -698,8 +675,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/husqvarna_automower/ @Thomas55555 | ||||
| /homeassistant/components/husqvarna_automower_ble/ @alistair23 | ||||
| /tests/components/husqvarna_automower_ble/ @alistair23 | ||||
| /homeassistant/components/huum/ @frwickst @vincentwolsink | ||||
| /tests/components/huum/ @frwickst @vincentwolsink | ||||
| /homeassistant/components/huum/ @frwickst | ||||
| /tests/components/huum/ @frwickst | ||||
| /homeassistant/components/hvv_departures/ @vigonotion | ||||
| /tests/components/hvv_departures/ @vigonotion | ||||
| /homeassistant/components/hydrawise/ @dknowles2 @thomaskistler @ptcryan | ||||
| @@ -727,12 +704,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/image_upload/ @home-assistant/core | ||||
| /homeassistant/components/imap/ @jbouwh | ||||
| /tests/components/imap/ @jbouwh | ||||
| /homeassistant/components/imeon_inverter/ @Imeon-Energy | ||||
| /tests/components/imeon_inverter/ @Imeon-Energy | ||||
| /homeassistant/components/imgw_pib/ @bieniu | ||||
| /tests/components/imgw_pib/ @bieniu | ||||
| /homeassistant/components/immich/ @mib1185 | ||||
| /tests/components/immich/ @mib1185 | ||||
| /homeassistant/components/improv_ble/ @emontnemery | ||||
| /tests/components/improv_ble/ @emontnemery | ||||
| /homeassistant/components/incomfort/ @jbouwh | ||||
| @@ -759,11 +732,11 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/integration/ @dgomes | ||||
| /homeassistant/components/intellifire/ @jeeftor | ||||
| /tests/components/intellifire/ @jeeftor | ||||
| /homeassistant/components/intent/ @home-assistant/core @synesthesiam @arturpragacz | ||||
| /tests/components/intent/ @home-assistant/core @synesthesiam @arturpragacz | ||||
| /homeassistant/components/intent/ @home-assistant/core @synesthesiam | ||||
| /tests/components/intent/ @home-assistant/core @synesthesiam | ||||
| /homeassistant/components/intesishome/ @jnimmo | ||||
| /homeassistant/components/iometer/ @jukrebs | ||||
| /tests/components/iometer/ @jukrebs | ||||
| /homeassistant/components/iometer/ @MaestroOnICe | ||||
| /tests/components/iometer/ @MaestroOnICe | ||||
| /homeassistant/components/ios/ @robbiet480 | ||||
| /tests/components/ios/ @robbiet480 | ||||
| /homeassistant/components/iotawatt/ @gtdiehl @jyavenard | ||||
| @@ -778,8 +751,6 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/iqvia/ @bachya | ||||
| /tests/components/iqvia/ @bachya | ||||
| /homeassistant/components/irish_rail_transport/ @ttroy50 | ||||
| /homeassistant/components/irm_kmi/ @jdejaegh | ||||
| /tests/components/irm_kmi/ @jdejaegh | ||||
| /homeassistant/components/iron_os/ @tr4nt0r | ||||
| /tests/components/iron_os/ @tr4nt0r | ||||
| /homeassistant/components/isal/ @bdraco | ||||
| @@ -804,6 +775,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/jellyfin/ @RunC0deRun @ctalkington | ||||
| /homeassistant/components/jewish_calendar/ @tsvi | ||||
| /tests/components/jewish_calendar/ @tsvi | ||||
| /homeassistant/components/juicenet/ @jesserockz | ||||
| /tests/components/juicenet/ @jesserockz | ||||
| /homeassistant/components/justnimbus/ @kvanzuijlen | ||||
| /tests/components/justnimbus/ @kvanzuijlen | ||||
| /homeassistant/components/jvc_projector/ @SteveEasley @msavazzi | ||||
| @@ -870,14 +843,14 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/lg_netcast/ @Drafteed @splinter98 | ||||
| /homeassistant/components/lg_thinq/ @LG-ThinQ-Integration | ||||
| /tests/components/lg_thinq/ @LG-ThinQ-Integration | ||||
| /homeassistant/components/libre_hardware_monitor/ @Sab44 | ||||
| /tests/components/libre_hardware_monitor/ @Sab44 | ||||
| /homeassistant/components/lidarr/ @tkdrob | ||||
| /tests/components/lidarr/ @tkdrob | ||||
| /homeassistant/components/lifx/ @Djelibeybi | ||||
| /tests/components/lifx/ @Djelibeybi | ||||
| /homeassistant/components/light/ @home-assistant/core | ||||
| /tests/components/light/ @home-assistant/core | ||||
| /homeassistant/components/linear_garage_door/ @IceBotYT | ||||
| /tests/components/linear_garage_door/ @IceBotYT | ||||
| /homeassistant/components/linkplay/ @Velleman | ||||
| /tests/components/linkplay/ @Velleman | ||||
| /homeassistant/components/linux_battery/ @fabaff | ||||
| @@ -910,8 +883,6 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/luci/ @mzdrale | ||||
| /homeassistant/components/luftdaten/ @fabaff @frenck | ||||
| /tests/components/luftdaten/ @fabaff @frenck | ||||
| /homeassistant/components/lunatone/ @MoonDevLT | ||||
| /tests/components/lunatone/ @MoonDevLT | ||||
| /homeassistant/components/lupusec/ @majuss @suaveolent | ||||
| /tests/components/lupusec/ @majuss @suaveolent | ||||
| /homeassistant/components/lutron/ @cdheiser @wilburCForce | ||||
| @@ -957,8 +928,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/met_eireann/ @DylanGore | ||||
| /homeassistant/components/meteo_france/ @hacf-fr @oncleben31 @Quentame | ||||
| /tests/components/meteo_france/ @hacf-fr @oncleben31 @Quentame | ||||
| /homeassistant/components/meteo_lt/ @xE1H | ||||
| /tests/components/meteo_lt/ @xE1H | ||||
| /homeassistant/components/meteoalarm/ @rolfberkenbosch | ||||
| /homeassistant/components/meteoclimatic/ @adrianmo | ||||
| /tests/components/meteoclimatic/ @adrianmo | ||||
| @@ -966,8 +935,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/metoffice/ @MrHarcombe @avee87 | ||||
| /homeassistant/components/microbees/ @microBeesTech | ||||
| /tests/components/microbees/ @microBeesTech | ||||
| /homeassistant/components/miele/ @astrandb | ||||
| /tests/components/miele/ @astrandb | ||||
| /homeassistant/components/mikrotik/ @engrbm87 | ||||
| /tests/components/mikrotik/ @engrbm87 | ||||
| /homeassistant/components/mill/ @danielhiversen | ||||
| @@ -1029,8 +996,7 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/nanoleaf/ @milanmeu @joostlek | ||||
| /homeassistant/components/nasweb/ @nasWebio | ||||
| /tests/components/nasweb/ @nasWebio | ||||
| /homeassistant/components/nederlandse_spoorwegen/ @YarmoM @heindrichpaul | ||||
| /tests/components/nederlandse_spoorwegen/ @YarmoM @heindrichpaul | ||||
| /homeassistant/components/nederlandse_spoorwegen/ @YarmoM | ||||
| /homeassistant/components/ness_alarm/ @nickw444 | ||||
| /tests/components/ness_alarm/ @nickw444 | ||||
| /homeassistant/components/nest/ @allenporter | ||||
| @@ -1065,8 +1031,6 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/nilu/ @hfurubotten | ||||
| /homeassistant/components/nina/ @DeerMaximum | ||||
| /tests/components/nina/ @DeerMaximum | ||||
| /homeassistant/components/nintendo_parental_controls/ @pantherale0 | ||||
| /tests/components/nintendo_parental_controls/ @pantherale0 | ||||
| /homeassistant/components/nissan_leaf/ @filcole | ||||
| /homeassistant/components/noaa_tides/ @jdelaney72 | ||||
| /homeassistant/components/nobo_hub/ @echoromeo @oyvindwe | ||||
| @@ -1083,8 +1047,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/nsw_fuel_station/ @nickw444 | ||||
| /homeassistant/components/nsw_rural_fire_service_feed/ @exxamalte | ||||
| /tests/components/nsw_rural_fire_service_feed/ @exxamalte | ||||
| /homeassistant/components/ntfy/ @tr4nt0r | ||||
| /tests/components/ntfy/ @tr4nt0r | ||||
| /homeassistant/components/nuheat/ @tstabrawa | ||||
| /tests/components/nuheat/ @tstabrawa | ||||
| /homeassistant/components/nuki/ @pschmitt @pvizeli @pree | ||||
| @@ -1113,6 +1075,8 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/ombi/ @larssont | ||||
| /homeassistant/components/onboarding/ @home-assistant/core | ||||
| /tests/components/onboarding/ @home-assistant/core | ||||
| /homeassistant/components/oncue/ @bdraco @peterager | ||||
| /tests/components/oncue/ @bdraco @peterager | ||||
| /homeassistant/components/ondilo_ico/ @JeromeHXP | ||||
| /tests/components/ondilo_ico/ @JeromeHXP | ||||
| /homeassistant/components/onedrive/ @zweckj | ||||
| @@ -1125,8 +1089,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/onvif/ @hunterjm @jterrace | ||||
| /homeassistant/components/open_meteo/ @frenck | ||||
| /tests/components/open_meteo/ @frenck | ||||
| /homeassistant/components/open_router/ @joostlek | ||||
| /tests/components/open_router/ @joostlek | ||||
| /homeassistant/components/openai_conversation/ @balloob | ||||
| /tests/components/openai_conversation/ @balloob | ||||
| /homeassistant/components/openerz/ @misialq | ||||
| /tests/components/openerz/ @misialq | ||||
| /homeassistant/components/openexchangerates/ @MartinHjelmare | ||||
| @@ -1141,8 +1105,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/opentherm_gw/ @mvn23 | ||||
| /homeassistant/components/openuv/ @bachya | ||||
| /tests/components/openuv/ @bachya | ||||
| /homeassistant/components/openweathermap/ @fabaff @freekode @nzapponi @wittypluck | ||||
| /tests/components/openweathermap/ @fabaff @freekode @nzapponi @wittypluck | ||||
| /homeassistant/components/openweathermap/ @fabaff @freekode @nzapponi | ||||
| /tests/components/openweathermap/ @fabaff @freekode @nzapponi | ||||
| /homeassistant/components/opnsense/ @mtreinish | ||||
| /tests/components/opnsense/ @mtreinish | ||||
| /homeassistant/components/opower/ @tronikos | ||||
| @@ -1168,8 +1132,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/palazzetti/ @dotvav | ||||
| /homeassistant/components/panel_custom/ @home-assistant/frontend | ||||
| /tests/components/panel_custom/ @home-assistant/frontend | ||||
| /homeassistant/components/paperless_ngx/ @fvgarrel | ||||
| /tests/components/paperless_ngx/ @fvgarrel | ||||
| /homeassistant/components/peblar/ @frenck | ||||
| /tests/components/peblar/ @frenck | ||||
| /homeassistant/components/peco/ @IceBotYT | ||||
| @@ -1192,28 +1154,22 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/ping/ @jpbede | ||||
| /homeassistant/components/plaato/ @JohNan | ||||
| /tests/components/plaato/ @JohNan | ||||
| /homeassistant/components/playstation_network/ @jackjpowell @tr4nt0r | ||||
| /tests/components/playstation_network/ @jackjpowell @tr4nt0r | ||||
| /homeassistant/components/plex/ @jjlawren | ||||
| /tests/components/plex/ @jjlawren | ||||
| /homeassistant/components/plugwise/ @CoMPaTech @bouwew | ||||
| /tests/components/plugwise/ @CoMPaTech @bouwew | ||||
| /homeassistant/components/plum_lightpad/ @ColinHarrington @prystupa | ||||
| /tests/components/plum_lightpad/ @ColinHarrington @prystupa | ||||
| /homeassistant/components/point/ @fredrike | ||||
| /tests/components/point/ @fredrike | ||||
| /homeassistant/components/pooldose/ @lmaertin | ||||
| /tests/components/pooldose/ @lmaertin | ||||
| /homeassistant/components/poolsense/ @haemishkyd | ||||
| /tests/components/poolsense/ @haemishkyd | ||||
| /homeassistant/components/portainer/ @erwindouna | ||||
| /tests/components/portainer/ @erwindouna | ||||
| /homeassistant/components/powerfox/ @klaasnicolaas | ||||
| /tests/components/powerfox/ @klaasnicolaas | ||||
| /homeassistant/components/powerwall/ @bdraco @jrester @daniel-simpson | ||||
| /tests/components/powerwall/ @bdraco @jrester @daniel-simpson | ||||
| /homeassistant/components/private_ble_device/ @Jc2k | ||||
| /tests/components/private_ble_device/ @Jc2k | ||||
| /homeassistant/components/probe_plus/ @pantherale0 | ||||
| /tests/components/probe_plus/ @pantherale0 | ||||
| /homeassistant/components/profiler/ @bdraco | ||||
| /tests/components/profiler/ @bdraco | ||||
| /homeassistant/components/progettihwsw/ @ardaseremet | ||||
| @@ -1225,6 +1181,8 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/proximity/ @mib1185 | ||||
| /tests/components/proximity/ @mib1185 | ||||
| /homeassistant/components/proxmoxve/ @jhollowe @Corbeno | ||||
| /homeassistant/components/prusalink/ @balloob | ||||
| /tests/components/prusalink/ @balloob | ||||
| /homeassistant/components/ps4/ @ktnrg45 | ||||
| /tests/components/ps4/ @ktnrg45 | ||||
| /homeassistant/components/pterodactyl/ @elmurato | ||||
| @@ -1258,7 +1216,6 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/qnap_qsw/ @Noltari | ||||
| /tests/components/qnap_qsw/ @Noltari | ||||
| /homeassistant/components/quantum_gateway/ @cisasteelersfan | ||||
| /tests/components/quantum_gateway/ @cisasteelersfan | ||||
| /homeassistant/components/qvr_pro/ @oblogic7 | ||||
| /homeassistant/components/qwikswitch/ @kellerza | ||||
| /tests/components/qwikswitch/ @kellerza | ||||
| @@ -1297,12 +1254,10 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/recovery_mode/ @home-assistant/core | ||||
| /homeassistant/components/refoss/ @ashionky | ||||
| /tests/components/refoss/ @ashionky | ||||
| /homeassistant/components/rehlko/ @bdraco @peterager | ||||
| /tests/components/rehlko/ @bdraco @peterager | ||||
| /homeassistant/components/remote/ @home-assistant/core | ||||
| /tests/components/remote/ @home-assistant/core | ||||
| /homeassistant/components/remote_calendar/ @Thomas55555 @allenporter | ||||
| /tests/components/remote_calendar/ @Thomas55555 @allenporter | ||||
| /homeassistant/components/remote_calendar/ @Thomas55555 | ||||
| /tests/components/remote_calendar/ @Thomas55555 | ||||
| /homeassistant/components/renault/ @epenet | ||||
| /tests/components/renault/ @epenet | ||||
| /homeassistant/components/renson/ @jimmyd-be | ||||
| @@ -1318,8 +1273,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/rflink/ @javicalle | ||||
| /homeassistant/components/rfxtrx/ @danielhiversen @elupus @RobBie1221 | ||||
| /tests/components/rfxtrx/ @danielhiversen @elupus @RobBie1221 | ||||
| /homeassistant/components/rhasspy/ @synesthesiam | ||||
| /tests/components/rhasspy/ @synesthesiam | ||||
| /homeassistant/components/rhasspy/ @balloob @synesthesiam | ||||
| /tests/components/rhasspy/ @balloob @synesthesiam | ||||
| /homeassistant/components/ridwell/ @bachya | ||||
| /tests/components/ridwell/ @bachya | ||||
| /homeassistant/components/ring/ @sdb9696 | ||||
| @@ -1340,12 +1295,12 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/roomba/ @pschmitt @cyr-ius @shenxn @Orhideous | ||||
| /homeassistant/components/roon/ @pavoni | ||||
| /tests/components/roon/ @pavoni | ||||
| /homeassistant/components/route_b_smart_meter/ @SeraphicRav | ||||
| /tests/components/route_b_smart_meter/ @SeraphicRav | ||||
| /homeassistant/components/rpi_power/ @shenxn @swetoast | ||||
| /tests/components/rpi_power/ @shenxn @swetoast | ||||
| /homeassistant/components/rss_feed_template/ @home-assistant/core | ||||
| /tests/components/rss_feed_template/ @home-assistant/core | ||||
| /homeassistant/components/rtsp_to_webrtc/ @allenporter | ||||
| /tests/components/rtsp_to_webrtc/ @allenporter | ||||
| /homeassistant/components/ruckus_unleashed/ @lanrat @ms264556 @gabe565 | ||||
| /tests/components/ruckus_unleashed/ @lanrat @ms264556 @gabe565 | ||||
| /homeassistant/components/russound_rio/ @noahhusby | ||||
| @@ -1364,8 +1319,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/samsungtv/ @chemelli74 @epenet | ||||
| /homeassistant/components/sanix/ @tomaszsluszniak | ||||
| /tests/components/sanix/ @tomaszsluszniak | ||||
| /homeassistant/components/satel_integra/ @Tommatheussen | ||||
| /tests/components/satel_integra/ @Tommatheussen | ||||
| /homeassistant/components/scene/ @home-assistant/core | ||||
| /tests/components/scene/ @home-assistant/core | ||||
| /homeassistant/components/schedule/ @home-assistant/core | ||||
| @@ -1411,14 +1364,12 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/seventeentrack/ @shaiu | ||||
| /homeassistant/components/sfr_box/ @epenet | ||||
| /tests/components/sfr_box/ @epenet | ||||
| /homeassistant/components/sftp_storage/ @maretodoric | ||||
| /tests/components/sftp_storage/ @maretodoric | ||||
| /homeassistant/components/sharkiq/ @JeffResc @funkybunch @TheOneOgre | ||||
| /tests/components/sharkiq/ @JeffResc @funkybunch @TheOneOgre | ||||
| /homeassistant/components/sharkiq/ @JeffResc @funkybunch | ||||
| /tests/components/sharkiq/ @JeffResc @funkybunch | ||||
| /homeassistant/components/shell_command/ @home-assistant/core | ||||
| /tests/components/shell_command/ @home-assistant/core | ||||
| /homeassistant/components/shelly/ @bieniu @thecode @chemelli74 @bdraco | ||||
| /tests/components/shelly/ @bieniu @thecode @chemelli74 @bdraco | ||||
| /homeassistant/components/shelly/ @balloob @bieniu @thecode @chemelli74 @bdraco | ||||
| /tests/components/shelly/ @balloob @bieniu @thecode @chemelli74 @bdraco | ||||
| /homeassistant/components/shodan/ @fabaff | ||||
| /homeassistant/components/sia/ @eavanvalkenburg | ||||
| /tests/components/sia/ @eavanvalkenburg | ||||
| @@ -1436,14 +1387,13 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/siren/ @home-assistant/core @raman325 | ||||
| /tests/components/siren/ @home-assistant/core @raman325 | ||||
| /homeassistant/components/sisyphus/ @jkeljo | ||||
| /homeassistant/components/sky_hub/ @rogerselwyn | ||||
| /homeassistant/components/sky_remote/ @dunnmj @saty9 | ||||
| /tests/components/sky_remote/ @dunnmj @saty9 | ||||
| /homeassistant/components/skybell/ @tkdrob | ||||
| /tests/components/skybell/ @tkdrob | ||||
| /homeassistant/components/slack/ @tkdrob @fletcherau | ||||
| /tests/components/slack/ @tkdrob @fletcherau | ||||
| /homeassistant/components/sleep_as_android/ @tr4nt0r | ||||
| /tests/components/sleep_as_android/ @tr4nt0r | ||||
| /homeassistant/components/sleepiq/ @mfugate1 @kbickar | ||||
| /tests/components/sleepiq/ @mfugate1 @kbickar | ||||
| /homeassistant/components/slide/ @ualex73 | ||||
| @@ -1455,8 +1405,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/sma/ @kellerza @rklomp @erwindouna | ||||
| /homeassistant/components/smappee/ @bsmappee | ||||
| /tests/components/smappee/ @bsmappee | ||||
| /homeassistant/components/smarla/ @explicatis @rlint-explicatis | ||||
| /tests/components/smarla/ @explicatis @rlint-explicatis | ||||
| /homeassistant/components/smart_meter_texas/ @grahamwetzler | ||||
| /tests/components/smart_meter_texas/ @grahamwetzler | ||||
| /homeassistant/components/smartthings/ @joostlek | ||||
| @@ -1479,15 +1427,15 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/snoo/ @Lash-L | ||||
| /homeassistant/components/snooz/ @AustinBrunkhorst | ||||
| /tests/components/snooz/ @AustinBrunkhorst | ||||
| /homeassistant/components/solaredge/ @frenck @bdraco @tronikos | ||||
| /tests/components/solaredge/ @frenck @bdraco @tronikos | ||||
| /homeassistant/components/solaredge/ @frenck @bdraco | ||||
| /tests/components/solaredge/ @frenck @bdraco | ||||
| /homeassistant/components/solaredge_local/ @drobtravels @scheric | ||||
| /homeassistant/components/solarlog/ @Ernst79 @dontinelli | ||||
| /tests/components/solarlog/ @Ernst79 @dontinelli | ||||
| /homeassistant/components/solax/ @squishykid @Darsstar | ||||
| /tests/components/solax/ @squishykid @Darsstar | ||||
| /homeassistant/components/soma/ @ratsept | ||||
| /tests/components/soma/ @ratsept | ||||
| /homeassistant/components/soma/ @ratsept @sebfortier2288 | ||||
| /tests/components/soma/ @ratsept @sebfortier2288 | ||||
| /homeassistant/components/sonarr/ @ctalkington | ||||
| /tests/components/sonarr/ @ctalkington | ||||
| /homeassistant/components/songpal/ @rytilahti @shenxn | ||||
| @@ -1519,8 +1467,7 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/steam_online/ @tkdrob | ||||
| /homeassistant/components/steamist/ @bdraco | ||||
| /tests/components/steamist/ @bdraco | ||||
| /homeassistant/components/stiebel_eltron/ @fucm @ThyMYthOS | ||||
| /tests/components/stiebel_eltron/ @fucm @ThyMYthOS | ||||
| /homeassistant/components/stiebel_eltron/ @fucm | ||||
| /homeassistant/components/stookwijzer/ @fwestenberg | ||||
| /tests/components/stookwijzer/ @fwestenberg | ||||
| /homeassistant/components/stream/ @hunterjm @uvjustin @allenporter | ||||
| @@ -1531,8 +1478,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/subaru/ @G-Two | ||||
| /homeassistant/components/suez_water/ @ooii @jb101010-2 | ||||
| /tests/components/suez_water/ @ooii @jb101010-2 | ||||
| /homeassistant/components/sun/ @home-assistant/core | ||||
| /tests/components/sun/ @home-assistant/core | ||||
| /homeassistant/components/sun/ @Swamp-Ig | ||||
| /tests/components/sun/ @Swamp-Ig | ||||
| /homeassistant/components/supla/ @mwegrzynek | ||||
| /homeassistant/components/surepetcare/ @benleb @danielhiversen | ||||
| /tests/components/surepetcare/ @benleb @danielhiversen | ||||
| @@ -1545,10 +1492,10 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/switch_as_x/ @home-assistant/core | ||||
| /homeassistant/components/switchbee/ @jafar-atili | ||||
| /tests/components/switchbee/ @jafar-atili | ||||
| /homeassistant/components/switchbot/ @danielhiversen @RenierM26 @murtas @Eloston @dsypniewski @zerzhang | ||||
| /tests/components/switchbot/ @danielhiversen @RenierM26 @murtas @Eloston @dsypniewski @zerzhang | ||||
| /homeassistant/components/switchbot_cloud/ @SeraphicRav @laurence-presland @Gigatrappeur @XiaoLing-git | ||||
| /tests/components/switchbot_cloud/ @SeraphicRav @laurence-presland @Gigatrappeur @XiaoLing-git | ||||
| /homeassistant/components/switchbot/ @danielhiversen @RenierM26 @murtas @Eloston @dsypniewski | ||||
| /tests/components/switchbot/ @danielhiversen @RenierM26 @murtas @Eloston @dsypniewski | ||||
| /homeassistant/components/switchbot_cloud/ @SeraphicRav @laurence-presland @Gigatrappeur | ||||
| /tests/components/switchbot_cloud/ @SeraphicRav @laurence-presland @Gigatrappeur | ||||
| /homeassistant/components/switcher_kis/ @thecode @YogevBokobza | ||||
| /tests/components/switcher_kis/ @thecode @YogevBokobza | ||||
| /homeassistant/components/switchmate/ @danielhiversen @qiz-li | ||||
| @@ -1565,8 +1512,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/systemmonitor/ @gjohansson-ST | ||||
| /homeassistant/components/tado/ @erwindouna | ||||
| /tests/components/tado/ @erwindouna | ||||
| /homeassistant/components/tag/ @home-assistant/core | ||||
| /tests/components/tag/ @home-assistant/core | ||||
| /homeassistant/components/tag/ @balloob @dmulcahey | ||||
| /tests/components/tag/ @balloob @dmulcahey | ||||
| /homeassistant/components/tailscale/ @frenck | ||||
| /tests/components/tailscale/ @frenck | ||||
| /homeassistant/components/tailwind/ @frenck | ||||
| @@ -1584,12 +1531,10 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/technove/ @Moustachauve | ||||
| /homeassistant/components/tedee/ @patrickhilker @zweckj | ||||
| /tests/components/tedee/ @patrickhilker @zweckj | ||||
| /homeassistant/components/telegram_bot/ @hanwg | ||||
| /tests/components/telegram_bot/ @hanwg | ||||
| /homeassistant/components/tellduslive/ @fredrike | ||||
| /tests/components/tellduslive/ @fredrike | ||||
| /homeassistant/components/template/ @Petro31 @home-assistant/core | ||||
| /tests/components/template/ @Petro31 @home-assistant/core | ||||
| /homeassistant/components/template/ @Petro31 @PhracturedBlue @home-assistant/core | ||||
| /tests/components/template/ @Petro31 @PhracturedBlue @home-assistant/core | ||||
| /homeassistant/components/tesla_fleet/ @Bre77 | ||||
| /tests/components/tesla_fleet/ @Bre77 | ||||
| /homeassistant/components/tesla_wall_connector/ @einarhauks | ||||
| @@ -1615,8 +1560,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/tile/ @bachya | ||||
| /homeassistant/components/tilt_ble/ @apt-itude | ||||
| /tests/components/tilt_ble/ @apt-itude | ||||
| /homeassistant/components/tilt_pi/ @michaelheyman | ||||
| /tests/components/tilt_pi/ @michaelheyman | ||||
| /homeassistant/components/time/ @home-assistant/core | ||||
| /tests/components/time/ @home-assistant/core | ||||
| /homeassistant/components/time_date/ @fabaff | ||||
| @@ -1626,8 +1569,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/todo/ @home-assistant/core | ||||
| /homeassistant/components/todoist/ @boralyl | ||||
| /tests/components/todoist/ @boralyl | ||||
| /homeassistant/components/togrill/ @elupus | ||||
| /tests/components/togrill/ @elupus | ||||
| /homeassistant/components/tolo/ @MatthiasLohr | ||||
| /tests/components/tolo/ @MatthiasLohr | ||||
| /homeassistant/components/tomorrowio/ @raman325 @lymanepp | ||||
| @@ -1642,6 +1583,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/tplink_omada/ @MarkGodwin | ||||
| /homeassistant/components/traccar/ @ludeeus | ||||
| /tests/components/traccar/ @ludeeus | ||||
| /homeassistant/components/traccar_server/ @ludeeus | ||||
| /tests/components/traccar_server/ @ludeeus | ||||
| /homeassistant/components/trace/ @home-assistant/core | ||||
| /tests/components/trace/ @home-assistant/core | ||||
| /homeassistant/components/tractive/ @Danielhiversen @zhulik @bieniu | ||||
| @@ -1689,12 +1632,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/upnp/ @StevenLooman | ||||
| /homeassistant/components/uptime/ @frenck | ||||
| /tests/components/uptime/ @frenck | ||||
| /homeassistant/components/uptime_kuma/ @tr4nt0r | ||||
| /tests/components/uptime_kuma/ @tr4nt0r | ||||
| /homeassistant/components/uptimerobot/ @ludeeus @chemelli74 | ||||
| /tests/components/uptimerobot/ @ludeeus @chemelli74 | ||||
| /homeassistant/components/usage_prediction/ @home-assistant/core | ||||
| /tests/components/usage_prediction/ @home-assistant/core | ||||
| /homeassistant/components/usb/ @bdraco | ||||
| /tests/components/usb/ @bdraco | ||||
| /homeassistant/components/usgs_earthquakes_feed/ @exxamalte | ||||
| @@ -1709,23 +1648,19 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/vallox/ @andre-richter @slovdahl @viiru- @yozik04 | ||||
| /homeassistant/components/valve/ @home-assistant/core | ||||
| /tests/components/valve/ @home-assistant/core | ||||
| /homeassistant/components/vegehub/ @ghowevege | ||||
| /tests/components/vegehub/ @ghowevege | ||||
| /homeassistant/components/velbus/ @Cereal2nd @brefra | ||||
| /tests/components/velbus/ @Cereal2nd @brefra | ||||
| /homeassistant/components/velux/ @Julius2342 @DeerMaximum @pawlizio @wollew | ||||
| /tests/components/velux/ @Julius2342 @DeerMaximum @pawlizio @wollew | ||||
| /homeassistant/components/velux/ @Julius2342 @DeerMaximum @pawlizio | ||||
| /tests/components/velux/ @Julius2342 @DeerMaximum @pawlizio | ||||
| /homeassistant/components/venstar/ @garbled1 @jhollowe | ||||
| /tests/components/venstar/ @garbled1 @jhollowe | ||||
| /homeassistant/components/versasense/ @imstevenxyz | ||||
| /homeassistant/components/version/ @ludeeus | ||||
| /tests/components/version/ @ludeeus | ||||
| /homeassistant/components/vesync/ @markperdue @webdjoe @thegardenmonkey @cdnninja @iprak @sapuseven | ||||
| /tests/components/vesync/ @markperdue @webdjoe @thegardenmonkey @cdnninja @iprak @sapuseven | ||||
| /homeassistant/components/vesync/ @markperdue @webdjoe @thegardenmonkey @cdnninja @iprak | ||||
| /tests/components/vesync/ @markperdue @webdjoe @thegardenmonkey @cdnninja @iprak | ||||
| /homeassistant/components/vicare/ @CFenner | ||||
| /tests/components/vicare/ @CFenner | ||||
| /homeassistant/components/victron_remote_monitoring/ @AndyTempel | ||||
| /tests/components/victron_remote_monitoring/ @AndyTempel | ||||
| /homeassistant/components/vilfo/ @ManneW | ||||
| /tests/components/vilfo/ @ManneW | ||||
| /homeassistant/components/vivotek/ @HarlemSquirrel | ||||
| @@ -1735,14 +1670,14 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/vlc_telnet/ @rodripf @MartinHjelmare | ||||
| /homeassistant/components/vodafone_station/ @paoloantinori @chemelli74 | ||||
| /tests/components/vodafone_station/ @paoloantinori @chemelli74 | ||||
| /homeassistant/components/voip/ @synesthesiam @jaminh | ||||
| /tests/components/voip/ @synesthesiam @jaminh | ||||
| /homeassistant/components/voip/ @balloob @synesthesiam | ||||
| /tests/components/voip/ @balloob @synesthesiam | ||||
| /homeassistant/components/volumio/ @OnFreund | ||||
| /tests/components/volumio/ @OnFreund | ||||
| /homeassistant/components/volvo/ @thomasddn | ||||
| /tests/components/volvo/ @thomasddn | ||||
| /homeassistant/components/volvooncall/ @molobrakos @svrooij | ||||
| /tests/components/volvooncall/ @molobrakos @svrooij | ||||
| /homeassistant/components/volvooncall/ @molobrakos | ||||
| /tests/components/volvooncall/ @molobrakos | ||||
| /homeassistant/components/vulcan/ @Antoni-Czaplicki | ||||
| /tests/components/vulcan/ @Antoni-Czaplicki | ||||
| /homeassistant/components/wake_on_lan/ @ntilley905 | ||||
| /tests/components/wake_on_lan/ @ntilley905 | ||||
| /homeassistant/components/wake_word/ @home-assistant/core @synesthesiam | ||||
| @@ -1793,8 +1728,8 @@ build.json @home-assistant/supervisor | ||||
| /homeassistant/components/wirelesstag/ @sergeymaysak | ||||
| /homeassistant/components/withings/ @joostlek | ||||
| /tests/components/withings/ @joostlek | ||||
| /homeassistant/components/wiz/ @sbidy @arturpragacz | ||||
| /tests/components/wiz/ @sbidy @arturpragacz | ||||
| /homeassistant/components/wiz/ @sbidy | ||||
| /tests/components/wiz/ @sbidy | ||||
| /homeassistant/components/wled/ @frenck | ||||
| /tests/components/wled/ @frenck | ||||
| /homeassistant/components/wmspro/ @mback2k | ||||
| @@ -1807,8 +1742,8 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/worldclock/ @fabaff | ||||
| /homeassistant/components/ws66i/ @ssaenger | ||||
| /tests/components/ws66i/ @ssaenger | ||||
| /homeassistant/components/wyoming/ @synesthesiam | ||||
| /tests/components/wyoming/ @synesthesiam | ||||
| /homeassistant/components/wyoming/ @balloob @synesthesiam | ||||
| /tests/components/wyoming/ @balloob @synesthesiam | ||||
| /homeassistant/components/xbox/ @hunterjm | ||||
| /tests/components/xbox/ @hunterjm | ||||
| /homeassistant/components/xiaomi_aqara/ @danielhiversen @syssi | ||||
| @@ -1853,8 +1788,6 @@ build.json @home-assistant/supervisor | ||||
| /tests/components/zeversolar/ @kvanzuijlen | ||||
| /homeassistant/components/zha/ @dmulcahey @adminiuga @puddly @TheJulianJES | ||||
| /tests/components/zha/ @dmulcahey @adminiuga @puddly @TheJulianJES | ||||
| /homeassistant/components/zimi/ @markhannon | ||||
| /tests/components/zimi/ @markhannon | ||||
| /homeassistant/components/zodiac/ @JulienTant | ||||
| /tests/components/zodiac/ @JulienTant | ||||
| /homeassistant/components/zone/ @home-assistant/core | ||||
|   | ||||
| @@ -14,8 +14,5 @@ Still interested? Then you should take a peek at the [developer documentation](h | ||||
|  | ||||
| ## Feature suggestions | ||||
|  | ||||
| If you want to suggest a new feature for Home Assistant (e.g. new integrations), please [start a discussion](https://github.com/orgs/home-assistant/discussions) on GitHub. | ||||
|  | ||||
| ## Issue Tracker | ||||
|  | ||||
| If you want to report an issue, please [create an issue](https://github.com/home-assistant/core/issues) on GitHub. | ||||
| If you want to suggest a new feature for Home Assistant (e.g., new integrations), please open a thread in our [Community Forum: Feature Requests](https://community.home-assistant.io/c/feature-requests). | ||||
| We use [GitHub for tracking issues](https://github.com/home-assistant/core/issues), not for tracking feature requests. | ||||
|   | ||||
							
								
								
									
										2
									
								
								Dockerfile
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2
									
								
								Dockerfile
									
									
									
										generated
									
									
									
								
							| @@ -31,7 +31,7 @@ RUN \ | ||||
|     && go2rtc --version | ||||
|  | ||||
| # Install uv | ||||
| RUN pip3 install uv==0.8.9 | ||||
| RUN pip3 install uv==0.6.10 | ||||
|  | ||||
| WORKDIR /usr/src | ||||
|  | ||||
|   | ||||
| @@ -1,9 +1,18 @@ | ||||
| FROM mcr.microsoft.com/vscode/devcontainers/base:debian | ||||
| FROM mcr.microsoft.com/devcontainers/python:1-3.13 | ||||
|  | ||||
| SHELL ["/bin/bash", "-o", "pipefail", "-c"] | ||||
|  | ||||
| # Uninstall pre-installed formatting and linting tools | ||||
| # They would conflict with our pinned versions | ||||
| RUN \ | ||||
|     apt-get update \ | ||||
|     pipx uninstall pydocstyle \ | ||||
|     && pipx uninstall pycodestyle \ | ||||
|     && pipx uninstall mypy \ | ||||
|     && pipx uninstall pylint | ||||
|  | ||||
| RUN \ | ||||
|     curl -sS https://dl.yarnpkg.com/debian/pubkey.gpg | apt-key add - \ | ||||
|     && apt-get update \ | ||||
|     && DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \ | ||||
|         # Additional library needed by some tests and accordingly by VScode Tests Discovery | ||||
|         bluez \ | ||||
| @@ -23,32 +32,29 @@ RUN \ | ||||
|         libxml2 \ | ||||
|         git \ | ||||
|         cmake \ | ||||
|         autoconf \ | ||||
|     && apt-get clean \ | ||||
|     && rm -rf /var/lib/apt/lists/* | ||||
|  | ||||
| # Add go2rtc binary | ||||
| COPY --from=ghcr.io/alexxit/go2rtc:latest /usr/local/bin/go2rtc /bin/go2rtc | ||||
|  | ||||
| # Install uv | ||||
| RUN pip3 install uv | ||||
|  | ||||
| WORKDIR /usr/src | ||||
|  | ||||
| COPY --from=ghcr.io/astral-sh/uv:latest /uv /usr/local/bin/uv | ||||
| # Setup hass-release | ||||
| RUN git clone --depth 1 https://github.com/home-assistant/hass-release \ | ||||
|     && uv pip install --system -e hass-release/ \ | ||||
|     && chown -R vscode /usr/src/hass-release/data | ||||
|  | ||||
| USER vscode | ||||
|  | ||||
| ENV UV_PYTHON=3.13.2 | ||||
| RUN uv python install | ||||
|  | ||||
| ENV VIRTUAL_ENV="/home/vscode/.local/ha-venv" | ||||
| RUN uv venv $VIRTUAL_ENV | ||||
| ENV PATH="$VIRTUAL_ENV/bin:$PATH" | ||||
|  | ||||
| WORKDIR /tmp | ||||
|  | ||||
| # Setup hass-release | ||||
| RUN git clone --depth 1 https://github.com/home-assistant/hass-release ~/hass-release \ | ||||
|     && uv pip install -e ~/hass-release/ | ||||
|  | ||||
| # Install Python dependencies from requirements | ||||
| COPY requirements.txt ./ | ||||
| COPY homeassistant/package_constraints.txt homeassistant/package_constraints.txt | ||||
| @@ -59,4 +65,4 @@ RUN uv pip install -r requirements_test.txt | ||||
| WORKDIR /workspaces | ||||
|  | ||||
| # Set the default shell to bash instead of sh | ||||
| ENV SHELL=/bin/bash | ||||
| ENV SHELL /bin/bash | ||||
|   | ||||
							
								
								
									
										10
									
								
								build.yaml
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								build.yaml
									
									
									
									
									
								
							| @@ -1,10 +1,10 @@ | ||||
| image: ghcr.io/home-assistant/{arch}-homeassistant | ||||
| build_from: | ||||
|   aarch64: ghcr.io/home-assistant/aarch64-homeassistant-base:2025.10.0 | ||||
|   armhf: ghcr.io/home-assistant/armhf-homeassistant-base:2025.10.0 | ||||
|   armv7: ghcr.io/home-assistant/armv7-homeassistant-base:2025.10.0 | ||||
|   amd64: ghcr.io/home-assistant/amd64-homeassistant-base:2025.10.0 | ||||
|   i386: ghcr.io/home-assistant/i386-homeassistant-base:2025.10.0 | ||||
|   aarch64: ghcr.io/home-assistant/aarch64-homeassistant-base:2025.02.1 | ||||
|   armhf: ghcr.io/home-assistant/armhf-homeassistant-base:2025.02.1 | ||||
|   armv7: ghcr.io/home-assistant/armv7-homeassistant-base:2025.02.1 | ||||
|   amd64: ghcr.io/home-assistant/amd64-homeassistant-base:2025.02.1 | ||||
|   i386: ghcr.io/home-assistant/i386-homeassistant-base:2025.02.1 | ||||
| codenotary: | ||||
|   signer: notary@home-assistant.io | ||||
|   base_image: notary@home-assistant.io | ||||
|   | ||||
| @@ -38,7 +38,8 @@ def validate_python() -> None: | ||||
|  | ||||
| def ensure_config_path(config_dir: str) -> None: | ||||
|     """Validate the configuration directory.""" | ||||
|     from . import config as config_util  # noqa: PLC0415 | ||||
|     # pylint: disable-next=import-outside-toplevel | ||||
|     from . import config as config_util | ||||
|  | ||||
|     lib_dir = os.path.join(config_dir, "deps") | ||||
|  | ||||
| @@ -79,7 +80,8 @@ def ensure_config_path(config_dir: str) -> None: | ||||
|  | ||||
| def get_arguments() -> argparse.Namespace: | ||||
|     """Get parsed passed in arguments.""" | ||||
|     from . import config as config_util  # noqa: PLC0415 | ||||
|     # pylint: disable-next=import-outside-toplevel | ||||
|     from . import config as config_util | ||||
|  | ||||
|     parser = argparse.ArgumentParser( | ||||
|         description="Home Assistant: Observe, Control, Automate.", | ||||
| @@ -175,7 +177,8 @@ def main() -> int: | ||||
|         validate_os() | ||||
|  | ||||
|     if args.script is not None: | ||||
|         from . import scripts  # noqa: PLC0415 | ||||
|         # pylint: disable-next=import-outside-toplevel | ||||
|         from . import scripts | ||||
|  | ||||
|         return scripts.run(args.script) | ||||
|  | ||||
| @@ -185,44 +188,39 @@ def main() -> int: | ||||
|  | ||||
|     ensure_config_path(config_dir) | ||||
|  | ||||
|     from . import config, runner  # noqa: PLC0415 | ||||
|     # pylint: disable-next=import-outside-toplevel | ||||
|     from . import config, runner | ||||
|  | ||||
|     # Ensure only one instance runs per config directory | ||||
|     with runner.ensure_single_execution(config_dir) as single_execution_lock: | ||||
|         # Check if another instance is already running | ||||
|         if single_execution_lock.exit_code is not None: | ||||
|             return single_execution_lock.exit_code | ||||
|     safe_mode = config.safe_mode_enabled(config_dir) | ||||
|  | ||||
|         safe_mode = config.safe_mode_enabled(config_dir) | ||||
|     runtime_conf = runner.RuntimeConfig( | ||||
|         config_dir=config_dir, | ||||
|         verbose=args.verbose, | ||||
|         log_rotate_days=args.log_rotate_days, | ||||
|         log_file=args.log_file, | ||||
|         log_no_color=args.log_no_color, | ||||
|         skip_pip=args.skip_pip, | ||||
|         skip_pip_packages=args.skip_pip_packages, | ||||
|         recovery_mode=args.recovery_mode, | ||||
|         debug=args.debug, | ||||
|         open_ui=args.open_ui, | ||||
|         safe_mode=safe_mode, | ||||
|     ) | ||||
|  | ||||
|         runtime_conf = runner.RuntimeConfig( | ||||
|             config_dir=config_dir, | ||||
|             verbose=args.verbose, | ||||
|             log_rotate_days=args.log_rotate_days, | ||||
|             log_file=args.log_file, | ||||
|             log_no_color=args.log_no_color, | ||||
|             skip_pip=args.skip_pip, | ||||
|             skip_pip_packages=args.skip_pip_packages, | ||||
|             recovery_mode=args.recovery_mode, | ||||
|             debug=args.debug, | ||||
|             open_ui=args.open_ui, | ||||
|             safe_mode=safe_mode, | ||||
|         ) | ||||
|     fault_file_name = os.path.join(config_dir, FAULT_LOG_FILENAME) | ||||
|     with open(fault_file_name, mode="a", encoding="utf8") as fault_file: | ||||
|         faulthandler.enable(fault_file) | ||||
|         exit_code = runner.run(runtime_conf) | ||||
|         faulthandler.disable() | ||||
|  | ||||
|         fault_file_name = os.path.join(config_dir, FAULT_LOG_FILENAME) | ||||
|         with open(fault_file_name, mode="a", encoding="utf8") as fault_file: | ||||
|             faulthandler.enable(fault_file) | ||||
|             exit_code = runner.run(runtime_conf) | ||||
|             faulthandler.disable() | ||||
|     # It's possible for the fault file to disappear, so suppress obvious errors | ||||
|     with suppress(FileNotFoundError): | ||||
|         if os.path.getsize(fault_file_name) == 0: | ||||
|             os.remove(fault_file_name) | ||||
|  | ||||
|         # It's possible for the fault file to disappear, so suppress obvious errors | ||||
|         with suppress(FileNotFoundError): | ||||
|             if os.path.getsize(fault_file_name) == 0: | ||||
|                 os.remove(fault_file_name) | ||||
|     check_threads() | ||||
|  | ||||
|         check_threads() | ||||
|  | ||||
|         return exit_code | ||||
|     return exit_code | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|   | ||||
| @@ -120,9 +120,6 @@ class AuthStore: | ||||
|  | ||||
|         new_user = models.User(**kwargs) | ||||
|  | ||||
|         while new_user.id in self._users: | ||||
|             new_user = models.User(**kwargs) | ||||
|  | ||||
|         self._users[new_user.id] = new_user | ||||
|  | ||||
|         if credentials is None: | ||||
|   | ||||
| @@ -27,7 +27,7 @@ from . import ( | ||||
|     SetupFlow, | ||||
| ) | ||||
|  | ||||
| REQUIREMENTS = ["pyotp==2.9.0"] | ||||
| REQUIREMENTS = ["pyotp==2.8.0"] | ||||
|  | ||||
| CONF_MESSAGE = "message" | ||||
|  | ||||
| @@ -52,28 +52,28 @@ _LOGGER = logging.getLogger(__name__) | ||||
|  | ||||
| def _generate_secret() -> str: | ||||
|     """Generate a secret.""" | ||||
|     import pyotp  # noqa: PLC0415 | ||||
|     import pyotp  # pylint: disable=import-outside-toplevel | ||||
|  | ||||
|     return str(pyotp.random_base32()) | ||||
|  | ||||
|  | ||||
| def _generate_random() -> int: | ||||
|     """Generate a 32 digit number.""" | ||||
|     import pyotp  # noqa: PLC0415 | ||||
|     import pyotp  # pylint: disable=import-outside-toplevel | ||||
|  | ||||
|     return int(pyotp.random_base32(length=32, chars=list("1234567890"))) | ||||
|  | ||||
|  | ||||
| def _generate_otp(secret: str, count: int) -> str: | ||||
|     """Generate one time password.""" | ||||
|     import pyotp  # noqa: PLC0415 | ||||
|     import pyotp  # pylint: disable=import-outside-toplevel | ||||
|  | ||||
|     return str(pyotp.HOTP(secret).at(count)) | ||||
|  | ||||
|  | ||||
| def _verify_otp(secret: str, otp: str, count: int) -> bool: | ||||
|     """Verify one time password.""" | ||||
|     import pyotp  # noqa: PLC0415 | ||||
|     import pyotp  # pylint: disable=import-outside-toplevel | ||||
|  | ||||
|     return bool(pyotp.HOTP(secret).verify(otp, count)) | ||||
|  | ||||
|   | ||||
| @@ -20,7 +20,7 @@ from . import ( | ||||
|     SetupFlow, | ||||
| ) | ||||
|  | ||||
| REQUIREMENTS = ["pyotp==2.9.0", "PyQRCode==1.2.1"] | ||||
| REQUIREMENTS = ["pyotp==2.8.0", "PyQRCode==1.2.1"] | ||||
|  | ||||
| CONFIG_SCHEMA = MULTI_FACTOR_AUTH_MODULE_SCHEMA.extend({}, extra=vol.PREVENT_EXTRA) | ||||
|  | ||||
| @@ -37,7 +37,7 @@ DUMMY_SECRET = "FPPTH34D4E3MI2HG" | ||||
|  | ||||
| def _generate_qr_code(data: str) -> str: | ||||
|     """Generate a base64 PNG string represent QR Code image of data.""" | ||||
|     import pyqrcode  # noqa: PLC0415 | ||||
|     import pyqrcode  # pylint: disable=import-outside-toplevel | ||||
|  | ||||
|     qr_code = pyqrcode.create(data) | ||||
|  | ||||
| @@ -59,7 +59,7 @@ def _generate_qr_code(data: str) -> str: | ||||
|  | ||||
| def _generate_secret_and_qr_code(username: str) -> tuple[str, str, str]: | ||||
|     """Generate a secret, url, and QR code.""" | ||||
|     import pyotp  # noqa: PLC0415 | ||||
|     import pyotp  # pylint: disable=import-outside-toplevel | ||||
|  | ||||
|     ota_secret = pyotp.random_base32() | ||||
|     url = pyotp.totp.TOTP(ota_secret).provisioning_uri( | ||||
| @@ -107,7 +107,7 @@ class TotpAuthModule(MultiFactorAuthModule): | ||||
|  | ||||
|     def _add_ota_secret(self, user_id: str, secret: str | None = None) -> str: | ||||
|         """Create a ota_secret for user.""" | ||||
|         import pyotp  # noqa: PLC0415 | ||||
|         import pyotp  # pylint: disable=import-outside-toplevel | ||||
|  | ||||
|         ota_secret: str = secret or pyotp.random_base32() | ||||
|  | ||||
| @@ -163,7 +163,7 @@ class TotpAuthModule(MultiFactorAuthModule): | ||||
|  | ||||
|     def _validate_2fa(self, user_id: str, code: str) -> bool: | ||||
|         """Validate two factor authentication code.""" | ||||
|         import pyotp  # noqa: PLC0415 | ||||
|         import pyotp  # pylint: disable=import-outside-toplevel | ||||
|  | ||||
|         if (ota_secret := self._users.get(user_id)) is None:  # type: ignore[union-attr] | ||||
|             # even we cannot find user, we still do verify | ||||
| @@ -196,7 +196,7 @@ class TotpSetupFlow(SetupFlow[TotpAuthModule]): | ||||
|         Return self.async_show_form(step_id='init') if user_input is None. | ||||
|         Return self.async_create_entry(data={'result': result}) if finish. | ||||
|         """ | ||||
|         import pyotp  # noqa: PLC0415 | ||||
|         import pyotp  # pylint: disable=import-outside-toplevel | ||||
|  | ||||
|         errors: dict[str, str] = {} | ||||
|  | ||||
|   | ||||
| @@ -33,10 +33,7 @@ class AuthFlowContext(FlowContext, total=False): | ||||
|     redirect_uri: str | ||||
|  | ||||
|  | ||||
| class AuthFlowResult(FlowResult[AuthFlowContext, tuple[str, str]], total=False): | ||||
|     """Typed result dict for auth flow.""" | ||||
|  | ||||
|     result: Credentials  # Only present if type is CREATE_ENTRY | ||||
| AuthFlowResult = FlowResult[AuthFlowContext, tuple[str, str]] | ||||
|  | ||||
|  | ||||
| @attr.s(slots=True) | ||||
|   | ||||
							
								
								
									
										29
									
								
								homeassistant/backports/enum.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								homeassistant/backports/enum.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| """Enum backports from standard lib. | ||||
|  | ||||
| This file contained the backport of the StrEnum of Python 3.11. | ||||
|  | ||||
| Since we have dropped support for Python 3.10, we can remove this backport. | ||||
| This file is kept for now to avoid breaking custom components that might | ||||
| import it. | ||||
| """ | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| from enum import StrEnum as _StrEnum | ||||
| from functools import partial | ||||
|  | ||||
| from homeassistant.helpers.deprecation import ( | ||||
|     DeprecatedAlias, | ||||
|     all_with_deprecated_constants, | ||||
|     check_if_deprecated_constant, | ||||
|     dir_with_deprecated_constants, | ||||
| ) | ||||
|  | ||||
| # StrEnum deprecated as of 2024.5 use enum.StrEnum instead. | ||||
| _DEPRECATED_StrEnum = DeprecatedAlias(_StrEnum, "enum.StrEnum", "2025.5") | ||||
|  | ||||
| __getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) | ||||
| __dir__ = partial( | ||||
|     dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] | ||||
| ) | ||||
| __all__ = all_with_deprecated_constants(globals()) | ||||
							
								
								
									
										31
									
								
								homeassistant/backports/functools.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								homeassistant/backports/functools.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| """Functools backports from standard lib. | ||||
|  | ||||
| This file contained the backport of the cached_property implementation of Python 3.12. | ||||
|  | ||||
| Since we have dropped support for Python 3.11, we can remove this backport. | ||||
| This file is kept for now to avoid breaking custom components that might | ||||
| import it. | ||||
| """ | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| # pylint: disable-next=hass-deprecated-import | ||||
| from functools import cached_property as _cached_property, partial | ||||
|  | ||||
| from homeassistant.helpers.deprecation import ( | ||||
|     DeprecatedAlias, | ||||
|     all_with_deprecated_constants, | ||||
|     check_if_deprecated_constant, | ||||
|     dir_with_deprecated_constants, | ||||
| ) | ||||
|  | ||||
| # cached_property deprecated as of 2024.5 use functools.cached_property instead. | ||||
| _DEPRECATED_cached_property = DeprecatedAlias( | ||||
|     _cached_property, "functools.cached_property", "2025.5" | ||||
| ) | ||||
|  | ||||
| __getattr__ = partial(check_if_deprecated_constant, module_globals=globals()) | ||||
| __dir__ = partial( | ||||
|     dir_with_deprecated_constants, module_globals_keys=[*globals().keys()] | ||||
| ) | ||||
| __all__ = all_with_deprecated_constants(globals()) | ||||
| @@ -53,7 +53,6 @@ from .components import ( | ||||
|     logbook as logbook_pre_import,  # noqa: F401 | ||||
|     lovelace as lovelace_pre_import,  # noqa: F401 | ||||
|     onboarding as onboarding_pre_import,  # noqa: F401 | ||||
|     person as person_pre_import,  # noqa: F401 | ||||
|     recorder as recorder_import,  # noqa: F401 - not named pre_import since it has requirements | ||||
|     repairs as repairs_pre_import,  # noqa: F401 | ||||
|     search as search_pre_import,  # noqa: F401 | ||||
| @@ -75,8 +74,8 @@ from .core_config import async_process_ha_core_config | ||||
| from .exceptions import HomeAssistantError | ||||
| from .helpers import ( | ||||
|     area_registry, | ||||
|     backup, | ||||
|     category_registry, | ||||
|     condition, | ||||
|     config_validation as cv, | ||||
|     device_registry, | ||||
|     entity, | ||||
| @@ -89,7 +88,6 @@ from .helpers import ( | ||||
|     restore_state, | ||||
|     template, | ||||
|     translation, | ||||
|     trigger, | ||||
| ) | ||||
| from .helpers.dispatcher import async_dispatcher_send_internal | ||||
| from .helpers.storage import get_internal_store_manager | ||||
| @@ -172,6 +170,8 @@ FRONTEND_INTEGRATIONS = { | ||||
| # Stage 0 is divided into substages. Each substage has a name, a set of integrations and a timeout. | ||||
| # The substage containing recorder should have no timeout, as it could cancel a database migration. | ||||
| # Recorder freezes "recorder" timeout during a migration, but it does not freeze other timeouts. | ||||
| # The substages preceding it should also have no timeout, until we ensure that the recorder | ||||
| # is not accidentally promoted as a dependency of any of the integrations in them. | ||||
| # If we add timeouts to the frontend substages, we should make sure they don't apply in recovery mode. | ||||
| STAGE_0_INTEGRATIONS = ( | ||||
|     # Load logging and http deps as soon as possible | ||||
| @@ -332,9 +332,6 @@ async def async_setup_hass( | ||||
|             if not is_virtual_env(): | ||||
|                 await async_mount_local_lib_path(runtime_config.config_dir) | ||||
|  | ||||
|             if hass.config.safe_mode: | ||||
|                 _LOGGER.info("Starting in safe mode") | ||||
|  | ||||
|             basic_setup_success = ( | ||||
|                 await async_from_config_dict(config_dict, hass) is not None | ||||
|             ) | ||||
| @@ -387,6 +384,8 @@ async def async_setup_hass( | ||||
|             {"recovery_mode": {}, "http": http_conf}, | ||||
|             hass, | ||||
|         ) | ||||
|     elif hass.config.safe_mode: | ||||
|         _LOGGER.info("Starting in safe mode") | ||||
|  | ||||
|     if runtime_config.open_ui: | ||||
|         hass.add_job(open_hass_ui, hass) | ||||
| @@ -396,7 +395,7 @@ async def async_setup_hass( | ||||
|  | ||||
| def open_hass_ui(hass: core.HomeAssistant) -> None: | ||||
|     """Open the UI.""" | ||||
|     import webbrowser  # noqa: PLC0415 | ||||
|     import webbrowser  # pylint: disable=import-outside-toplevel | ||||
|  | ||||
|     if hass.config.api is None or "frontend" not in hass.config.components: | ||||
|         _LOGGER.warning("Cannot launch the UI because frontend not loaded") | ||||
| @@ -454,8 +453,6 @@ async def async_load_base_functionality(hass: core.HomeAssistant) -> None: | ||||
|         create_eager_task(restore_state.async_load(hass)), | ||||
|         create_eager_task(hass.config_entries.async_initialize()), | ||||
|         create_eager_task(async_get_system_info(hass)), | ||||
|         create_eager_task(condition.async_setup(hass)), | ||||
|         create_eager_task(trigger.async_setup(hass)), | ||||
|     ) | ||||
|  | ||||
|  | ||||
| @@ -565,7 +562,8 @@ async def async_enable_logging( | ||||
|  | ||||
|     if not log_no_color: | ||||
|         try: | ||||
|             from colorlog import ColoredFormatter  # noqa: PLC0415 | ||||
|             # pylint: disable-next=import-outside-toplevel | ||||
|             from colorlog import ColoredFormatter | ||||
|  | ||||
|             # basicConfig must be called after importing colorlog in order to | ||||
|             # ensure that the handlers it sets up wraps the correct streams. | ||||
| @@ -616,34 +614,34 @@ async def async_enable_logging( | ||||
|         ), | ||||
|     ) | ||||
|  | ||||
|     logger = logging.getLogger() | ||||
|     logger.setLevel(logging.INFO if verbose else logging.WARNING) | ||||
|  | ||||
|     # Log errors to a file if we have write access to file or config dir | ||||
|     if log_file is None: | ||||
|         default_log_path = hass.config.path(ERROR_LOG_FILENAME) | ||||
|         if "SUPERVISOR" in os.environ: | ||||
|             _LOGGER.info("Running in Supervisor, not logging to file") | ||||
|             # Rename the default log file if it exists, since previous versions created | ||||
|             # it even on Supervisor | ||||
|             if os.path.isfile(default_log_path): | ||||
|                 with contextlib.suppress(OSError): | ||||
|                     os.rename(default_log_path, f"{default_log_path}.old") | ||||
|             err_log_path = None | ||||
|         else: | ||||
|             err_log_path = default_log_path | ||||
|         err_log_path = hass.config.path(ERROR_LOG_FILENAME) | ||||
|     else: | ||||
|         err_log_path = os.path.abspath(log_file) | ||||
|  | ||||
|     if err_log_path: | ||||
|     err_path_exists = os.path.isfile(err_log_path) | ||||
|     err_dir = os.path.dirname(err_log_path) | ||||
|  | ||||
|     # Check if we can write to the error log if it exists or that | ||||
|     # we can create files in the containing directory if not. | ||||
|     if (err_path_exists and os.access(err_log_path, os.W_OK)) or ( | ||||
|         not err_path_exists and os.access(err_dir, os.W_OK) | ||||
|     ): | ||||
|         err_handler = await hass.async_add_executor_job( | ||||
|             _create_log_file, err_log_path, log_rotate_days | ||||
|         ) | ||||
|  | ||||
|         err_handler.setFormatter(logging.Formatter(fmt, datefmt=FORMAT_DATETIME)) | ||||
|  | ||||
|         logger = logging.getLogger() | ||||
|         logger.addHandler(err_handler) | ||||
|         logger.setLevel(logging.INFO if verbose else logging.WARNING) | ||||
|  | ||||
|         # Save the log file location for access by other components. | ||||
|         hass.data[DATA_LOGGING] = err_log_path | ||||
|     else: | ||||
|         _LOGGER.error("Unable to set up error log %s (access denied)", err_log_path) | ||||
|  | ||||
|     async_activate_log_queue_handler(hass) | ||||
|  | ||||
| @@ -695,10 +693,10 @@ async def async_mount_local_lib_path(config_dir: str) -> str: | ||||
|  | ||||
| def _get_domains(hass: core.HomeAssistant, config: dict[str, Any]) -> set[str]: | ||||
|     """Get domains of components to set up.""" | ||||
|     # The common config section [homeassistant] could be filtered here, | ||||
|     # but that is not necessary, since it corresponds to the core integration, | ||||
|     # that is always unconditionally loaded. | ||||
|     domains = {cv.domain_key(key) for key in config} | ||||
|     # Filter out the repeating and common config section [homeassistant] | ||||
|     domains = { | ||||
|         domain for key in config if (domain := cv.domain_key(key)) != core.DOMAIN | ||||
|     } | ||||
|  | ||||
|     # Add config entry and default domains | ||||
|     if not hass.config.recovery_mode: | ||||
| @@ -726,28 +724,34 @@ async def _async_resolve_domains_and_preload( | ||||
|       together with all their dependencies. | ||||
|     """ | ||||
|     domains_to_setup = _get_domains(hass, config) | ||||
|  | ||||
|     # Also process all base platforms since we do not require the manifest | ||||
|     # to list them as dependencies. | ||||
|     # We want to later avoid lock contention when multiple integrations try to load | ||||
|     # their manifests at once. | ||||
|     platform_integrations = conf_util.extract_platform_integrations( | ||||
|         config, BASE_PLATFORMS | ||||
|     ) | ||||
|     # Ensure base platforms that have platform integrations are added to `domains`, | ||||
|     # so they can be setup first instead of discovering them later when a config | ||||
|     # entry setup task notices that it's needed and there is already a long line | ||||
|     # to use the import executor. | ||||
|     # | ||||
|     # Additionally process integrations that are defined under base platforms | ||||
|     # to speed things up. | ||||
|     # For example if we have | ||||
|     # sensor: | ||||
|     #   - platform: template | ||||
|     # | ||||
|     # `template` has to be loaded to validate the config for sensor. | ||||
|     # The more platforms under `sensor:`, the longer | ||||
|     # `template` has to be loaded to validate the config for sensor | ||||
|     # so we want to start loading `sensor` as soon as we know | ||||
|     # it will be needed. The more platforms under `sensor:`, the longer | ||||
|     # it will take to finish setup for `sensor` because each of these | ||||
|     # platforms has to be imported before we can validate the config. | ||||
|     # | ||||
|     # Thankfully we are migrating away from the platform pattern | ||||
|     # so this will be less of a problem in the future. | ||||
|     platform_integrations = conf_util.extract_platform_integrations( | ||||
|         config, BASE_PLATFORMS | ||||
|     ) | ||||
|     domains_to_setup.update(platform_integrations) | ||||
|  | ||||
|     # Additionally process base platforms since we do not require the manifest | ||||
|     # to list them as dependencies. | ||||
|     # We want to later avoid lock contention when multiple integrations try to load | ||||
|     # their manifests at once. | ||||
|     # Also process integrations that are defined under base platforms | ||||
|     # to speed things up. | ||||
|     additional_domains_to_process = { | ||||
|         *BASE_PLATFORMS, | ||||
|         *chain.from_iterable(platform_integrations.values()), | ||||
| @@ -855,27 +859,23 @@ async def _async_set_up_integrations( | ||||
|     integrations, all_integrations = await _async_resolve_domains_and_preload( | ||||
|         hass, config | ||||
|     ) | ||||
|     # Detect all cycles | ||||
|     integrations_after_dependencies = ( | ||||
|         await loader.resolve_integrations_after_dependencies( | ||||
|             hass, all_integrations.values(), set(all_integrations) | ||||
|         ) | ||||
|     ) | ||||
|     all_domains = set(integrations_after_dependencies) | ||||
|     domains = set(integrations) & all_domains | ||||
|     all_domains = set(all_integrations) | ||||
|     domains = set(integrations) | ||||
|  | ||||
|     _LOGGER.info( | ||||
|         "Domains to be set up: %s\nDependencies: %s", | ||||
|         domains or "{}", | ||||
|         (all_domains - domains) or "{}", | ||||
|         "Domains to be set up: %s | %s", | ||||
|         domains, | ||||
|         all_domains - domains, | ||||
|     ) | ||||
|  | ||||
|     async_set_domains_to_be_loaded(hass, all_domains) | ||||
|  | ||||
|     # Initialize recorder | ||||
|     if "recorder" in all_domains: | ||||
|         recorder.async_initialize_recorder(hass) | ||||
|  | ||||
|     # Initialize backup | ||||
|     if "backup" in all_domains: | ||||
|         backup.async_initialize_backup(hass) | ||||
|  | ||||
|     stages: list[tuple[str, set[str], int | None]] = [ | ||||
|         *( | ||||
|             (name, domain_group, timeout) | ||||
| @@ -900,32 +900,41 @@ async def _async_set_up_integrations( | ||||
|         stage_dep_domains_unfiltered = { | ||||
|             dep | ||||
|             for domain in stage_domains | ||||
|             for dep in integrations_after_dependencies[domain] | ||||
|             for dep in all_integrations[domain].all_dependencies | ||||
|             if dep not in stage_domains | ||||
|         } | ||||
|         stage_dep_domains = stage_dep_domains_unfiltered - hass.config.components | ||||
|  | ||||
|         stage_all_domains = stage_domains | stage_dep_domains | ||||
|         stage_all_integrations = { | ||||
|             domain: all_integrations[domain] for domain in stage_all_domains | ||||
|         } | ||||
|         # Detect all cycles | ||||
|         stage_integrations_after_dependencies = ( | ||||
|             await loader.resolve_integrations_after_dependencies( | ||||
|                 hass, stage_all_integrations.values(), stage_all_domains | ||||
|             ) | ||||
|         ) | ||||
|         stage_all_domains = set(stage_integrations_after_dependencies) | ||||
|         stage_domains &= stage_all_domains | ||||
|         stage_dep_domains &= stage_all_domains | ||||
|  | ||||
|         _LOGGER.info( | ||||
|             "Setting up stage %s: %s; already set up: %s\n" | ||||
|             "Dependencies: %s; already set up: %s", | ||||
|             "Setting up stage %s: %s | %s\nDependencies: %s | %s", | ||||
|             name, | ||||
|             stage_domains, | ||||
|             (stage_domains_unfiltered - stage_domains) or "{}", | ||||
|             stage_dep_domains or "{}", | ||||
|             (stage_dep_domains_unfiltered - stage_dep_domains) or "{}", | ||||
|             stage_domains_unfiltered - stage_domains, | ||||
|             stage_dep_domains, | ||||
|             stage_dep_domains_unfiltered - stage_dep_domains, | ||||
|         ) | ||||
|  | ||||
|         async_set_domains_to_be_loaded(hass, stage_all_domains) | ||||
|  | ||||
|         if timeout is None: | ||||
|             await _async_setup_multi_components(hass, stage_all_domains, config) | ||||
|             continue | ||||
|         try: | ||||
|             async with hass.timeout.async_timeout( | ||||
|                 timeout, | ||||
|                 cool_down=COOLDOWN_TIME, | ||||
|                 cancel_message=f"Bootstrap stage {name} timeout", | ||||
|             ): | ||||
|             async with hass.timeout.async_timeout(timeout, cool_down=COOLDOWN_TIME): | ||||
|                 await _async_setup_multi_components(hass, stage_all_domains, config) | ||||
|         except TimeoutError: | ||||
|             _LOGGER.warning( | ||||
| @@ -937,11 +946,7 @@ async def _async_set_up_integrations( | ||||
|     # Wrap up startup | ||||
|     _LOGGER.debug("Waiting for startup to wrap up") | ||||
|     try: | ||||
|         async with hass.timeout.async_timeout( | ||||
|             WRAP_UP_TIMEOUT, | ||||
|             cool_down=COOLDOWN_TIME, | ||||
|             cancel_message="Bootstrap startup wrap up timeout", | ||||
|         ): | ||||
|         async with hass.timeout.async_timeout(WRAP_UP_TIMEOUT, cool_down=COOLDOWN_TIME): | ||||
|             await hass.async_block_till_done() | ||||
|     except TimeoutError: | ||||
|         _LOGGER.warning( | ||||
|   | ||||
| @@ -1,13 +1,5 @@ | ||||
| { | ||||
|   "domain": "amazon", | ||||
|   "name": "Amazon", | ||||
|   "integrations": [ | ||||
|     "alexa", | ||||
|     "alexa_devices", | ||||
|     "amazon_polly", | ||||
|     "aws", | ||||
|     "aws_s3", | ||||
|     "fire_tv", | ||||
|     "route53" | ||||
|   ] | ||||
|   "integrations": ["alexa", "amazon_polly", "aws", "fire_tv", "route53"] | ||||
| } | ||||
|   | ||||
| @@ -1,5 +0,0 @@ | ||||
| { | ||||
|   "domain": "eltako", | ||||
|   "name": "Eltako", | ||||
|   "iot_standards": ["matter"] | ||||
| } | ||||
| @@ -1,5 +0,0 @@ | ||||
| { | ||||
|   "domain": "eve", | ||||
|   "name": "Eve", | ||||
|   "iot_standards": ["matter"] | ||||
| } | ||||
| @@ -1,5 +0,0 @@ | ||||
| { | ||||
|   "domain": "frient", | ||||
|   "name": "Frient", | ||||
|   "iot_standards": ["zigbee"] | ||||
| } | ||||
| @@ -1,5 +1,5 @@ | ||||
| { | ||||
|   "domain": "fritzbox", | ||||
|   "name": "FRITZ!", | ||||
|   "name": "FRITZ!Box", | ||||
|   "integrations": ["fritz", "fritzbox", "fritzbox_callmonitor"] | ||||
| } | ||||
|   | ||||
							
								
								
									
										5
									
								
								homeassistant/brands/ibm.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								homeassistant/brands/ibm.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| { | ||||
|   "domain": "ibm", | ||||
|   "name": "IBM", | ||||
|   "integrations": ["watson_iot", "watson_tts"] | ||||
| } | ||||
| @@ -1,5 +0,0 @@ | ||||
| { | ||||
|   "domain": "konnected", | ||||
|   "name": "Konnected", | ||||
|   "integrations": ["konnected", "konnected_esphome"] | ||||
| } | ||||
| @@ -1,5 +0,0 @@ | ||||
| { | ||||
|   "domain": "level", | ||||
|   "name": "Level", | ||||
|   "iot_standards": ["matter"] | ||||
| } | ||||
| @@ -1,6 +0,0 @@ | ||||
| { | ||||
|   "domain": "nuki", | ||||
|   "name": "Nuki", | ||||
|   "integrations": ["nuki"], | ||||
|   "iot_standards": ["matter"] | ||||
| } | ||||
| @@ -1,6 +0,0 @@ | ||||
| { | ||||
|   "domain": "shelly", | ||||
|   "name": "shelly", | ||||
|   "integrations": ["shelly"], | ||||
|   "iot_standards": ["zwave"] | ||||
| } | ||||
| @@ -1,11 +1,5 @@ | ||||
| { | ||||
|   "domain": "sony", | ||||
|   "name": "Sony", | ||||
|   "integrations": [ | ||||
|     "braviatv", | ||||
|     "ps4", | ||||
|     "sony_projector", | ||||
|     "songpal", | ||||
|     "playstation_network" | ||||
|   ] | ||||
|   "integrations": ["braviatv", "ps4", "sony_projector", "songpal"] | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| { | ||||
|   "domain": "switchbot", | ||||
|   "name": "SwitchBot", | ||||
|   "integrations": ["switchbot", "switchbot_cloud"], | ||||
|   "iot_standards": ["matter"] | ||||
|   "integrations": ["switchbot", "switchbot_cloud"] | ||||
| } | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| { | ||||
|   "domain": "third_reality", | ||||
|   "name": "Third Reality", | ||||
|   "iot_standards": ["matter", "zigbee"] | ||||
|   "iot_standards": ["zigbee"] | ||||
| } | ||||
|   | ||||
| @@ -1,5 +0,0 @@ | ||||
| { | ||||
|   "domain": "tilt", | ||||
|   "name": "Tilt", | ||||
|   "integrations": ["tilt_ble", "tilt_pi"] | ||||
| } | ||||
| @@ -1,5 +1,5 @@ | ||||
| { | ||||
|   "domain": "ubiquiti", | ||||
|   "name": "Ubiquiti", | ||||
|   "integrations": ["airos", "unifi", "unifi_direct", "unifiled", "unifiprotect"] | ||||
|   "integrations": ["unifi", "unifi_direct", "unifiled", "unifiprotect"] | ||||
| } | ||||
|   | ||||
| @@ -14,24 +14,30 @@ from jaraco.abode.exceptions import ( | ||||
| ) | ||||
| from jaraco.abode.helpers.timeline import Groups as GROUPS | ||||
| from requests.exceptions import ConnectTimeout, HTTPError | ||||
| import voluptuous as vol | ||||
|  | ||||
| from homeassistant.config_entries import ConfigEntry | ||||
| from homeassistant.const import ( | ||||
|     ATTR_DATE, | ||||
|     ATTR_DEVICE_ID, | ||||
|     ATTR_ENTITY_ID, | ||||
|     ATTR_TIME, | ||||
|     CONF_PASSWORD, | ||||
|     CONF_USERNAME, | ||||
|     EVENT_HOMEASSISTANT_STOP, | ||||
|     Platform, | ||||
| ) | ||||
| from homeassistant.core import CALLBACK_TYPE, Event, HomeAssistant | ||||
| from homeassistant.core import CALLBACK_TYPE, Event, HomeAssistant, ServiceCall | ||||
| from homeassistant.exceptions import ConfigEntryAuthFailed, ConfigEntryNotReady | ||||
| from homeassistant.helpers import config_validation as cv | ||||
| from homeassistant.helpers.dispatcher import dispatcher_send | ||||
| from homeassistant.helpers.typing import ConfigType | ||||
|  | ||||
| from .const import CONF_POLLING, DOMAIN, LOGGER | ||||
| from .services import async_setup_services | ||||
|  | ||||
| SERVICE_SETTINGS = "change_setting" | ||||
| SERVICE_CAPTURE_IMAGE = "capture_image" | ||||
| SERVICE_TRIGGER_AUTOMATION = "trigger_automation" | ||||
|  | ||||
| ATTR_DEVICE_NAME = "device_name" | ||||
| ATTR_DEVICE_TYPE = "device_type" | ||||
| @@ -39,12 +45,22 @@ ATTR_EVENT_CODE = "event_code" | ||||
| ATTR_EVENT_NAME = "event_name" | ||||
| ATTR_EVENT_TYPE = "event_type" | ||||
| ATTR_EVENT_UTC = "event_utc" | ||||
| ATTR_SETTING = "setting" | ||||
| ATTR_USER_NAME = "user_name" | ||||
| ATTR_APP_TYPE = "app_type" | ||||
| ATTR_EVENT_BY = "event_by" | ||||
| ATTR_VALUE = "value" | ||||
|  | ||||
| CONFIG_SCHEMA = cv.removed(DOMAIN, raise_if_present=False) | ||||
|  | ||||
| CHANGE_SETTING_SCHEMA = vol.Schema( | ||||
|     {vol.Required(ATTR_SETTING): cv.string, vol.Required(ATTR_VALUE): cv.string} | ||||
| ) | ||||
|  | ||||
| CAPTURE_IMAGE_SCHEMA = vol.Schema({ATTR_ENTITY_ID: cv.entity_ids}) | ||||
|  | ||||
| AUTOMATION_SCHEMA = vol.Schema({ATTR_ENTITY_ID: cv.entity_ids}) | ||||
|  | ||||
| PLATFORMS = [ | ||||
|     Platform.ALARM_CONTROL_PANEL, | ||||
|     Platform.BINARY_SENSOR, | ||||
| @@ -69,7 +85,7 @@ class AbodeSystem: | ||||
|  | ||||
| async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool: | ||||
|     """Set up the Abode component.""" | ||||
|     async_setup_services(hass) | ||||
|     setup_hass_services(hass) | ||||
|     return True | ||||
|  | ||||
|  | ||||
| @@ -122,6 +138,60 @@ async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: | ||||
|     return unload_ok | ||||
|  | ||||
|  | ||||
| def setup_hass_services(hass: HomeAssistant) -> None: | ||||
|     """Home Assistant services.""" | ||||
|  | ||||
|     def change_setting(call: ServiceCall) -> None: | ||||
|         """Change an Abode system setting.""" | ||||
|         setting = call.data[ATTR_SETTING] | ||||
|         value = call.data[ATTR_VALUE] | ||||
|  | ||||
|         try: | ||||
|             hass.data[DOMAIN].abode.set_setting(setting, value) | ||||
|         except AbodeException as ex: | ||||
|             LOGGER.warning(ex) | ||||
|  | ||||
|     def capture_image(call: ServiceCall) -> None: | ||||
|         """Capture a new image.""" | ||||
|         entity_ids = call.data[ATTR_ENTITY_ID] | ||||
|  | ||||
|         target_entities = [ | ||||
|             entity_id | ||||
|             for entity_id in hass.data[DOMAIN].entity_ids | ||||
|             if entity_id in entity_ids | ||||
|         ] | ||||
|  | ||||
|         for entity_id in target_entities: | ||||
|             signal = f"abode_camera_capture_{entity_id}" | ||||
|             dispatcher_send(hass, signal) | ||||
|  | ||||
|     def trigger_automation(call: ServiceCall) -> None: | ||||
|         """Trigger an Abode automation.""" | ||||
|         entity_ids = call.data[ATTR_ENTITY_ID] | ||||
|  | ||||
|         target_entities = [ | ||||
|             entity_id | ||||
|             for entity_id in hass.data[DOMAIN].entity_ids | ||||
|             if entity_id in entity_ids | ||||
|         ] | ||||
|  | ||||
|         for entity_id in target_entities: | ||||
|             signal = f"abode_trigger_automation_{entity_id}" | ||||
|             dispatcher_send(hass, signal) | ||||
|  | ||||
|     hass.services.async_register( | ||||
|         DOMAIN, SERVICE_SETTINGS, change_setting, schema=CHANGE_SETTING_SCHEMA | ||||
|     ) | ||||
|  | ||||
|     hass.services.async_register( | ||||
|         DOMAIN, SERVICE_CAPTURE_IMAGE, capture_image, schema=CAPTURE_IMAGE_SCHEMA | ||||
|     ) | ||||
|  | ||||
|     hass.services.async_register( | ||||
|         DOMAIN, SERVICE_TRIGGER_AUTOMATION, trigger_automation, schema=AUTOMATION_SCHEMA | ||||
|     ) | ||||
|  | ||||
|  | ||||
| async def setup_hass_events(hass: HomeAssistant) -> None: | ||||
|     """Home Assistant start and stop callbacks.""" | ||||
|  | ||||
|   | ||||
| @@ -1,90 +0,0 @@ | ||||
| """Support for the Abode Security System.""" | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| from jaraco.abode.exceptions import Exception as AbodeException | ||||
| import voluptuous as vol | ||||
|  | ||||
| from homeassistant.const import ATTR_ENTITY_ID | ||||
| from homeassistant.core import HomeAssistant, ServiceCall, callback | ||||
| from homeassistant.helpers import config_validation as cv | ||||
| from homeassistant.helpers.dispatcher import dispatcher_send | ||||
|  | ||||
| from .const import DOMAIN, LOGGER | ||||
|  | ||||
| SERVICE_SETTINGS = "change_setting" | ||||
| SERVICE_CAPTURE_IMAGE = "capture_image" | ||||
| SERVICE_TRIGGER_AUTOMATION = "trigger_automation" | ||||
|  | ||||
| ATTR_SETTING = "setting" | ||||
| ATTR_VALUE = "value" | ||||
|  | ||||
|  | ||||
| CHANGE_SETTING_SCHEMA = vol.Schema( | ||||
|     {vol.Required(ATTR_SETTING): cv.string, vol.Required(ATTR_VALUE): cv.string} | ||||
| ) | ||||
|  | ||||
| CAPTURE_IMAGE_SCHEMA = vol.Schema({ATTR_ENTITY_ID: cv.entity_ids}) | ||||
|  | ||||
| AUTOMATION_SCHEMA = vol.Schema({ATTR_ENTITY_ID: cv.entity_ids}) | ||||
|  | ||||
|  | ||||
| def _change_setting(call: ServiceCall) -> None: | ||||
|     """Change an Abode system setting.""" | ||||
|     setting = call.data[ATTR_SETTING] | ||||
|     value = call.data[ATTR_VALUE] | ||||
|  | ||||
|     try: | ||||
|         call.hass.data[DOMAIN].abode.set_setting(setting, value) | ||||
|     except AbodeException as ex: | ||||
|         LOGGER.warning(ex) | ||||
|  | ||||
|  | ||||
| def _capture_image(call: ServiceCall) -> None: | ||||
|     """Capture a new image.""" | ||||
|     entity_ids = call.data[ATTR_ENTITY_ID] | ||||
|  | ||||
|     target_entities = [ | ||||
|         entity_id | ||||
|         for entity_id in call.hass.data[DOMAIN].entity_ids | ||||
|         if entity_id in entity_ids | ||||
|     ] | ||||
|  | ||||
|     for entity_id in target_entities: | ||||
|         signal = f"abode_camera_capture_{entity_id}" | ||||
|         dispatcher_send(call.hass, signal) | ||||
|  | ||||
|  | ||||
| def _trigger_automation(call: ServiceCall) -> None: | ||||
|     """Trigger an Abode automation.""" | ||||
|     entity_ids = call.data[ATTR_ENTITY_ID] | ||||
|  | ||||
|     target_entities = [ | ||||
|         entity_id | ||||
|         for entity_id in call.hass.data[DOMAIN].entity_ids | ||||
|         if entity_id in entity_ids | ||||
|     ] | ||||
|  | ||||
|     for entity_id in target_entities: | ||||
|         signal = f"abode_trigger_automation_{entity_id}" | ||||
|         dispatcher_send(call.hass, signal) | ||||
|  | ||||
|  | ||||
| @callback | ||||
| def async_setup_services(hass: HomeAssistant) -> None: | ||||
|     """Home Assistant services.""" | ||||
|  | ||||
|     hass.services.async_register( | ||||
|         DOMAIN, SERVICE_SETTINGS, _change_setting, schema=CHANGE_SETTING_SCHEMA | ||||
|     ) | ||||
|  | ||||
|     hass.services.async_register( | ||||
|         DOMAIN, SERVICE_CAPTURE_IMAGE, _capture_image, schema=CAPTURE_IMAGE_SCHEMA | ||||
|     ) | ||||
|  | ||||
|     hass.services.async_register( | ||||
|         DOMAIN, | ||||
|         SERVICE_TRIGGER_AUTOMATION, | ||||
|         _trigger_automation, | ||||
|         schema=AUTOMATION_SCHEMA, | ||||
|     ) | ||||
| @@ -8,17 +8,14 @@ import logging | ||||
| from aioacaia.acaiascale import AcaiaScale | ||||
| from aioacaia.exceptions import AcaiaDeviceNotFound, AcaiaError | ||||
|  | ||||
| from homeassistant.components.bluetooth import async_get_scanner | ||||
| from homeassistant.config_entries import ConfigEntry | ||||
| from homeassistant.const import CONF_ADDRESS | ||||
| from homeassistant.core import HomeAssistant | ||||
| from homeassistant.helpers.debounce import Debouncer | ||||
| from homeassistant.helpers.update_coordinator import DataUpdateCoordinator | ||||
|  | ||||
| from .const import CONF_IS_NEW_STYLE_SCALE | ||||
|  | ||||
| SCAN_INTERVAL = timedelta(seconds=15) | ||||
| UPDATE_DEBOUNCE_TIME = 0.2 | ||||
|  | ||||
| _LOGGER = logging.getLogger(__name__) | ||||
|  | ||||
| @@ -40,20 +37,11 @@ class AcaiaCoordinator(DataUpdateCoordinator[None]): | ||||
|             config_entry=entry, | ||||
|         ) | ||||
|  | ||||
|         debouncer = Debouncer( | ||||
|             hass=hass, | ||||
|             logger=_LOGGER, | ||||
|             cooldown=UPDATE_DEBOUNCE_TIME, | ||||
|             immediate=True, | ||||
|             function=self.async_update_listeners, | ||||
|         ) | ||||
|  | ||||
|         self._scale = AcaiaScale( | ||||
|             address_or_ble_device=entry.data[CONF_ADDRESS], | ||||
|             name=entry.title, | ||||
|             is_new_style_scale=entry.data[CONF_IS_NEW_STYLE_SCALE], | ||||
|             notify_callback=debouncer.async_schedule_call, | ||||
|             scanner=async_get_scanner(hass), | ||||
|             notify_callback=self.async_update_listeners, | ||||
|         ) | ||||
|  | ||||
|     @property | ||||
|   | ||||
| @@ -26,5 +26,5 @@ | ||||
|   "iot_class": "local_push", | ||||
|   "loggers": ["aioacaia"], | ||||
|   "quality_scale": "platinum", | ||||
|   "requirements": ["aioacaia==0.1.17"] | ||||
|   "requirements": ["aioacaia==0.1.14"] | ||||
| } | ||||
|   | ||||
| @@ -2,23 +2,21 @@ | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| import asyncio | ||||
| import logging | ||||
|  | ||||
| from accuweather import AccuWeather | ||||
|  | ||||
| from homeassistant.components.sensor import DOMAIN as SENSOR_PLATFORM | ||||
| from homeassistant.const import CONF_API_KEY, Platform | ||||
| from homeassistant.const import CONF_API_KEY, CONF_NAME, Platform | ||||
| from homeassistant.core import HomeAssistant | ||||
| from homeassistant.helpers import entity_registry as er | ||||
| from homeassistant.helpers.aiohttp_client import async_get_clientsession | ||||
|  | ||||
| from .const import DOMAIN | ||||
| from .const import DOMAIN, UPDATE_INTERVAL_DAILY_FORECAST, UPDATE_INTERVAL_OBSERVATION | ||||
| from .coordinator import ( | ||||
|     AccuWeatherConfigEntry, | ||||
|     AccuWeatherDailyForecastDataUpdateCoordinator, | ||||
|     AccuWeatherData, | ||||
|     AccuWeatherHourlyForecastDataUpdateCoordinator, | ||||
|     AccuWeatherObservationDataUpdateCoordinator, | ||||
| ) | ||||
|  | ||||
| @@ -30,6 +28,7 @@ PLATFORMS = [Platform.SENSOR, Platform.WEATHER] | ||||
| async def async_setup_entry(hass: HomeAssistant, entry: AccuWeatherConfigEntry) -> bool: | ||||
|     """Set up AccuWeather as config entry.""" | ||||
|     api_key: str = entry.data[CONF_API_KEY] | ||||
|     name: str = entry.data[CONF_NAME] | ||||
|  | ||||
|     location_key = entry.unique_id | ||||
|  | ||||
| @@ -42,28 +41,26 @@ async def async_setup_entry(hass: HomeAssistant, entry: AccuWeatherConfigEntry) | ||||
|         hass, | ||||
|         entry, | ||||
|         accuweather, | ||||
|         name, | ||||
|         "observation", | ||||
|         UPDATE_INTERVAL_OBSERVATION, | ||||
|     ) | ||||
|  | ||||
|     coordinator_daily_forecast = AccuWeatherDailyForecastDataUpdateCoordinator( | ||||
|         hass, | ||||
|         entry, | ||||
|         accuweather, | ||||
|     ) | ||||
|     coordinator_hourly_forecast = AccuWeatherHourlyForecastDataUpdateCoordinator( | ||||
|         hass, | ||||
|         entry, | ||||
|         accuweather, | ||||
|         name, | ||||
|         "daily forecast", | ||||
|         UPDATE_INTERVAL_DAILY_FORECAST, | ||||
|     ) | ||||
|  | ||||
|     await asyncio.gather( | ||||
|         coordinator_observation.async_config_entry_first_refresh(), | ||||
|         coordinator_daily_forecast.async_config_entry_first_refresh(), | ||||
|         coordinator_hourly_forecast.async_config_entry_first_refresh(), | ||||
|     ) | ||||
|     await coordinator_observation.async_config_entry_first_refresh() | ||||
|     await coordinator_daily_forecast.async_config_entry_first_refresh() | ||||
|  | ||||
|     entry.runtime_data = AccuWeatherData( | ||||
|         coordinator_observation=coordinator_observation, | ||||
|         coordinator_daily_forecast=coordinator_daily_forecast, | ||||
|         coordinator_hourly_forecast=coordinator_hourly_forecast, | ||||
|     ) | ||||
|  | ||||
|     await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS) | ||||
|   | ||||
| @@ -3,7 +3,6 @@ | ||||
| from __future__ import annotations | ||||
|  | ||||
| from asyncio import timeout | ||||
| from collections.abc import Mapping | ||||
| from typing import Any | ||||
|  | ||||
| from accuweather import AccuWeather, ApiError, InvalidApiKeyError, RequestsExceededError | ||||
| @@ -23,8 +22,6 @@ class AccuWeatherFlowHandler(ConfigFlow, domain=DOMAIN): | ||||
|     """Config flow for AccuWeather.""" | ||||
|  | ||||
|     VERSION = 1 | ||||
|     _latitude: float | None = None | ||||
|     _longitude: float | None = None | ||||
|  | ||||
|     async def async_step_user( | ||||
|         self, user_input: dict[str, Any] | None = None | ||||
| @@ -53,7 +50,6 @@ class AccuWeatherFlowHandler(ConfigFlow, domain=DOMAIN): | ||||
|                 await self.async_set_unique_id( | ||||
|                     accuweather.location_key, raise_on_progress=False | ||||
|                 ) | ||||
|                 self._abort_if_unique_id_configured() | ||||
|  | ||||
|                 return self.async_create_entry( | ||||
|                     title=user_input[CONF_NAME], data=user_input | ||||
| @@ -77,46 +73,3 @@ class AccuWeatherFlowHandler(ConfigFlow, domain=DOMAIN): | ||||
|             ), | ||||
|             errors=errors, | ||||
|         ) | ||||
|  | ||||
|     async def async_step_reauth( | ||||
|         self, entry_data: Mapping[str, Any] | ||||
|     ) -> ConfigFlowResult: | ||||
|         """Handle configuration by re-auth.""" | ||||
|         self._latitude = entry_data[CONF_LATITUDE] | ||||
|         self._longitude = entry_data[CONF_LONGITUDE] | ||||
|  | ||||
|         return await self.async_step_reauth_confirm() | ||||
|  | ||||
|     async def async_step_reauth_confirm( | ||||
|         self, user_input: dict[str, Any] | None = None | ||||
|     ) -> ConfigFlowResult: | ||||
|         """Dialog that informs the user that reauth is required.""" | ||||
|         errors: dict[str, str] = {} | ||||
|  | ||||
|         if user_input is not None: | ||||
|             websession = async_get_clientsession(self.hass) | ||||
|             try: | ||||
|                 async with timeout(10): | ||||
|                     accuweather = AccuWeather( | ||||
|                         user_input[CONF_API_KEY], | ||||
|                         websession, | ||||
|                         latitude=self._latitude, | ||||
|                         longitude=self._longitude, | ||||
|                     ) | ||||
|                     await accuweather.async_get_location() | ||||
|             except (ApiError, ClientConnectorError, TimeoutError, ClientError): | ||||
|                 errors["base"] = "cannot_connect" | ||||
|             except InvalidApiKeyError: | ||||
|                 errors["base"] = "invalid_api_key" | ||||
|             except RequestsExceededError: | ||||
|                 errors["base"] = "requests_exceeded" | ||||
|             else: | ||||
|                 return self.async_update_reload_and_abort( | ||||
|                     self._get_reauth_entry(), data_updates=user_input | ||||
|                 ) | ||||
|  | ||||
|         return self.async_show_form( | ||||
|             step_id="reauth_confirm", | ||||
|             data_schema=vol.Schema({vol.Required(CONF_API_KEY): str}), | ||||
|             errors=errors, | ||||
|         ) | ||||
|   | ||||
| @@ -67,8 +67,6 @@ POLLEN_CATEGORY_MAP = { | ||||
|     2: "moderate", | ||||
|     3: "high", | ||||
|     4: "very_high", | ||||
|     5: "extreme", | ||||
| } | ||||
| UPDATE_INTERVAL_OBSERVATION = timedelta(minutes=10) | ||||
| UPDATE_INTERVAL_OBSERVATION = timedelta(minutes=40) | ||||
| UPDATE_INTERVAL_DAILY_FORECAST = timedelta(hours=6) | ||||
| UPDATE_INTERVAL_HOURLY_FORECAST = timedelta(minutes=30) | ||||
|   | ||||
| @@ -3,7 +3,6 @@ | ||||
| from __future__ import annotations | ||||
|  | ||||
| from asyncio import timeout | ||||
| from collections.abc import Awaitable, Callable | ||||
| from dataclasses import dataclass | ||||
| from datetime import timedelta | ||||
| import logging | ||||
| @@ -13,9 +12,7 @@ from accuweather import AccuWeather, ApiError, InvalidApiKeyError, RequestsExcee | ||||
| from aiohttp.client_exceptions import ClientConnectorError | ||||
|  | ||||
| from homeassistant.config_entries import ConfigEntry | ||||
| from homeassistant.const import CONF_NAME | ||||
| from homeassistant.core import HomeAssistant | ||||
| from homeassistant.exceptions import ConfigEntryAuthFailed | ||||
| from homeassistant.helpers.device_registry import DeviceEntryType, DeviceInfo | ||||
| from homeassistant.helpers.update_coordinator import ( | ||||
|     DataUpdateCoordinator, | ||||
| @@ -23,15 +20,9 @@ from homeassistant.helpers.update_coordinator import ( | ||||
|     UpdateFailed, | ||||
| ) | ||||
|  | ||||
| from .const import ( | ||||
|     DOMAIN, | ||||
|     MANUFACTURER, | ||||
|     UPDATE_INTERVAL_DAILY_FORECAST, | ||||
|     UPDATE_INTERVAL_HOURLY_FORECAST, | ||||
|     UPDATE_INTERVAL_OBSERVATION, | ||||
| ) | ||||
| from .const import DOMAIN, MANUFACTURER | ||||
|  | ||||
| EXCEPTIONS = (ApiError, ClientConnectorError, RequestsExceededError) | ||||
| EXCEPTIONS = (ApiError, ClientConnectorError, InvalidApiKeyError, RequestsExceededError) | ||||
|  | ||||
| _LOGGER = logging.getLogger(__name__) | ||||
|  | ||||
| @@ -42,7 +33,6 @@ class AccuWeatherData: | ||||
|  | ||||
|     coordinator_observation: AccuWeatherObservationDataUpdateCoordinator | ||||
|     coordinator_daily_forecast: AccuWeatherDailyForecastDataUpdateCoordinator | ||||
|     coordinator_hourly_forecast: AccuWeatherHourlyForecastDataUpdateCoordinator | ||||
|  | ||||
|  | ||||
| type AccuWeatherConfigEntry = ConfigEntry[AccuWeatherData] | ||||
| @@ -53,18 +43,18 @@ class AccuWeatherObservationDataUpdateCoordinator( | ||||
| ): | ||||
|     """Class to manage fetching AccuWeather data API.""" | ||||
|  | ||||
|     config_entry: AccuWeatherConfigEntry | ||||
|  | ||||
|     def __init__( | ||||
|         self, | ||||
|         hass: HomeAssistant, | ||||
|         config_entry: AccuWeatherConfigEntry, | ||||
|         accuweather: AccuWeather, | ||||
|         name: str, | ||||
|         coordinator_type: str, | ||||
|         update_interval: timedelta, | ||||
|     ) -> None: | ||||
|         """Initialize.""" | ||||
|         self.accuweather = accuweather | ||||
|         self.location_key = accuweather.location_key | ||||
|         name = config_entry.data[CONF_NAME] | ||||
|  | ||||
|         if TYPE_CHECKING: | ||||
|             assert self.location_key is not None | ||||
| @@ -75,8 +65,8 @@ class AccuWeatherObservationDataUpdateCoordinator( | ||||
|             hass, | ||||
|             _LOGGER, | ||||
|             config_entry=config_entry, | ||||
|             name=f"{name} (observation)", | ||||
|             update_interval=UPDATE_INTERVAL_OBSERVATION, | ||||
|             name=f"{name} ({coordinator_type})", | ||||
|             update_interval=update_interval, | ||||
|         ) | ||||
|  | ||||
|     async def _async_update_data(self) -> dict[str, Any]: | ||||
| @@ -90,39 +80,29 @@ class AccuWeatherObservationDataUpdateCoordinator( | ||||
|                 translation_key="current_conditions_update_error", | ||||
|                 translation_placeholders={"error": repr(error)}, | ||||
|             ) from error | ||||
|         except InvalidApiKeyError as err: | ||||
|             raise ConfigEntryAuthFailed( | ||||
|                 translation_domain=DOMAIN, | ||||
|                 translation_key="auth_error", | ||||
|                 translation_placeholders={"entry": self.config_entry.title}, | ||||
|             ) from err | ||||
|  | ||||
|         _LOGGER.debug("Requests remaining: %d", self.accuweather.requests_remaining) | ||||
|  | ||||
|         return result | ||||
|  | ||||
|  | ||||
| class AccuWeatherForecastDataUpdateCoordinator( | ||||
| class AccuWeatherDailyForecastDataUpdateCoordinator( | ||||
|     TimestampDataUpdateCoordinator[list[dict[str, Any]]] | ||||
| ): | ||||
|     """Base class for AccuWeather forecast.""" | ||||
|  | ||||
|     config_entry: AccuWeatherConfigEntry | ||||
|     """Class to manage fetching AccuWeather data API.""" | ||||
|  | ||||
|     def __init__( | ||||
|         self, | ||||
|         hass: HomeAssistant, | ||||
|         config_entry: AccuWeatherConfigEntry, | ||||
|         accuweather: AccuWeather, | ||||
|         name: str, | ||||
|         coordinator_type: str, | ||||
|         update_interval: timedelta, | ||||
|         fetch_method: Callable[..., Awaitable[list[dict[str, Any]]]], | ||||
|     ) -> None: | ||||
|         """Initialize.""" | ||||
|         self.accuweather = accuweather | ||||
|         self.location_key = accuweather.location_key | ||||
|         self._fetch_method = fetch_method | ||||
|         name = config_entry.data[CONF_NAME] | ||||
|  | ||||
|         if TYPE_CHECKING: | ||||
|             assert self.location_key is not None | ||||
| @@ -138,71 +118,24 @@ class AccuWeatherForecastDataUpdateCoordinator( | ||||
|         ) | ||||
|  | ||||
|     async def _async_update_data(self) -> list[dict[str, Any]]: | ||||
|         """Update forecast data via library.""" | ||||
|         """Update data via library.""" | ||||
|         try: | ||||
|             async with timeout(10): | ||||
|                 result = await self._fetch_method(language=self.hass.config.language) | ||||
|                 result = await self.accuweather.async_get_daily_forecast( | ||||
|                     language=self.hass.config.language | ||||
|                 ) | ||||
|         except EXCEPTIONS as error: | ||||
|             raise UpdateFailed( | ||||
|                 translation_domain=DOMAIN, | ||||
|                 translation_key="forecast_update_error", | ||||
|                 translation_placeholders={"error": repr(error)}, | ||||
|             ) from error | ||||
|         except InvalidApiKeyError as err: | ||||
|             raise ConfigEntryAuthFailed( | ||||
|                 translation_domain=DOMAIN, | ||||
|                 translation_key="auth_error", | ||||
|                 translation_placeholders={"entry": self.config_entry.title}, | ||||
|             ) from err | ||||
|  | ||||
|         _LOGGER.debug("Requests remaining: %d", self.accuweather.requests_remaining) | ||||
|  | ||||
|         return result | ||||
|  | ||||
|  | ||||
| class AccuWeatherDailyForecastDataUpdateCoordinator( | ||||
|     AccuWeatherForecastDataUpdateCoordinator | ||||
| ): | ||||
|     """Coordinator for daily forecast.""" | ||||
|  | ||||
|     def __init__( | ||||
|         self, | ||||
|         hass: HomeAssistant, | ||||
|         config_entry: AccuWeatherConfigEntry, | ||||
|         accuweather: AccuWeather, | ||||
|     ) -> None: | ||||
|         """Initialize.""" | ||||
|         super().__init__( | ||||
|             hass, | ||||
|             config_entry, | ||||
|             accuweather, | ||||
|             "daily forecast", | ||||
|             UPDATE_INTERVAL_DAILY_FORECAST, | ||||
|             fetch_method=accuweather.async_get_daily_forecast, | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class AccuWeatherHourlyForecastDataUpdateCoordinator( | ||||
|     AccuWeatherForecastDataUpdateCoordinator | ||||
| ): | ||||
|     """Coordinator for hourly forecast.""" | ||||
|  | ||||
|     def __init__( | ||||
|         self, | ||||
|         hass: HomeAssistant, | ||||
|         config_entry: AccuWeatherConfigEntry, | ||||
|         accuweather: AccuWeather, | ||||
|     ) -> None: | ||||
|         """Initialize.""" | ||||
|         super().__init__( | ||||
|             hass, | ||||
|             config_entry, | ||||
|             accuweather, | ||||
|             "hourly forecast", | ||||
|             UPDATE_INTERVAL_HOURLY_FORECAST, | ||||
|             fetch_method=accuweather.async_get_hourly_forecast, | ||||
|         ) | ||||
|  | ||||
|  | ||||
| def _get_device_info(location_key: str, name: str) -> DeviceInfo: | ||||
|     """Get device info.""" | ||||
|     return DeviceInfo( | ||||
|   | ||||
| @@ -1,9 +1,6 @@ | ||||
| { | ||||
|   "entity": { | ||||
|     "sensor": { | ||||
|       "air_quality": { | ||||
|         "default": "mdi:air-filter" | ||||
|       }, | ||||
|       "cloud_ceiling": { | ||||
|         "default": "mdi:weather-fog" | ||||
|       }, | ||||
| @@ -37,6 +34,9 @@ | ||||
|       "thunderstorm_probability_night": { | ||||
|         "default": "mdi:weather-lightning" | ||||
|       }, | ||||
|       "translation_key": { | ||||
|         "default": "mdi:air-filter" | ||||
|       }, | ||||
|       "tree_pollen": { | ||||
|         "default": "mdi:tree-outline" | ||||
|       }, | ||||
|   | ||||
| @@ -7,5 +7,6 @@ | ||||
|   "integration_type": "service", | ||||
|   "iot_class": "cloud_polling", | ||||
|   "loggers": ["accuweather"], | ||||
|   "requirements": ["accuweather==4.2.2"] | ||||
|   "requirements": ["accuweather==4.2.0"], | ||||
|   "single_config_entry": true | ||||
| } | ||||
|   | ||||
| @@ -7,17 +7,6 @@ | ||||
|           "api_key": "[%key:common::config_flow::data::api_key%]", | ||||
|           "latitude": "[%key:common::config_flow::data::latitude%]", | ||||
|           "longitude": "[%key:common::config_flow::data::longitude%]" | ||||
|         }, | ||||
|         "data_description": { | ||||
|           "api_key": "API key generated in the AccuWeather APIs portal." | ||||
|         } | ||||
|       }, | ||||
|       "reauth_confirm": { | ||||
|         "data": { | ||||
|           "api_key": "[%key:common::config_flow::data::api_key%]" | ||||
|         }, | ||||
|         "data_description": { | ||||
|           "api_key": "[%key:component::accuweather::config::step::user::data_description::api_key%]" | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
| @@ -28,10 +17,6 @@ | ||||
|       "cannot_connect": "[%key:common::config_flow::error::cannot_connect%]", | ||||
|       "invalid_api_key": "[%key:common::config_flow::error::invalid_api_key%]", | ||||
|       "requests_exceeded": "The allowed number of requests to the AccuWeather API has been exceeded. You have to wait or change the API key." | ||||
|     }, | ||||
|     "abort": { | ||||
|       "already_configured": "[%key:common::config_flow::abort::already_configured_location%]", | ||||
|       "reauth_successful": "[%key:common::config_flow::abort::reauth_successful%]" | ||||
|     } | ||||
|   }, | ||||
|   "entity": { | ||||
| @@ -87,11 +72,10 @@ | ||||
|           "level": { | ||||
|             "name": "Level", | ||||
|             "state": { | ||||
|               "extreme": "Extreme", | ||||
|               "high": "[%key:common::state::high%]", | ||||
|               "low": "[%key:common::state::low%]", | ||||
|               "high": "High", | ||||
|               "low": "Low", | ||||
|               "moderate": "Moderate", | ||||
|               "very_high": "[%key:common::state::very_high%]" | ||||
|               "very_high": "Very high" | ||||
|             } | ||||
|           } | ||||
|         } | ||||
| @@ -105,11 +89,10 @@ | ||||
|           "level": { | ||||
|             "name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]", | ||||
|             "state": { | ||||
|               "extreme": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::extreme%]", | ||||
|               "high": "[%key:common::state::high%]", | ||||
|               "low": "[%key:common::state::low%]", | ||||
|               "high": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::high%]", | ||||
|               "low": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::low%]", | ||||
|               "moderate": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::moderate%]", | ||||
|               "very_high": "[%key:common::state::very_high%]" | ||||
|               "very_high": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::very_high%]" | ||||
|             } | ||||
|           } | ||||
|         } | ||||
| @@ -140,11 +123,10 @@ | ||||
|           "level": { | ||||
|             "name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]", | ||||
|             "state": { | ||||
|               "extreme": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::extreme%]", | ||||
|               "high": "[%key:common::state::high%]", | ||||
|               "low": "[%key:common::state::low%]", | ||||
|               "high": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::high%]", | ||||
|               "low": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::low%]", | ||||
|               "moderate": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::moderate%]", | ||||
|               "very_high": "[%key:common::state::very_high%]" | ||||
|               "very_high": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::very_high%]" | ||||
|             } | ||||
|           } | ||||
|         } | ||||
| @@ -185,11 +167,10 @@ | ||||
|           "level": { | ||||
|             "name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]", | ||||
|             "state": { | ||||
|               "extreme": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::extreme%]", | ||||
|               "high": "[%key:common::state::high%]", | ||||
|               "low": "[%key:common::state::low%]", | ||||
|               "high": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::high%]", | ||||
|               "low": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::low%]", | ||||
|               "moderate": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::moderate%]", | ||||
|               "very_high": "[%key:common::state::very_high%]" | ||||
|               "very_high": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::very_high%]" | ||||
|             } | ||||
|           } | ||||
|         } | ||||
| @@ -200,11 +181,10 @@ | ||||
|           "level": { | ||||
|             "name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]", | ||||
|             "state": { | ||||
|               "extreme": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::extreme%]", | ||||
|               "high": "[%key:common::state::high%]", | ||||
|               "low": "[%key:common::state::low%]", | ||||
|               "high": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::high%]", | ||||
|               "low": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::low%]", | ||||
|               "moderate": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::moderate%]", | ||||
|               "very_high": "[%key:common::state::very_high%]" | ||||
|               "very_high": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::very_high%]" | ||||
|             } | ||||
|           } | ||||
|         } | ||||
| @@ -215,11 +195,10 @@ | ||||
|           "level": { | ||||
|             "name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]", | ||||
|             "state": { | ||||
|               "extreme": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::extreme%]", | ||||
|               "high": "[%key:common::state::high%]", | ||||
|               "low": "[%key:common::state::low%]", | ||||
|               "high": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::high%]", | ||||
|               "low": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::low%]", | ||||
|               "moderate": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::moderate%]", | ||||
|               "very_high": "[%key:common::state::very_high%]" | ||||
|               "very_high": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::very_high%]" | ||||
|             } | ||||
|           } | ||||
|         } | ||||
| @@ -251,9 +230,6 @@ | ||||
|     } | ||||
|   }, | ||||
|   "exceptions": { | ||||
|     "auth_error": { | ||||
|       "message": "Authentication failed for {entry}, please update your API key" | ||||
|     }, | ||||
|     "current_conditions_update_error": { | ||||
|       "message": "An error occurred while retrieving weather current conditions data from the AccuWeather API: {error}" | ||||
|     }, | ||||
|   | ||||
| @@ -45,7 +45,6 @@ from .coordinator import ( | ||||
|     AccuWeatherConfigEntry, | ||||
|     AccuWeatherDailyForecastDataUpdateCoordinator, | ||||
|     AccuWeatherData, | ||||
|     AccuWeatherHourlyForecastDataUpdateCoordinator, | ||||
|     AccuWeatherObservationDataUpdateCoordinator, | ||||
| ) | ||||
|  | ||||
| @@ -65,7 +64,6 @@ class AccuWeatherEntity( | ||||
|     CoordinatorWeatherEntity[ | ||||
|         AccuWeatherObservationDataUpdateCoordinator, | ||||
|         AccuWeatherDailyForecastDataUpdateCoordinator, | ||||
|         AccuWeatherHourlyForecastDataUpdateCoordinator, | ||||
|     ] | ||||
| ): | ||||
|     """Define an AccuWeather entity.""" | ||||
| @@ -78,7 +76,6 @@ class AccuWeatherEntity( | ||||
|         super().__init__( | ||||
|             observation_coordinator=accuweather_data.coordinator_observation, | ||||
|             daily_coordinator=accuweather_data.coordinator_daily_forecast, | ||||
|             hourly_coordinator=accuweather_data.coordinator_hourly_forecast, | ||||
|         ) | ||||
|  | ||||
|         self._attr_native_precipitation_unit = UnitOfPrecipitationDepth.MILLIMETERS | ||||
| @@ -89,13 +86,10 @@ class AccuWeatherEntity( | ||||
|         self._attr_unique_id = accuweather_data.coordinator_observation.location_key | ||||
|         self._attr_attribution = ATTRIBUTION | ||||
|         self._attr_device_info = accuweather_data.coordinator_observation.device_info | ||||
|         self._attr_supported_features = ( | ||||
|             WeatherEntityFeature.FORECAST_DAILY | WeatherEntityFeature.FORECAST_HOURLY | ||||
|         ) | ||||
|         self._attr_supported_features = WeatherEntityFeature.FORECAST_DAILY | ||||
|  | ||||
|         self.observation_coordinator = accuweather_data.coordinator_observation | ||||
|         self.daily_coordinator = accuweather_data.coordinator_daily_forecast | ||||
|         self.hourly_coordinator = accuweather_data.coordinator_hourly_forecast | ||||
|  | ||||
|     @property | ||||
|     def condition(self) -> str | None: | ||||
| @@ -213,32 +207,3 @@ class AccuWeatherEntity( | ||||
|             } | ||||
|             for item in self.daily_coordinator.data | ||||
|         ] | ||||
|  | ||||
|     @callback | ||||
|     def _async_forecast_hourly(self) -> list[Forecast] | None: | ||||
|         """Return the hourly forecast in native units.""" | ||||
|         return [ | ||||
|             { | ||||
|                 ATTR_FORECAST_TIME: utc_from_timestamp( | ||||
|                     item["EpochDateTime"] | ||||
|                 ).isoformat(), | ||||
|                 ATTR_FORECAST_CLOUD_COVERAGE: item["CloudCover"], | ||||
|                 ATTR_FORECAST_HUMIDITY: item["RelativeHumidity"], | ||||
|                 ATTR_FORECAST_NATIVE_TEMP: item["Temperature"][ATTR_VALUE], | ||||
|                 ATTR_FORECAST_NATIVE_APPARENT_TEMP: item["RealFeelTemperature"][ | ||||
|                     ATTR_VALUE | ||||
|                 ], | ||||
|                 ATTR_FORECAST_NATIVE_PRECIPITATION: item["TotalLiquid"][ATTR_VALUE], | ||||
|                 ATTR_FORECAST_PRECIPITATION_PROBABILITY: item[ | ||||
|                     "PrecipitationProbability" | ||||
|                 ], | ||||
|                 ATTR_FORECAST_NATIVE_WIND_SPEED: item["Wind"][ATTR_SPEED][ATTR_VALUE], | ||||
|                 ATTR_FORECAST_NATIVE_WIND_GUST_SPEED: item["WindGust"][ATTR_SPEED][ | ||||
|                     ATTR_VALUE | ||||
|                 ], | ||||
|                 ATTR_FORECAST_UV_INDEX: item["UVIndex"], | ||||
|                 ATTR_FORECAST_WIND_BEARING: item["Wind"][ATTR_DIRECTION]["Degrees"], | ||||
|                 ATTR_FORECAST_CONDITION: CONDITION_MAP.get(item["WeatherIcon"]), | ||||
|             } | ||||
|             for item in self.hourly_coordinator.data | ||||
|         ] | ||||
|   | ||||
| @@ -40,10 +40,9 @@ class AcmedaFlowHandler(ConfigFlow, domain=DOMAIN): | ||||
|             entry.unique_id for entry in self._async_current_entries() | ||||
|         } | ||||
|  | ||||
|         hubs: list[aiopulse.Hub] = [] | ||||
|         with suppress(TimeoutError): | ||||
|             async with timeout(5): | ||||
|                 hubs = [ | ||||
|                 hubs: list[aiopulse.Hub] = [ | ||||
|                     hub | ||||
|                     async for hub in aiopulse.Hub.discover() | ||||
|                     if hub.id not in already_configured | ||||
|   | ||||
| @@ -2,38 +2,25 @@ | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| from homeassistant.config_entries import ConfigEntry | ||||
| from homeassistant.const import Platform | ||||
| from homeassistant.core import HomeAssistant | ||||
|  | ||||
| from .const import CONNECTION_TYPE, LOCAL | ||||
| from .coordinator import AdaxCloudCoordinator, AdaxConfigEntry, AdaxLocalCoordinator | ||||
|  | ||||
| PLATFORMS = [Platform.CLIMATE, Platform.SENSOR] | ||||
| PLATFORMS = [Platform.CLIMATE] | ||||
|  | ||||
|  | ||||
| async def async_setup_entry(hass: HomeAssistant, entry: AdaxConfigEntry) -> bool: | ||||
| async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: | ||||
|     """Set up Adax from a config entry.""" | ||||
|     if entry.data.get(CONNECTION_TYPE) == LOCAL: | ||||
|         local_coordinator = AdaxLocalCoordinator(hass, entry) | ||||
|         entry.runtime_data = local_coordinator | ||||
|     else: | ||||
|         cloud_coordinator = AdaxCloudCoordinator(hass, entry) | ||||
|         entry.runtime_data = cloud_coordinator | ||||
|  | ||||
|     await entry.runtime_data.async_config_entry_first_refresh() | ||||
|  | ||||
|     await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS) | ||||
|     return True | ||||
|  | ||||
|  | ||||
| async def async_unload_entry(hass: HomeAssistant, entry: AdaxConfigEntry) -> bool: | ||||
| async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: | ||||
|     """Unload a config entry.""" | ||||
|     return await hass.config_entries.async_unload_platforms(entry, PLATFORMS) | ||||
|  | ||||
|  | ||||
| async def async_migrate_entry( | ||||
|     hass: HomeAssistant, config_entry: AdaxConfigEntry | ||||
| ) -> bool: | ||||
| async def async_migrate_entry(hass: HomeAssistant, config_entry: ConfigEntry) -> bool: | ||||
|     """Migrate old entry.""" | ||||
|     # convert title and unique_id to string | ||||
|     if config_entry.version == 1: | ||||
|   | ||||
| @@ -12,42 +12,57 @@ from homeassistant.components.climate import ( | ||||
|     ClimateEntityFeature, | ||||
|     HVACMode, | ||||
| ) | ||||
| from homeassistant.config_entries import ConfigEntry | ||||
| from homeassistant.const import ( | ||||
|     ATTR_TEMPERATURE, | ||||
|     CONF_IP_ADDRESS, | ||||
|     CONF_PASSWORD, | ||||
|     CONF_TOKEN, | ||||
|     CONF_UNIQUE_ID, | ||||
|     PRECISION_WHOLE, | ||||
|     UnitOfTemperature, | ||||
| ) | ||||
| from homeassistant.core import HomeAssistant, callback | ||||
| from homeassistant.core import HomeAssistant | ||||
| from homeassistant.helpers.aiohttp_client import async_get_clientsession | ||||
| from homeassistant.helpers.device_registry import DeviceInfo | ||||
| from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | ||||
| from homeassistant.helpers.update_coordinator import CoordinatorEntity | ||||
|  | ||||
| from . import AdaxConfigEntry | ||||
| from .const import CONNECTION_TYPE, DOMAIN, LOCAL | ||||
| from .coordinator import AdaxCloudCoordinator, AdaxLocalCoordinator | ||||
| from .const import ACCOUNT_ID, CONNECTION_TYPE, DOMAIN, LOCAL | ||||
|  | ||||
|  | ||||
| async def async_setup_entry( | ||||
|     hass: HomeAssistant, | ||||
|     entry: AdaxConfigEntry, | ||||
|     entry: ConfigEntry, | ||||
|     async_add_entities: AddConfigEntryEntitiesCallback, | ||||
| ) -> None: | ||||
|     """Set up the Adax thermostat with config flow.""" | ||||
|     if entry.data.get(CONNECTION_TYPE) == LOCAL: | ||||
|         local_coordinator = cast(AdaxLocalCoordinator, entry.runtime_data) | ||||
|         async_add_entities( | ||||
|             [LocalAdaxDevice(local_coordinator, entry.data[CONF_UNIQUE_ID])], | ||||
|         adax_data_handler = AdaxLocal( | ||||
|             entry.data[CONF_IP_ADDRESS], | ||||
|             entry.data[CONF_TOKEN], | ||||
|             websession=async_get_clientsession(hass, verify_ssl=False), | ||||
|         ) | ||||
|     else: | ||||
|         cloud_coordinator = cast(AdaxCloudCoordinator, entry.runtime_data) | ||||
|         async_add_entities( | ||||
|             AdaxDevice(cloud_coordinator, device_id) | ||||
|             for device_id in cloud_coordinator.data | ||||
|             [LocalAdaxDevice(adax_data_handler, entry.data[CONF_UNIQUE_ID])], True | ||||
|         ) | ||||
|         return | ||||
|  | ||||
|     adax_data_handler = Adax( | ||||
|         entry.data[ACCOUNT_ID], | ||||
|         entry.data[CONF_PASSWORD], | ||||
|         websession=async_get_clientsession(hass), | ||||
|     ) | ||||
|  | ||||
|     async_add_entities( | ||||
|         ( | ||||
|             AdaxDevice(room, adax_data_handler) | ||||
|             for room in await adax_data_handler.get_rooms() | ||||
|         ), | ||||
|         True, | ||||
|     ) | ||||
|  | ||||
|  | ||||
| class AdaxDevice(CoordinatorEntity[AdaxCloudCoordinator], ClimateEntity): | ||||
| class AdaxDevice(ClimateEntity): | ||||
|     """Representation of a heater.""" | ||||
|  | ||||
|     _attr_hvac_modes = [HVACMode.HEAT, HVACMode.OFF] | ||||
| @@ -61,37 +76,20 @@ class AdaxDevice(CoordinatorEntity[AdaxCloudCoordinator], ClimateEntity): | ||||
|     _attr_target_temperature_step = PRECISION_WHOLE | ||||
|     _attr_temperature_unit = UnitOfTemperature.CELSIUS | ||||
|  | ||||
|     def __init__( | ||||
|         self, | ||||
|         coordinator: AdaxCloudCoordinator, | ||||
|         device_id: str, | ||||
|     ) -> None: | ||||
|     def __init__(self, heater_data: dict[str, Any], adax_data_handler: Adax) -> None: | ||||
|         """Initialize the heater.""" | ||||
|         super().__init__(coordinator) | ||||
|         self._adax_data_handler: Adax = coordinator.adax_data_handler | ||||
|         self._device_id = device_id | ||||
|         self._device_id = heater_data["id"] | ||||
|         self._adax_data_handler = adax_data_handler | ||||
|  | ||||
|         self._attr_name = self.room["name"] | ||||
|         self._attr_unique_id = f"{self.room['homeId']}_{self._device_id}" | ||||
|         self._attr_unique_id = f"{heater_data['homeId']}_{heater_data['id']}" | ||||
|         self._attr_device_info = DeviceInfo( | ||||
|             identifiers={(DOMAIN, device_id)}, | ||||
|             identifiers={(DOMAIN, heater_data["id"])}, | ||||
|             # Instead of setting the device name to the entity name, adax | ||||
|             # should be updated to set has_entity_name = True, and set the entity | ||||
|             # name to None | ||||
|             name=cast(str | None, self.name), | ||||
|             manufacturer="Adax", | ||||
|         ) | ||||
|         self._apply_data(self.room) | ||||
|  | ||||
|     @property | ||||
|     def available(self) -> bool: | ||||
|         """Whether the entity is available or not.""" | ||||
|         return super().available and self._device_id in self.coordinator.data | ||||
|  | ||||
|     @property | ||||
|     def room(self) -> dict[str, Any]: | ||||
|         """Gets the data for this particular device.""" | ||||
|         return self.coordinator.data[self._device_id] | ||||
|  | ||||
|     async def async_set_hvac_mode(self, hvac_mode: HVACMode) -> None: | ||||
|         """Set hvac mode.""" | ||||
| @@ -106,9 +104,7 @@ class AdaxDevice(CoordinatorEntity[AdaxCloudCoordinator], ClimateEntity): | ||||
|             ) | ||||
|         else: | ||||
|             return | ||||
|  | ||||
|         # Request data refresh from source to verify that update was successful | ||||
|         await self.coordinator.async_request_refresh() | ||||
|         await self._adax_data_handler.update() | ||||
|  | ||||
|     async def async_set_temperature(self, **kwargs: Any) -> None: | ||||
|         """Set new target temperature.""" | ||||
| @@ -118,31 +114,28 @@ class AdaxDevice(CoordinatorEntity[AdaxCloudCoordinator], ClimateEntity): | ||||
|             self._device_id, temperature, True | ||||
|         ) | ||||
|  | ||||
|     @callback | ||||
|     def _handle_coordinator_update(self) -> None: | ||||
|         """Handle updated data from the coordinator.""" | ||||
|         if room := self.room: | ||||
|             self._apply_data(room) | ||||
|         super()._handle_coordinator_update() | ||||
|  | ||||
|     def _apply_data(self, room: dict[str, Any]) -> None: | ||||
|         """Update the appropriate attributues based on received data.""" | ||||
|         self._attr_current_temperature = room.get("temperature") | ||||
|         self._attr_target_temperature = room.get("targetTemperature") | ||||
|         if room["heatingEnabled"]: | ||||
|             self._attr_hvac_mode = HVACMode.HEAT | ||||
|             self._attr_icon = "mdi:radiator" | ||||
|         else: | ||||
|             self._attr_hvac_mode = HVACMode.OFF | ||||
|             self._attr_icon = "mdi:radiator-off" | ||||
|     async def async_update(self) -> None: | ||||
|         """Get the latest data.""" | ||||
|         for room in await self._adax_data_handler.get_rooms(): | ||||
|             if room["id"] != self._device_id: | ||||
|                 continue | ||||
|             self._attr_name = room["name"] | ||||
|             self._attr_current_temperature = room.get("temperature") | ||||
|             self._attr_target_temperature = room.get("targetTemperature") | ||||
|             if room["heatingEnabled"]: | ||||
|                 self._attr_hvac_mode = HVACMode.HEAT | ||||
|                 self._attr_icon = "mdi:radiator" | ||||
|             else: | ||||
|                 self._attr_hvac_mode = HVACMode.OFF | ||||
|                 self._attr_icon = "mdi:radiator-off" | ||||
|             return | ||||
|  | ||||
|  | ||||
| class LocalAdaxDevice(CoordinatorEntity[AdaxLocalCoordinator], ClimateEntity): | ||||
| class LocalAdaxDevice(ClimateEntity): | ||||
|     """Representation of a heater.""" | ||||
|  | ||||
|     _attr_hvac_modes = [HVACMode.HEAT, HVACMode.OFF] | ||||
|     _attr_hvac_mode = HVACMode.OFF | ||||
|     _attr_icon = "mdi:radiator-off" | ||||
|     _attr_hvac_mode = HVACMode.HEAT | ||||
|     _attr_max_temp = 35 | ||||
|     _attr_min_temp = 5 | ||||
|     _attr_supported_features = ( | ||||
| @@ -153,10 +146,9 @@ class LocalAdaxDevice(CoordinatorEntity[AdaxLocalCoordinator], ClimateEntity): | ||||
|     _attr_target_temperature_step = PRECISION_WHOLE | ||||
|     _attr_temperature_unit = UnitOfTemperature.CELSIUS | ||||
|  | ||||
|     def __init__(self, coordinator: AdaxLocalCoordinator, unique_id: str) -> None: | ||||
|     def __init__(self, adax_data_handler: AdaxLocal, unique_id: str) -> None: | ||||
|         """Initialize the heater.""" | ||||
|         super().__init__(coordinator) | ||||
|         self._adax_data_handler: AdaxLocal = coordinator.adax_data_handler | ||||
|         self._adax_data_handler = adax_data_handler | ||||
|         self._attr_unique_id = unique_id | ||||
|         self._attr_device_info = DeviceInfo( | ||||
|             identifiers={(DOMAIN, unique_id)}, | ||||
| @@ -177,20 +169,17 @@ class LocalAdaxDevice(CoordinatorEntity[AdaxLocalCoordinator], ClimateEntity): | ||||
|             return | ||||
|         await self._adax_data_handler.set_target_temperature(temperature) | ||||
|  | ||||
|     @callback | ||||
|     def _handle_coordinator_update(self) -> None: | ||||
|         """Handle updated data from the coordinator.""" | ||||
|         if data := self.coordinator.data: | ||||
|             self._attr_current_temperature = data["current_temperature"] | ||||
|             self._attr_available = self._attr_current_temperature is not None | ||||
|             if (target_temp := data["target_temperature"]) == 0: | ||||
|                 self._attr_hvac_mode = HVACMode.OFF | ||||
|                 self._attr_icon = "mdi:radiator-off" | ||||
|                 if target_temp == 0: | ||||
|                     self._attr_target_temperature = self._attr_min_temp | ||||
|             else: | ||||
|                 self._attr_hvac_mode = HVACMode.HEAT | ||||
|                 self._attr_icon = "mdi:radiator" | ||||
|                 self._attr_target_temperature = target_temp | ||||
|  | ||||
|         super()._handle_coordinator_update() | ||||
|     async def async_update(self) -> None: | ||||
|         """Get the latest data.""" | ||||
|         data = await self._adax_data_handler.get_status() | ||||
|         self._attr_current_temperature = data["current_temperature"] | ||||
|         self._attr_available = self._attr_current_temperature is not None | ||||
|         if (target_temp := data["target_temperature"]) == 0: | ||||
|             self._attr_hvac_mode = HVACMode.OFF | ||||
|             self._attr_icon = "mdi:radiator-off" | ||||
|             if target_temp == 0: | ||||
|                 self._attr_target_temperature = self._attr_min_temp | ||||
|         else: | ||||
|             self._attr_hvac_mode = HVACMode.HEAT | ||||
|             self._attr_icon = "mdi:radiator" | ||||
|             self._attr_target_temperature = target_temp | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| """Constants for the Adax integration.""" | ||||
|  | ||||
| import datetime | ||||
| from typing import Final | ||||
|  | ||||
| ACCOUNT_ID: Final = "account_id" | ||||
| @@ -10,5 +9,3 @@ DOMAIN: Final = "adax" | ||||
| LOCAL = "Local" | ||||
| WIFI_SSID = "wifi_ssid" | ||||
| WIFI_PSWD = "wifi_pswd" | ||||
|  | ||||
| SCAN_INTERVAL = datetime.timedelta(seconds=60) | ||||
|   | ||||
| @@ -1,94 +0,0 @@ | ||||
| """DataUpdateCoordinator for the Adax component.""" | ||||
|  | ||||
| import logging | ||||
| from typing import Any, cast | ||||
|  | ||||
| from adax import Adax | ||||
| from adax_local import Adax as AdaxLocal | ||||
|  | ||||
| from homeassistant.config_entries import ConfigEntry | ||||
| from homeassistant.const import CONF_IP_ADDRESS, CONF_PASSWORD, CONF_TOKEN | ||||
| from homeassistant.core import HomeAssistant | ||||
| from homeassistant.helpers.aiohttp_client import async_get_clientsession | ||||
| from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed | ||||
|  | ||||
| from .const import ACCOUNT_ID, SCAN_INTERVAL | ||||
|  | ||||
| _LOGGER = logging.getLogger(__name__) | ||||
|  | ||||
|  | ||||
| type AdaxConfigEntry = ConfigEntry[AdaxCloudCoordinator | AdaxLocalCoordinator] | ||||
|  | ||||
|  | ||||
| class AdaxCloudCoordinator(DataUpdateCoordinator[dict[str, dict[str, Any]]]): | ||||
|     """Coordinator for updating data to and from Adax (cloud).""" | ||||
|  | ||||
|     def __init__(self, hass: HomeAssistant, entry: AdaxConfigEntry) -> None: | ||||
|         """Initialize the Adax coordinator used for Cloud mode.""" | ||||
|         super().__init__( | ||||
|             hass, | ||||
|             config_entry=entry, | ||||
|             logger=_LOGGER, | ||||
|             name="AdaxCloud", | ||||
|             update_interval=SCAN_INTERVAL, | ||||
|         ) | ||||
|  | ||||
|         self.adax_data_handler = Adax( | ||||
|             entry.data[ACCOUNT_ID], | ||||
|             entry.data[CONF_PASSWORD], | ||||
|             websession=async_get_clientsession(hass), | ||||
|         ) | ||||
|  | ||||
|     async def _async_update_data(self) -> dict[str, dict[str, Any]]: | ||||
|         """Fetch data from the Adax.""" | ||||
|         try: | ||||
|             if hasattr(self.adax_data_handler, "fetch_rooms_info"): | ||||
|                 rooms = await self.adax_data_handler.fetch_rooms_info() or [] | ||||
|                 _LOGGER.debug("fetch_rooms_info returned: %s", rooms) | ||||
|             else: | ||||
|                 _LOGGER.debug("fetch_rooms_info method not available, using get_rooms") | ||||
|                 rooms = [] | ||||
|  | ||||
|             if not rooms: | ||||
|                 _LOGGER.debug( | ||||
|                     "No rooms from fetch_rooms_info, trying get_rooms as fallback" | ||||
|                 ) | ||||
|                 rooms = await self.adax_data_handler.get_rooms() or [] | ||||
|                 _LOGGER.debug("get_rooms fallback returned: %s", rooms) | ||||
|  | ||||
|             if not rooms: | ||||
|                 raise UpdateFailed("No rooms available from Adax API") | ||||
|  | ||||
|         except OSError as e: | ||||
|             raise UpdateFailed(f"Error communicating with API: {e}") from e | ||||
|  | ||||
|         for room in rooms: | ||||
|             room["energyWh"] = int(room.get("energyWh", 0)) | ||||
|  | ||||
|         return {r["id"]: r for r in rooms} | ||||
|  | ||||
|  | ||||
| class AdaxLocalCoordinator(DataUpdateCoordinator[dict[str, Any] | None]): | ||||
|     """Coordinator for updating data to and from Adax (local).""" | ||||
|  | ||||
|     def __init__(self, hass: HomeAssistant, entry: AdaxConfigEntry) -> None: | ||||
|         """Initialize the Adax coordinator used for Local mode.""" | ||||
|         super().__init__( | ||||
|             hass, | ||||
|             config_entry=entry, | ||||
|             logger=_LOGGER, | ||||
|             name="AdaxLocal", | ||||
|             update_interval=SCAN_INTERVAL, | ||||
|         ) | ||||
|  | ||||
|         self.adax_data_handler = AdaxLocal( | ||||
|             entry.data[CONF_IP_ADDRESS], | ||||
|             entry.data[CONF_TOKEN], | ||||
|             websession=async_get_clientsession(hass, verify_ssl=False), | ||||
|         ) | ||||
|  | ||||
|     async def _async_update_data(self) -> dict[str, Any]: | ||||
|         """Fetch data from the Adax.""" | ||||
|         if result := await self.adax_data_handler.get_status(): | ||||
|             return cast(dict[str, Any], result) | ||||
|         raise UpdateFailed("Got invalid status from device") | ||||
| @@ -1,7 +1,7 @@ | ||||
| { | ||||
|   "domain": "adax", | ||||
|   "name": "Adax", | ||||
|   "codeowners": ["@danielhiversen", "@lazytarget"], | ||||
|   "codeowners": ["@danielhiversen"], | ||||
|   "config_flow": true, | ||||
|   "documentation": "https://www.home-assistant.io/integrations/adax", | ||||
|   "iot_class": "local_polling", | ||||
|   | ||||
| @@ -1,77 +0,0 @@ | ||||
| """Support for Adax energy sensors.""" | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| from typing import cast | ||||
|  | ||||
| from homeassistant.components.sensor import ( | ||||
|     SensorDeviceClass, | ||||
|     SensorEntity, | ||||
|     SensorStateClass, | ||||
| ) | ||||
| from homeassistant.const import UnitOfEnergy | ||||
| from homeassistant.core import HomeAssistant | ||||
| from homeassistant.helpers.device_registry import DeviceInfo | ||||
| from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | ||||
| from homeassistant.helpers.update_coordinator import CoordinatorEntity | ||||
|  | ||||
| from . import AdaxConfigEntry | ||||
| from .const import CONNECTION_TYPE, DOMAIN, LOCAL | ||||
| from .coordinator import AdaxCloudCoordinator | ||||
|  | ||||
|  | ||||
| async def async_setup_entry( | ||||
|     hass: HomeAssistant, | ||||
|     entry: AdaxConfigEntry, | ||||
|     async_add_entities: AddConfigEntryEntitiesCallback, | ||||
| ) -> None: | ||||
|     """Set up the Adax energy sensors with config flow.""" | ||||
|     if entry.data.get(CONNECTION_TYPE) != LOCAL: | ||||
|         cloud_coordinator = cast(AdaxCloudCoordinator, entry.runtime_data) | ||||
|  | ||||
|         # Create individual energy sensors for each device | ||||
|         async_add_entities( | ||||
|             AdaxEnergySensor(cloud_coordinator, device_id) | ||||
|             for device_id in cloud_coordinator.data | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class AdaxEnergySensor(CoordinatorEntity[AdaxCloudCoordinator], SensorEntity): | ||||
|     """Representation of an Adax energy sensor.""" | ||||
|  | ||||
|     _attr_has_entity_name = True | ||||
|     _attr_translation_key = "energy" | ||||
|     _attr_device_class = SensorDeviceClass.ENERGY | ||||
|     _attr_native_unit_of_measurement = UnitOfEnergy.WATT_HOUR | ||||
|     _attr_suggested_unit_of_measurement = UnitOfEnergy.KILO_WATT_HOUR | ||||
|     _attr_state_class = SensorStateClass.TOTAL_INCREASING | ||||
|     _attr_suggested_display_precision = 3 | ||||
|  | ||||
|     def __init__( | ||||
|         self, | ||||
|         coordinator: AdaxCloudCoordinator, | ||||
|         device_id: str, | ||||
|     ) -> None: | ||||
|         """Initialize the energy sensor.""" | ||||
|         super().__init__(coordinator) | ||||
|         self._device_id = device_id | ||||
|         room = coordinator.data[device_id] | ||||
|  | ||||
|         self._attr_unique_id = f"{room['homeId']}_{device_id}_energy" | ||||
|         self._attr_device_info = DeviceInfo( | ||||
|             identifiers={(DOMAIN, device_id)}, | ||||
|             name=room["name"], | ||||
|             manufacturer="Adax", | ||||
|         ) | ||||
|  | ||||
|     @property | ||||
|     def available(self) -> bool: | ||||
|         """Return True if entity is available.""" | ||||
|         return ( | ||||
|             super().available and "energyWh" in self.coordinator.data[self._device_id] | ||||
|         ) | ||||
|  | ||||
|     @property | ||||
|     def native_value(self) -> int: | ||||
|         """Return the native value of the sensor.""" | ||||
|         return int(self.coordinator.data[self._device_id]["energyWh"]) | ||||
| @@ -8,7 +8,7 @@ from homeassistant.helpers.device_registry import DeviceInfo | ||||
| from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | ||||
|  | ||||
| from . import AdvantageAirDataConfigEntry | ||||
| from .const import ADVANTAGE_AIR_STATE_ON, DOMAIN | ||||
| from .const import ADVANTAGE_AIR_STATE_ON, DOMAIN as ADVANTAGE_AIR_DOMAIN | ||||
| from .entity import AdvantageAirEntity, AdvantageAirThingEntity | ||||
| from .models import AdvantageAirData | ||||
|  | ||||
| @@ -52,8 +52,8 @@ class AdvantageAirLight(AdvantageAirEntity, LightEntity): | ||||
|         self._id: str = light["id"] | ||||
|         self._attr_unique_id += f"-{self._id}" | ||||
|         self._attr_device_info = DeviceInfo( | ||||
|             identifiers={(DOMAIN, self._attr_unique_id)}, | ||||
|             via_device=(DOMAIN, self.coordinator.data["system"]["rid"]), | ||||
|             identifiers={(ADVANTAGE_AIR_DOMAIN, self._attr_unique_id)}, | ||||
|             via_device=(ADVANTAGE_AIR_DOMAIN, self.coordinator.data["system"]["rid"]), | ||||
|             manufacturer="Advantage Air", | ||||
|             model=light.get("moduleType"), | ||||
|             name=light["name"], | ||||
|   | ||||
| @@ -6,7 +6,7 @@ from homeassistant.helpers.device_registry import DeviceInfo | ||||
| from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | ||||
|  | ||||
| from . import AdvantageAirDataConfigEntry | ||||
| from .const import DOMAIN | ||||
| from .const import DOMAIN as ADVANTAGE_AIR_DOMAIN | ||||
| from .entity import AdvantageAirEntity | ||||
| from .models import AdvantageAirData | ||||
|  | ||||
| @@ -32,7 +32,9 @@ class AdvantageAirApp(AdvantageAirEntity, UpdateEntity): | ||||
|         """Initialize the Advantage Air App.""" | ||||
|         super().__init__(instance) | ||||
|         self._attr_device_info = DeviceInfo( | ||||
|             identifiers={(DOMAIN, self.coordinator.data["system"]["rid"])}, | ||||
|             identifiers={ | ||||
|                 (ADVANTAGE_AIR_DOMAIN, self.coordinator.data["system"]["rid"]) | ||||
|             }, | ||||
|             manufacturer="Advantage Air", | ||||
|             model=self.coordinator.data["system"]["sysType"], | ||||
|             name=self.coordinator.data["system"]["name"], | ||||
|   | ||||
| @@ -71,14 +71,7 @@ class AemetConfigFlow(ConfigFlow, domain=DOMAIN): | ||||
|             } | ||||
|         ) | ||||
|  | ||||
|         return self.async_show_form( | ||||
|             step_id="user", | ||||
|             data_schema=schema, | ||||
|             errors=errors, | ||||
|             description_placeholders={ | ||||
|                 "api_key_url": "https://opendata.aemet.es/centrodedescargas/altaUsuario" | ||||
|             }, | ||||
|         ) | ||||
|         return self.async_show_form(step_id="user", data_schema=schema, errors=errors) | ||||
|  | ||||
|     @staticmethod | ||||
|     @callback | ||||
|   | ||||
| @@ -185,7 +185,6 @@ FORECAST_SENSORS: Final[tuple[AemetSensorEntityDescription, ...]] = ( | ||||
|         keys=[AOD_TOWN, AOD_FORECAST_DAILY, AOD_FORECAST_CURRENT, AOD_WIND_DIRECTION], | ||||
|         name="Daily forecast wind bearing", | ||||
|         native_unit_of_measurement=DEGREE, | ||||
|         device_class=SensorDeviceClass.WIND_DIRECTION, | ||||
|     ), | ||||
|     AemetSensorEntityDescription( | ||||
|         entity_registry_enabled_default=False, | ||||
| @@ -193,7 +192,6 @@ FORECAST_SENSORS: Final[tuple[AemetSensorEntityDescription, ...]] = ( | ||||
|         keys=[AOD_TOWN, AOD_FORECAST_HOURLY, AOD_FORECAST_CURRENT, AOD_WIND_DIRECTION], | ||||
|         name="Hourly forecast wind bearing", | ||||
|         native_unit_of_measurement=DEGREE, | ||||
|         device_class=SensorDeviceClass.WIND_DIRECTION, | ||||
|     ), | ||||
|     AemetSensorEntityDescription( | ||||
|         entity_registry_enabled_default=False, | ||||
| @@ -336,8 +334,7 @@ WEATHER_SENSORS: Final[tuple[AemetSensorEntityDescription, ...]] = ( | ||||
|         keys=[AOD_WEATHER, AOD_WIND_DIRECTION], | ||||
|         name="Wind bearing", | ||||
|         native_unit_of_measurement=DEGREE, | ||||
|         state_class=SensorStateClass.MEASUREMENT_ANGLE, | ||||
|         device_class=SensorDeviceClass.WIND_DIRECTION, | ||||
|         state_class=SensorStateClass.MEASUREMENT, | ||||
|     ), | ||||
|     AemetSensorEntityDescription( | ||||
|         key=ATTR_API_WIND_MAX_SPEED, | ||||
|   | ||||
| @@ -14,7 +14,7 @@ | ||||
|           "longitude": "[%key:common::config_flow::data::longitude%]", | ||||
|           "name": "Name of the integration" | ||||
|         }, | ||||
|         "description": "To generate API key go to {api_key_url}" | ||||
|         "description": "To generate API key go to https://opendata.aemet.es/centrodedescargas/altaUsuario" | ||||
|       } | ||||
|     } | ||||
|   }, | ||||
|   | ||||
| @@ -10,7 +10,7 @@ from homeassistant.exceptions import ConfigEntryNotReady | ||||
| from homeassistant.helpers import device_registry as dr | ||||
| from homeassistant.helpers.aiohttp_client import async_get_clientsession | ||||
|  | ||||
| from .const import DOMAIN, SERVER_URL | ||||
| from .const import DOMAIN as AGENT_DOMAIN, SERVER_URL | ||||
|  | ||||
| ATTRIBUTION = "ispyconnect.com" | ||||
| DEFAULT_BRAND = "Agent DVR by ispyconnect.com" | ||||
| @@ -46,7 +46,7 @@ async def async_setup_entry( | ||||
|  | ||||
|     device_registry.async_get_or_create( | ||||
|         config_entry_id=config_entry.entry_id, | ||||
|         identifiers={(DOMAIN, agent_client.unique)}, | ||||
|         identifiers={(AGENT_DOMAIN, agent_client.unique)}, | ||||
|         manufacturer="iSpyConnect", | ||||
|         name=f"Agent {agent_client.name}", | ||||
|         model="Agent DVR", | ||||
|   | ||||
| @@ -12,7 +12,7 @@ from homeassistant.helpers.device_registry import DeviceInfo | ||||
| from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback | ||||
|  | ||||
| from . import AgentDVRConfigEntry | ||||
| from .const import DOMAIN | ||||
| from .const import DOMAIN as AGENT_DOMAIN | ||||
|  | ||||
| CONF_HOME_MODE_NAME = "home" | ||||
| CONF_AWAY_MODE_NAME = "away" | ||||
| @@ -47,7 +47,7 @@ class AgentBaseStation(AlarmControlPanelEntity): | ||||
|         self._client = client | ||||
|         self._attr_unique_id = f"{client.unique}_CP" | ||||
|         self._attr_device_info = DeviceInfo( | ||||
|             identifiers={(DOMAIN, client.unique)}, | ||||
|             identifiers={(AGENT_DOMAIN, client.unique)}, | ||||
|             name=f"{client.name} {CONST_ALARM_CONTROL_PANEL_NAME}", | ||||
|             manufacturer="Agent", | ||||
|             model=CONST_ALARM_CONTROL_PANEL_NAME, | ||||
|   | ||||
| @@ -15,7 +15,7 @@ from homeassistant.helpers.entity_platform import ( | ||||
| ) | ||||
|  | ||||
| from . import AgentDVRConfigEntry | ||||
| from .const import ATTRIBUTION, CAMERA_SCAN_INTERVAL_SECS, DOMAIN | ||||
| from .const import ATTRIBUTION, CAMERA_SCAN_INTERVAL_SECS, DOMAIN as AGENT_DOMAIN | ||||
|  | ||||
| SCAN_INTERVAL = timedelta(seconds=CAMERA_SCAN_INTERVAL_SECS) | ||||
|  | ||||
| @@ -82,7 +82,7 @@ class AgentCamera(MjpegCamera): | ||||
|             still_image_url=f"{device.client._server_url}{device.still_image_url}&size={device.mjpegStreamWidth}x{device.mjpegStreamHeight}",  # noqa: SLF001 | ||||
|         ) | ||||
|         self._attr_device_info = DeviceInfo( | ||||
|             identifiers={(DOMAIN, self.unique_id)}, | ||||
|             identifiers={(AGENT_DOMAIN, self.unique_id)}, | ||||
|             manufacturer="Agent", | ||||
|             model="Camera", | ||||
|             name=f"{device.client.name} {device.name}", | ||||
|   | ||||
| @@ -1,204 +0,0 @@ | ||||
| """Integration to offer AI tasks to Home Assistant.""" | ||||
|  | ||||
| import logging | ||||
| from typing import Any | ||||
|  | ||||
| import voluptuous as vol | ||||
|  | ||||
| from homeassistant.config_entries import ConfigEntry | ||||
| from homeassistant.const import ATTR_ENTITY_ID, CONF_DESCRIPTION, CONF_SELECTOR | ||||
| from homeassistant.core import ( | ||||
|     HassJobType, | ||||
|     HomeAssistant, | ||||
|     ServiceCall, | ||||
|     ServiceResponse, | ||||
|     SupportsResponse, | ||||
|     callback, | ||||
| ) | ||||
| from homeassistant.helpers import config_validation as cv, selector, storage | ||||
| from homeassistant.helpers.entity_component import EntityComponent | ||||
| from homeassistant.helpers.typing import UNDEFINED, ConfigType, UndefinedType | ||||
|  | ||||
| from .const import ( | ||||
|     ATTR_ATTACHMENTS, | ||||
|     ATTR_INSTRUCTIONS, | ||||
|     ATTR_REQUIRED, | ||||
|     ATTR_STRUCTURE, | ||||
|     ATTR_TASK_NAME, | ||||
|     DATA_COMPONENT, | ||||
|     DATA_PREFERENCES, | ||||
|     DOMAIN, | ||||
|     SERVICE_GENERATE_DATA, | ||||
|     SERVICE_GENERATE_IMAGE, | ||||
|     AITaskEntityFeature, | ||||
| ) | ||||
| from .entity import AITaskEntity | ||||
| from .http import async_setup as async_setup_http | ||||
| from .task import ( | ||||
|     GenDataTask, | ||||
|     GenDataTaskResult, | ||||
|     GenImageTask, | ||||
|     GenImageTaskResult, | ||||
|     async_generate_data, | ||||
|     async_generate_image, | ||||
| ) | ||||
|  | ||||
| __all__ = [ | ||||
|     "DOMAIN", | ||||
|     "AITaskEntity", | ||||
|     "AITaskEntityFeature", | ||||
|     "GenDataTask", | ||||
|     "GenDataTaskResult", | ||||
|     "GenImageTask", | ||||
|     "GenImageTaskResult", | ||||
|     "async_generate_data", | ||||
|     "async_generate_image", | ||||
|     "async_setup", | ||||
|     "async_setup_entry", | ||||
|     "async_unload_entry", | ||||
| ] | ||||
|  | ||||
| _LOGGER = logging.getLogger(__name__) | ||||
|  | ||||
| CONFIG_SCHEMA = cv.config_entry_only_config_schema(DOMAIN) | ||||
|  | ||||
| STRUCTURE_FIELD_SCHEMA = vol.Schema( | ||||
|     { | ||||
|         vol.Optional(CONF_DESCRIPTION): str, | ||||
|         vol.Optional(ATTR_REQUIRED): bool, | ||||
|         vol.Required(CONF_SELECTOR): selector.validate_selector, | ||||
|     } | ||||
| ) | ||||
|  | ||||
|  | ||||
| def _validate_structure_fields(value: dict[str, Any]) -> vol.Schema: | ||||
|     """Validate the structure fields as a voluptuous Schema.""" | ||||
|     if not isinstance(value, dict): | ||||
|         raise vol.Invalid("Structure must be a dictionary") | ||||
|     fields = {} | ||||
|     for k, v in value.items(): | ||||
|         field_class = vol.Required if v.get(ATTR_REQUIRED, False) else vol.Optional | ||||
|         fields[field_class(k, description=v.get(CONF_DESCRIPTION))] = selector.selector( | ||||
|             v[CONF_SELECTOR] | ||||
|         ) | ||||
|     return vol.Schema(fields, extra=vol.PREVENT_EXTRA) | ||||
|  | ||||
|  | ||||
| async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool: | ||||
|     """Register the process service.""" | ||||
|     entity_component = EntityComponent[AITaskEntity](_LOGGER, DOMAIN, hass) | ||||
|     hass.data[DATA_COMPONENT] = entity_component | ||||
|     hass.data[DATA_PREFERENCES] = AITaskPreferences(hass) | ||||
|     await hass.data[DATA_PREFERENCES].async_load() | ||||
|     async_setup_http(hass) | ||||
|     hass.services.async_register( | ||||
|         DOMAIN, | ||||
|         SERVICE_GENERATE_DATA, | ||||
|         async_service_generate_data, | ||||
|         schema=vol.Schema( | ||||
|             { | ||||
|                 vol.Required(ATTR_TASK_NAME): cv.string, | ||||
|                 vol.Optional(ATTR_ENTITY_ID): cv.entity_id, | ||||
|                 vol.Required(ATTR_INSTRUCTIONS): cv.string, | ||||
|                 vol.Optional(ATTR_STRUCTURE): vol.All( | ||||
|                     vol.Schema({str: STRUCTURE_FIELD_SCHEMA}), | ||||
|                     _validate_structure_fields, | ||||
|                 ), | ||||
|                 vol.Optional(ATTR_ATTACHMENTS): vol.All( | ||||
|                     cv.ensure_list, [selector.MediaSelector({"accept": ["*/*"]})] | ||||
|                 ), | ||||
|             } | ||||
|         ), | ||||
|         supports_response=SupportsResponse.ONLY, | ||||
|         job_type=HassJobType.Coroutinefunction, | ||||
|     ) | ||||
|     hass.services.async_register( | ||||
|         DOMAIN, | ||||
|         SERVICE_GENERATE_IMAGE, | ||||
|         async_service_generate_image, | ||||
|         schema=vol.Schema( | ||||
|             { | ||||
|                 vol.Required(ATTR_TASK_NAME): cv.string, | ||||
|                 vol.Optional(ATTR_ENTITY_ID): cv.entity_id, | ||||
|                 vol.Required(ATTR_INSTRUCTIONS): cv.string, | ||||
|                 vol.Optional(ATTR_ATTACHMENTS): vol.All( | ||||
|                     cv.ensure_list, [selector.MediaSelector({"accept": ["*/*"]})] | ||||
|                 ), | ||||
|             } | ||||
|         ), | ||||
|         supports_response=SupportsResponse.ONLY, | ||||
|         job_type=HassJobType.Coroutinefunction, | ||||
|     ) | ||||
|     return True | ||||
|  | ||||
|  | ||||
| async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: | ||||
|     """Set up a config entry.""" | ||||
|     return await hass.data[DATA_COMPONENT].async_setup_entry(entry) | ||||
|  | ||||
|  | ||||
| async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: | ||||
|     """Unload a config entry.""" | ||||
|     return await hass.data[DATA_COMPONENT].async_unload_entry(entry) | ||||
|  | ||||
|  | ||||
| async def async_service_generate_data(call: ServiceCall) -> ServiceResponse: | ||||
|     """Run the data task service.""" | ||||
|     result = await async_generate_data(hass=call.hass, **call.data) | ||||
|     return result.as_dict() | ||||
|  | ||||
|  | ||||
| async def async_service_generate_image(call: ServiceCall) -> ServiceResponse: | ||||
|     """Run the image task service.""" | ||||
|     return await async_generate_image(hass=call.hass, **call.data) | ||||
|  | ||||
|  | ||||
| class AITaskPreferences: | ||||
|     """AI Task preferences.""" | ||||
|  | ||||
|     KEYS = ("gen_data_entity_id", "gen_image_entity_id") | ||||
|  | ||||
|     gen_data_entity_id: str | None = None | ||||
|     gen_image_entity_id: str | None = None | ||||
|  | ||||
|     def __init__(self, hass: HomeAssistant) -> None: | ||||
|         """Initialize the preferences.""" | ||||
|         self._store: storage.Store[dict[str, str | None]] = storage.Store( | ||||
|             hass, 1, DOMAIN | ||||
|         ) | ||||
|  | ||||
|     async def async_load(self) -> None: | ||||
|         """Load the data from the store.""" | ||||
|         data = await self._store.async_load() | ||||
|         if data is None: | ||||
|             return | ||||
|         for key in self.KEYS: | ||||
|             setattr(self, key, data.get(key)) | ||||
|  | ||||
|     @callback | ||||
|     def async_set_preferences( | ||||
|         self, | ||||
|         *, | ||||
|         gen_data_entity_id: str | None | UndefinedType = UNDEFINED, | ||||
|         gen_image_entity_id: str | None | UndefinedType = UNDEFINED, | ||||
|     ) -> None: | ||||
|         """Set the preferences.""" | ||||
|         changed = False | ||||
|         for key, value in ( | ||||
|             ("gen_data_entity_id", gen_data_entity_id), | ||||
|             ("gen_image_entity_id", gen_image_entity_id), | ||||
|         ): | ||||
|             if value is not UNDEFINED: | ||||
|                 if getattr(self, key) != value: | ||||
|                     setattr(self, key, value) | ||||
|                     changed = True | ||||
|  | ||||
|         if not changed: | ||||
|             return | ||||
|  | ||||
|         self._store.async_delay_save(self.as_dict, 10) | ||||
|  | ||||
|     @callback | ||||
|     def as_dict(self) -> dict[str, str | None]: | ||||
|         """Get the current preferences.""" | ||||
|         return {key: getattr(self, key) for key in self.KEYS} | ||||
| @@ -1,49 +0,0 @@ | ||||
| """Constants for the AI Task integration.""" | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| from enum import IntFlag | ||||
| from typing import TYPE_CHECKING, Final | ||||
|  | ||||
| from homeassistant.util.hass_dict import HassKey | ||||
|  | ||||
| if TYPE_CHECKING: | ||||
|     from homeassistant.components.media_source import local_source | ||||
|     from homeassistant.helpers.entity_component import EntityComponent | ||||
|  | ||||
|     from . import AITaskPreferences | ||||
|     from .entity import AITaskEntity | ||||
|  | ||||
| DOMAIN = "ai_task" | ||||
| DATA_COMPONENT: HassKey[EntityComponent[AITaskEntity]] = HassKey(DOMAIN) | ||||
| DATA_PREFERENCES: HassKey[AITaskPreferences] = HassKey(f"{DOMAIN}_preferences") | ||||
| DATA_MEDIA_SOURCE: HassKey[local_source.LocalSource] = HassKey(f"{DOMAIN}_media_source") | ||||
|  | ||||
| IMAGE_DIR: Final = "image" | ||||
| IMAGE_EXPIRY_TIME = 60 * 60  # 1 hour | ||||
|  | ||||
| SERVICE_GENERATE_DATA = "generate_data" | ||||
| SERVICE_GENERATE_IMAGE = "generate_image" | ||||
|  | ||||
| ATTR_INSTRUCTIONS: Final = "instructions" | ||||
| ATTR_TASK_NAME: Final = "task_name" | ||||
| ATTR_STRUCTURE: Final = "structure" | ||||
| ATTR_REQUIRED: Final = "required" | ||||
| ATTR_ATTACHMENTS: Final = "attachments" | ||||
|  | ||||
| DEFAULT_SYSTEM_PROMPT = ( | ||||
|     "You are a Home Assistant expert and help users with their tasks." | ||||
| ) | ||||
|  | ||||
|  | ||||
| class AITaskEntityFeature(IntFlag): | ||||
|     """Supported features of the AI task entity.""" | ||||
|  | ||||
|     GENERATE_DATA = 1 | ||||
|     """Generate data based on instructions.""" | ||||
|  | ||||
|     SUPPORT_ATTACHMENTS = 2 | ||||
|     """Support attachments with generate data.""" | ||||
|  | ||||
|     GENERATE_IMAGE = 4 | ||||
|     """Generate images based on instructions.""" | ||||
| @@ -1,131 +0,0 @@ | ||||
| """Entity for the AI Task integration.""" | ||||
|  | ||||
| from collections.abc import AsyncGenerator | ||||
| import contextlib | ||||
| from typing import final | ||||
|  | ||||
| from propcache.api import cached_property | ||||
|  | ||||
| from homeassistant.components.conversation import ( | ||||
|     ChatLog, | ||||
|     UserContent, | ||||
|     async_get_chat_log, | ||||
| ) | ||||
| from homeassistant.const import STATE_UNAVAILABLE, STATE_UNKNOWN | ||||
| from homeassistant.helpers import llm | ||||
| from homeassistant.helpers.chat_session import ChatSession | ||||
| from homeassistant.helpers.restore_state import RestoreEntity | ||||
| from homeassistant.util import dt as dt_util | ||||
|  | ||||
| from .const import DEFAULT_SYSTEM_PROMPT, DOMAIN, AITaskEntityFeature | ||||
| from .task import GenDataTask, GenDataTaskResult, GenImageTask, GenImageTaskResult | ||||
|  | ||||
|  | ||||
| class AITaskEntity(RestoreEntity): | ||||
|     """Entity that supports conversations.""" | ||||
|  | ||||
|     _attr_should_poll = False | ||||
|     _attr_supported_features = AITaskEntityFeature(0) | ||||
|     __last_activity: str | None = None | ||||
|  | ||||
|     @property | ||||
|     @final | ||||
|     def state(self) -> str | None: | ||||
|         """Return the state of the entity.""" | ||||
|         if self.__last_activity is None: | ||||
|             return None | ||||
|         return self.__last_activity | ||||
|  | ||||
|     @cached_property | ||||
|     def supported_features(self) -> AITaskEntityFeature: | ||||
|         """Flag supported features.""" | ||||
|         return self._attr_supported_features | ||||
|  | ||||
|     async def async_internal_added_to_hass(self) -> None: | ||||
|         """Call when the entity is added to hass.""" | ||||
|         await super().async_internal_added_to_hass() | ||||
|         state = await self.async_get_last_state() | ||||
|         if ( | ||||
|             state is not None | ||||
|             and state.state is not None | ||||
|             and state.state not in (STATE_UNAVAILABLE, STATE_UNKNOWN) | ||||
|         ): | ||||
|             self.__last_activity = state.state | ||||
|  | ||||
|     @final | ||||
|     @contextlib.asynccontextmanager | ||||
|     async def _async_get_ai_task_chat_log( | ||||
|         self, | ||||
|         session: ChatSession, | ||||
|         task: GenDataTask | GenImageTask, | ||||
|     ) -> AsyncGenerator[ChatLog]: | ||||
|         """Context manager used to manage the ChatLog used during an AI Task.""" | ||||
|         user_llm_hass_api: llm.API | None = None | ||||
|         if isinstance(task, GenDataTask): | ||||
|             user_llm_hass_api = task.llm_api | ||||
|  | ||||
|         # pylint: disable-next=contextmanager-generator-missing-cleanup | ||||
|         with ( | ||||
|             async_get_chat_log( | ||||
|                 self.hass, | ||||
|                 session, | ||||
|                 None, | ||||
|             ) as chat_log, | ||||
|         ): | ||||
|             await chat_log.async_provide_llm_data( | ||||
|                 llm.LLMContext( | ||||
|                     platform=self.platform.domain, | ||||
|                     context=None, | ||||
|                     language=None, | ||||
|                     assistant=DOMAIN, | ||||
|                     device_id=None, | ||||
|                 ), | ||||
|                 user_llm_prompt=DEFAULT_SYSTEM_PROMPT, | ||||
|                 user_llm_hass_api=user_llm_hass_api, | ||||
|             ) | ||||
|  | ||||
|             chat_log.async_add_user_content( | ||||
|                 UserContent(task.instructions, attachments=task.attachments) | ||||
|             ) | ||||
|  | ||||
|             yield chat_log | ||||
|  | ||||
|     @final | ||||
|     async def internal_async_generate_data( | ||||
|         self, | ||||
|         session: ChatSession, | ||||
|         task: GenDataTask, | ||||
|     ) -> GenDataTaskResult: | ||||
|         """Run a gen data task.""" | ||||
|         self.__last_activity = dt_util.utcnow().isoformat() | ||||
|         self.async_write_ha_state() | ||||
|         async with self._async_get_ai_task_chat_log(session, task) as chat_log: | ||||
|             return await self._async_generate_data(task, chat_log) | ||||
|  | ||||
|     async def _async_generate_data( | ||||
|         self, | ||||
|         task: GenDataTask, | ||||
|         chat_log: ChatLog, | ||||
|     ) -> GenDataTaskResult: | ||||
|         """Handle a gen data task.""" | ||||
|         raise NotImplementedError | ||||
|  | ||||
|     @final | ||||
|     async def internal_async_generate_image( | ||||
|         self, | ||||
|         session: ChatSession, | ||||
|         task: GenImageTask, | ||||
|     ) -> GenImageTaskResult: | ||||
|         """Run a gen image task.""" | ||||
|         self.__last_activity = dt_util.utcnow().isoformat() | ||||
|         self.async_write_ha_state() | ||||
|         async with self._async_get_ai_task_chat_log(session, task) as chat_log: | ||||
|             return await self._async_generate_image(task, chat_log) | ||||
|  | ||||
|     async def _async_generate_image( | ||||
|         self, | ||||
|         task: GenImageTask, | ||||
|         chat_log: ChatLog, | ||||
|     ) -> GenImageTaskResult: | ||||
|         """Handle a gen image task.""" | ||||
|         raise NotImplementedError | ||||
| @@ -1,55 +0,0 @@ | ||||
| """HTTP endpoint for AI Task integration.""" | ||||
|  | ||||
| from typing import Any | ||||
|  | ||||
| import voluptuous as vol | ||||
|  | ||||
| from homeassistant.components import websocket_api | ||||
| from homeassistant.core import HomeAssistant, callback | ||||
|  | ||||
| from .const import DATA_PREFERENCES | ||||
|  | ||||
|  | ||||
| @callback | ||||
| def async_setup(hass: HomeAssistant) -> None: | ||||
|     """Set up the HTTP API for the conversation integration.""" | ||||
|     websocket_api.async_register_command(hass, websocket_get_preferences) | ||||
|     websocket_api.async_register_command(hass, websocket_set_preferences) | ||||
|  | ||||
|  | ||||
| @websocket_api.websocket_command( | ||||
|     { | ||||
|         vol.Required("type"): "ai_task/preferences/get", | ||||
|     } | ||||
| ) | ||||
| @callback | ||||
| def websocket_get_preferences( | ||||
|     hass: HomeAssistant, | ||||
|     connection: websocket_api.ActiveConnection, | ||||
|     msg: dict[str, Any], | ||||
| ) -> None: | ||||
|     """Get AI task preferences.""" | ||||
|     preferences = hass.data[DATA_PREFERENCES] | ||||
|     connection.send_result(msg["id"], preferences.as_dict()) | ||||
|  | ||||
|  | ||||
| @websocket_api.websocket_command( | ||||
|     { | ||||
|         vol.Required("type"): "ai_task/preferences/set", | ||||
|         vol.Optional("gen_data_entity_id"): vol.Any(str, None), | ||||
|         vol.Optional("gen_image_entity_id"): vol.Any(str, None), | ||||
|     } | ||||
| ) | ||||
| @websocket_api.require_admin | ||||
| @callback | ||||
| def websocket_set_preferences( | ||||
|     hass: HomeAssistant, | ||||
|     connection: websocket_api.ActiveConnection, | ||||
|     msg: dict[str, Any], | ||||
| ) -> None: | ||||
|     """Set AI task preferences.""" | ||||
|     preferences = hass.data[DATA_PREFERENCES] | ||||
|     msg.pop("type") | ||||
|     msg_id = msg.pop("id") | ||||
|     preferences.async_set_preferences(**msg) | ||||
|     connection.send_result(msg_id, preferences.as_dict()) | ||||
| @@ -1,15 +0,0 @@ | ||||
| { | ||||
|   "entity_component": { | ||||
|     "_": { | ||||
|       "default": "mdi:star-four-points" | ||||
|     } | ||||
|   }, | ||||
|   "services": { | ||||
|     "generate_data": { | ||||
|       "service": "mdi:file-star-four-points-outline" | ||||
|     }, | ||||
|     "generate_image": { | ||||
|       "service": "mdi:star-four-points-box-outline" | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,10 +0,0 @@ | ||||
| { | ||||
|   "domain": "ai_task", | ||||
|   "name": "AI Task", | ||||
|   "after_dependencies": ["camera"], | ||||
|   "codeowners": ["@home-assistant/core"], | ||||
|   "dependencies": ["conversation", "media_source"], | ||||
|   "documentation": "https://www.home-assistant.io/integrations/ai_task", | ||||
|   "integration_type": "entity", | ||||
|   "quality_scale": "internal" | ||||
| } | ||||
| @@ -1,32 +0,0 @@ | ||||
| """Expose images as media sources.""" | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| from pathlib import Path | ||||
|  | ||||
| from homeassistant.components.media_source import MediaSource, local_source | ||||
| from homeassistant.core import HomeAssistant | ||||
| from homeassistant.exceptions import HomeAssistantError | ||||
|  | ||||
| from .const import DATA_MEDIA_SOURCE, DOMAIN, IMAGE_DIR | ||||
|  | ||||
|  | ||||
| async def async_get_media_source(hass: HomeAssistant) -> MediaSource: | ||||
|     """Set up local media source.""" | ||||
|     media_dirs = list(hass.config.media_dirs.values()) | ||||
|  | ||||
|     if not media_dirs: | ||||
|         raise HomeAssistantError( | ||||
|             "AI Task media source requires at least one media directory configured" | ||||
|         ) | ||||
|  | ||||
|     media_dir = Path(media_dirs[0]) / DOMAIN / IMAGE_DIR | ||||
|  | ||||
|     hass.data[DATA_MEDIA_SOURCE] = source = local_source.LocalSource( | ||||
|         hass, | ||||
|         DOMAIN, | ||||
|         "AI Generated Images", | ||||
|         {IMAGE_DIR: str(media_dir)}, | ||||
|         f"/{DOMAIN}", | ||||
|     ) | ||||
|     return source | ||||
| @@ -1,61 +0,0 @@ | ||||
| generate_data: | ||||
|   fields: | ||||
|     task_name: | ||||
|       example: "home summary" | ||||
|       required: true | ||||
|       selector: | ||||
|         text: | ||||
|     instructions: | ||||
|       example: "Generate a funny notification that the garage door was left open" | ||||
|       required: true | ||||
|       selector: | ||||
|         text: | ||||
|           multiline: true | ||||
|     entity_id: | ||||
|       required: false | ||||
|       selector: | ||||
|         entity: | ||||
|           filter: | ||||
|             domain: ai_task | ||||
|             supported_features: | ||||
|               - ai_task.AITaskEntityFeature.GENERATE_DATA | ||||
|     structure: | ||||
|       required: false | ||||
|       example: '{ "name": { "selector": { "text": }, "description": "Name of the user", "required": "True" } } }, "age": { "selector": { "number": }, "description": "Age of the user" } }' | ||||
|       selector: | ||||
|         object: | ||||
|     attachments: | ||||
|       required: false | ||||
|       selector: | ||||
|         media: | ||||
|           accept: | ||||
|             - "*" | ||||
|           multiple: true | ||||
| generate_image: | ||||
|   fields: | ||||
|     task_name: | ||||
|       example: "picture of a dog" | ||||
|       required: true | ||||
|       selector: | ||||
|         text: | ||||
|     instructions: | ||||
|       example: "Generate a high quality square image of a dog on transparent background" | ||||
|       required: true | ||||
|       selector: | ||||
|         text: | ||||
|           multiline: true | ||||
|     entity_id: | ||||
|       required: true | ||||
|       selector: | ||||
|         entity: | ||||
|           filter: | ||||
|             domain: ai_task | ||||
|             supported_features: | ||||
|               - ai_task.AITaskEntityFeature.GENERATE_IMAGE | ||||
|     attachments: | ||||
|       required: false | ||||
|       selector: | ||||
|         media: | ||||
|           accept: | ||||
|             - "*" | ||||
|           multiple: true | ||||
| @@ -1,52 +0,0 @@ | ||||
| { | ||||
|   "services": { | ||||
|     "generate_data": { | ||||
|       "name": "Generate data", | ||||
|       "description": "Uses AI to run a task that generates data.", | ||||
|       "fields": { | ||||
|         "task_name": { | ||||
|           "name": "Task name", | ||||
|           "description": "Name of the task." | ||||
|         }, | ||||
|         "instructions": { | ||||
|           "name": "Instructions", | ||||
|           "description": "Instructions on what needs to be done." | ||||
|         }, | ||||
|         "entity_id": { | ||||
|           "name": "Entity ID", | ||||
|           "description": "Entity ID to run the task on. If not provided, the preferred entity will be used." | ||||
|         }, | ||||
|         "structure": { | ||||
|           "name": "Structured output", | ||||
|           "description": "When set, the AI Task will output fields with this in structure. The structure is a dictionary where the keys are the field names and the values contain a 'description', a 'selector', and an optional 'required' field." | ||||
|         }, | ||||
|         "attachments": { | ||||
|           "name": "Attachments", | ||||
|           "description": "List of files to attach for multi-modal AI analysis." | ||||
|         } | ||||
|       } | ||||
|     }, | ||||
|     "generate_image": { | ||||
|       "name": "Generate image", | ||||
|       "description": "Uses AI to generate image.", | ||||
|       "fields": { | ||||
|         "task_name": { | ||||
|           "name": "Task name", | ||||
|           "description": "Name of the task." | ||||
|         }, | ||||
|         "instructions": { | ||||
|           "name": "Instructions", | ||||
|           "description": "Instructions that explains the image to be generated." | ||||
|         }, | ||||
|         "entity_id": { | ||||
|           "name": "Entity ID", | ||||
|           "description": "Entity ID to run the task on." | ||||
|         }, | ||||
|         "attachments": { | ||||
|           "name": "Attachments", | ||||
|           "description": "List of files to attach for using as references." | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @@ -1,349 +0,0 @@ | ||||
| """AI tasks to be handled by agents.""" | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| from dataclasses import dataclass | ||||
| from datetime import datetime, timedelta | ||||
| import io | ||||
| import mimetypes | ||||
| from pathlib import Path | ||||
| import tempfile | ||||
| from typing import Any | ||||
|  | ||||
| import voluptuous as vol | ||||
|  | ||||
| from homeassistant.components import camera, conversation, image, media_source | ||||
| from homeassistant.components.http.auth import async_sign_path | ||||
| from homeassistant.core import HomeAssistant, ServiceResponse, callback | ||||
| from homeassistant.exceptions import HomeAssistantError | ||||
| from homeassistant.helpers import llm | ||||
| from homeassistant.helpers.chat_session import ChatSession, async_get_chat_session | ||||
| from homeassistant.util import RE_SANITIZE_FILENAME, slugify | ||||
|  | ||||
| from .const import ( | ||||
|     DATA_COMPONENT, | ||||
|     DATA_MEDIA_SOURCE, | ||||
|     DATA_PREFERENCES, | ||||
|     DOMAIN, | ||||
|     IMAGE_DIR, | ||||
|     IMAGE_EXPIRY_TIME, | ||||
|     AITaskEntityFeature, | ||||
| ) | ||||
|  | ||||
|  | ||||
| def _save_camera_snapshot(image_data: camera.Image | image.Image) -> Path: | ||||
|     """Save camera snapshot to temp file.""" | ||||
|     with tempfile.NamedTemporaryFile( | ||||
|         mode="wb", | ||||
|         suffix=mimetypes.guess_extension(image_data.content_type, False), | ||||
|         delete=False, | ||||
|     ) as temp_file: | ||||
|         temp_file.write(image_data.content) | ||||
|         return Path(temp_file.name) | ||||
|  | ||||
|  | ||||
| async def _resolve_attachments( | ||||
|     hass: HomeAssistant, | ||||
|     session: ChatSession, | ||||
|     attachments: list[dict] | None = None, | ||||
| ) -> list[conversation.Attachment]: | ||||
|     """Resolve attachments for a task.""" | ||||
|     resolved_attachments: list[conversation.Attachment] = [] | ||||
|     created_files: list[Path] = [] | ||||
|  | ||||
|     for attachment in attachments or []: | ||||
|         media_content_id = attachment["media_content_id"] | ||||
|  | ||||
|         # Special case for certain media sources | ||||
|         for integration in camera, image: | ||||
|             media_source_prefix = f"media-source://{integration.DOMAIN}/" | ||||
|             if not media_content_id.startswith(media_source_prefix): | ||||
|                 continue | ||||
|  | ||||
|             # Extract entity_id from the media content ID | ||||
|             entity_id = media_content_id.removeprefix(media_source_prefix) | ||||
|  | ||||
|             # Get snapshot from entity | ||||
|             image_data = await integration.async_get_image(hass, entity_id) | ||||
|  | ||||
|             temp_filename = await hass.async_add_executor_job( | ||||
|                 _save_camera_snapshot, image_data | ||||
|             ) | ||||
|             created_files.append(temp_filename) | ||||
|  | ||||
|             resolved_attachments.append( | ||||
|                 conversation.Attachment( | ||||
|                     media_content_id=media_content_id, | ||||
|                     mime_type=image_data.content_type, | ||||
|                     path=temp_filename, | ||||
|                 ) | ||||
|             ) | ||||
|             break | ||||
|         else: | ||||
|             # Handle regular media sources | ||||
|             media = await media_source.async_resolve_media(hass, media_content_id, None) | ||||
|             if media.path is None: | ||||
|                 raise HomeAssistantError( | ||||
|                     "Only local attachments are currently supported" | ||||
|                 ) | ||||
|             resolved_attachments.append( | ||||
|                 conversation.Attachment( | ||||
|                     media_content_id=media_content_id, | ||||
|                     mime_type=media.mime_type, | ||||
|                     path=media.path, | ||||
|                 ) | ||||
|             ) | ||||
|  | ||||
|     if not created_files: | ||||
|         return resolved_attachments | ||||
|  | ||||
|     def cleanup_files() -> None: | ||||
|         """Cleanup temporary files.""" | ||||
|         for file in created_files: | ||||
|             file.unlink(missing_ok=True) | ||||
|  | ||||
|     @callback | ||||
|     def cleanup_files_callback() -> None: | ||||
|         """Cleanup temporary files.""" | ||||
|         hass.async_add_executor_job(cleanup_files) | ||||
|  | ||||
|     session.async_on_cleanup(cleanup_files_callback) | ||||
|  | ||||
|     return resolved_attachments | ||||
|  | ||||
|  | ||||
| async def async_generate_data( | ||||
|     hass: HomeAssistant, | ||||
|     *, | ||||
|     task_name: str, | ||||
|     entity_id: str | None = None, | ||||
|     instructions: str, | ||||
|     structure: vol.Schema | None = None, | ||||
|     attachments: list[dict] | None = None, | ||||
|     llm_api: llm.API | None = None, | ||||
| ) -> GenDataTaskResult: | ||||
|     """Run a data generation task in the AI Task integration.""" | ||||
|     if entity_id is None: | ||||
|         entity_id = hass.data[DATA_PREFERENCES].gen_data_entity_id | ||||
|  | ||||
|     if entity_id is None: | ||||
|         raise HomeAssistantError("No entity_id provided and no preferred entity set") | ||||
|  | ||||
|     entity = hass.data[DATA_COMPONENT].get_entity(entity_id) | ||||
|     if entity is None: | ||||
|         raise HomeAssistantError(f"AI Task entity {entity_id} not found") | ||||
|  | ||||
|     if AITaskEntityFeature.GENERATE_DATA not in entity.supported_features: | ||||
|         raise HomeAssistantError( | ||||
|             f"AI Task entity {entity_id} does not support generating data" | ||||
|         ) | ||||
|  | ||||
|     if ( | ||||
|         attachments | ||||
|         and AITaskEntityFeature.SUPPORT_ATTACHMENTS not in entity.supported_features | ||||
|     ): | ||||
|         raise HomeAssistantError( | ||||
|             f"AI Task entity {entity_id} does not support attachments" | ||||
|         ) | ||||
|  | ||||
|     with async_get_chat_session(hass) as session: | ||||
|         resolved_attachments = await _resolve_attachments(hass, session, attachments) | ||||
|  | ||||
|         return await entity.internal_async_generate_data( | ||||
|             session, | ||||
|             GenDataTask( | ||||
|                 name=task_name, | ||||
|                 instructions=instructions, | ||||
|                 structure=structure, | ||||
|                 attachments=resolved_attachments or None, | ||||
|                 llm_api=llm_api, | ||||
|             ), | ||||
|         ) | ||||
|  | ||||
|  | ||||
| async def async_generate_image( | ||||
|     hass: HomeAssistant, | ||||
|     *, | ||||
|     task_name: str, | ||||
|     entity_id: str | None = None, | ||||
|     instructions: str, | ||||
|     attachments: list[dict] | None = None, | ||||
| ) -> ServiceResponse: | ||||
|     """Run an image generation task in the AI Task integration.""" | ||||
|     if entity_id is None: | ||||
|         entity_id = hass.data[DATA_PREFERENCES].gen_image_entity_id | ||||
|  | ||||
|     if entity_id is None: | ||||
|         raise HomeAssistantError("No entity_id provided and no preferred entity set") | ||||
|  | ||||
|     entity = hass.data[DATA_COMPONENT].get_entity(entity_id) | ||||
|     if entity is None: | ||||
|         raise HomeAssistantError(f"AI Task entity {entity_id} not found") | ||||
|  | ||||
|     if AITaskEntityFeature.GENERATE_IMAGE not in entity.supported_features: | ||||
|         raise HomeAssistantError( | ||||
|             f"AI Task entity {entity_id} does not support generating images" | ||||
|         ) | ||||
|  | ||||
|     if ( | ||||
|         attachments | ||||
|         and AITaskEntityFeature.SUPPORT_ATTACHMENTS not in entity.supported_features | ||||
|     ): | ||||
|         raise HomeAssistantError( | ||||
|             f"AI Task entity {entity_id} does not support attachments" | ||||
|         ) | ||||
|  | ||||
|     with async_get_chat_session(hass) as session: | ||||
|         resolved_attachments = await _resolve_attachments(hass, session, attachments) | ||||
|  | ||||
|         task_result = await entity.internal_async_generate_image( | ||||
|             session, | ||||
|             GenImageTask( | ||||
|                 name=task_name, | ||||
|                 instructions=instructions, | ||||
|                 attachments=resolved_attachments or None, | ||||
|             ), | ||||
|         ) | ||||
|  | ||||
|     service_result = task_result.as_dict() | ||||
|     image_data = service_result.pop("image_data") | ||||
|     if service_result.get("revised_prompt") is None: | ||||
|         service_result["revised_prompt"] = instructions | ||||
|  | ||||
|     source = hass.data[DATA_MEDIA_SOURCE] | ||||
|  | ||||
|     current_time = datetime.now() | ||||
|     ext = mimetypes.guess_extension(task_result.mime_type, False) or ".png" | ||||
|     sanitized_task_name = RE_SANITIZE_FILENAME.sub("", slugify(task_name)) | ||||
|  | ||||
|     image_file = ImageData( | ||||
|         filename=f"{current_time.strftime('%Y-%m-%d_%H%M%S')}_{sanitized_task_name}{ext}", | ||||
|         file=io.BytesIO(image_data), | ||||
|         content_type=task_result.mime_type, | ||||
|     ) | ||||
|  | ||||
|     target_folder = media_source.MediaSourceItem.from_uri( | ||||
|         hass, f"media-source://{DOMAIN}/{IMAGE_DIR}", None | ||||
|     ) | ||||
|  | ||||
|     service_result["media_source_id"] = await source.async_upload_media( | ||||
|         target_folder, image_file | ||||
|     ) | ||||
|  | ||||
|     item = media_source.MediaSourceItem.from_uri( | ||||
|         hass, service_result["media_source_id"], None | ||||
|     ) | ||||
|     service_result["url"] = async_sign_path( | ||||
|         hass, | ||||
|         (await source.async_resolve_media(item)).url, | ||||
|         timedelta(seconds=IMAGE_EXPIRY_TIME), | ||||
|     ) | ||||
|  | ||||
|     return service_result | ||||
|  | ||||
|  | ||||
| @dataclass(slots=True) | ||||
| class GenDataTask: | ||||
|     """Gen data task to be processed.""" | ||||
|  | ||||
|     name: str | ||||
|     """Name of the task.""" | ||||
|  | ||||
|     instructions: str | ||||
|     """Instructions on what needs to be done.""" | ||||
|  | ||||
|     structure: vol.Schema | None = None | ||||
|     """Optional structure for the data to be generated.""" | ||||
|  | ||||
|     attachments: list[conversation.Attachment] | None = None | ||||
|     """List of attachments to go along the instructions.""" | ||||
|  | ||||
|     llm_api: llm.API | None = None | ||||
|     """API to provide to the LLM.""" | ||||
|  | ||||
|     def __str__(self) -> str: | ||||
|         """Return task as a string.""" | ||||
|         return f"<GenDataTask {self.name}: {id(self)}>" | ||||
|  | ||||
|  | ||||
| @dataclass(slots=True) | ||||
| class GenDataTaskResult: | ||||
|     """Result of gen data task.""" | ||||
|  | ||||
|     conversation_id: str | ||||
|     """Unique identifier for the conversation.""" | ||||
|  | ||||
|     data: Any | ||||
|     """Data generated by the task.""" | ||||
|  | ||||
|     def as_dict(self) -> dict[str, Any]: | ||||
|         """Return result as a dict.""" | ||||
|         return { | ||||
|             "conversation_id": self.conversation_id, | ||||
|             "data": self.data, | ||||
|         } | ||||
|  | ||||
|  | ||||
| @dataclass(slots=True) | ||||
| class GenImageTask: | ||||
|     """Gen image task to be processed.""" | ||||
|  | ||||
|     name: str | ||||
|     """Name of the task.""" | ||||
|  | ||||
|     instructions: str | ||||
|     """Instructions on what needs to be done.""" | ||||
|  | ||||
|     attachments: list[conversation.Attachment] | None = None | ||||
|     """List of attachments to go along the instructions.""" | ||||
|  | ||||
|     def __str__(self) -> str: | ||||
|         """Return task as a string.""" | ||||
|         return f"<GenImageTask {self.name}: {id(self)}>" | ||||
|  | ||||
|  | ||||
| @dataclass(slots=True) | ||||
| class GenImageTaskResult: | ||||
|     """Result of gen image task.""" | ||||
|  | ||||
|     image_data: bytes | ||||
|     """Raw image data generated by the model.""" | ||||
|  | ||||
|     conversation_id: str | ||||
|     """Unique identifier for the conversation.""" | ||||
|  | ||||
|     mime_type: str | ||||
|     """MIME type of the generated image.""" | ||||
|  | ||||
|     width: int | None = None | ||||
|     """Width of the generated image, if available.""" | ||||
|  | ||||
|     height: int | None = None | ||||
|     """Height of the generated image, if available.""" | ||||
|  | ||||
|     model: str | None = None | ||||
|     """Model used to generate the image, if available.""" | ||||
|  | ||||
|     revised_prompt: str | None = None | ||||
|     """Revised prompt used to generate the image, if applicable.""" | ||||
|  | ||||
|     def as_dict(self) -> dict[str, Any]: | ||||
|         """Return result as a dict.""" | ||||
|         return { | ||||
|             "image_data": self.image_data, | ||||
|             "conversation_id": self.conversation_id, | ||||
|             "mime_type": self.mime_type, | ||||
|             "width": self.width, | ||||
|             "height": self.height, | ||||
|             "model": self.model, | ||||
|             "revised_prompt": self.revised_prompt, | ||||
|         } | ||||
|  | ||||
|  | ||||
| @dataclass(slots=True) | ||||
| class ImageData: | ||||
|     """Implementation of media_source.local_source.UploadedFile protocol.""" | ||||
|  | ||||
|     filename: str | ||||
|     file: io.IOBase | ||||
|     content_type: str | ||||
| @@ -51,16 +51,9 @@ class AirGradientCoordinator(DataUpdateCoordinator[AirGradientData]): | ||||
|  | ||||
|     async def _async_setup(self) -> None: | ||||
|         """Set up the coordinator.""" | ||||
|         try: | ||||
|             self._current_version = ( | ||||
|                 await self.client.get_current_measures() | ||||
|             ).firmware_version | ||||
|         except AirGradientError as error: | ||||
|             raise UpdateFailed( | ||||
|                 translation_domain=DOMAIN, | ||||
|                 translation_key="update_error", | ||||
|                 translation_placeholders={"error": str(error)}, | ||||
|             ) from error | ||||
|         self._current_version = ( | ||||
|             await self.client.get_current_measures() | ||||
|         ).firmware_version | ||||
|  | ||||
|     async def _async_update_data(self) -> AirGradientData: | ||||
|         try: | ||||
|   | ||||
| @@ -6,7 +6,6 @@ from typing import Any, Concatenate | ||||
| from airgradient import AirGradientConnectionError, AirGradientError, get_model_name | ||||
|  | ||||
| from homeassistant.exceptions import HomeAssistantError | ||||
| from homeassistant.helpers import device_registry as dr | ||||
| from homeassistant.helpers.device_registry import DeviceInfo | ||||
| from homeassistant.helpers.update_coordinator import CoordinatorEntity | ||||
|  | ||||
| @@ -30,7 +29,6 @@ class AirGradientEntity(CoordinatorEntity[AirGradientCoordinator]): | ||||
|             model_id=measures.model, | ||||
|             serial_number=coordinator.serial_number, | ||||
|             sw_version=measures.firmware_version, | ||||
|             connections={(dr.CONNECTION_NETWORK_MAC, coordinator.serial_number)}, | ||||
|         ) | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -6,7 +6,6 @@ | ||||
|   "documentation": "https://www.home-assistant.io/integrations/airgradient", | ||||
|   "integration_type": "device", | ||||
|   "iot_class": "local_polling", | ||||
|   "quality_scale": "platinum", | ||||
|   "requirements": ["airgradient==0.9.2"], | ||||
|   "zeroconf": ["_airgradient._tcp.local."] | ||||
| } | ||||
|   | ||||
| @@ -14,9 +14,9 @@ rules: | ||||
|     status: exempt | ||||
|     comment: | | ||||
|       This integration does not provide additional actions. | ||||
|   docs-high-level-description: done | ||||
|   docs-installation-instructions: done | ||||
|   docs-removal-instructions: done | ||||
|   docs-high-level-description: todo | ||||
|   docs-installation-instructions: todo | ||||
|   docs-removal-instructions: todo | ||||
|   entity-event-setup: | ||||
|     status: exempt | ||||
|     comment: | | ||||
| @@ -34,7 +34,7 @@ rules: | ||||
|   docs-configuration-parameters: | ||||
|     status: exempt | ||||
|     comment: No options to configure | ||||
|   docs-installation-parameters: done | ||||
|   docs-installation-parameters: todo | ||||
|   entity-unavailable: done | ||||
|   integration-owner: done | ||||
|   log-when-unavailable: done | ||||
| @@ -43,19 +43,23 @@ rules: | ||||
|     status: exempt | ||||
|     comment: | | ||||
|       This integration does not require authentication. | ||||
|   test-coverage: done | ||||
|   test-coverage: todo | ||||
|   # Gold | ||||
|   devices: done | ||||
|   diagnostics: done | ||||
|   discovery-update-info: done | ||||
|   discovery: done | ||||
|   docs-data-update: done | ||||
|   docs-examples: done | ||||
|   docs-known-limitations: done | ||||
|   docs-supported-devices: done | ||||
|   docs-supported-functions: done | ||||
|   docs-troubleshooting: done | ||||
|   docs-use-cases: done | ||||
|   discovery-update-info: | ||||
|     status: todo | ||||
|     comment: DHCP is still possible | ||||
|   discovery: | ||||
|     status: todo | ||||
|     comment: DHCP is still possible | ||||
|   docs-data-update: todo | ||||
|   docs-examples: todo | ||||
|   docs-known-limitations: todo | ||||
|   docs-supported-devices: todo | ||||
|   docs-supported-functions: todo | ||||
|   docs-troubleshooting: todo | ||||
|   docs-use-cases: todo | ||||
|   dynamic-devices: | ||||
|     status: exempt | ||||
|     comment: | | ||||
|   | ||||
| @@ -61,7 +61,7 @@ | ||||
|       "display_pm_standard": { | ||||
|         "name": "Display PM standard", | ||||
|         "state": { | ||||
|           "ugm3": "μg/m³", | ||||
|           "ugm3": "µg/m³", | ||||
|           "us_aqi": "US AQI" | ||||
|         } | ||||
|       }, | ||||
|   | ||||
| @@ -1,9 +1,7 @@ | ||||
| """Airgradient Update platform.""" | ||||
|  | ||||
| from datetime import timedelta | ||||
| import logging | ||||
|  | ||||
| from airgradient import AirGradientConnectionError | ||||
| from propcache.api import cached_property | ||||
|  | ||||
| from homeassistant.components.update import UpdateDeviceClass, UpdateEntity | ||||
| @@ -15,7 +13,6 @@ from .entity import AirGradientEntity | ||||
|  | ||||
| PARALLEL_UPDATES = 1 | ||||
| SCAN_INTERVAL = timedelta(hours=1) | ||||
| _LOGGER = logging.getLogger(__name__) | ||||
|  | ||||
|  | ||||
| async def async_setup_entry( | ||||
| @@ -34,7 +31,6 @@ class AirGradientUpdate(AirGradientEntity, UpdateEntity): | ||||
|     """Representation of Airgradient Update.""" | ||||
|  | ||||
|     _attr_device_class = UpdateDeviceClass.FIRMWARE | ||||
|     _server_unreachable_logged = False | ||||
|  | ||||
|     def __init__(self, coordinator: AirGradientCoordinator) -> None: | ||||
|         """Initialize the entity.""" | ||||
| @@ -51,27 +47,10 @@ class AirGradientUpdate(AirGradientEntity, UpdateEntity): | ||||
|         """Return the installed version of the entity.""" | ||||
|         return self.coordinator.data.measures.firmware_version | ||||
|  | ||||
|     @property | ||||
|     def available(self) -> bool: | ||||
|         """Return if entity is available.""" | ||||
|         return super().available and self._attr_available | ||||
|  | ||||
|     async def async_update(self) -> None: | ||||
|         """Update the entity.""" | ||||
|         try: | ||||
|             self._attr_latest_version = ( | ||||
|                 await self.coordinator.client.get_latest_firmware_version( | ||||
|                     self.coordinator.serial_number | ||||
|                 ) | ||||
|         self._attr_latest_version = ( | ||||
|             await self.coordinator.client.get_latest_firmware_version( | ||||
|                 self.coordinator.serial_number | ||||
|             ) | ||||
|         except AirGradientConnectionError: | ||||
|             self._attr_latest_version = None | ||||
|             self._attr_available = False | ||||
|             if not self._server_unreachable_logged: | ||||
|                 _LOGGER.error( | ||||
|                     "Unable to connect to AirGradient server to check for updates" | ||||
|                 ) | ||||
|                 self._server_unreachable_logged = True | ||||
|         else: | ||||
|             self._server_unreachable_logged = False | ||||
|             self._attr_available = True | ||||
|         ) | ||||
|   | ||||
| @@ -18,10 +18,6 @@ from homeassistant.helpers.aiohttp_client import async_get_clientsession | ||||
|  | ||||
| from .const import CONF_USE_NEAREST, DOMAIN, NO_AIRLY_SENSORS | ||||
|  | ||||
| DESCRIPTION_PLACEHOLDERS = { | ||||
|     "developer_registration_url": "https://developer.airly.eu/register", | ||||
| } | ||||
|  | ||||
|  | ||||
| class AirlyFlowHandler(ConfigFlow, domain=DOMAIN): | ||||
|     """Config flow for Airly.""" | ||||
| @@ -43,14 +39,14 @@ class AirlyFlowHandler(ConfigFlow, domain=DOMAIN): | ||||
|             ) | ||||
|             self._abort_if_unique_id_configured() | ||||
|             try: | ||||
|                 location_point_valid = await check_location( | ||||
|                 location_point_valid = await test_location( | ||||
|                     websession, | ||||
|                     user_input["api_key"], | ||||
|                     user_input["latitude"], | ||||
|                     user_input["longitude"], | ||||
|                 ) | ||||
|                 if not location_point_valid: | ||||
|                     location_nearest_valid = await check_location( | ||||
|                     location_nearest_valid = await test_location( | ||||
|                         websession, | ||||
|                         user_input["api_key"], | ||||
|                         user_input["latitude"], | ||||
| @@ -89,11 +85,10 @@ class AirlyFlowHandler(ConfigFlow, domain=DOMAIN): | ||||
|                 } | ||||
|             ), | ||||
|             errors=errors, | ||||
|             description_placeholders=DESCRIPTION_PLACEHOLDERS, | ||||
|         ) | ||||
|  | ||||
|  | ||||
| async def check_location( | ||||
| async def test_location( | ||||
|     client: ClientSession, | ||||
|     api_key: str, | ||||
|     latitude: float, | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
|   "config": { | ||||
|     "step": { | ||||
|       "user": { | ||||
|         "description": "To generate API key go to {developer_registration_url}", | ||||
|         "description": "To generate API key go to https://developer.airly.eu/register", | ||||
|         "data": { | ||||
|           "name": "[%key:common::config_flow::data::name%]", | ||||
|           "api_key": "[%key:common::config_flow::data::api_key%]", | ||||
|   | ||||
| @@ -45,6 +45,9 @@ async def async_setup_entry(hass: HomeAssistant, entry: AirNowConfigEntry) -> bo | ||||
|     # Store Entity and Initialize Platforms | ||||
|     entry.runtime_data = coordinator | ||||
|  | ||||
|     # Listen for option changes | ||||
|     entry.async_on_unload(entry.add_update_listener(update_listener)) | ||||
|  | ||||
|     await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS) | ||||
|  | ||||
|     # Clean up unused device entries with no entities | ||||
| @@ -85,3 +88,8 @@ async def async_migrate_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool: | ||||
| async def async_unload_entry(hass: HomeAssistant, entry: AirNowConfigEntry) -> bool: | ||||
|     """Unload a config entry.""" | ||||
|     return await hass.config_entries.async_unload_platforms(entry, PLATFORMS) | ||||
|  | ||||
|  | ||||
| async def update_listener(hass: HomeAssistant, entry: ConfigEntry) -> None: | ||||
|     """Handle options update.""" | ||||
|     await hass.config_entries.async_reload(entry.entry_id) | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user