mirror of
https://github.com/home-assistant/core.git
synced 2025-10-24 11:09:37 +00:00
Compare commits
3 Commits
epenet-pat
...
mqtt-clien
Author | SHA1 | Date | |
---|---|---|---|
![]() |
1edea46a4d | ||
![]() |
c476e92bdc | ||
![]() |
8dcd9945e8 |
@@ -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
|
# Extra components that trigger the full suite
|
||||||
components: &components
|
components: &components
|
||||||
- homeassistant/components/alexa/**
|
- homeassistant/components/alexa/**
|
||||||
- homeassistant/components/analytics/**
|
|
||||||
- homeassistant/components/application_credentials/**
|
- homeassistant/components/application_credentials/**
|
||||||
- homeassistant/components/assist_pipeline/**
|
- homeassistant/components/assist_pipeline/**
|
||||||
- homeassistant/components/auth/**
|
- homeassistant/components/auth/**
|
||||||
|
@@ -8,9 +8,6 @@
|
|||||||
"PYTHONASYNCIODEBUG": "1"
|
"PYTHONASYNCIODEBUG": "1"
|
||||||
},
|
},
|
||||||
"features": {
|
"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": {}
|
"ghcr.io/devcontainers/features/github-cli:1": {}
|
||||||
},
|
},
|
||||||
// Port 5683 udp is used by Shelly integration
|
// Port 5683 udp is used by Shelly integration
|
||||||
|
@@ -14,8 +14,7 @@ tests
|
|||||||
|
|
||||||
# Other virtualization methods
|
# Other virtualization methods
|
||||||
venv
|
venv
|
||||||
.venv
|
|
||||||
.vagrant
|
.vagrant
|
||||||
|
|
||||||
# Temporary files
|
# 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: |
|
value: |
|
||||||
This issue form is for reporting bugs only!
|
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
|
- type: textarea
|
||||||
validations:
|
validations:
|
||||||
required: true
|
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
|
url: https://www.home-assistant.io/help
|
||||||
about: We use GitHub for tracking bugs, check our website for resources on getting help.
|
about: We use GitHub for tracking bugs, check our website for resources on getting help.
|
||||||
- name: Feature Request
|
- name: Feature Request
|
||||||
url: https://github.com/orgs/home-assistant/discussions
|
url: https://community.home-assistant.io/c/feature-requests
|
||||||
about: Please use this link to request new features or enhancements to existing features.
|
about: Please use our Community Forum for making feature requests.
|
||||||
- name: I'm unsure where to go
|
- name: I'm unsure where to go
|
||||||
url: https://www.home-assistant.io/join-chat
|
url: https://www.home-assistant.io/join-chat
|
||||||
about: If you are unsure where to go, then joining our chat is recommended; Just ask!
|
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.
|
creating the PR. If you're unsure about any of them, don't hesitate to ask.
|
||||||
We're here to help! This is simply a reminder of what we are going to look
|
We're here to help! This is simply a reminder of what we are going to look
|
||||||
for before merging your code.
|
for before merging your code.
|
||||||
|
|
||||||
AI tools are welcome, but contributors are responsible for *fully*
|
|
||||||
understanding the code before submitting a PR.
|
|
||||||
-->
|
-->
|
||||||
|
|
||||||
- [ ] I understand the code I am submitting and can explain how it works.
|
|
||||||
- [ ] The code change is tested and works locally.
|
- [ ] The code change is tested and works locally.
|
||||||
- [ ] Local tests pass. **Your PR cannot be merged unless tests pass**
|
- [ ] Local tests pass. **Your PR cannot be merged unless tests pass**
|
||||||
- [ ] There is no commented out code in this PR.
|
- [ ] There is no commented out code in this PR.
|
||||||
@@ -68,7 +64,6 @@
|
|||||||
- [ ] I have followed the [perfect PR recommendations][perfect-pr]
|
- [ ] I have followed the [perfect PR recommendations][perfect-pr]
|
||||||
- [ ] The code has been formatted using Ruff (`ruff format homeassistant tests`)
|
- [ ] The code has been formatted using Ruff (`ruff format homeassistant tests`)
|
||||||
- [ ] Tests have been added to verify that the new code works.
|
- [ ] Tests have been added to verify that the new code works.
|
||||||
- [ ] Any generated code has been carefully reviewed for correctness and compliance with project standards.
|
|
||||||
|
|
||||||
If user exposed functionality or configuration variables are added/changed:
|
If user exposed functionality or configuration variables are added/changed:
|
||||||
|
|
||||||
|
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
|
interval: daily
|
||||||
time: "06:00"
|
time: "06:00"
|
||||||
open-pull-requests-limit: 10
|
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 }}
|
publish: ${{ steps.version.outputs.publish }}
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0
|
uses: actions/checkout@v4.2.2
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
|
|
||||||
- name: Set up Python ${{ env.DEFAULT_PYTHON }}
|
- name: Set up Python ${{ env.DEFAULT_PYTHON }}
|
||||||
uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0
|
uses: actions/setup-python@v5.4.0
|
||||||
with:
|
with:
|
||||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||||
|
|
||||||
@@ -69,7 +69,7 @@ jobs:
|
|||||||
run: find ./homeassistant/components/*/translations -name "*.json" | tar zcvf translations.tar.gz -T -
|
run: find ./homeassistant/components/*/translations -name "*.json" | tar zcvf translations.tar.gz -T -
|
||||||
|
|
||||||
- name: Upload translations
|
- name: Upload translations
|
||||||
uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2
|
uses: actions/upload-artifact@v4.6.1
|
||||||
with:
|
with:
|
||||||
name: translations
|
name: translations
|
||||||
path: translations.tar.gz
|
path: translations.tar.gz
|
||||||
@@ -90,11 +90,11 @@ jobs:
|
|||||||
arch: ${{ fromJson(needs.init.outputs.architectures) }}
|
arch: ${{ fromJson(needs.init.outputs.architectures) }}
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0
|
uses: actions/checkout@v4.2.2
|
||||||
|
|
||||||
- name: Download nightly wheels of frontend
|
- name: Download nightly wheels of frontend
|
||||||
if: needs.init.outputs.channel == 'dev'
|
if: needs.init.outputs.channel == 'dev'
|
||||||
uses: dawidd6/action-download-artifact@ac66b43f0e6a346234dd65d4d0c8fbb31cb316e5 # v11
|
uses: dawidd6/action-download-artifact@v8
|
||||||
with:
|
with:
|
||||||
github_token: ${{secrets.GITHUB_TOKEN}}
|
github_token: ${{secrets.GITHUB_TOKEN}}
|
||||||
repo: home-assistant/frontend
|
repo: home-assistant/frontend
|
||||||
@@ -105,10 +105,10 @@ jobs:
|
|||||||
|
|
||||||
- name: Download nightly wheels of intents
|
- name: Download nightly wheels of intents
|
||||||
if: needs.init.outputs.channel == 'dev'
|
if: needs.init.outputs.channel == 'dev'
|
||||||
uses: dawidd6/action-download-artifact@ac66b43f0e6a346234dd65d4d0c8fbb31cb316e5 # v11
|
uses: dawidd6/action-download-artifact@v8
|
||||||
with:
|
with:
|
||||||
github_token: ${{secrets.GITHUB_TOKEN}}
|
github_token: ${{secrets.GITHUB_TOKEN}}
|
||||||
repo: OHF-Voice/intents-package
|
repo: home-assistant/intents-package
|
||||||
branch: main
|
branch: main
|
||||||
workflow: nightly.yaml
|
workflow: nightly.yaml
|
||||||
workflow_conclusion: success
|
workflow_conclusion: success
|
||||||
@@ -116,7 +116,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Set up Python ${{ env.DEFAULT_PYTHON }}
|
- name: Set up Python ${{ env.DEFAULT_PYTHON }}
|
||||||
if: needs.init.outputs.channel == 'dev'
|
if: needs.init.outputs.channel == 'dev'
|
||||||
uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0
|
uses: actions/setup-python@v5.4.0
|
||||||
with:
|
with:
|
||||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||||
|
|
||||||
@@ -175,7 +175,7 @@ jobs:
|
|||||||
sed -i "s|pykrakenapi|# pykrakenapi|g" requirements_all.txt
|
sed -i "s|pykrakenapi|# pykrakenapi|g" requirements_all.txt
|
||||||
|
|
||||||
- name: Download translations
|
- name: Download translations
|
||||||
uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0
|
uses: actions/download-artifact@v4.1.9
|
||||||
with:
|
with:
|
||||||
name: translations
|
name: translations
|
||||||
|
|
||||||
@@ -190,15 +190,14 @@ jobs:
|
|||||||
echo "${{ github.sha }};${{ github.ref }};${{ github.event_name }};${{ github.actor }}" > rootfs/OFFICIAL_IMAGE
|
echo "${{ github.sha }};${{ github.ref }};${{ github.event_name }};${{ github.actor }}" > rootfs/OFFICIAL_IMAGE
|
||||||
|
|
||||||
- name: Login to GitHub Container Registry
|
- name: Login to GitHub Container Registry
|
||||||
uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0
|
uses: docker/login-action@v3.3.0
|
||||||
with:
|
with:
|
||||||
registry: ghcr.io
|
registry: ghcr.io
|
||||||
username: ${{ github.repository_owner }}
|
username: ${{ github.repository_owner }}
|
||||||
password: ${{ secrets.GITHUB_TOKEN }}
|
password: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
# home-assistant/builder doesn't support sha pinning
|
|
||||||
- name: Build base image
|
- name: Build base image
|
||||||
uses: home-assistant/builder@2025.09.0
|
uses: home-assistant/builder@2025.02.0
|
||||||
with:
|
with:
|
||||||
args: |
|
args: |
|
||||||
$BUILD_ARGS \
|
$BUILD_ARGS \
|
||||||
@@ -243,7 +242,7 @@ jobs:
|
|||||||
- green
|
- green
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0
|
uses: actions/checkout@v4.2.2
|
||||||
|
|
||||||
- name: Set build additional args
|
- name: Set build additional args
|
||||||
run: |
|
run: |
|
||||||
@@ -257,15 +256,14 @@ jobs:
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
- name: Login to GitHub Container Registry
|
- name: Login to GitHub Container Registry
|
||||||
uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0
|
uses: docker/login-action@v3.3.0
|
||||||
with:
|
with:
|
||||||
registry: ghcr.io
|
registry: ghcr.io
|
||||||
username: ${{ github.repository_owner }}
|
username: ${{ github.repository_owner }}
|
||||||
password: ${{ secrets.GITHUB_TOKEN }}
|
password: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
# home-assistant/builder doesn't support sha pinning
|
|
||||||
- name: Build base image
|
- name: Build base image
|
||||||
uses: home-assistant/builder@2025.09.0
|
uses: home-assistant/builder@2025.02.0
|
||||||
with:
|
with:
|
||||||
args: |
|
args: |
|
||||||
$BUILD_ARGS \
|
$BUILD_ARGS \
|
||||||
@@ -281,7 +279,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0
|
uses: actions/checkout@v4.2.2
|
||||||
|
|
||||||
- name: Initialize git
|
- name: Initialize git
|
||||||
uses: home-assistant/actions/helpers/git-init@master
|
uses: home-assistant/actions/helpers/git-init@master
|
||||||
@@ -323,23 +321,23 @@ jobs:
|
|||||||
registry: ["ghcr.io/home-assistant", "docker.io/homeassistant"]
|
registry: ["ghcr.io/home-assistant", "docker.io/homeassistant"]
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0
|
uses: actions/checkout@v4.2.2
|
||||||
|
|
||||||
- name: Install Cosign
|
- name: Install Cosign
|
||||||
uses: sigstore/cosign-installer@faadad0cce49287aee09b3a48701e75088a2c6ad # v4.0.0
|
uses: sigstore/cosign-installer@v3.8.1
|
||||||
with:
|
with:
|
||||||
cosign-release: "v2.2.3"
|
cosign-release: "v2.2.3"
|
||||||
|
|
||||||
- name: Login to DockerHub
|
- name: Login to DockerHub
|
||||||
if: matrix.registry == 'docker.io/homeassistant'
|
if: matrix.registry == 'docker.io/homeassistant'
|
||||||
uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0
|
uses: docker/login-action@v3.3.0
|
||||||
with:
|
with:
|
||||||
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
||||||
password: ${{ secrets.DOCKERHUB_TOKEN }}
|
password: ${{ secrets.DOCKERHUB_TOKEN }}
|
||||||
|
|
||||||
- name: Login to GitHub Container Registry
|
- name: Login to GitHub Container Registry
|
||||||
if: matrix.registry == 'ghcr.io/home-assistant'
|
if: matrix.registry == 'ghcr.io/home-assistant'
|
||||||
uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0
|
uses: docker/login-action@v3.3.0
|
||||||
with:
|
with:
|
||||||
registry: ghcr.io
|
registry: ghcr.io
|
||||||
username: ${{ github.repository_owner }}
|
username: ${{ github.repository_owner }}
|
||||||
@@ -456,15 +454,15 @@ jobs:
|
|||||||
if: github.repository_owner == 'home-assistant' && needs.init.outputs.publish == 'true'
|
if: github.repository_owner == 'home-assistant' && needs.init.outputs.publish == 'true'
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0
|
uses: actions/checkout@v4.2.2
|
||||||
|
|
||||||
- name: Set up Python ${{ env.DEFAULT_PYTHON }}
|
- name: Set up Python ${{ env.DEFAULT_PYTHON }}
|
||||||
uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0
|
uses: actions/setup-python@v5.4.0
|
||||||
with:
|
with:
|
||||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||||
|
|
||||||
- name: Download translations
|
- name: Download translations
|
||||||
uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0
|
uses: actions/download-artifact@v4.1.9
|
||||||
with:
|
with:
|
||||||
name: translations
|
name: translations
|
||||||
|
|
||||||
@@ -482,7 +480,7 @@ jobs:
|
|||||||
python -m build
|
python -m build
|
||||||
|
|
||||||
- name: Upload package to PyPI
|
- 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:
|
with:
|
||||||
skip-existing: true
|
skip-existing: true
|
||||||
|
|
||||||
@@ -501,17 +499,17 @@ jobs:
|
|||||||
HASSFEST_IMAGE_TAG: ghcr.io/home-assistant/hassfest:${{ needs.init.outputs.version }}
|
HASSFEST_IMAGE_TAG: ghcr.io/home-assistant/hassfest:${{ needs.init.outputs.version }}
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0
|
uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
|
||||||
|
|
||||||
- name: Login to GitHub Container Registry
|
- name: Login to GitHub Container Registry
|
||||||
uses: docker/login-action@5e57cd118135c172c3672efd75eb46360885c0ef # v3.6.0
|
uses: docker/login-action@9780b0c442fbb1117ed29e0efdff1e18412f7567 # v3.3.0
|
||||||
with:
|
with:
|
||||||
registry: ghcr.io
|
registry: ghcr.io
|
||||||
username: ${{ github.repository_owner }}
|
username: ${{ github.repository_owner }}
|
||||||
password: ${{ secrets.GITHUB_TOKEN }}
|
password: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
- name: Build Docker image
|
- name: Build Docker image
|
||||||
uses: docker/build-push-action@263435318d21b8e681c14492fe198d362a7d2c83 # v6.18.0
|
uses: docker/build-push-action@471d1dc4e07e5cdedd4c2171150001c434f0b7a4 # v6.15.0
|
||||||
with:
|
with:
|
||||||
context: . # So action will not pull the repository again
|
context: . # So action will not pull the repository again
|
||||||
file: ./script/hassfest/docker/Dockerfile
|
file: ./script/hassfest/docker/Dockerfile
|
||||||
@@ -524,7 +522,7 @@ jobs:
|
|||||||
- name: Push Docker image
|
- name: Push Docker image
|
||||||
if: needs.init.outputs.channel != 'dev' && needs.init.outputs.publish == 'true'
|
if: needs.init.outputs.channel != 'dev' && needs.init.outputs.publish == 'true'
|
||||||
id: push
|
id: push
|
||||||
uses: docker/build-push-action@263435318d21b8e681c14492fe198d362a7d2c83 # v6.18.0
|
uses: docker/build-push-action@471d1dc4e07e5cdedd4c2171150001c434f0b7a4 # v6.15.0
|
||||||
with:
|
with:
|
||||||
context: . # So action will not pull the repository again
|
context: . # So action will not pull the repository again
|
||||||
file: ./script/hassfest/docker/Dockerfile
|
file: ./script/hassfest/docker/Dockerfile
|
||||||
@@ -533,7 +531,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Generate artifact attestation
|
- name: Generate artifact attestation
|
||||||
if: needs.init.outputs.channel != 'dev' && needs.init.outputs.publish == 'true'
|
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@f9eaf234fc1c2e333c1eca18177db0f44fa6ba52 # v2.2.1
|
||||||
with:
|
with:
|
||||||
subject-name: ${{ env.HASSFEST_IMAGE_NAME }}
|
subject-name: ${{ env.HASSFEST_IMAGE_NAME }}
|
||||||
subject-digest: ${{ steps.push.outputs.digest }}
|
subject-digest: ${{ steps.push.outputs.digest }}
|
||||||
|
851
.github/workflows/ci.yaml
vendored
851
.github/workflows/ci.yaml
vendored
File diff suppressed because it is too large
Load Diff
6
.github/workflows/codeql.yml
vendored
6
.github/workflows/codeql.yml
vendored
@@ -21,14 +21,14 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0
|
uses: actions/checkout@v4.2.2
|
||||||
|
|
||||||
- name: Initialize CodeQL
|
- name: Initialize CodeQL
|
||||||
uses: github/codeql-action/init@f443b600d91635bebf5b0d9ebc620189c0d6fba5 # v4.30.8
|
uses: github/codeql-action/init@v3.28.10
|
||||||
with:
|
with:
|
||||||
languages: python
|
languages: python
|
||||||
|
|
||||||
- name: Perform CodeQL Analysis
|
- name: Perform CodeQL Analysis
|
||||||
uses: github/codeql-action/analyze@f443b600d91635bebf5b0d9ebc620189c0d6fba5 # v4.30.8
|
uses: github/codeql-action/analyze@v3.28.10
|
||||||
with:
|
with:
|
||||||
category: "/language:python"
|
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'
|
if: github.repository_owner == 'home-assistant'
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: dessant/lock-threads@1bf7ec25051fe7c00bdd17e6a7cf3d7bfb7dc771 # v5.0.1
|
- uses: dessant/lock-threads@v5.0.1
|
||||||
with:
|
with:
|
||||||
github-token: ${{ github.token }}
|
github-token: ${{ github.token }}
|
||||||
issue-inactive-days: "30"
|
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 PRs marked as no-stale
|
||||||
# - No issues (-1)
|
# - No issues (-1)
|
||||||
- name: 60 days stale PRs policy
|
- name: 60 days stale PRs policy
|
||||||
uses: actions/stale@5f858e3efba33a5ca4407a664cc011ad407f2008 # v10.1.0
|
uses: actions/stale@v9.1.0
|
||||||
with:
|
with:
|
||||||
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
days-before-stale: 60
|
days-before-stale: 60
|
||||||
@@ -57,7 +57,7 @@ jobs:
|
|||||||
# - No issues marked as no-stale or help-wanted
|
# - No issues marked as no-stale or help-wanted
|
||||||
# - No PRs (-1)
|
# - No PRs (-1)
|
||||||
- name: 90 days stale issues
|
- name: 90 days stale issues
|
||||||
uses: actions/stale@5f858e3efba33a5ca4407a664cc011ad407f2008 # v10.1.0
|
uses: actions/stale@v9.1.0
|
||||||
with:
|
with:
|
||||||
repo-token: ${{ steps.token.outputs.token }}
|
repo-token: ${{ steps.token.outputs.token }}
|
||||||
days-before-stale: 90
|
days-before-stale: 90
|
||||||
@@ -87,7 +87,7 @@ jobs:
|
|||||||
# - No Issues marked as no-stale or help-wanted
|
# - No Issues marked as no-stale or help-wanted
|
||||||
# - No PRs (-1)
|
# - No PRs (-1)
|
||||||
- name: Needs more information stale issues policy
|
- name: Needs more information stale issues policy
|
||||||
uses: actions/stale@5f858e3efba33a5ca4407a664cc011ad407f2008 # v10.1.0
|
uses: actions/stale@v9.1.0
|
||||||
with:
|
with:
|
||||||
repo-token: ${{ steps.token.outputs.token }}
|
repo-token: ${{ steps.token.outputs.token }}
|
||||||
only-labels: "needs-more-information"
|
only-labels: "needs-more-information"
|
||||||
|
4
.github/workflows/translations.yml
vendored
4
.github/workflows/translations.yml
vendored
@@ -19,10 +19,10 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0
|
uses: actions/checkout@v4.2.2
|
||||||
|
|
||||||
- name: Set up Python ${{ env.DEFAULT_PYTHON }}
|
- name: Set up Python ${{ env.DEFAULT_PYTHON }}
|
||||||
uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0
|
uses: actions/setup-python@v5.4.0
|
||||||
with:
|
with:
|
||||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||||
|
|
||||||
|
76
.github/workflows/wheels.yml
vendored
76
.github/workflows/wheels.yml
vendored
@@ -32,11 +32,11 @@ jobs:
|
|||||||
architectures: ${{ steps.info.outputs.architectures }}
|
architectures: ${{ steps.info.outputs.architectures }}
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0
|
uses: actions/checkout@v4.2.2
|
||||||
|
|
||||||
- name: Set up Python ${{ env.DEFAULT_PYTHON }}
|
- name: Set up Python ${{ env.DEFAULT_PYTHON }}
|
||||||
id: python
|
id: python
|
||||||
uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0
|
uses: actions/setup-python@v5.4.0
|
||||||
with:
|
with:
|
||||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
python-version: ${{ env.DEFAULT_PYTHON }}
|
||||||
check-latest: true
|
check-latest: true
|
||||||
@@ -91,7 +91,7 @@ jobs:
|
|||||||
) > build_constraints.txt
|
) > build_constraints.txt
|
||||||
|
|
||||||
- name: Upload env_file
|
- name: Upload env_file
|
||||||
uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2
|
uses: actions/upload-artifact@v4.6.1
|
||||||
with:
|
with:
|
||||||
name: env_file
|
name: env_file
|
||||||
path: ./.env_file
|
path: ./.env_file
|
||||||
@@ -99,14 +99,14 @@ jobs:
|
|||||||
overwrite: true
|
overwrite: true
|
||||||
|
|
||||||
- name: Upload build_constraints
|
- name: Upload build_constraints
|
||||||
uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2
|
uses: actions/upload-artifact@v4.6.1
|
||||||
with:
|
with:
|
||||||
name: build_constraints
|
name: build_constraints
|
||||||
path: ./build_constraints.txt
|
path: ./build_constraints.txt
|
||||||
overwrite: true
|
overwrite: true
|
||||||
|
|
||||||
- name: Upload requirements_diff
|
- name: Upload requirements_diff
|
||||||
uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2
|
uses: actions/upload-artifact@v4.6.1
|
||||||
with:
|
with:
|
||||||
name: requirements_diff
|
name: requirements_diff
|
||||||
path: ./requirements_diff.txt
|
path: ./requirements_diff.txt
|
||||||
@@ -118,7 +118,7 @@ jobs:
|
|||||||
python -m script.gen_requirements_all ci
|
python -m script.gen_requirements_all ci
|
||||||
|
|
||||||
- name: Upload requirements_all_wheels
|
- name: Upload requirements_all_wheels
|
||||||
uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2
|
uses: actions/upload-artifact@v4.6.1
|
||||||
with:
|
with:
|
||||||
name: requirements_all_wheels
|
name: requirements_all_wheels
|
||||||
path: ./requirements_all_wheels_*.txt
|
path: ./requirements_all_wheels_*.txt
|
||||||
@@ -135,20 +135,20 @@ jobs:
|
|||||||
arch: ${{ fromJson(needs.init.outputs.architectures) }}
|
arch: ${{ fromJson(needs.init.outputs.architectures) }}
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0
|
uses: actions/checkout@v4.2.2
|
||||||
|
|
||||||
- name: Download env_file
|
- name: Download env_file
|
||||||
uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0
|
uses: actions/download-artifact@v4.1.9
|
||||||
with:
|
with:
|
||||||
name: env_file
|
name: env_file
|
||||||
|
|
||||||
- name: Download build_constraints
|
- name: Download build_constraints
|
||||||
uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0
|
uses: actions/download-artifact@v4.1.9
|
||||||
with:
|
with:
|
||||||
name: build_constraints
|
name: build_constraints
|
||||||
|
|
||||||
- name: Download requirements_diff
|
- name: Download requirements_diff
|
||||||
uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0
|
uses: actions/download-artifact@v4.1.9
|
||||||
with:
|
with:
|
||||||
name: requirements_diff
|
name: requirements_diff
|
||||||
|
|
||||||
@@ -158,9 +158,8 @@ jobs:
|
|||||||
sed -i "/uv/d" requirements.txt
|
sed -i "/uv/d" requirements.txt
|
||||||
sed -i "/uv/d" requirements_diff.txt
|
sed -i "/uv/d" requirements_diff.txt
|
||||||
|
|
||||||
# home-assistant/wheels doesn't support sha pinning
|
|
||||||
- name: Build wheels
|
- name: Build wheels
|
||||||
uses: home-assistant/wheels@2025.09.1
|
uses: home-assistant/wheels@2024.11.0
|
||||||
with:
|
with:
|
||||||
abi: ${{ matrix.abi }}
|
abi: ${{ matrix.abi }}
|
||||||
tag: musllinux_1_2
|
tag: musllinux_1_2
|
||||||
@@ -185,25 +184,25 @@ jobs:
|
|||||||
arch: ${{ fromJson(needs.init.outputs.architectures) }}
|
arch: ${{ fromJson(needs.init.outputs.architectures) }}
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0
|
uses: actions/checkout@v4.2.2
|
||||||
|
|
||||||
- name: Download env_file
|
- name: Download env_file
|
||||||
uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0
|
uses: actions/download-artifact@v4.1.9
|
||||||
with:
|
with:
|
||||||
name: env_file
|
name: env_file
|
||||||
|
|
||||||
- name: Download build_constraints
|
- name: Download build_constraints
|
||||||
uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0
|
uses: actions/download-artifact@v4.1.9
|
||||||
with:
|
with:
|
||||||
name: build_constraints
|
name: build_constraints
|
||||||
|
|
||||||
- name: Download requirements_diff
|
- name: Download requirements_diff
|
||||||
uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0
|
uses: actions/download-artifact@v4.1.9
|
||||||
with:
|
with:
|
||||||
name: requirements_diff
|
name: requirements_diff
|
||||||
|
|
||||||
- name: Download requirements_all_wheels
|
- name: Download requirements_all_wheels
|
||||||
uses: actions/download-artifact@634f93cb2916e3fdff6788551b99b062d0335ce0 # v5.0.0
|
uses: actions/download-artifact@v4.1.9
|
||||||
with:
|
with:
|
||||||
name: requirements_all_wheels
|
name: requirements_all_wheels
|
||||||
|
|
||||||
@@ -219,9 +218,16 @@ jobs:
|
|||||||
sed -i "/uv/d" requirements.txt
|
sed -i "/uv/d" requirements.txt
|
||||||
sed -i "/uv/d" requirements_diff.txt
|
sed -i "/uv/d" requirements_diff.txt
|
||||||
|
|
||||||
# home-assistant/wheels doesn't support sha pinning
|
- name: Split requirements all
|
||||||
- name: Build wheels
|
run: |
|
||||||
uses: home-assistant/wheels@2025.09.1
|
# We split requirements all into multiple files.
|
||||||
|
# This is to prevent the build from running out of memory when
|
||||||
|
# resolving packages on 32-bits systems (like armhf, armv7).
|
||||||
|
|
||||||
|
split -l $(expr $(expr $(cat requirements_all.txt | wc -l) + 1) / 3) requirements_all_wheels_${{ matrix.arch }}.txt requirements_all.txt
|
||||||
|
|
||||||
|
- name: Build wheels (part 1)
|
||||||
|
uses: home-assistant/wheels@2024.11.0
|
||||||
with:
|
with:
|
||||||
abi: ${{ matrix.abi }}
|
abi: ${{ matrix.abi }}
|
||||||
tag: musllinux_1_2
|
tag: musllinux_1_2
|
||||||
@@ -232,4 +238,32 @@ jobs:
|
|||||||
skip-binary: aiohttp;charset-normalizer;grpcio;multidict;SQLAlchemy;propcache;protobuf;pymicro-vad;yarl
|
skip-binary: aiohttp;charset-normalizer;grpcio;multidict;SQLAlchemy;propcache;protobuf;pymicro-vad;yarl
|
||||||
constraints: "homeassistant/package_constraints.txt"
|
constraints: "homeassistant/package_constraints.txt"
|
||||||
requirements-diff: "requirements_diff.txt"
|
requirements-diff: "requirements_diff.txt"
|
||||||
requirements: "requirements_all.txt"
|
requirements: "requirements_all.txtaa"
|
||||||
|
|
||||||
|
- name: Build wheels (part 2)
|
||||||
|
uses: home-assistant/wheels@2024.11.0
|
||||||
|
with:
|
||||||
|
abi: ${{ matrix.abi }}
|
||||||
|
tag: musllinux_1_2
|
||||||
|
arch: ${{ matrix.arch }}
|
||||||
|
wheels-key: ${{ secrets.WHEELS_KEY }}
|
||||||
|
env-file: true
|
||||||
|
apk: "bluez-dev;libffi-dev;openssl-dev;glib-dev;eudev-dev;libxml2-dev;libxslt-dev;libpng-dev;libjpeg-turbo-dev;tiff-dev;cups-dev;gmp-dev;mpfr-dev;mpc1-dev;ffmpeg-dev;gammu-dev;yaml-dev;openblas-dev;fftw-dev;lapack-dev;gfortran;blas-dev;eigen-dev;freetype-dev;glew-dev;harfbuzz-dev;hdf5-dev;libdc1394-dev;libtbb-dev;mesa-dev;openexr-dev;openjpeg-dev;uchardet-dev;nasm;zlib-ng-dev"
|
||||||
|
skip-binary: aiohttp;charset-normalizer;grpcio;multidict;SQLAlchemy;propcache;protobuf;pymicro-vad;yarl
|
||||||
|
constraints: "homeassistant/package_constraints.txt"
|
||||||
|
requirements-diff: "requirements_diff.txt"
|
||||||
|
requirements: "requirements_all.txtab"
|
||||||
|
|
||||||
|
- name: Build wheels (part 3)
|
||||||
|
uses: home-assistant/wheels@2024.11.0
|
||||||
|
with:
|
||||||
|
abi: ${{ matrix.abi }}
|
||||||
|
tag: musllinux_1_2
|
||||||
|
arch: ${{ matrix.arch }}
|
||||||
|
wheels-key: ${{ secrets.WHEELS_KEY }}
|
||||||
|
env-file: true
|
||||||
|
apk: "bluez-dev;libffi-dev;openssl-dev;glib-dev;eudev-dev;libxml2-dev;libxslt-dev;libpng-dev;libjpeg-turbo-dev;tiff-dev;cups-dev;gmp-dev;mpfr-dev;mpc1-dev;ffmpeg-dev;gammu-dev;yaml-dev;openblas-dev;fftw-dev;lapack-dev;gfortran;blas-dev;eigen-dev;freetype-dev;glew-dev;harfbuzz-dev;hdf5-dev;libdc1394-dev;libtbb-dev;mesa-dev;openexr-dev;openjpeg-dev;uchardet-dev;nasm;zlib-ng-dev"
|
||||||
|
skip-binary: aiohttp;charset-normalizer;grpcio;multidict;SQLAlchemy;propcache;protobuf;pymicro-vad;yarl
|
||||||
|
constraints: "homeassistant/package_constraints.txt"
|
||||||
|
requirements-diff: "requirements_diff.txt"
|
||||||
|
requirements: "requirements_all.txtac"
|
||||||
|
8
.gitignore
vendored
8
.gitignore
vendored
@@ -69,7 +69,6 @@ test-reports/
|
|||||||
test-results.xml
|
test-results.xml
|
||||||
test-output.xml
|
test-output.xml
|
||||||
pytest-*.txt
|
pytest-*.txt
|
||||||
junit.xml
|
|
||||||
|
|
||||||
# Translations
|
# Translations
|
||||||
*.mo
|
*.mo
|
||||||
@@ -79,6 +78,7 @@ junit.xml
|
|||||||
.project
|
.project
|
||||||
.pydevproject
|
.pydevproject
|
||||||
|
|
||||||
|
.python-version
|
||||||
.tool-versions
|
.tool-versions
|
||||||
|
|
||||||
# emacs auto backups
|
# emacs auto backups
|
||||||
@@ -136,8 +136,4 @@ tmp_cache
|
|||||||
.ropeproject
|
.ropeproject
|
||||||
|
|
||||||
# Will be created from script/split_tests.py
|
# Will be created from script/split_tests.py
|
||||||
pytest_buckets.txt
|
pytest_buckets.txt
|
||||||
|
|
||||||
# AI tooling
|
|
||||||
.claude/settings.local.json
|
|
||||||
|
|
@@ -1,8 +1,8 @@
|
|||||||
repos:
|
repos:
|
||||||
- repo: https://github.com/astral-sh/ruff-pre-commit
|
- repo: https://github.com/astral-sh/ruff-pre-commit
|
||||||
rev: v0.13.0
|
rev: v0.9.8
|
||||||
hooks:
|
hooks:
|
||||||
- id: ruff-check
|
- id: ruff
|
||||||
args:
|
args:
|
||||||
- --fix
|
- --fix
|
||||||
- id: ruff-format
|
- id: ruff-format
|
||||||
@@ -18,7 +18,7 @@ repos:
|
|||||||
exclude_types: [csv, json, html]
|
exclude_types: [csv, json, html]
|
||||||
exclude: ^tests/fixtures/|homeassistant/generated/|tests/components/.*/snapshots/
|
exclude: ^tests/fixtures/|homeassistant/generated/|tests/components/.*/snapshots/
|
||||||
- repo: https://github.com/pre-commit/pre-commit-hooks
|
- repo: https://github.com/pre-commit/pre-commit-hooks
|
||||||
rev: v6.0.0
|
rev: v5.0.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: check-executables-have-shebangs
|
- id: check-executables-have-shebangs
|
||||||
stages: [manual]
|
stages: [manual]
|
||||||
@@ -30,7 +30,7 @@ repos:
|
|||||||
- --branch=master
|
- --branch=master
|
||||||
- --branch=rc
|
- --branch=rc
|
||||||
- repo: https://github.com/adrienverge/yamllint.git
|
- repo: https://github.com/adrienverge/yamllint.git
|
||||||
rev: v1.37.1
|
rev: v1.35.1
|
||||||
hooks:
|
hooks:
|
||||||
- id: yamllint
|
- id: yamllint
|
||||||
- repo: https://github.com/pre-commit/mirrors-prettier
|
- repo: https://github.com/pre-commit/mirrors-prettier
|
||||||
|
@@ -1 +0,0 @@
|
|||||||
3.13
|
|
@@ -53,7 +53,6 @@ homeassistant.components.air_quality.*
|
|||||||
homeassistant.components.airgradient.*
|
homeassistant.components.airgradient.*
|
||||||
homeassistant.components.airly.*
|
homeassistant.components.airly.*
|
||||||
homeassistant.components.airnow.*
|
homeassistant.components.airnow.*
|
||||||
homeassistant.components.airos.*
|
|
||||||
homeassistant.components.airq.*
|
homeassistant.components.airq.*
|
||||||
homeassistant.components.airthings.*
|
homeassistant.components.airthings.*
|
||||||
homeassistant.components.airthings_ble.*
|
homeassistant.components.airthings_ble.*
|
||||||
@@ -66,9 +65,7 @@ homeassistant.components.aladdin_connect.*
|
|||||||
homeassistant.components.alarm_control_panel.*
|
homeassistant.components.alarm_control_panel.*
|
||||||
homeassistant.components.alert.*
|
homeassistant.components.alert.*
|
||||||
homeassistant.components.alexa.*
|
homeassistant.components.alexa.*
|
||||||
homeassistant.components.alexa_devices.*
|
|
||||||
homeassistant.components.alpha_vantage.*
|
homeassistant.components.alpha_vantage.*
|
||||||
homeassistant.components.altruist.*
|
|
||||||
homeassistant.components.amazon_polly.*
|
homeassistant.components.amazon_polly.*
|
||||||
homeassistant.components.amberelectric.*
|
homeassistant.components.amberelectric.*
|
||||||
homeassistant.components.ambient_network.*
|
homeassistant.components.ambient_network.*
|
||||||
@@ -122,7 +119,6 @@ homeassistant.components.bluetooth_adapters.*
|
|||||||
homeassistant.components.bluetooth_tracker.*
|
homeassistant.components.bluetooth_tracker.*
|
||||||
homeassistant.components.bmw_connected_drive.*
|
homeassistant.components.bmw_connected_drive.*
|
||||||
homeassistant.components.bond.*
|
homeassistant.components.bond.*
|
||||||
homeassistant.components.bosch_alarm.*
|
|
||||||
homeassistant.components.braviatv.*
|
homeassistant.components.braviatv.*
|
||||||
homeassistant.components.bring.*
|
homeassistant.components.bring.*
|
||||||
homeassistant.components.brother.*
|
homeassistant.components.brother.*
|
||||||
@@ -140,9 +136,7 @@ homeassistant.components.clicksend.*
|
|||||||
homeassistant.components.climate.*
|
homeassistant.components.climate.*
|
||||||
homeassistant.components.cloud.*
|
homeassistant.components.cloud.*
|
||||||
homeassistant.components.co2signal.*
|
homeassistant.components.co2signal.*
|
||||||
homeassistant.components.comelit.*
|
|
||||||
homeassistant.components.command_line.*
|
homeassistant.components.command_line.*
|
||||||
homeassistant.components.compit.*
|
|
||||||
homeassistant.components.config.*
|
homeassistant.components.config.*
|
||||||
homeassistant.components.configurator.*
|
homeassistant.components.configurator.*
|
||||||
homeassistant.components.cookidoo.*
|
homeassistant.components.cookidoo.*
|
||||||
@@ -170,7 +164,6 @@ homeassistant.components.dnsip.*
|
|||||||
homeassistant.components.doorbird.*
|
homeassistant.components.doorbird.*
|
||||||
homeassistant.components.dormakaba_dkey.*
|
homeassistant.components.dormakaba_dkey.*
|
||||||
homeassistant.components.downloader.*
|
homeassistant.components.downloader.*
|
||||||
homeassistant.components.droplet.*
|
|
||||||
homeassistant.components.dsmr.*
|
homeassistant.components.dsmr.*
|
||||||
homeassistant.components.duckdns.*
|
homeassistant.components.duckdns.*
|
||||||
homeassistant.components.dunehd.*
|
homeassistant.components.dunehd.*
|
||||||
@@ -203,7 +196,6 @@ homeassistant.components.feedreader.*
|
|||||||
homeassistant.components.file_upload.*
|
homeassistant.components.file_upload.*
|
||||||
homeassistant.components.filesize.*
|
homeassistant.components.filesize.*
|
||||||
homeassistant.components.filter.*
|
homeassistant.components.filter.*
|
||||||
homeassistant.components.firefly_iii.*
|
|
||||||
homeassistant.components.fitbit.*
|
homeassistant.components.fitbit.*
|
||||||
homeassistant.components.flexit_bacnet.*
|
homeassistant.components.flexit_bacnet.*
|
||||||
homeassistant.components.flux_led.*
|
homeassistant.components.flux_led.*
|
||||||
@@ -221,7 +213,6 @@ homeassistant.components.generic_thermostat.*
|
|||||||
homeassistant.components.geo_location.*
|
homeassistant.components.geo_location.*
|
||||||
homeassistant.components.geocaching.*
|
homeassistant.components.geocaching.*
|
||||||
homeassistant.components.gios.*
|
homeassistant.components.gios.*
|
||||||
homeassistant.components.github.*
|
|
||||||
homeassistant.components.glances.*
|
homeassistant.components.glances.*
|
||||||
homeassistant.components.go2rtc.*
|
homeassistant.components.go2rtc.*
|
||||||
homeassistant.components.goalzero.*
|
homeassistant.components.goalzero.*
|
||||||
@@ -277,7 +268,6 @@ homeassistant.components.image_processing.*
|
|||||||
homeassistant.components.image_upload.*
|
homeassistant.components.image_upload.*
|
||||||
homeassistant.components.imap.*
|
homeassistant.components.imap.*
|
||||||
homeassistant.components.imgw_pib.*
|
homeassistant.components.imgw_pib.*
|
||||||
homeassistant.components.immich.*
|
|
||||||
homeassistant.components.incomfort.*
|
homeassistant.components.incomfort.*
|
||||||
homeassistant.components.input_button.*
|
homeassistant.components.input_button.*
|
||||||
homeassistant.components.input_select.*
|
homeassistant.components.input_select.*
|
||||||
@@ -299,7 +289,6 @@ homeassistant.components.kaleidescape.*
|
|||||||
homeassistant.components.knocki.*
|
homeassistant.components.knocki.*
|
||||||
homeassistant.components.knx.*
|
homeassistant.components.knx.*
|
||||||
homeassistant.components.kraken.*
|
homeassistant.components.kraken.*
|
||||||
homeassistant.components.kulersky.*
|
|
||||||
homeassistant.components.lacrosse.*
|
homeassistant.components.lacrosse.*
|
||||||
homeassistant.components.lacrosse_view.*
|
homeassistant.components.lacrosse_view.*
|
||||||
homeassistant.components.lamarzocco.*
|
homeassistant.components.lamarzocco.*
|
||||||
@@ -311,10 +300,10 @@ homeassistant.components.ld2410_ble.*
|
|||||||
homeassistant.components.led_ble.*
|
homeassistant.components.led_ble.*
|
||||||
homeassistant.components.lektrico.*
|
homeassistant.components.lektrico.*
|
||||||
homeassistant.components.letpot.*
|
homeassistant.components.letpot.*
|
||||||
homeassistant.components.libre_hardware_monitor.*
|
|
||||||
homeassistant.components.lidarr.*
|
homeassistant.components.lidarr.*
|
||||||
homeassistant.components.lifx.*
|
homeassistant.components.lifx.*
|
||||||
homeassistant.components.light.*
|
homeassistant.components.light.*
|
||||||
|
homeassistant.components.linear_garage_door.*
|
||||||
homeassistant.components.linkplay.*
|
homeassistant.components.linkplay.*
|
||||||
homeassistant.components.litejet.*
|
homeassistant.components.litejet.*
|
||||||
homeassistant.components.litterrobot.*
|
homeassistant.components.litterrobot.*
|
||||||
@@ -327,7 +316,6 @@ homeassistant.components.london_underground.*
|
|||||||
homeassistant.components.lookin.*
|
homeassistant.components.lookin.*
|
||||||
homeassistant.components.lovelace.*
|
homeassistant.components.lovelace.*
|
||||||
homeassistant.components.luftdaten.*
|
homeassistant.components.luftdaten.*
|
||||||
homeassistant.components.lunatone.*
|
|
||||||
homeassistant.components.madvr.*
|
homeassistant.components.madvr.*
|
||||||
homeassistant.components.manual.*
|
homeassistant.components.manual.*
|
||||||
homeassistant.components.mastodon.*
|
homeassistant.components.mastodon.*
|
||||||
@@ -341,7 +329,6 @@ homeassistant.components.media_player.*
|
|||||||
homeassistant.components.media_source.*
|
homeassistant.components.media_source.*
|
||||||
homeassistant.components.met_eireann.*
|
homeassistant.components.met_eireann.*
|
||||||
homeassistant.components.metoffice.*
|
homeassistant.components.metoffice.*
|
||||||
homeassistant.components.miele.*
|
|
||||||
homeassistant.components.mikrotik.*
|
homeassistant.components.mikrotik.*
|
||||||
homeassistant.components.min_max.*
|
homeassistant.components.min_max.*
|
||||||
homeassistant.components.minecraft_server.*
|
homeassistant.components.minecraft_server.*
|
||||||
@@ -373,23 +360,18 @@ homeassistant.components.no_ip.*
|
|||||||
homeassistant.components.nordpool.*
|
homeassistant.components.nordpool.*
|
||||||
homeassistant.components.notify.*
|
homeassistant.components.notify.*
|
||||||
homeassistant.components.notion.*
|
homeassistant.components.notion.*
|
||||||
homeassistant.components.ntfy.*
|
|
||||||
homeassistant.components.number.*
|
homeassistant.components.number.*
|
||||||
homeassistant.components.nut.*
|
homeassistant.components.nut.*
|
||||||
homeassistant.components.ohme.*
|
|
||||||
homeassistant.components.onboarding.*
|
homeassistant.components.onboarding.*
|
||||||
homeassistant.components.oncue.*
|
homeassistant.components.oncue.*
|
||||||
homeassistant.components.onedrive.*
|
homeassistant.components.onedrive.*
|
||||||
homeassistant.components.onewire.*
|
homeassistant.components.onewire.*
|
||||||
homeassistant.components.onkyo.*
|
homeassistant.components.onkyo.*
|
||||||
homeassistant.components.open_meteo.*
|
homeassistant.components.open_meteo.*
|
||||||
homeassistant.components.open_router.*
|
|
||||||
homeassistant.components.openai_conversation.*
|
homeassistant.components.openai_conversation.*
|
||||||
homeassistant.components.openexchangerates.*
|
homeassistant.components.openexchangerates.*
|
||||||
homeassistant.components.opensky.*
|
homeassistant.components.opensky.*
|
||||||
homeassistant.components.openuv.*
|
homeassistant.components.openuv.*
|
||||||
homeassistant.components.opnsense.*
|
|
||||||
homeassistant.components.opower.*
|
|
||||||
homeassistant.components.oralb.*
|
homeassistant.components.oralb.*
|
||||||
homeassistant.components.otbr.*
|
homeassistant.components.otbr.*
|
||||||
homeassistant.components.overkiz.*
|
homeassistant.components.overkiz.*
|
||||||
@@ -397,16 +379,13 @@ homeassistant.components.overseerr.*
|
|||||||
homeassistant.components.p1_monitor.*
|
homeassistant.components.p1_monitor.*
|
||||||
homeassistant.components.pandora.*
|
homeassistant.components.pandora.*
|
||||||
homeassistant.components.panel_custom.*
|
homeassistant.components.panel_custom.*
|
||||||
homeassistant.components.paperless_ngx.*
|
|
||||||
homeassistant.components.peblar.*
|
homeassistant.components.peblar.*
|
||||||
homeassistant.components.peco.*
|
homeassistant.components.peco.*
|
||||||
homeassistant.components.pegel_online.*
|
|
||||||
homeassistant.components.persistent_notification.*
|
homeassistant.components.persistent_notification.*
|
||||||
homeassistant.components.person.*
|
homeassistant.components.person.*
|
||||||
homeassistant.components.pi_hole.*
|
homeassistant.components.pi_hole.*
|
||||||
homeassistant.components.ping.*
|
homeassistant.components.ping.*
|
||||||
homeassistant.components.plugwise.*
|
homeassistant.components.plugwise.*
|
||||||
homeassistant.components.portainer.*
|
|
||||||
homeassistant.components.powerfox.*
|
homeassistant.components.powerfox.*
|
||||||
homeassistant.components.powerwall.*
|
homeassistant.components.powerwall.*
|
||||||
homeassistant.components.private_ble_device.*
|
homeassistant.components.private_ble_device.*
|
||||||
@@ -417,7 +396,6 @@ homeassistant.components.pure_energie.*
|
|||||||
homeassistant.components.purpleair.*
|
homeassistant.components.purpleair.*
|
||||||
homeassistant.components.pushbullet.*
|
homeassistant.components.pushbullet.*
|
||||||
homeassistant.components.pvoutput.*
|
homeassistant.components.pvoutput.*
|
||||||
homeassistant.components.pyload.*
|
|
||||||
homeassistant.components.python_script.*
|
homeassistant.components.python_script.*
|
||||||
homeassistant.components.qbus.*
|
homeassistant.components.qbus.*
|
||||||
homeassistant.components.qnap_qsw.*
|
homeassistant.components.qnap_qsw.*
|
||||||
@@ -432,7 +410,6 @@ homeassistant.components.recollect_waste.*
|
|||||||
homeassistant.components.recorder.*
|
homeassistant.components.recorder.*
|
||||||
homeassistant.components.remember_the_milk.*
|
homeassistant.components.remember_the_milk.*
|
||||||
homeassistant.components.remote.*
|
homeassistant.components.remote.*
|
||||||
homeassistant.components.remote_calendar.*
|
|
||||||
homeassistant.components.renault.*
|
homeassistant.components.renault.*
|
||||||
homeassistant.components.reolink.*
|
homeassistant.components.reolink.*
|
||||||
homeassistant.components.repairs.*
|
homeassistant.components.repairs.*
|
||||||
@@ -446,9 +423,9 @@ homeassistant.components.rituals_perfume_genie.*
|
|||||||
homeassistant.components.roborock.*
|
homeassistant.components.roborock.*
|
||||||
homeassistant.components.roku.*
|
homeassistant.components.roku.*
|
||||||
homeassistant.components.romy.*
|
homeassistant.components.romy.*
|
||||||
homeassistant.components.route_b_smart_meter.*
|
|
||||||
homeassistant.components.rpi_power.*
|
homeassistant.components.rpi_power.*
|
||||||
homeassistant.components.rss_feed_template.*
|
homeassistant.components.rss_feed_template.*
|
||||||
|
homeassistant.components.rtsp_to_webrtc.*
|
||||||
homeassistant.components.russound_rio.*
|
homeassistant.components.russound_rio.*
|
||||||
homeassistant.components.ruuvi_gateway.*
|
homeassistant.components.ruuvi_gateway.*
|
||||||
homeassistant.components.ruuvitag_ble.*
|
homeassistant.components.ruuvitag_ble.*
|
||||||
@@ -467,7 +444,6 @@ homeassistant.components.sensorpush_cloud.*
|
|||||||
homeassistant.components.sensoterra.*
|
homeassistant.components.sensoterra.*
|
||||||
homeassistant.components.senz.*
|
homeassistant.components.senz.*
|
||||||
homeassistant.components.sfr_box.*
|
homeassistant.components.sfr_box.*
|
||||||
homeassistant.components.sftp_storage.*
|
|
||||||
homeassistant.components.shell_command.*
|
homeassistant.components.shell_command.*
|
||||||
homeassistant.components.shelly.*
|
homeassistant.components.shelly.*
|
||||||
homeassistant.components.shopping_list.*
|
homeassistant.components.shopping_list.*
|
||||||
@@ -476,11 +452,9 @@ homeassistant.components.simplisafe.*
|
|||||||
homeassistant.components.siren.*
|
homeassistant.components.siren.*
|
||||||
homeassistant.components.skybell.*
|
homeassistant.components.skybell.*
|
||||||
homeassistant.components.slack.*
|
homeassistant.components.slack.*
|
||||||
homeassistant.components.sleep_as_android.*
|
|
||||||
homeassistant.components.sleepiq.*
|
homeassistant.components.sleepiq.*
|
||||||
homeassistant.components.smhi.*
|
homeassistant.components.smhi.*
|
||||||
homeassistant.components.smlight.*
|
homeassistant.components.smlight.*
|
||||||
homeassistant.components.smtp.*
|
|
||||||
homeassistant.components.snooz.*
|
homeassistant.components.snooz.*
|
||||||
homeassistant.components.solarlog.*
|
homeassistant.components.solarlog.*
|
||||||
homeassistant.components.sonarr.*
|
homeassistant.components.sonarr.*
|
||||||
@@ -512,12 +486,10 @@ homeassistant.components.tag.*
|
|||||||
homeassistant.components.tailscale.*
|
homeassistant.components.tailscale.*
|
||||||
homeassistant.components.tailwind.*
|
homeassistant.components.tailwind.*
|
||||||
homeassistant.components.tami4.*
|
homeassistant.components.tami4.*
|
||||||
homeassistant.components.tankerkoenig.*
|
|
||||||
homeassistant.components.tautulli.*
|
homeassistant.components.tautulli.*
|
||||||
homeassistant.components.tcp.*
|
homeassistant.components.tcp.*
|
||||||
homeassistant.components.technove.*
|
homeassistant.components.technove.*
|
||||||
homeassistant.components.tedee.*
|
homeassistant.components.tedee.*
|
||||||
homeassistant.components.telegram_bot.*
|
|
||||||
homeassistant.components.text.*
|
homeassistant.components.text.*
|
||||||
homeassistant.components.thethingsnetwork.*
|
homeassistant.components.thethingsnetwork.*
|
||||||
homeassistant.components.threshold.*
|
homeassistant.components.threshold.*
|
||||||
@@ -548,7 +520,6 @@ homeassistant.components.unifiprotect.*
|
|||||||
homeassistant.components.upcloud.*
|
homeassistant.components.upcloud.*
|
||||||
homeassistant.components.update.*
|
homeassistant.components.update.*
|
||||||
homeassistant.components.uptime.*
|
homeassistant.components.uptime.*
|
||||||
homeassistant.components.uptime_kuma.*
|
|
||||||
homeassistant.components.uptimerobot.*
|
homeassistant.components.uptimerobot.*
|
||||||
homeassistant.components.usb.*
|
homeassistant.components.usb.*
|
||||||
homeassistant.components.uvc.*
|
homeassistant.components.uvc.*
|
||||||
@@ -556,10 +527,7 @@ homeassistant.components.vacuum.*
|
|||||||
homeassistant.components.vallox.*
|
homeassistant.components.vallox.*
|
||||||
homeassistant.components.valve.*
|
homeassistant.components.valve.*
|
||||||
homeassistant.components.velbus.*
|
homeassistant.components.velbus.*
|
||||||
homeassistant.components.vivotek.*
|
|
||||||
homeassistant.components.vlc_telnet.*
|
homeassistant.components.vlc_telnet.*
|
||||||
homeassistant.components.vodafone_station.*
|
|
||||||
homeassistant.components.volvo.*
|
|
||||||
homeassistant.components.wake_on_lan.*
|
homeassistant.components.wake_on_lan.*
|
||||||
homeassistant.components.wake_word.*
|
homeassistant.components.wake_word.*
|
||||||
homeassistant.components.wallbox.*
|
homeassistant.components.wallbox.*
|
||||||
|
4
.vscode/tasks.json
vendored
4
.vscode/tasks.json
vendored
@@ -4,7 +4,7 @@
|
|||||||
{
|
{
|
||||||
"label": "Run Home Assistant Core",
|
"label": "Run Home Assistant Core",
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
"command": "${command:python.interpreterPath} -m homeassistant -c ./config",
|
"command": "hass -c ./config",
|
||||||
"group": "test",
|
"group": "test",
|
||||||
"presentation": {
|
"presentation": {
|
||||||
"reveal": "always",
|
"reveal": "always",
|
||||||
@@ -45,7 +45,7 @@
|
|||||||
{
|
{
|
||||||
"label": "Ruff",
|
"label": "Ruff",
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
"command": "pre-commit run ruff-check --all-files",
|
"command": "pre-commit run ruff --all-files",
|
||||||
"group": {
|
"group": {
|
||||||
"kind": "test",
|
"kind": "test",
|
||||||
"isDefault": true
|
"isDefault": true
|
||||||
|
269
CODEOWNERS
generated
269
CODEOWNERS
generated
@@ -46,10 +46,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/accuweather/ @bieniu
|
/tests/components/accuweather/ @bieniu
|
||||||
/homeassistant/components/acmeda/ @atmurray
|
/homeassistant/components/acmeda/ @atmurray
|
||||||
/tests/components/acmeda/ @atmurray
|
/tests/components/acmeda/ @atmurray
|
||||||
/homeassistant/components/actron_air/ @kclif9 @JagadishDhanamjayam
|
/homeassistant/components/adax/ @danielhiversen
|
||||||
/tests/components/actron_air/ @kclif9 @JagadishDhanamjayam
|
/tests/components/adax/ @danielhiversen
|
||||||
/homeassistant/components/adax/ @danielhiversen @lazytarget
|
|
||||||
/tests/components/adax/ @danielhiversen @lazytarget
|
|
||||||
/homeassistant/components/adguard/ @frenck
|
/homeassistant/components/adguard/ @frenck
|
||||||
/tests/components/adguard/ @frenck
|
/tests/components/adguard/ @frenck
|
||||||
/homeassistant/components/ads/ @mrpasztoradam
|
/homeassistant/components/ads/ @mrpasztoradam
|
||||||
@@ -59,8 +57,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/aemet/ @Noltari
|
/tests/components/aemet/ @Noltari
|
||||||
/homeassistant/components/agent_dvr/ @ispysoftware
|
/homeassistant/components/agent_dvr/ @ispysoftware
|
||||||
/tests/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
|
/homeassistant/components/air_quality/ @home-assistant/core
|
||||||
/tests/components/air_quality/ @home-assistant/core
|
/tests/components/air_quality/ @home-assistant/core
|
||||||
/homeassistant/components/airgradient/ @airgradienthq @joostlek
|
/homeassistant/components/airgradient/ @airgradienthq @joostlek
|
||||||
@@ -69,8 +65,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/airly/ @bieniu
|
/tests/components/airly/ @bieniu
|
||||||
/homeassistant/components/airnow/ @asymworks
|
/homeassistant/components/airnow/ @asymworks
|
||||||
/tests/components/airnow/ @asymworks
|
/tests/components/airnow/ @asymworks
|
||||||
/homeassistant/components/airos/ @CoMPaTech
|
|
||||||
/tests/components/airos/ @CoMPaTech
|
|
||||||
/homeassistant/components/airq/ @Sibgatulin @dl2080
|
/homeassistant/components/airq/ @Sibgatulin @dl2080
|
||||||
/tests/components/airq/ @Sibgatulin @dl2080
|
/tests/components/airq/ @Sibgatulin @dl2080
|
||||||
/homeassistant/components/airthings/ @danielhiversen @LaStrada
|
/homeassistant/components/airthings/ @danielhiversen @LaStrada
|
||||||
@@ -89,18 +83,12 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/airzone/ @Noltari
|
/tests/components/airzone/ @Noltari
|
||||||
/homeassistant/components/airzone_cloud/ @Noltari
|
/homeassistant/components/airzone_cloud/ @Noltari
|
||||||
/tests/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
|
/homeassistant/components/alarm_control_panel/ @home-assistant/core
|
||||||
/tests/components/alarm_control_panel/ @home-assistant/core
|
/tests/components/alarm_control_panel/ @home-assistant/core
|
||||||
/homeassistant/components/alert/ @home-assistant/core @frenck
|
/homeassistant/components/alert/ @home-assistant/core @frenck
|
||||||
/tests/components/alert/ @home-assistant/core @frenck
|
/tests/components/alert/ @home-assistant/core @frenck
|
||||||
/homeassistant/components/alexa/ @home-assistant/cloud @ochlocracy @jbouwh
|
/homeassistant/components/alexa/ @home-assistant/cloud @ochlocracy @jbouwh
|
||||||
/tests/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/amazon_polly/ @jschlyter
|
||||||
/homeassistant/components/amberelectric/ @madpilot
|
/homeassistant/components/amberelectric/ @madpilot
|
||||||
/tests/components/amberelectric/ @madpilot
|
/tests/components/amberelectric/ @madpilot
|
||||||
@@ -109,8 +97,8 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/ambient_station/ @bachya
|
/homeassistant/components/ambient_station/ @bachya
|
||||||
/tests/components/ambient_station/ @bachya
|
/tests/components/ambient_station/ @bachya
|
||||||
/homeassistant/components/amcrest/ @flacjacket
|
/homeassistant/components/amcrest/ @flacjacket
|
||||||
/homeassistant/components/analytics/ @home-assistant/core
|
/homeassistant/components/analytics/ @home-assistant/core @ludeeus
|
||||||
/tests/components/analytics/ @home-assistant/core
|
/tests/components/analytics/ @home-assistant/core @ludeeus
|
||||||
/homeassistant/components/analytics_insights/ @joostlek
|
/homeassistant/components/analytics_insights/ @joostlek
|
||||||
/tests/components/analytics_insights/ @joostlek
|
/tests/components/analytics_insights/ @joostlek
|
||||||
/homeassistant/components/android_ip_webcam/ @engrbm87
|
/homeassistant/components/android_ip_webcam/ @engrbm87
|
||||||
@@ -156,12 +144,12 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/arve/ @ikalnyi
|
/tests/components/arve/ @ikalnyi
|
||||||
/homeassistant/components/aseko_pool_live/ @milanmeu
|
/homeassistant/components/aseko_pool_live/ @milanmeu
|
||||||
/tests/components/aseko_pool_live/ @milanmeu
|
/tests/components/aseko_pool_live/ @milanmeu
|
||||||
/homeassistant/components/assist_pipeline/ @synesthesiam @arturpragacz
|
/homeassistant/components/assist_pipeline/ @balloob @synesthesiam
|
||||||
/tests/components/assist_pipeline/ @synesthesiam @arturpragacz
|
/tests/components/assist_pipeline/ @balloob @synesthesiam
|
||||||
/homeassistant/components/assist_satellite/ @home-assistant/core @synesthesiam @arturpragacz
|
/homeassistant/components/assist_satellite/ @home-assistant/core @synesthesiam
|
||||||
/tests/components/assist_satellite/ @home-assistant/core @synesthesiam @arturpragacz
|
/tests/components/assist_satellite/ @home-assistant/core @synesthesiam
|
||||||
/homeassistant/components/asuswrt/ @kennedyshead @ollo69 @Vaskivskyi
|
/homeassistant/components/asuswrt/ @kennedyshead @ollo69
|
||||||
/tests/components/asuswrt/ @kennedyshead @ollo69 @Vaskivskyi
|
/tests/components/asuswrt/ @kennedyshead @ollo69
|
||||||
/homeassistant/components/atag/ @MatsNL
|
/homeassistant/components/atag/ @MatsNL
|
||||||
/tests/components/atag/ @MatsNL
|
/tests/components/atag/ @MatsNL
|
||||||
/homeassistant/components/aten_pe/ @mtdcr
|
/homeassistant/components/aten_pe/ @mtdcr
|
||||||
@@ -183,8 +171,6 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/avea/ @pattyland
|
/homeassistant/components/avea/ @pattyland
|
||||||
/homeassistant/components/awair/ @ahayworth @danielsjf
|
/homeassistant/components/awair/ @ahayworth @danielsjf
|
||||||
/tests/components/awair/ @ahayworth @danielsjf
|
/tests/components/awair/ @ahayworth @danielsjf
|
||||||
/homeassistant/components/aws_s3/ @tomasbedrich
|
|
||||||
/tests/components/aws_s3/ @tomasbedrich
|
|
||||||
/homeassistant/components/axis/ @Kane610
|
/homeassistant/components/axis/ @Kane610
|
||||||
/tests/components/axis/ @Kane610
|
/tests/components/axis/ @Kane610
|
||||||
/homeassistant/components/azure_data_explorer/ @kaareseras
|
/homeassistant/components/azure_data_explorer/ @kaareseras
|
||||||
@@ -214,8 +200,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/blebox/ @bbx-a @swistakm
|
/tests/components/blebox/ @bbx-a @swistakm
|
||||||
/homeassistant/components/blink/ @fronzbot @mkmer
|
/homeassistant/components/blink/ @fronzbot @mkmer
|
||||||
/tests/components/blink/ @fronzbot @mkmer
|
/tests/components/blink/ @fronzbot @mkmer
|
||||||
/homeassistant/components/blue_current/ @gleeuwen @NickKoepr @jtodorova23
|
/homeassistant/components/blue_current/ @Floris272 @gleeuwen
|
||||||
/tests/components/blue_current/ @gleeuwen @NickKoepr @jtodorova23
|
/tests/components/blue_current/ @Floris272 @gleeuwen
|
||||||
/homeassistant/components/bluemaestro/ @bdraco
|
/homeassistant/components/bluemaestro/ @bdraco
|
||||||
/tests/components/bluemaestro/ @bdraco
|
/tests/components/bluemaestro/ @bdraco
|
||||||
/homeassistant/components/blueprint/ @home-assistant/core
|
/homeassistant/components/blueprint/ @home-assistant/core
|
||||||
@@ -230,8 +216,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/bmw_connected_drive/ @gerard33 @rikroe
|
/tests/components/bmw_connected_drive/ @gerard33 @rikroe
|
||||||
/homeassistant/components/bond/ @bdraco @prystupa @joshs85 @marciogranzotto
|
/homeassistant/components/bond/ @bdraco @prystupa @joshs85 @marciogranzotto
|
||||||
/tests/components/bond/ @bdraco @prystupa @joshs85 @marciogranzotto
|
/tests/components/bond/ @bdraco @prystupa @joshs85 @marciogranzotto
|
||||||
/homeassistant/components/bosch_alarm/ @mag1024 @sanjay900
|
|
||||||
/tests/components/bosch_alarm/ @mag1024 @sanjay900
|
|
||||||
/homeassistant/components/bosch_shc/ @tschamm
|
/homeassistant/components/bosch_shc/ @tschamm
|
||||||
/tests/components/bosch_shc/ @tschamm
|
/tests/components/bosch_shc/ @tschamm
|
||||||
/homeassistant/components/braviatv/ @bieniu @Drafteed
|
/homeassistant/components/braviatv/ @bieniu @Drafteed
|
||||||
@@ -294,16 +278,14 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/command_line/ @gjohansson-ST
|
/tests/components/command_line/ @gjohansson-ST
|
||||||
/homeassistant/components/compensation/ @Petro31
|
/homeassistant/components/compensation/ @Petro31
|
||||||
/tests/components/compensation/ @Petro31
|
/tests/components/compensation/ @Petro31
|
||||||
/homeassistant/components/compit/ @Przemko92
|
|
||||||
/tests/components/compit/ @Przemko92
|
|
||||||
/homeassistant/components/config/ @home-assistant/core
|
/homeassistant/components/config/ @home-assistant/core
|
||||||
/tests/components/config/ @home-assistant/core
|
/tests/components/config/ @home-assistant/core
|
||||||
/homeassistant/components/configurator/ @home-assistant/core
|
/homeassistant/components/configurator/ @home-assistant/core
|
||||||
/tests/components/configurator/ @home-assistant/core
|
/tests/components/configurator/ @home-assistant/core
|
||||||
/homeassistant/components/control4/ @lawtancool
|
/homeassistant/components/control4/ @lawtancool
|
||||||
/tests/components/control4/ @lawtancool
|
/tests/components/control4/ @lawtancool
|
||||||
/homeassistant/components/conversation/ @home-assistant/core @synesthesiam @arturpragacz
|
/homeassistant/components/conversation/ @home-assistant/core @synesthesiam
|
||||||
/tests/components/conversation/ @home-assistant/core @synesthesiam @arturpragacz
|
/tests/components/conversation/ @home-assistant/core @synesthesiam
|
||||||
/homeassistant/components/cookidoo/ @miaucl
|
/homeassistant/components/cookidoo/ @miaucl
|
||||||
/tests/components/cookidoo/ @miaucl
|
/tests/components/cookidoo/ @miaucl
|
||||||
/homeassistant/components/coolmaster/ @OnFreund
|
/homeassistant/components/coolmaster/ @OnFreund
|
||||||
@@ -317,9 +299,6 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/crownstone/ @Crownstone @RicArch97
|
/homeassistant/components/crownstone/ @Crownstone @RicArch97
|
||||||
/tests/components/crownstone/ @Crownstone @RicArch97
|
/tests/components/crownstone/ @Crownstone @RicArch97
|
||||||
/homeassistant/components/cups/ @fabaff
|
/homeassistant/components/cups/ @fabaff
|
||||||
/tests/components/cups/ @fabaff
|
|
||||||
/homeassistant/components/cync/ @Kinachi249
|
|
||||||
/tests/components/cync/ @Kinachi249
|
|
||||||
/homeassistant/components/daikin/ @fredrike
|
/homeassistant/components/daikin/ @fredrike
|
||||||
/tests/components/daikin/ @fredrike
|
/tests/components/daikin/ @fredrike
|
||||||
/homeassistant/components/date/ @home-assistant/core
|
/homeassistant/components/date/ @home-assistant/core
|
||||||
@@ -341,8 +320,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/demo/ @home-assistant/core
|
/tests/components/demo/ @home-assistant/core
|
||||||
/homeassistant/components/denonavr/ @ol-iver @starkillerOG
|
/homeassistant/components/denonavr/ @ol-iver @starkillerOG
|
||||||
/tests/components/denonavr/ @ol-iver @starkillerOG
|
/tests/components/denonavr/ @ol-iver @starkillerOG
|
||||||
/homeassistant/components/derivative/ @afaucogney @karwosts
|
/homeassistant/components/derivative/ @afaucogney
|
||||||
/tests/components/derivative/ @afaucogney @karwosts
|
/tests/components/derivative/ @afaucogney
|
||||||
/homeassistant/components/devialet/ @fwestenberg
|
/homeassistant/components/devialet/ @fwestenberg
|
||||||
/tests/components/devialet/ @fwestenberg
|
/tests/components/devialet/ @fwestenberg
|
||||||
/homeassistant/components/device_automation/ @home-assistant/core
|
/homeassistant/components/device_automation/ @home-assistant/core
|
||||||
@@ -383,8 +362,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/dremel_3d_printer/ @tkdrob
|
/tests/components/dremel_3d_printer/ @tkdrob
|
||||||
/homeassistant/components/drop_connect/ @ChandlerSystems @pfrazer
|
/homeassistant/components/drop_connect/ @ChandlerSystems @pfrazer
|
||||||
/tests/components/drop_connect/ @ChandlerSystems @pfrazer
|
/tests/components/drop_connect/ @ChandlerSystems @pfrazer
|
||||||
/homeassistant/components/droplet/ @sarahseidman
|
|
||||||
/tests/components/droplet/ @sarahseidman
|
|
||||||
/homeassistant/components/dsmr/ @Robbie1221
|
/homeassistant/components/dsmr/ @Robbie1221
|
||||||
/tests/components/dsmr/ @Robbie1221
|
/tests/components/dsmr/ @Robbie1221
|
||||||
/homeassistant/components/dsmr_reader/ @sorted-bits @glodenox @erwindouna
|
/homeassistant/components/dsmr_reader/ @sorted-bits @glodenox @erwindouna
|
||||||
@@ -414,8 +391,6 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/egardia/ @jeroenterheerdt
|
/homeassistant/components/egardia/ @jeroenterheerdt
|
||||||
/homeassistant/components/eheimdigital/ @autinerd
|
/homeassistant/components/eheimdigital/ @autinerd
|
||||||
/tests/components/eheimdigital/ @autinerd
|
/tests/components/eheimdigital/ @autinerd
|
||||||
/homeassistant/components/ekeybionyx/ @richardpolzer
|
|
||||||
/tests/components/ekeybionyx/ @richardpolzer
|
|
||||||
/homeassistant/components/electrasmart/ @jafar-atili
|
/homeassistant/components/electrasmart/ @jafar-atili
|
||||||
/tests/components/electrasmart/ @jafar-atili
|
/tests/components/electrasmart/ @jafar-atili
|
||||||
/homeassistant/components/electric_kiwi/ @mikey0000
|
/homeassistant/components/electric_kiwi/ @mikey0000
|
||||||
@@ -434,8 +409,6 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/emby/ @mezz64
|
/homeassistant/components/emby/ @mezz64
|
||||||
/homeassistant/components/emoncms/ @borpin @alexandrecuer
|
/homeassistant/components/emoncms/ @borpin @alexandrecuer
|
||||||
/tests/components/emoncms/ @borpin @alexandrecuer
|
/tests/components/emoncms/ @borpin @alexandrecuer
|
||||||
/homeassistant/components/emoncms_history/ @alexandrecuer
|
|
||||||
/tests/components/emoncms_history/ @alexandrecuer
|
|
||||||
/homeassistant/components/emonitor/ @bdraco
|
/homeassistant/components/emonitor/ @bdraco
|
||||||
/tests/components/emonitor/ @bdraco
|
/tests/components/emonitor/ @bdraco
|
||||||
/homeassistant/components/emulated_hue/ @bdraco @Tho85
|
/homeassistant/components/emulated_hue/ @bdraco @Tho85
|
||||||
@@ -450,12 +423,14 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/energyzero/ @klaasnicolaas
|
/tests/components/energyzero/ @klaasnicolaas
|
||||||
/homeassistant/components/enigma2/ @autinerd
|
/homeassistant/components/enigma2/ @autinerd
|
||||||
/tests/components/enigma2/ @autinerd
|
/tests/components/enigma2/ @autinerd
|
||||||
/homeassistant/components/enphase_envoy/ @bdraco @cgarwood @catsmanac
|
/homeassistant/components/enocean/ @bdurrer
|
||||||
/tests/components/enphase_envoy/ @bdraco @cgarwood @catsmanac
|
/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/entur_public_transport/ @hfurubotten
|
||||||
/homeassistant/components/environment_canada/ @gwww @michaeldavie
|
/homeassistant/components/environment_canada/ @gwww @michaeldavie
|
||||||
/tests/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
|
/homeassistant/components/epic_games_store/ @hacf-fr @Quentame
|
||||||
/tests/components/epic_games_store/ @hacf-fr @Quentame
|
/tests/components/epic_games_store/ @hacf-fr @Quentame
|
||||||
/homeassistant/components/epion/ @lhgravendeel
|
/homeassistant/components/epion/ @lhgravendeel
|
||||||
@@ -466,16 +441,18 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/eq3btsmart/ @eulemitkeule @dbuezas
|
/tests/components/eq3btsmart/ @eulemitkeule @dbuezas
|
||||||
/homeassistant/components/escea/ @lazdavila
|
/homeassistant/components/escea/ @lazdavila
|
||||||
/tests/components/escea/ @lazdavila
|
/tests/components/escea/ @lazdavila
|
||||||
/homeassistant/components/esphome/ @jesserockz @kbx81 @bdraco
|
/homeassistant/components/esphome/ @OttoWinter @jesserockz @kbx81 @bdraco
|
||||||
/tests/components/esphome/ @jesserockz @kbx81 @bdraco
|
/tests/components/esphome/ @OttoWinter @jesserockz @kbx81 @bdraco
|
||||||
/homeassistant/components/eufylife_ble/ @bdr99
|
/homeassistant/components/eufylife_ble/ @bdr99
|
||||||
/tests/components/eufylife_ble/ @bdr99
|
/tests/components/eufylife_ble/ @bdr99
|
||||||
/homeassistant/components/event/ @home-assistant/core
|
/homeassistant/components/event/ @home-assistant/core
|
||||||
/tests/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
|
/homeassistant/components/evohome/ @zxdavb
|
||||||
/tests/components/evohome/ @zxdavb
|
/tests/components/evohome/ @zxdavb
|
||||||
/homeassistant/components/ezviz/ @RenierM26
|
/homeassistant/components/ezviz/ @RenierM26 @baqs
|
||||||
/tests/components/ezviz/ @RenierM26
|
/tests/components/ezviz/ @RenierM26 @baqs
|
||||||
/homeassistant/components/faa_delays/ @ntilley905
|
/homeassistant/components/faa_delays/ @ntilley905
|
||||||
/tests/components/faa_delays/ @ntilley905
|
/tests/components/faa_delays/ @ntilley905
|
||||||
/homeassistant/components/fan/ @home-assistant/core
|
/homeassistant/components/fan/ @home-assistant/core
|
||||||
@@ -494,8 +471,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/filesize/ @gjohansson-ST
|
/tests/components/filesize/ @gjohansson-ST
|
||||||
/homeassistant/components/filter/ @dgomes
|
/homeassistant/components/filter/ @dgomes
|
||||||
/tests/components/filter/ @dgomes
|
/tests/components/filter/ @dgomes
|
||||||
/homeassistant/components/firefly_iii/ @erwindouna
|
|
||||||
/tests/components/firefly_iii/ @erwindouna
|
|
||||||
/homeassistant/components/fireservicerota/ @cyberjunky
|
/homeassistant/components/fireservicerota/ @cyberjunky
|
||||||
/tests/components/fireservicerota/ @cyberjunky
|
/tests/components/fireservicerota/ @cyberjunky
|
||||||
/homeassistant/components/firmata/ @DaAwesomeP
|
/homeassistant/components/firmata/ @DaAwesomeP
|
||||||
@@ -523,8 +498,8 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/forked_daapd/ @uvjustin
|
/homeassistant/components/forked_daapd/ @uvjustin
|
||||||
/tests/components/forked_daapd/ @uvjustin
|
/tests/components/forked_daapd/ @uvjustin
|
||||||
/homeassistant/components/fortios/ @kimfrellsen
|
/homeassistant/components/fortios/ @kimfrellsen
|
||||||
/homeassistant/components/foscam/ @Foscam-wangzhengyu
|
/homeassistant/components/foscam/ @krmarien
|
||||||
/tests/components/foscam/ @Foscam-wangzhengyu
|
/tests/components/foscam/ @krmarien
|
||||||
/homeassistant/components/freebox/ @hacf-fr @Quentame
|
/homeassistant/components/freebox/ @hacf-fr @Quentame
|
||||||
/tests/components/freebox/ @hacf-fr @Quentame
|
/tests/components/freebox/ @hacf-fr @Quentame
|
||||||
/homeassistant/components/freedompro/ @stefano055415
|
/homeassistant/components/freedompro/ @stefano055415
|
||||||
@@ -595,8 +570,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/google_cloud/ @lufton @tronikos
|
/tests/components/google_cloud/ @lufton @tronikos
|
||||||
/homeassistant/components/google_drive/ @tronikos
|
/homeassistant/components/google_drive/ @tronikos
|
||||||
/tests/components/google_drive/ @tronikos
|
/tests/components/google_drive/ @tronikos
|
||||||
/homeassistant/components/google_generative_ai_conversation/ @tronikos @ivanlh
|
/homeassistant/components/google_generative_ai_conversation/ @tronikos
|
||||||
/tests/components/google_generative_ai_conversation/ @tronikos @ivanlh
|
/tests/components/google_generative_ai_conversation/ @tronikos
|
||||||
/homeassistant/components/google_mail/ @tkdrob
|
/homeassistant/components/google_mail/ @tkdrob
|
||||||
/tests/components/google_mail/ @tkdrob
|
/tests/components/google_mail/ @tkdrob
|
||||||
/homeassistant/components/google_photos/ @allenporter
|
/homeassistant/components/google_photos/ @allenporter
|
||||||
@@ -619,8 +594,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/greeneye_monitor/ @jkeljo
|
/tests/components/greeneye_monitor/ @jkeljo
|
||||||
/homeassistant/components/group/ @home-assistant/core
|
/homeassistant/components/group/ @home-assistant/core
|
||||||
/tests/components/group/ @home-assistant/core
|
/tests/components/group/ @home-assistant/core
|
||||||
/homeassistant/components/growatt_server/ @johanzander
|
|
||||||
/tests/components/growatt_server/ @johanzander
|
|
||||||
/homeassistant/components/guardian/ @bachya
|
/homeassistant/components/guardian/ @bachya
|
||||||
/tests/components/guardian/ @bachya
|
/tests/components/guardian/ @bachya
|
||||||
/homeassistant/components/habitica/ @tr4nt0r
|
/homeassistant/components/habitica/ @tr4nt0r
|
||||||
@@ -660,8 +633,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/homeassistant/ @home-assistant/core
|
/tests/components/homeassistant/ @home-assistant/core
|
||||||
/homeassistant/components/homeassistant_alerts/ @home-assistant/core
|
/homeassistant/components/homeassistant_alerts/ @home-assistant/core
|
||||||
/tests/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
|
/homeassistant/components/homeassistant_green/ @home-assistant/core
|
||||||
/tests/components/homeassistant_green/ @home-assistant/core
|
/tests/components/homeassistant_green/ @home-assistant/core
|
||||||
/homeassistant/components/homeassistant_hardware/ @home-assistant/core
|
/homeassistant/components/homeassistant_hardware/ @home-assistant/core
|
||||||
@@ -690,8 +661,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/http/ @home-assistant/core
|
/tests/components/http/ @home-assistant/core
|
||||||
/homeassistant/components/huawei_lte/ @scop @fphammerle
|
/homeassistant/components/huawei_lte/ @scop @fphammerle
|
||||||
/tests/components/huawei_lte/ @scop @fphammerle
|
/tests/components/huawei_lte/ @scop @fphammerle
|
||||||
/homeassistant/components/hue/ @marcelveldt
|
/homeassistant/components/hue/ @balloob @marcelveldt
|
||||||
/tests/components/hue/ @marcelveldt
|
/tests/components/hue/ @balloob @marcelveldt
|
||||||
/homeassistant/components/huisbaasje/ @dennisschroer
|
/homeassistant/components/huisbaasje/ @dennisschroer
|
||||||
/tests/components/huisbaasje/ @dennisschroer
|
/tests/components/huisbaasje/ @dennisschroer
|
||||||
/homeassistant/components/humidifier/ @home-assistant/core @Shulyaka
|
/homeassistant/components/humidifier/ @home-assistant/core @Shulyaka
|
||||||
@@ -702,8 +673,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/husqvarna_automower/ @Thomas55555
|
/tests/components/husqvarna_automower/ @Thomas55555
|
||||||
/homeassistant/components/husqvarna_automower_ble/ @alistair23
|
/homeassistant/components/husqvarna_automower_ble/ @alistair23
|
||||||
/tests/components/husqvarna_automower_ble/ @alistair23
|
/tests/components/husqvarna_automower_ble/ @alistair23
|
||||||
/homeassistant/components/huum/ @frwickst @vincentwolsink
|
/homeassistant/components/huum/ @frwickst
|
||||||
/tests/components/huum/ @frwickst @vincentwolsink
|
/tests/components/huum/ @frwickst
|
||||||
/homeassistant/components/hvv_departures/ @vigonotion
|
/homeassistant/components/hvv_departures/ @vigonotion
|
||||||
/tests/components/hvv_departures/ @vigonotion
|
/tests/components/hvv_departures/ @vigonotion
|
||||||
/homeassistant/components/hydrawise/ @dknowles2 @thomaskistler @ptcryan
|
/homeassistant/components/hydrawise/ @dknowles2 @thomaskistler @ptcryan
|
||||||
@@ -731,12 +702,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/image_upload/ @home-assistant/core
|
/tests/components/image_upload/ @home-assistant/core
|
||||||
/homeassistant/components/imap/ @jbouwh
|
/homeassistant/components/imap/ @jbouwh
|
||||||
/tests/components/imap/ @jbouwh
|
/tests/components/imap/ @jbouwh
|
||||||
/homeassistant/components/imeon_inverter/ @Imeon-Energy
|
|
||||||
/tests/components/imeon_inverter/ @Imeon-Energy
|
|
||||||
/homeassistant/components/imgw_pib/ @bieniu
|
/homeassistant/components/imgw_pib/ @bieniu
|
||||||
/tests/components/imgw_pib/ @bieniu
|
/tests/components/imgw_pib/ @bieniu
|
||||||
/homeassistant/components/immich/ @mib1185
|
|
||||||
/tests/components/immich/ @mib1185
|
|
||||||
/homeassistant/components/improv_ble/ @emontnemery
|
/homeassistant/components/improv_ble/ @emontnemery
|
||||||
/tests/components/improv_ble/ @emontnemery
|
/tests/components/improv_ble/ @emontnemery
|
||||||
/homeassistant/components/incomfort/ @jbouwh
|
/homeassistant/components/incomfort/ @jbouwh
|
||||||
@@ -763,11 +730,11 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/integration/ @dgomes
|
/tests/components/integration/ @dgomes
|
||||||
/homeassistant/components/intellifire/ @jeeftor
|
/homeassistant/components/intellifire/ @jeeftor
|
||||||
/tests/components/intellifire/ @jeeftor
|
/tests/components/intellifire/ @jeeftor
|
||||||
/homeassistant/components/intent/ @home-assistant/core @synesthesiam @arturpragacz
|
/homeassistant/components/intent/ @home-assistant/core @synesthesiam
|
||||||
/tests/components/intent/ @home-assistant/core @synesthesiam @arturpragacz
|
/tests/components/intent/ @home-assistant/core @synesthesiam
|
||||||
/homeassistant/components/intesishome/ @jnimmo
|
/homeassistant/components/intesishome/ @jnimmo
|
||||||
/homeassistant/components/iometer/ @jukrebs
|
/homeassistant/components/iometer/ @MaestroOnICe
|
||||||
/tests/components/iometer/ @jukrebs
|
/tests/components/iometer/ @MaestroOnICe
|
||||||
/homeassistant/components/ios/ @robbiet480
|
/homeassistant/components/ios/ @robbiet480
|
||||||
/tests/components/ios/ @robbiet480
|
/tests/components/ios/ @robbiet480
|
||||||
/homeassistant/components/iotawatt/ @gtdiehl @jyavenard
|
/homeassistant/components/iotawatt/ @gtdiehl @jyavenard
|
||||||
@@ -782,8 +749,6 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/iqvia/ @bachya
|
/homeassistant/components/iqvia/ @bachya
|
||||||
/tests/components/iqvia/ @bachya
|
/tests/components/iqvia/ @bachya
|
||||||
/homeassistant/components/irish_rail_transport/ @ttroy50
|
/homeassistant/components/irish_rail_transport/ @ttroy50
|
||||||
/homeassistant/components/irm_kmi/ @jdejaegh
|
|
||||||
/tests/components/irm_kmi/ @jdejaegh
|
|
||||||
/homeassistant/components/iron_os/ @tr4nt0r
|
/homeassistant/components/iron_os/ @tr4nt0r
|
||||||
/tests/components/iron_os/ @tr4nt0r
|
/tests/components/iron_os/ @tr4nt0r
|
||||||
/homeassistant/components/isal/ @bdraco
|
/homeassistant/components/isal/ @bdraco
|
||||||
@@ -808,6 +773,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/jellyfin/ @RunC0deRun @ctalkington
|
/tests/components/jellyfin/ @RunC0deRun @ctalkington
|
||||||
/homeassistant/components/jewish_calendar/ @tsvi
|
/homeassistant/components/jewish_calendar/ @tsvi
|
||||||
/tests/components/jewish_calendar/ @tsvi
|
/tests/components/jewish_calendar/ @tsvi
|
||||||
|
/homeassistant/components/juicenet/ @jesserockz
|
||||||
|
/tests/components/juicenet/ @jesserockz
|
||||||
/homeassistant/components/justnimbus/ @kvanzuijlen
|
/homeassistant/components/justnimbus/ @kvanzuijlen
|
||||||
/tests/components/justnimbus/ @kvanzuijlen
|
/tests/components/justnimbus/ @kvanzuijlen
|
||||||
/homeassistant/components/jvc_projector/ @SteveEasley @msavazzi
|
/homeassistant/components/jvc_projector/ @SteveEasley @msavazzi
|
||||||
@@ -874,14 +841,14 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/lg_netcast/ @Drafteed @splinter98
|
/tests/components/lg_netcast/ @Drafteed @splinter98
|
||||||
/homeassistant/components/lg_thinq/ @LG-ThinQ-Integration
|
/homeassistant/components/lg_thinq/ @LG-ThinQ-Integration
|
||||||
/tests/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
|
/homeassistant/components/lidarr/ @tkdrob
|
||||||
/tests/components/lidarr/ @tkdrob
|
/tests/components/lidarr/ @tkdrob
|
||||||
/homeassistant/components/lifx/ @Djelibeybi
|
/homeassistant/components/lifx/ @Djelibeybi
|
||||||
/tests/components/lifx/ @Djelibeybi
|
/tests/components/lifx/ @Djelibeybi
|
||||||
/homeassistant/components/light/ @home-assistant/core
|
/homeassistant/components/light/ @home-assistant/core
|
||||||
/tests/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
|
/homeassistant/components/linkplay/ @Velleman
|
||||||
/tests/components/linkplay/ @Velleman
|
/tests/components/linkplay/ @Velleman
|
||||||
/homeassistant/components/linux_battery/ @fabaff
|
/homeassistant/components/linux_battery/ @fabaff
|
||||||
@@ -914,8 +881,6 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/luci/ @mzdrale
|
/homeassistant/components/luci/ @mzdrale
|
||||||
/homeassistant/components/luftdaten/ @fabaff @frenck
|
/homeassistant/components/luftdaten/ @fabaff @frenck
|
||||||
/tests/components/luftdaten/ @fabaff @frenck
|
/tests/components/luftdaten/ @fabaff @frenck
|
||||||
/homeassistant/components/lunatone/ @MoonDevLT
|
|
||||||
/tests/components/lunatone/ @MoonDevLT
|
|
||||||
/homeassistant/components/lupusec/ @majuss @suaveolent
|
/homeassistant/components/lupusec/ @majuss @suaveolent
|
||||||
/tests/components/lupusec/ @majuss @suaveolent
|
/tests/components/lupusec/ @majuss @suaveolent
|
||||||
/homeassistant/components/lutron/ @cdheiser @wilburCForce
|
/homeassistant/components/lutron/ @cdheiser @wilburCForce
|
||||||
@@ -961,8 +926,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/met_eireann/ @DylanGore
|
/tests/components/met_eireann/ @DylanGore
|
||||||
/homeassistant/components/meteo_france/ @hacf-fr @oncleben31 @Quentame
|
/homeassistant/components/meteo_france/ @hacf-fr @oncleben31 @Quentame
|
||||||
/tests/components/meteo_france/ @hacf-fr @oncleben31 @Quentame
|
/tests/components/meteo_france/ @hacf-fr @oncleben31 @Quentame
|
||||||
/homeassistant/components/meteo_lt/ @xE1H
|
|
||||||
/tests/components/meteo_lt/ @xE1H
|
|
||||||
/homeassistant/components/meteoalarm/ @rolfberkenbosch
|
/homeassistant/components/meteoalarm/ @rolfberkenbosch
|
||||||
/homeassistant/components/meteoclimatic/ @adrianmo
|
/homeassistant/components/meteoclimatic/ @adrianmo
|
||||||
/tests/components/meteoclimatic/ @adrianmo
|
/tests/components/meteoclimatic/ @adrianmo
|
||||||
@@ -970,8 +933,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/metoffice/ @MrHarcombe @avee87
|
/tests/components/metoffice/ @MrHarcombe @avee87
|
||||||
/homeassistant/components/microbees/ @microBeesTech
|
/homeassistant/components/microbees/ @microBeesTech
|
||||||
/tests/components/microbees/ @microBeesTech
|
/tests/components/microbees/ @microBeesTech
|
||||||
/homeassistant/components/miele/ @astrandb
|
|
||||||
/tests/components/miele/ @astrandb
|
|
||||||
/homeassistant/components/mikrotik/ @engrbm87
|
/homeassistant/components/mikrotik/ @engrbm87
|
||||||
/tests/components/mikrotik/ @engrbm87
|
/tests/components/mikrotik/ @engrbm87
|
||||||
/homeassistant/components/mill/ @danielhiversen
|
/homeassistant/components/mill/ @danielhiversen
|
||||||
@@ -1033,8 +994,7 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/nanoleaf/ @milanmeu @joostlek
|
/tests/components/nanoleaf/ @milanmeu @joostlek
|
||||||
/homeassistant/components/nasweb/ @nasWebio
|
/homeassistant/components/nasweb/ @nasWebio
|
||||||
/tests/components/nasweb/ @nasWebio
|
/tests/components/nasweb/ @nasWebio
|
||||||
/homeassistant/components/nederlandse_spoorwegen/ @YarmoM @heindrichpaul
|
/homeassistant/components/nederlandse_spoorwegen/ @YarmoM
|
||||||
/tests/components/nederlandse_spoorwegen/ @YarmoM @heindrichpaul
|
|
||||||
/homeassistant/components/ness_alarm/ @nickw444
|
/homeassistant/components/ness_alarm/ @nickw444
|
||||||
/tests/components/ness_alarm/ @nickw444
|
/tests/components/ness_alarm/ @nickw444
|
||||||
/homeassistant/components/nest/ @allenporter
|
/homeassistant/components/nest/ @allenporter
|
||||||
@@ -1069,8 +1029,6 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/nilu/ @hfurubotten
|
/homeassistant/components/nilu/ @hfurubotten
|
||||||
/homeassistant/components/nina/ @DeerMaximum
|
/homeassistant/components/nina/ @DeerMaximum
|
||||||
/tests/components/nina/ @DeerMaximum
|
/tests/components/nina/ @DeerMaximum
|
||||||
/homeassistant/components/nintendo_parental_controls/ @pantherale0
|
|
||||||
/tests/components/nintendo_parental_controls/ @pantherale0
|
|
||||||
/homeassistant/components/nissan_leaf/ @filcole
|
/homeassistant/components/nissan_leaf/ @filcole
|
||||||
/homeassistant/components/noaa_tides/ @jdelaney72
|
/homeassistant/components/noaa_tides/ @jdelaney72
|
||||||
/homeassistant/components/nobo_hub/ @echoromeo @oyvindwe
|
/homeassistant/components/nobo_hub/ @echoromeo @oyvindwe
|
||||||
@@ -1087,8 +1045,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/nsw_fuel_station/ @nickw444
|
/tests/components/nsw_fuel_station/ @nickw444
|
||||||
/homeassistant/components/nsw_rural_fire_service_feed/ @exxamalte
|
/homeassistant/components/nsw_rural_fire_service_feed/ @exxamalte
|
||||||
/tests/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
|
/homeassistant/components/nuheat/ @tstabrawa
|
||||||
/tests/components/nuheat/ @tstabrawa
|
/tests/components/nuheat/ @tstabrawa
|
||||||
/homeassistant/components/nuki/ @pschmitt @pvizeli @pree
|
/homeassistant/components/nuki/ @pschmitt @pvizeli @pree
|
||||||
@@ -1117,6 +1073,8 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/ombi/ @larssont
|
/homeassistant/components/ombi/ @larssont
|
||||||
/homeassistant/components/onboarding/ @home-assistant/core
|
/homeassistant/components/onboarding/ @home-assistant/core
|
||||||
/tests/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
|
/homeassistant/components/ondilo_ico/ @JeromeHXP
|
||||||
/tests/components/ondilo_ico/ @JeromeHXP
|
/tests/components/ondilo_ico/ @JeromeHXP
|
||||||
/homeassistant/components/onedrive/ @zweckj
|
/homeassistant/components/onedrive/ @zweckj
|
||||||
@@ -1129,8 +1087,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/onvif/ @hunterjm @jterrace
|
/tests/components/onvif/ @hunterjm @jterrace
|
||||||
/homeassistant/components/open_meteo/ @frenck
|
/homeassistant/components/open_meteo/ @frenck
|
||||||
/tests/components/open_meteo/ @frenck
|
/tests/components/open_meteo/ @frenck
|
||||||
/homeassistant/components/open_router/ @joostlek
|
/homeassistant/components/openai_conversation/ @balloob
|
||||||
/tests/components/open_router/ @joostlek
|
/tests/components/openai_conversation/ @balloob
|
||||||
/homeassistant/components/openerz/ @misialq
|
/homeassistant/components/openerz/ @misialq
|
||||||
/tests/components/openerz/ @misialq
|
/tests/components/openerz/ @misialq
|
||||||
/homeassistant/components/openexchangerates/ @MartinHjelmare
|
/homeassistant/components/openexchangerates/ @MartinHjelmare
|
||||||
@@ -1139,16 +1097,14 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/opengarage/ @danielhiversen
|
/tests/components/opengarage/ @danielhiversen
|
||||||
/homeassistant/components/openhome/ @bazwilliams
|
/homeassistant/components/openhome/ @bazwilliams
|
||||||
/tests/components/openhome/ @bazwilliams
|
/tests/components/openhome/ @bazwilliams
|
||||||
/homeassistant/components/openrgb/ @felipecrs
|
|
||||||
/tests/components/openrgb/ @felipecrs
|
|
||||||
/homeassistant/components/opensky/ @joostlek
|
/homeassistant/components/opensky/ @joostlek
|
||||||
/tests/components/opensky/ @joostlek
|
/tests/components/opensky/ @joostlek
|
||||||
/homeassistant/components/opentherm_gw/ @mvn23
|
/homeassistant/components/opentherm_gw/ @mvn23
|
||||||
/tests/components/opentherm_gw/ @mvn23
|
/tests/components/opentherm_gw/ @mvn23
|
||||||
/homeassistant/components/openuv/ @bachya
|
/homeassistant/components/openuv/ @bachya
|
||||||
/tests/components/openuv/ @bachya
|
/tests/components/openuv/ @bachya
|
||||||
/homeassistant/components/openweathermap/ @fabaff @freekode @nzapponi @wittypluck
|
/homeassistant/components/openweathermap/ @fabaff @freekode @nzapponi
|
||||||
/tests/components/openweathermap/ @fabaff @freekode @nzapponi @wittypluck
|
/tests/components/openweathermap/ @fabaff @freekode @nzapponi
|
||||||
/homeassistant/components/opnsense/ @mtreinish
|
/homeassistant/components/opnsense/ @mtreinish
|
||||||
/tests/components/opnsense/ @mtreinish
|
/tests/components/opnsense/ @mtreinish
|
||||||
/homeassistant/components/opower/ @tronikos
|
/homeassistant/components/opower/ @tronikos
|
||||||
@@ -1174,8 +1130,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/palazzetti/ @dotvav
|
/tests/components/palazzetti/ @dotvav
|
||||||
/homeassistant/components/panel_custom/ @home-assistant/frontend
|
/homeassistant/components/panel_custom/ @home-assistant/frontend
|
||||||
/tests/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
|
/homeassistant/components/peblar/ @frenck
|
||||||
/tests/components/peblar/ @frenck
|
/tests/components/peblar/ @frenck
|
||||||
/homeassistant/components/peco/ @IceBotYT
|
/homeassistant/components/peco/ @IceBotYT
|
||||||
@@ -1198,28 +1152,22 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/ping/ @jpbede
|
/tests/components/ping/ @jpbede
|
||||||
/homeassistant/components/plaato/ @JohNan
|
/homeassistant/components/plaato/ @JohNan
|
||||||
/tests/components/plaato/ @JohNan
|
/tests/components/plaato/ @JohNan
|
||||||
/homeassistant/components/playstation_network/ @jackjpowell @tr4nt0r
|
|
||||||
/tests/components/playstation_network/ @jackjpowell @tr4nt0r
|
|
||||||
/homeassistant/components/plex/ @jjlawren
|
/homeassistant/components/plex/ @jjlawren
|
||||||
/tests/components/plex/ @jjlawren
|
/tests/components/plex/ @jjlawren
|
||||||
/homeassistant/components/plugwise/ @CoMPaTech @bouwew
|
/homeassistant/components/plugwise/ @CoMPaTech @bouwew
|
||||||
/tests/components/plugwise/ @CoMPaTech @bouwew
|
/tests/components/plugwise/ @CoMPaTech @bouwew
|
||||||
|
/homeassistant/components/plum_lightpad/ @ColinHarrington @prystupa
|
||||||
|
/tests/components/plum_lightpad/ @ColinHarrington @prystupa
|
||||||
/homeassistant/components/point/ @fredrike
|
/homeassistant/components/point/ @fredrike
|
||||||
/tests/components/point/ @fredrike
|
/tests/components/point/ @fredrike
|
||||||
/homeassistant/components/pooldose/ @lmaertin
|
|
||||||
/tests/components/pooldose/ @lmaertin
|
|
||||||
/homeassistant/components/poolsense/ @haemishkyd
|
/homeassistant/components/poolsense/ @haemishkyd
|
||||||
/tests/components/poolsense/ @haemishkyd
|
/tests/components/poolsense/ @haemishkyd
|
||||||
/homeassistant/components/portainer/ @erwindouna
|
|
||||||
/tests/components/portainer/ @erwindouna
|
|
||||||
/homeassistant/components/powerfox/ @klaasnicolaas
|
/homeassistant/components/powerfox/ @klaasnicolaas
|
||||||
/tests/components/powerfox/ @klaasnicolaas
|
/tests/components/powerfox/ @klaasnicolaas
|
||||||
/homeassistant/components/powerwall/ @bdraco @jrester @daniel-simpson
|
/homeassistant/components/powerwall/ @bdraco @jrester @daniel-simpson
|
||||||
/tests/components/powerwall/ @bdraco @jrester @daniel-simpson
|
/tests/components/powerwall/ @bdraco @jrester @daniel-simpson
|
||||||
/homeassistant/components/private_ble_device/ @Jc2k
|
/homeassistant/components/private_ble_device/ @Jc2k
|
||||||
/tests/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
|
/homeassistant/components/profiler/ @bdraco
|
||||||
/tests/components/profiler/ @bdraco
|
/tests/components/profiler/ @bdraco
|
||||||
/homeassistant/components/progettihwsw/ @ardaseremet
|
/homeassistant/components/progettihwsw/ @ardaseremet
|
||||||
@@ -1231,10 +1179,10 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/proximity/ @mib1185
|
/homeassistant/components/proximity/ @mib1185
|
||||||
/tests/components/proximity/ @mib1185
|
/tests/components/proximity/ @mib1185
|
||||||
/homeassistant/components/proxmoxve/ @jhollowe @Corbeno
|
/homeassistant/components/proxmoxve/ @jhollowe @Corbeno
|
||||||
|
/homeassistant/components/prusalink/ @balloob
|
||||||
|
/tests/components/prusalink/ @balloob
|
||||||
/homeassistant/components/ps4/ @ktnrg45
|
/homeassistant/components/ps4/ @ktnrg45
|
||||||
/tests/components/ps4/ @ktnrg45
|
/tests/components/ps4/ @ktnrg45
|
||||||
/homeassistant/components/pterodactyl/ @elmurato
|
|
||||||
/tests/components/pterodactyl/ @elmurato
|
|
||||||
/homeassistant/components/pure_energie/ @klaasnicolaas
|
/homeassistant/components/pure_energie/ @klaasnicolaas
|
||||||
/tests/components/pure_energie/ @klaasnicolaas
|
/tests/components/pure_energie/ @klaasnicolaas
|
||||||
/homeassistant/components/purpleair/ @bachya
|
/homeassistant/components/purpleair/ @bachya
|
||||||
@@ -1264,7 +1212,6 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/qnap_qsw/ @Noltari
|
/homeassistant/components/qnap_qsw/ @Noltari
|
||||||
/tests/components/qnap_qsw/ @Noltari
|
/tests/components/qnap_qsw/ @Noltari
|
||||||
/homeassistant/components/quantum_gateway/ @cisasteelersfan
|
/homeassistant/components/quantum_gateway/ @cisasteelersfan
|
||||||
/tests/components/quantum_gateway/ @cisasteelersfan
|
|
||||||
/homeassistant/components/qvr_pro/ @oblogic7
|
/homeassistant/components/qvr_pro/ @oblogic7
|
||||||
/homeassistant/components/qwikswitch/ @kellerza
|
/homeassistant/components/qwikswitch/ @kellerza
|
||||||
/tests/components/qwikswitch/ @kellerza
|
/tests/components/qwikswitch/ @kellerza
|
||||||
@@ -1303,12 +1250,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/recovery_mode/ @home-assistant/core
|
/tests/components/recovery_mode/ @home-assistant/core
|
||||||
/homeassistant/components/refoss/ @ashionky
|
/homeassistant/components/refoss/ @ashionky
|
||||||
/tests/components/refoss/ @ashionky
|
/tests/components/refoss/ @ashionky
|
||||||
/homeassistant/components/rehlko/ @bdraco @peterager
|
|
||||||
/tests/components/rehlko/ @bdraco @peterager
|
|
||||||
/homeassistant/components/remote/ @home-assistant/core
|
/homeassistant/components/remote/ @home-assistant/core
|
||||||
/tests/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/renault/ @epenet
|
/homeassistant/components/renault/ @epenet
|
||||||
/tests/components/renault/ @epenet
|
/tests/components/renault/ @epenet
|
||||||
/homeassistant/components/renson/ @jimmyd-be
|
/homeassistant/components/renson/ @jimmyd-be
|
||||||
@@ -1324,8 +1267,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/rflink/ @javicalle
|
/tests/components/rflink/ @javicalle
|
||||||
/homeassistant/components/rfxtrx/ @danielhiversen @elupus @RobBie1221
|
/homeassistant/components/rfxtrx/ @danielhiversen @elupus @RobBie1221
|
||||||
/tests/components/rfxtrx/ @danielhiversen @elupus @RobBie1221
|
/tests/components/rfxtrx/ @danielhiversen @elupus @RobBie1221
|
||||||
/homeassistant/components/rhasspy/ @synesthesiam
|
/homeassistant/components/rhasspy/ @balloob @synesthesiam
|
||||||
/tests/components/rhasspy/ @synesthesiam
|
/tests/components/rhasspy/ @balloob @synesthesiam
|
||||||
/homeassistant/components/ridwell/ @bachya
|
/homeassistant/components/ridwell/ @bachya
|
||||||
/tests/components/ridwell/ @bachya
|
/tests/components/ridwell/ @bachya
|
||||||
/homeassistant/components/ring/ @sdb9696
|
/homeassistant/components/ring/ @sdb9696
|
||||||
@@ -1346,12 +1289,12 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/roomba/ @pschmitt @cyr-ius @shenxn @Orhideous
|
/tests/components/roomba/ @pschmitt @cyr-ius @shenxn @Orhideous
|
||||||
/homeassistant/components/roon/ @pavoni
|
/homeassistant/components/roon/ @pavoni
|
||||||
/tests/components/roon/ @pavoni
|
/tests/components/roon/ @pavoni
|
||||||
/homeassistant/components/route_b_smart_meter/ @SeraphicRav
|
|
||||||
/tests/components/route_b_smart_meter/ @SeraphicRav
|
|
||||||
/homeassistant/components/rpi_power/ @shenxn @swetoast
|
/homeassistant/components/rpi_power/ @shenxn @swetoast
|
||||||
/tests/components/rpi_power/ @shenxn @swetoast
|
/tests/components/rpi_power/ @shenxn @swetoast
|
||||||
/homeassistant/components/rss_feed_template/ @home-assistant/core
|
/homeassistant/components/rss_feed_template/ @home-assistant/core
|
||||||
/tests/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
|
/homeassistant/components/ruckus_unleashed/ @lanrat @ms264556 @gabe565
|
||||||
/tests/components/ruckus_unleashed/ @lanrat @ms264556 @gabe565
|
/tests/components/ruckus_unleashed/ @lanrat @ms264556 @gabe565
|
||||||
/homeassistant/components/russound_rio/ @noahhusby
|
/homeassistant/components/russound_rio/ @noahhusby
|
||||||
@@ -1370,8 +1313,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/samsungtv/ @chemelli74 @epenet
|
/tests/components/samsungtv/ @chemelli74 @epenet
|
||||||
/homeassistant/components/sanix/ @tomaszsluszniak
|
/homeassistant/components/sanix/ @tomaszsluszniak
|
||||||
/tests/components/sanix/ @tomaszsluszniak
|
/tests/components/sanix/ @tomaszsluszniak
|
||||||
/homeassistant/components/satel_integra/ @Tommatheussen
|
|
||||||
/tests/components/satel_integra/ @Tommatheussen
|
|
||||||
/homeassistant/components/scene/ @home-assistant/core
|
/homeassistant/components/scene/ @home-assistant/core
|
||||||
/tests/components/scene/ @home-assistant/core
|
/tests/components/scene/ @home-assistant/core
|
||||||
/homeassistant/components/schedule/ @home-assistant/core
|
/homeassistant/components/schedule/ @home-assistant/core
|
||||||
@@ -1417,14 +1358,12 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/seventeentrack/ @shaiu
|
/tests/components/seventeentrack/ @shaiu
|
||||||
/homeassistant/components/sfr_box/ @epenet
|
/homeassistant/components/sfr_box/ @epenet
|
||||||
/tests/components/sfr_box/ @epenet
|
/tests/components/sfr_box/ @epenet
|
||||||
/homeassistant/components/sftp_storage/ @maretodoric
|
/homeassistant/components/sharkiq/ @JeffResc @funkybunch
|
||||||
/tests/components/sftp_storage/ @maretodoric
|
/tests/components/sharkiq/ @JeffResc @funkybunch
|
||||||
/homeassistant/components/sharkiq/ @JeffResc @funkybunch @TheOneOgre
|
|
||||||
/tests/components/sharkiq/ @JeffResc @funkybunch @TheOneOgre
|
|
||||||
/homeassistant/components/shell_command/ @home-assistant/core
|
/homeassistant/components/shell_command/ @home-assistant/core
|
||||||
/tests/components/shell_command/ @home-assistant/core
|
/tests/components/shell_command/ @home-assistant/core
|
||||||
/homeassistant/components/shelly/ @bieniu @thecode @chemelli74 @bdraco
|
/homeassistant/components/shelly/ @balloob @bieniu @thecode @chemelli74 @bdraco
|
||||||
/tests/components/shelly/ @bieniu @thecode @chemelli74 @bdraco
|
/tests/components/shelly/ @balloob @bieniu @thecode @chemelli74 @bdraco
|
||||||
/homeassistant/components/shodan/ @fabaff
|
/homeassistant/components/shodan/ @fabaff
|
||||||
/homeassistant/components/sia/ @eavanvalkenburg
|
/homeassistant/components/sia/ @eavanvalkenburg
|
||||||
/tests/components/sia/ @eavanvalkenburg
|
/tests/components/sia/ @eavanvalkenburg
|
||||||
@@ -1442,14 +1381,13 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/siren/ @home-assistant/core @raman325
|
/homeassistant/components/siren/ @home-assistant/core @raman325
|
||||||
/tests/components/siren/ @home-assistant/core @raman325
|
/tests/components/siren/ @home-assistant/core @raman325
|
||||||
/homeassistant/components/sisyphus/ @jkeljo
|
/homeassistant/components/sisyphus/ @jkeljo
|
||||||
|
/homeassistant/components/sky_hub/ @rogerselwyn
|
||||||
/homeassistant/components/sky_remote/ @dunnmj @saty9
|
/homeassistant/components/sky_remote/ @dunnmj @saty9
|
||||||
/tests/components/sky_remote/ @dunnmj @saty9
|
/tests/components/sky_remote/ @dunnmj @saty9
|
||||||
/homeassistant/components/skybell/ @tkdrob
|
/homeassistant/components/skybell/ @tkdrob
|
||||||
/tests/components/skybell/ @tkdrob
|
/tests/components/skybell/ @tkdrob
|
||||||
/homeassistant/components/slack/ @tkdrob @fletcherau
|
/homeassistant/components/slack/ @tkdrob @fletcherau
|
||||||
/tests/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
|
/homeassistant/components/sleepiq/ @mfugate1 @kbickar
|
||||||
/tests/components/sleepiq/ @mfugate1 @kbickar
|
/tests/components/sleepiq/ @mfugate1 @kbickar
|
||||||
/homeassistant/components/slide/ @ualex73
|
/homeassistant/components/slide/ @ualex73
|
||||||
@@ -1461,8 +1399,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/sma/ @kellerza @rklomp @erwindouna
|
/tests/components/sma/ @kellerza @rklomp @erwindouna
|
||||||
/homeassistant/components/smappee/ @bsmappee
|
/homeassistant/components/smappee/ @bsmappee
|
||||||
/tests/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
|
/homeassistant/components/smart_meter_texas/ @grahamwetzler
|
||||||
/tests/components/smart_meter_texas/ @grahamwetzler
|
/tests/components/smart_meter_texas/ @grahamwetzler
|
||||||
/homeassistant/components/smartthings/ @joostlek
|
/homeassistant/components/smartthings/ @joostlek
|
||||||
@@ -1485,15 +1421,15 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/snoo/ @Lash-L
|
/tests/components/snoo/ @Lash-L
|
||||||
/homeassistant/components/snooz/ @AustinBrunkhorst
|
/homeassistant/components/snooz/ @AustinBrunkhorst
|
||||||
/tests/components/snooz/ @AustinBrunkhorst
|
/tests/components/snooz/ @AustinBrunkhorst
|
||||||
/homeassistant/components/solaredge/ @frenck @bdraco @tronikos
|
/homeassistant/components/solaredge/ @frenck @bdraco
|
||||||
/tests/components/solaredge/ @frenck @bdraco @tronikos
|
/tests/components/solaredge/ @frenck @bdraco
|
||||||
/homeassistant/components/solaredge_local/ @drobtravels @scheric
|
/homeassistant/components/solaredge_local/ @drobtravels @scheric
|
||||||
/homeassistant/components/solarlog/ @Ernst79 @dontinelli
|
/homeassistant/components/solarlog/ @Ernst79 @dontinelli
|
||||||
/tests/components/solarlog/ @Ernst79 @dontinelli
|
/tests/components/solarlog/ @Ernst79 @dontinelli
|
||||||
/homeassistant/components/solax/ @squishykid @Darsstar
|
/homeassistant/components/solax/ @squishykid @Darsstar
|
||||||
/tests/components/solax/ @squishykid @Darsstar
|
/tests/components/solax/ @squishykid @Darsstar
|
||||||
/homeassistant/components/soma/ @ratsept
|
/homeassistant/components/soma/ @ratsept @sebfortier2288
|
||||||
/tests/components/soma/ @ratsept
|
/tests/components/soma/ @ratsept @sebfortier2288
|
||||||
/homeassistant/components/sonarr/ @ctalkington
|
/homeassistant/components/sonarr/ @ctalkington
|
||||||
/tests/components/sonarr/ @ctalkington
|
/tests/components/sonarr/ @ctalkington
|
||||||
/homeassistant/components/songpal/ @rytilahti @shenxn
|
/homeassistant/components/songpal/ @rytilahti @shenxn
|
||||||
@@ -1525,8 +1461,7 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/steam_online/ @tkdrob
|
/tests/components/steam_online/ @tkdrob
|
||||||
/homeassistant/components/steamist/ @bdraco
|
/homeassistant/components/steamist/ @bdraco
|
||||||
/tests/components/steamist/ @bdraco
|
/tests/components/steamist/ @bdraco
|
||||||
/homeassistant/components/stiebel_eltron/ @fucm @ThyMYthOS
|
/homeassistant/components/stiebel_eltron/ @fucm
|
||||||
/tests/components/stiebel_eltron/ @fucm @ThyMYthOS
|
|
||||||
/homeassistant/components/stookwijzer/ @fwestenberg
|
/homeassistant/components/stookwijzer/ @fwestenberg
|
||||||
/tests/components/stookwijzer/ @fwestenberg
|
/tests/components/stookwijzer/ @fwestenberg
|
||||||
/homeassistant/components/stream/ @hunterjm @uvjustin @allenporter
|
/homeassistant/components/stream/ @hunterjm @uvjustin @allenporter
|
||||||
@@ -1537,8 +1472,10 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/subaru/ @G-Two
|
/tests/components/subaru/ @G-Two
|
||||||
/homeassistant/components/suez_water/ @ooii @jb101010-2
|
/homeassistant/components/suez_water/ @ooii @jb101010-2
|
||||||
/tests/components/suez_water/ @ooii @jb101010-2
|
/tests/components/suez_water/ @ooii @jb101010-2
|
||||||
/homeassistant/components/sun/ @home-assistant/core
|
/homeassistant/components/sun/ @Swamp-Ig
|
||||||
/tests/components/sun/ @home-assistant/core
|
/tests/components/sun/ @Swamp-Ig
|
||||||
|
/homeassistant/components/sunweg/ @rokam
|
||||||
|
/tests/components/sunweg/ @rokam
|
||||||
/homeassistant/components/supla/ @mwegrzynek
|
/homeassistant/components/supla/ @mwegrzynek
|
||||||
/homeassistant/components/surepetcare/ @benleb @danielhiversen
|
/homeassistant/components/surepetcare/ @benleb @danielhiversen
|
||||||
/tests/components/surepetcare/ @benleb @danielhiversen
|
/tests/components/surepetcare/ @benleb @danielhiversen
|
||||||
@@ -1551,10 +1488,10 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/switch_as_x/ @home-assistant/core
|
/tests/components/switch_as_x/ @home-assistant/core
|
||||||
/homeassistant/components/switchbee/ @jafar-atili
|
/homeassistant/components/switchbee/ @jafar-atili
|
||||||
/tests/components/switchbee/ @jafar-atili
|
/tests/components/switchbee/ @jafar-atili
|
||||||
/homeassistant/components/switchbot/ @danielhiversen @RenierM26 @murtas @Eloston @dsypniewski @zerzhang
|
/homeassistant/components/switchbot/ @danielhiversen @RenierM26 @murtas @Eloston @dsypniewski
|
||||||
/tests/components/switchbot/ @danielhiversen @RenierM26 @murtas @Eloston @dsypniewski @zerzhang
|
/tests/components/switchbot/ @danielhiversen @RenierM26 @murtas @Eloston @dsypniewski
|
||||||
/homeassistant/components/switchbot_cloud/ @SeraphicRav @laurence-presland @Gigatrappeur @XiaoLing-git
|
/homeassistant/components/switchbot_cloud/ @SeraphicRav @laurence-presland @Gigatrappeur
|
||||||
/tests/components/switchbot_cloud/ @SeraphicRav @laurence-presland @Gigatrappeur @XiaoLing-git
|
/tests/components/switchbot_cloud/ @SeraphicRav @laurence-presland @Gigatrappeur
|
||||||
/homeassistant/components/switcher_kis/ @thecode @YogevBokobza
|
/homeassistant/components/switcher_kis/ @thecode @YogevBokobza
|
||||||
/tests/components/switcher_kis/ @thecode @YogevBokobza
|
/tests/components/switcher_kis/ @thecode @YogevBokobza
|
||||||
/homeassistant/components/switchmate/ @danielhiversen @qiz-li
|
/homeassistant/components/switchmate/ @danielhiversen @qiz-li
|
||||||
@@ -1571,8 +1508,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/systemmonitor/ @gjohansson-ST
|
/tests/components/systemmonitor/ @gjohansson-ST
|
||||||
/homeassistant/components/tado/ @erwindouna
|
/homeassistant/components/tado/ @erwindouna
|
||||||
/tests/components/tado/ @erwindouna
|
/tests/components/tado/ @erwindouna
|
||||||
/homeassistant/components/tag/ @home-assistant/core
|
/homeassistant/components/tag/ @balloob @dmulcahey
|
||||||
/tests/components/tag/ @home-assistant/core
|
/tests/components/tag/ @balloob @dmulcahey
|
||||||
/homeassistant/components/tailscale/ @frenck
|
/homeassistant/components/tailscale/ @frenck
|
||||||
/tests/components/tailscale/ @frenck
|
/tests/components/tailscale/ @frenck
|
||||||
/homeassistant/components/tailwind/ @frenck
|
/homeassistant/components/tailwind/ @frenck
|
||||||
@@ -1590,12 +1527,10 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/technove/ @Moustachauve
|
/tests/components/technove/ @Moustachauve
|
||||||
/homeassistant/components/tedee/ @patrickhilker @zweckj
|
/homeassistant/components/tedee/ @patrickhilker @zweckj
|
||||||
/tests/components/tedee/ @patrickhilker @zweckj
|
/tests/components/tedee/ @patrickhilker @zweckj
|
||||||
/homeassistant/components/telegram_bot/ @hanwg
|
|
||||||
/tests/components/telegram_bot/ @hanwg
|
|
||||||
/homeassistant/components/tellduslive/ @fredrike
|
/homeassistant/components/tellduslive/ @fredrike
|
||||||
/tests/components/tellduslive/ @fredrike
|
/tests/components/tellduslive/ @fredrike
|
||||||
/homeassistant/components/template/ @Petro31 @home-assistant/core
|
/homeassistant/components/template/ @PhracturedBlue @home-assistant/core
|
||||||
/tests/components/template/ @Petro31 @home-assistant/core
|
/tests/components/template/ @PhracturedBlue @home-assistant/core
|
||||||
/homeassistant/components/tesla_fleet/ @Bre77
|
/homeassistant/components/tesla_fleet/ @Bre77
|
||||||
/tests/components/tesla_fleet/ @Bre77
|
/tests/components/tesla_fleet/ @Bre77
|
||||||
/homeassistant/components/tesla_wall_connector/ @einarhauks
|
/homeassistant/components/tesla_wall_connector/ @einarhauks
|
||||||
@@ -1621,8 +1556,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/tile/ @bachya
|
/tests/components/tile/ @bachya
|
||||||
/homeassistant/components/tilt_ble/ @apt-itude
|
/homeassistant/components/tilt_ble/ @apt-itude
|
||||||
/tests/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
|
/homeassistant/components/time/ @home-assistant/core
|
||||||
/tests/components/time/ @home-assistant/core
|
/tests/components/time/ @home-assistant/core
|
||||||
/homeassistant/components/time_date/ @fabaff
|
/homeassistant/components/time_date/ @fabaff
|
||||||
@@ -1632,8 +1565,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/todo/ @home-assistant/core
|
/tests/components/todo/ @home-assistant/core
|
||||||
/homeassistant/components/todoist/ @boralyl
|
/homeassistant/components/todoist/ @boralyl
|
||||||
/tests/components/todoist/ @boralyl
|
/tests/components/todoist/ @boralyl
|
||||||
/homeassistant/components/togrill/ @elupus
|
|
||||||
/tests/components/togrill/ @elupus
|
|
||||||
/homeassistant/components/tolo/ @MatthiasLohr
|
/homeassistant/components/tolo/ @MatthiasLohr
|
||||||
/tests/components/tolo/ @MatthiasLohr
|
/tests/components/tolo/ @MatthiasLohr
|
||||||
/homeassistant/components/tomorrowio/ @raman325 @lymanepp
|
/homeassistant/components/tomorrowio/ @raman325 @lymanepp
|
||||||
@@ -1648,6 +1579,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/tplink_omada/ @MarkGodwin
|
/tests/components/tplink_omada/ @MarkGodwin
|
||||||
/homeassistant/components/traccar/ @ludeeus
|
/homeassistant/components/traccar/ @ludeeus
|
||||||
/tests/components/traccar/ @ludeeus
|
/tests/components/traccar/ @ludeeus
|
||||||
|
/homeassistant/components/traccar_server/ @ludeeus
|
||||||
|
/tests/components/traccar_server/ @ludeeus
|
||||||
/homeassistant/components/trace/ @home-assistant/core
|
/homeassistant/components/trace/ @home-assistant/core
|
||||||
/tests/components/trace/ @home-assistant/core
|
/tests/components/trace/ @home-assistant/core
|
||||||
/homeassistant/components/tractive/ @Danielhiversen @zhulik @bieniu
|
/homeassistant/components/tractive/ @Danielhiversen @zhulik @bieniu
|
||||||
@@ -1695,12 +1628,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/upnp/ @StevenLooman
|
/tests/components/upnp/ @StevenLooman
|
||||||
/homeassistant/components/uptime/ @frenck
|
/homeassistant/components/uptime/ @frenck
|
||||||
/tests/components/uptime/ @frenck
|
/tests/components/uptime/ @frenck
|
||||||
/homeassistant/components/uptime_kuma/ @tr4nt0r
|
|
||||||
/tests/components/uptime_kuma/ @tr4nt0r
|
|
||||||
/homeassistant/components/uptimerobot/ @ludeeus @chemelli74
|
/homeassistant/components/uptimerobot/ @ludeeus @chemelli74
|
||||||
/tests/components/uptimerobot/ @ludeeus @chemelli74
|
/tests/components/uptimerobot/ @ludeeus @chemelli74
|
||||||
/homeassistant/components/usage_prediction/ @home-assistant/core
|
|
||||||
/tests/components/usage_prediction/ @home-assistant/core
|
|
||||||
/homeassistant/components/usb/ @bdraco
|
/homeassistant/components/usb/ @bdraco
|
||||||
/tests/components/usb/ @bdraco
|
/tests/components/usb/ @bdraco
|
||||||
/homeassistant/components/usgs_earthquakes_feed/ @exxamalte
|
/homeassistant/components/usgs_earthquakes_feed/ @exxamalte
|
||||||
@@ -1715,23 +1644,19 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/vallox/ @andre-richter @slovdahl @viiru- @yozik04
|
/tests/components/vallox/ @andre-richter @slovdahl @viiru- @yozik04
|
||||||
/homeassistant/components/valve/ @home-assistant/core
|
/homeassistant/components/valve/ @home-assistant/core
|
||||||
/tests/components/valve/ @home-assistant/core
|
/tests/components/valve/ @home-assistant/core
|
||||||
/homeassistant/components/vegehub/ @ghowevege
|
|
||||||
/tests/components/vegehub/ @ghowevege
|
|
||||||
/homeassistant/components/velbus/ @Cereal2nd @brefra
|
/homeassistant/components/velbus/ @Cereal2nd @brefra
|
||||||
/tests/components/velbus/ @Cereal2nd @brefra
|
/tests/components/velbus/ @Cereal2nd @brefra
|
||||||
/homeassistant/components/velux/ @Julius2342 @DeerMaximum @pawlizio @wollew
|
/homeassistant/components/velux/ @Julius2342 @DeerMaximum @pawlizio
|
||||||
/tests/components/velux/ @Julius2342 @DeerMaximum @pawlizio @wollew
|
/tests/components/velux/ @Julius2342 @DeerMaximum @pawlizio
|
||||||
/homeassistant/components/venstar/ @garbled1 @jhollowe
|
/homeassistant/components/venstar/ @garbled1 @jhollowe
|
||||||
/tests/components/venstar/ @garbled1 @jhollowe
|
/tests/components/venstar/ @garbled1 @jhollowe
|
||||||
/homeassistant/components/versasense/ @imstevenxyz
|
/homeassistant/components/versasense/ @imstevenxyz
|
||||||
/homeassistant/components/version/ @ludeeus
|
/homeassistant/components/version/ @ludeeus
|
||||||
/tests/components/version/ @ludeeus
|
/tests/components/version/ @ludeeus
|
||||||
/homeassistant/components/vesync/ @markperdue @webdjoe @thegardenmonkey @cdnninja @iprak @sapuseven
|
/homeassistant/components/vesync/ @markperdue @webdjoe @thegardenmonkey @cdnninja @iprak
|
||||||
/tests/components/vesync/ @markperdue @webdjoe @thegardenmonkey @cdnninja @iprak @sapuseven
|
/tests/components/vesync/ @markperdue @webdjoe @thegardenmonkey @cdnninja @iprak
|
||||||
/homeassistant/components/vicare/ @CFenner
|
/homeassistant/components/vicare/ @CFenner
|
||||||
/tests/components/vicare/ @CFenner
|
/tests/components/vicare/ @CFenner
|
||||||
/homeassistant/components/victron_remote_monitoring/ @AndyTempel
|
|
||||||
/tests/components/victron_remote_monitoring/ @AndyTempel
|
|
||||||
/homeassistant/components/vilfo/ @ManneW
|
/homeassistant/components/vilfo/ @ManneW
|
||||||
/tests/components/vilfo/ @ManneW
|
/tests/components/vilfo/ @ManneW
|
||||||
/homeassistant/components/vivotek/ @HarlemSquirrel
|
/homeassistant/components/vivotek/ @HarlemSquirrel
|
||||||
@@ -1741,14 +1666,14 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/vlc_telnet/ @rodripf @MartinHjelmare
|
/tests/components/vlc_telnet/ @rodripf @MartinHjelmare
|
||||||
/homeassistant/components/vodafone_station/ @paoloantinori @chemelli74
|
/homeassistant/components/vodafone_station/ @paoloantinori @chemelli74
|
||||||
/tests/components/vodafone_station/ @paoloantinori @chemelli74
|
/tests/components/vodafone_station/ @paoloantinori @chemelli74
|
||||||
/homeassistant/components/voip/ @synesthesiam @jaminh
|
/homeassistant/components/voip/ @balloob @synesthesiam
|
||||||
/tests/components/voip/ @synesthesiam @jaminh
|
/tests/components/voip/ @balloob @synesthesiam
|
||||||
/homeassistant/components/volumio/ @OnFreund
|
/homeassistant/components/volumio/ @OnFreund
|
||||||
/tests/components/volumio/ @OnFreund
|
/tests/components/volumio/ @OnFreund
|
||||||
/homeassistant/components/volvo/ @thomasddn
|
/homeassistant/components/volvooncall/ @molobrakos
|
||||||
/tests/components/volvo/ @thomasddn
|
/tests/components/volvooncall/ @molobrakos
|
||||||
/homeassistant/components/volvooncall/ @molobrakos @svrooij
|
/homeassistant/components/vulcan/ @Antoni-Czaplicki
|
||||||
/tests/components/volvooncall/ @molobrakos @svrooij
|
/tests/components/vulcan/ @Antoni-Czaplicki
|
||||||
/homeassistant/components/wake_on_lan/ @ntilley905
|
/homeassistant/components/wake_on_lan/ @ntilley905
|
||||||
/tests/components/wake_on_lan/ @ntilley905
|
/tests/components/wake_on_lan/ @ntilley905
|
||||||
/homeassistant/components/wake_word/ @home-assistant/core @synesthesiam
|
/homeassistant/components/wake_word/ @home-assistant/core @synesthesiam
|
||||||
@@ -1799,8 +1724,8 @@ build.json @home-assistant/supervisor
|
|||||||
/homeassistant/components/wirelesstag/ @sergeymaysak
|
/homeassistant/components/wirelesstag/ @sergeymaysak
|
||||||
/homeassistant/components/withings/ @joostlek
|
/homeassistant/components/withings/ @joostlek
|
||||||
/tests/components/withings/ @joostlek
|
/tests/components/withings/ @joostlek
|
||||||
/homeassistant/components/wiz/ @sbidy @arturpragacz
|
/homeassistant/components/wiz/ @sbidy
|
||||||
/tests/components/wiz/ @sbidy @arturpragacz
|
/tests/components/wiz/ @sbidy
|
||||||
/homeassistant/components/wled/ @frenck
|
/homeassistant/components/wled/ @frenck
|
||||||
/tests/components/wled/ @frenck
|
/tests/components/wled/ @frenck
|
||||||
/homeassistant/components/wmspro/ @mback2k
|
/homeassistant/components/wmspro/ @mback2k
|
||||||
@@ -1813,8 +1738,8 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/worldclock/ @fabaff
|
/tests/components/worldclock/ @fabaff
|
||||||
/homeassistant/components/ws66i/ @ssaenger
|
/homeassistant/components/ws66i/ @ssaenger
|
||||||
/tests/components/ws66i/ @ssaenger
|
/tests/components/ws66i/ @ssaenger
|
||||||
/homeassistant/components/wyoming/ @synesthesiam
|
/homeassistant/components/wyoming/ @balloob @synesthesiam
|
||||||
/tests/components/wyoming/ @synesthesiam
|
/tests/components/wyoming/ @balloob @synesthesiam
|
||||||
/homeassistant/components/xbox/ @hunterjm
|
/homeassistant/components/xbox/ @hunterjm
|
||||||
/tests/components/xbox/ @hunterjm
|
/tests/components/xbox/ @hunterjm
|
||||||
/homeassistant/components/xiaomi_aqara/ @danielhiversen @syssi
|
/homeassistant/components/xiaomi_aqara/ @danielhiversen @syssi
|
||||||
@@ -1859,8 +1784,6 @@ build.json @home-assistant/supervisor
|
|||||||
/tests/components/zeversolar/ @kvanzuijlen
|
/tests/components/zeversolar/ @kvanzuijlen
|
||||||
/homeassistant/components/zha/ @dmulcahey @adminiuga @puddly @TheJulianJES
|
/homeassistant/components/zha/ @dmulcahey @adminiuga @puddly @TheJulianJES
|
||||||
/tests/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
|
/homeassistant/components/zodiac/ @JulienTant
|
||||||
/tests/components/zodiac/ @JulienTant
|
/tests/components/zodiac/ @JulienTant
|
||||||
/homeassistant/components/zone/ @home-assistant/core
|
/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
|
## 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.
|
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.
|
||||||
## Issue Tracker
|
|
||||||
|
|
||||||
If you want to report an issue, please [create an issue](https://github.com/home-assistant/core/issues) on GitHub.
|
|
||||||
|
4
Dockerfile
generated
4
Dockerfile
generated
@@ -25,13 +25,13 @@ RUN \
|
|||||||
"armv7") go2rtc_suffix='arm' ;; \
|
"armv7") go2rtc_suffix='arm' ;; \
|
||||||
*) go2rtc_suffix=${BUILD_ARCH} ;; \
|
*) go2rtc_suffix=${BUILD_ARCH} ;; \
|
||||||
esac \
|
esac \
|
||||||
&& curl -L https://github.com/AlexxIT/go2rtc/releases/download/v1.9.9/go2rtc_linux_${go2rtc_suffix} --output /bin/go2rtc \
|
&& curl -L https://github.com/AlexxIT/go2rtc/releases/download/v1.9.8/go2rtc_linux_${go2rtc_suffix} --output /bin/go2rtc \
|
||||||
&& chmod +x /bin/go2rtc \
|
&& chmod +x /bin/go2rtc \
|
||||||
# Verify go2rtc can be executed
|
# Verify go2rtc can be executed
|
||||||
&& go2rtc --version
|
&& go2rtc --version
|
||||||
|
|
||||||
# Install uv
|
# Install uv
|
||||||
RUN pip3 install uv==0.8.9
|
RUN pip3 install uv==0.6.1
|
||||||
|
|
||||||
WORKDIR /usr/src
|
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"]
|
SHELL ["/bin/bash", "-o", "pipefail", "-c"]
|
||||||
|
|
||||||
|
# Uninstall pre-installed formatting and linting tools
|
||||||
|
# They would conflict with our pinned versions
|
||||||
RUN \
|
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 \
|
&& DEBIAN_FRONTEND=noninteractive apt-get install -y --no-install-recommends \
|
||||||
# Additional library needed by some tests and accordingly by VScode Tests Discovery
|
# Additional library needed by some tests and accordingly by VScode Tests Discovery
|
||||||
bluez \
|
bluez \
|
||||||
@@ -23,32 +32,29 @@ RUN \
|
|||||||
libxml2 \
|
libxml2 \
|
||||||
git \
|
git \
|
||||||
cmake \
|
cmake \
|
||||||
autoconf \
|
|
||||||
&& apt-get clean \
|
&& apt-get clean \
|
||||||
&& rm -rf /var/lib/apt/lists/*
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
# Add go2rtc binary
|
# Add go2rtc binary
|
||||||
COPY --from=ghcr.io/alexxit/go2rtc:latest /usr/local/bin/go2rtc /bin/go2rtc
|
COPY --from=ghcr.io/alexxit/go2rtc:latest /usr/local/bin/go2rtc /bin/go2rtc
|
||||||
|
|
||||||
|
# Install uv
|
||||||
|
RUN pip3 install uv
|
||||||
|
|
||||||
WORKDIR /usr/src
|
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
|
USER vscode
|
||||||
|
|
||||||
COPY .python-version ./
|
|
||||||
RUN uv python install
|
|
||||||
|
|
||||||
ENV VIRTUAL_ENV="/home/vscode/.local/ha-venv"
|
ENV VIRTUAL_ENV="/home/vscode/.local/ha-venv"
|
||||||
RUN uv venv $VIRTUAL_ENV
|
RUN uv venv $VIRTUAL_ENV
|
||||||
ENV PATH="$VIRTUAL_ENV/bin:$PATH"
|
ENV PATH="$VIRTUAL_ENV/bin:$PATH"
|
||||||
|
|
||||||
WORKDIR /tmp
|
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
|
# Install Python dependencies from requirements
|
||||||
COPY requirements.txt ./
|
COPY requirements.txt ./
|
||||||
COPY homeassistant/package_constraints.txt homeassistant/package_constraints.txt
|
COPY homeassistant/package_constraints.txt homeassistant/package_constraints.txt
|
||||||
@@ -59,4 +65,4 @@ RUN uv pip install -r requirements_test.txt
|
|||||||
WORKDIR /workspaces
|
WORKDIR /workspaces
|
||||||
|
|
||||||
# Set the default shell to bash instead of sh
|
# Set the default shell to bash instead of sh
|
||||||
ENV SHELL=/bin/bash
|
ENV SHELL /bin/bash
|
||||||
|
12
build.yaml
12
build.yaml
@@ -1,10 +1,10 @@
|
|||||||
image: ghcr.io/home-assistant/{arch}-homeassistant
|
image: ghcr.io/home-assistant/{arch}-homeassistant
|
||||||
build_from:
|
build_from:
|
||||||
aarch64: ghcr.io/home-assistant/aarch64-homeassistant-base:2025.10.1
|
aarch64: ghcr.io/home-assistant/aarch64-homeassistant-base:2025.02.1
|
||||||
armhf: ghcr.io/home-assistant/armhf-homeassistant-base:2025.10.1
|
armhf: ghcr.io/home-assistant/armhf-homeassistant-base:2025.02.1
|
||||||
armv7: ghcr.io/home-assistant/armv7-homeassistant-base:2025.10.1
|
armv7: ghcr.io/home-assistant/armv7-homeassistant-base:2025.02.1
|
||||||
amd64: ghcr.io/home-assistant/amd64-homeassistant-base:2025.10.1
|
amd64: ghcr.io/home-assistant/amd64-homeassistant-base:2025.02.1
|
||||||
i386: ghcr.io/home-assistant/i386-homeassistant-base:2025.10.1
|
i386: ghcr.io/home-assistant/i386-homeassistant-base:2025.02.1
|
||||||
codenotary:
|
codenotary:
|
||||||
signer: notary@home-assistant.io
|
signer: notary@home-assistant.io
|
||||||
base_image: notary@home-assistant.io
|
base_image: notary@home-assistant.io
|
||||||
@@ -19,4 +19,4 @@ labels:
|
|||||||
org.opencontainers.image.authors: The Home Assistant Authors
|
org.opencontainers.image.authors: The Home Assistant Authors
|
||||||
org.opencontainers.image.url: https://www.home-assistant.io/
|
org.opencontainers.image.url: https://www.home-assistant.io/
|
||||||
org.opencontainers.image.documentation: https://www.home-assistant.io/docs/
|
org.opencontainers.image.documentation: https://www.home-assistant.io/docs/
|
||||||
org.opencontainers.image.licenses: Apache-2.0
|
org.opencontainers.image.licenses: Apache License 2.0
|
||||||
|
@@ -38,7 +38,8 @@ def validate_python() -> None:
|
|||||||
|
|
||||||
def ensure_config_path(config_dir: str) -> None:
|
def ensure_config_path(config_dir: str) -> None:
|
||||||
"""Validate the configuration directory."""
|
"""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")
|
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:
|
def get_arguments() -> argparse.Namespace:
|
||||||
"""Get parsed passed in arguments."""
|
"""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(
|
parser = argparse.ArgumentParser(
|
||||||
description="Home Assistant: Observe, Control, Automate.",
|
description="Home Assistant: Observe, Control, Automate.",
|
||||||
@@ -175,7 +177,8 @@ def main() -> int:
|
|||||||
validate_os()
|
validate_os()
|
||||||
|
|
||||||
if args.script is not None:
|
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)
|
return scripts.run(args.script)
|
||||||
|
|
||||||
@@ -185,44 +188,39 @@ def main() -> int:
|
|||||||
|
|
||||||
ensure_config_path(config_dir)
|
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
|
safe_mode = config.safe_mode_enabled(config_dir)
|
||||||
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)
|
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(
|
fault_file_name = os.path.join(config_dir, FAULT_LOG_FILENAME)
|
||||||
config_dir=config_dir,
|
with open(fault_file_name, mode="a", encoding="utf8") as fault_file:
|
||||||
verbose=args.verbose,
|
faulthandler.enable(fault_file)
|
||||||
log_rotate_days=args.log_rotate_days,
|
exit_code = runner.run(runtime_conf)
|
||||||
log_file=args.log_file,
|
faulthandler.disable()
|
||||||
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)
|
# It's possible for the fault file to disappear, so suppress obvious errors
|
||||||
with open(fault_file_name, mode="a", encoding="utf8") as fault_file:
|
with suppress(FileNotFoundError):
|
||||||
faulthandler.enable(fault_file)
|
if os.path.getsize(fault_file_name) == 0:
|
||||||
exit_code = runner.run(runtime_conf)
|
os.remove(fault_file_name)
|
||||||
faulthandler.disable()
|
|
||||||
|
|
||||||
# It's possible for the fault file to disappear, so suppress obvious errors
|
check_threads()
|
||||||
with suppress(FileNotFoundError):
|
|
||||||
if os.path.getsize(fault_file_name) == 0:
|
|
||||||
os.remove(fault_file_name)
|
|
||||||
|
|
||||||
check_threads()
|
return exit_code
|
||||||
|
|
||||||
return exit_code
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
@@ -120,9 +120,6 @@ class AuthStore:
|
|||||||
|
|
||||||
new_user = models.User(**kwargs)
|
new_user = models.User(**kwargs)
|
||||||
|
|
||||||
while new_user.id in self._users:
|
|
||||||
new_user = models.User(**kwargs)
|
|
||||||
|
|
||||||
self._users[new_user.id] = new_user
|
self._users[new_user.id] = new_user
|
||||||
|
|
||||||
if credentials is None:
|
if credentials is None:
|
||||||
|
@@ -27,7 +27,7 @@ from . import (
|
|||||||
SetupFlow,
|
SetupFlow,
|
||||||
)
|
)
|
||||||
|
|
||||||
REQUIREMENTS = ["pyotp==2.9.0"]
|
REQUIREMENTS = ["pyotp==2.8.0"]
|
||||||
|
|
||||||
CONF_MESSAGE = "message"
|
CONF_MESSAGE = "message"
|
||||||
|
|
||||||
@@ -52,28 +52,28 @@ _LOGGER = logging.getLogger(__name__)
|
|||||||
|
|
||||||
def _generate_secret() -> str:
|
def _generate_secret() -> str:
|
||||||
"""Generate a secret."""
|
"""Generate a secret."""
|
||||||
import pyotp # noqa: PLC0415
|
import pyotp # pylint: disable=import-outside-toplevel
|
||||||
|
|
||||||
return str(pyotp.random_base32())
|
return str(pyotp.random_base32())
|
||||||
|
|
||||||
|
|
||||||
def _generate_random() -> int:
|
def _generate_random() -> int:
|
||||||
"""Generate a 32 digit number."""
|
"""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")))
|
return int(pyotp.random_base32(length=32, chars=list("1234567890")))
|
||||||
|
|
||||||
|
|
||||||
def _generate_otp(secret: str, count: int) -> str:
|
def _generate_otp(secret: str, count: int) -> str:
|
||||||
"""Generate one time password."""
|
"""Generate one time password."""
|
||||||
import pyotp # noqa: PLC0415
|
import pyotp # pylint: disable=import-outside-toplevel
|
||||||
|
|
||||||
return str(pyotp.HOTP(secret).at(count))
|
return str(pyotp.HOTP(secret).at(count))
|
||||||
|
|
||||||
|
|
||||||
def _verify_otp(secret: str, otp: str, count: int) -> bool:
|
def _verify_otp(secret: str, otp: str, count: int) -> bool:
|
||||||
"""Verify one time password."""
|
"""Verify one time password."""
|
||||||
import pyotp # noqa: PLC0415
|
import pyotp # pylint: disable=import-outside-toplevel
|
||||||
|
|
||||||
return bool(pyotp.HOTP(secret).verify(otp, count))
|
return bool(pyotp.HOTP(secret).verify(otp, count))
|
||||||
|
|
||||||
|
@@ -20,7 +20,7 @@ from . import (
|
|||||||
SetupFlow,
|
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)
|
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:
|
def _generate_qr_code(data: str) -> str:
|
||||||
"""Generate a base64 PNG string represent QR Code image of data."""
|
"""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)
|
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]:
|
def _generate_secret_and_qr_code(username: str) -> tuple[str, str, str]:
|
||||||
"""Generate a secret, url, and QR code."""
|
"""Generate a secret, url, and QR code."""
|
||||||
import pyotp # noqa: PLC0415
|
import pyotp # pylint: disable=import-outside-toplevel
|
||||||
|
|
||||||
ota_secret = pyotp.random_base32()
|
ota_secret = pyotp.random_base32()
|
||||||
url = pyotp.totp.TOTP(ota_secret).provisioning_uri(
|
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:
|
def _add_ota_secret(self, user_id: str, secret: str | None = None) -> str:
|
||||||
"""Create a ota_secret for user."""
|
"""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()
|
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:
|
def _validate_2fa(self, user_id: str, code: str) -> bool:
|
||||||
"""Validate two factor authentication code."""
|
"""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]
|
if (ota_secret := self._users.get(user_id)) is None: # type: ignore[union-attr]
|
||||||
# even we cannot find user, we still do verify
|
# 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_show_form(step_id='init') if user_input is None.
|
||||||
Return self.async_create_entry(data={'result': result}) if finish.
|
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] = {}
|
errors: dict[str, str] = {}
|
||||||
|
|
||||||
|
@@ -33,10 +33,7 @@ class AuthFlowContext(FlowContext, total=False):
|
|||||||
redirect_uri: str
|
redirect_uri: str
|
||||||
|
|
||||||
|
|
||||||
class AuthFlowResult(FlowResult[AuthFlowContext, tuple[str, str]], total=False):
|
AuthFlowResult = FlowResult[AuthFlowContext, tuple[str, str]]
|
||||||
"""Typed result dict for auth flow."""
|
|
||||||
|
|
||||||
result: Credentials # Only present if type is CREATE_ENTRY
|
|
||||||
|
|
||||||
|
|
||||||
@attr.s(slots=True)
|
@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())
|
@@ -178,15 +178,6 @@ _BLOCKING_CALLS: tuple[BlockingCall, ...] = (
|
|||||||
strict_core=False,
|
strict_core=False,
|
||||||
skip_for_tests=True,
|
skip_for_tests=True,
|
||||||
),
|
),
|
||||||
BlockingCall(
|
|
||||||
original_func=SSLContext.set_default_verify_paths,
|
|
||||||
object=SSLContext,
|
|
||||||
function="set_default_verify_paths",
|
|
||||||
check_allowed=None,
|
|
||||||
strict=False,
|
|
||||||
strict_core=False,
|
|
||||||
skip_for_tests=True,
|
|
||||||
),
|
|
||||||
BlockingCall(
|
BlockingCall(
|
||||||
original_func=Path.open,
|
original_func=Path.open,
|
||||||
object=Path,
|
object=Path,
|
||||||
|
@@ -53,7 +53,6 @@ from .components import (
|
|||||||
logbook as logbook_pre_import, # noqa: F401
|
logbook as logbook_pre_import, # noqa: F401
|
||||||
lovelace as lovelace_pre_import, # noqa: F401
|
lovelace as lovelace_pre_import, # noqa: F401
|
||||||
onboarding as onboarding_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
|
recorder as recorder_import, # noqa: F401 - not named pre_import since it has requirements
|
||||||
repairs as repairs_pre_import, # noqa: F401
|
repairs as repairs_pre_import, # noqa: F401
|
||||||
search as search_pre_import, # noqa: F401
|
search as search_pre_import, # noqa: F401
|
||||||
@@ -75,27 +74,24 @@ from .core_config import async_process_ha_core_config
|
|||||||
from .exceptions import HomeAssistantError
|
from .exceptions import HomeAssistantError
|
||||||
from .helpers import (
|
from .helpers import (
|
||||||
area_registry,
|
area_registry,
|
||||||
|
backup,
|
||||||
category_registry,
|
category_registry,
|
||||||
condition,
|
|
||||||
config_validation as cv,
|
config_validation as cv,
|
||||||
device_registry,
|
device_registry,
|
||||||
entity,
|
entity,
|
||||||
entity_registry,
|
entity_registry,
|
||||||
floor_registry,
|
floor_registry,
|
||||||
frame,
|
|
||||||
issue_registry,
|
issue_registry,
|
||||||
label_registry,
|
label_registry,
|
||||||
recorder,
|
recorder,
|
||||||
restore_state,
|
restore_state,
|
||||||
template,
|
template,
|
||||||
translation,
|
translation,
|
||||||
trigger,
|
|
||||||
)
|
)
|
||||||
from .helpers.dispatcher import async_dispatcher_send_internal
|
from .helpers.dispatcher import async_dispatcher_send_internal
|
||||||
from .helpers.storage import get_internal_store_manager
|
from .helpers.storage import get_internal_store_manager
|
||||||
from .helpers.system_info import async_get_system_info
|
from .helpers.system_info import async_get_system_info
|
||||||
from .helpers.typing import ConfigType
|
from .helpers.typing import ConfigType
|
||||||
from .loader import Integration
|
|
||||||
from .setup import (
|
from .setup import (
|
||||||
# _setup_started is marked as protected to make it clear
|
# _setup_started is marked as protected to make it clear
|
||||||
# that it is not part of the public API and should not be used
|
# that it is not part of the public API and should not be used
|
||||||
@@ -172,6 +168,8 @@ FRONTEND_INTEGRATIONS = {
|
|||||||
# Stage 0 is divided into substages. Each substage has a name, a set of integrations and a timeout.
|
# 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.
|
# 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.
|
# 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.
|
# If we add timeouts to the frontend substages, we should make sure they don't apply in recovery mode.
|
||||||
STAGE_0_INTEGRATIONS = (
|
STAGE_0_INTEGRATIONS = (
|
||||||
# Load logging and http deps as soon as possible
|
# Load logging and http deps as soon as possible
|
||||||
@@ -300,6 +298,14 @@ async def async_setup_hass(
|
|||||||
|
|
||||||
return hass
|
return hass
|
||||||
|
|
||||||
|
async def stop_hass(hass: core.HomeAssistant) -> None:
|
||||||
|
"""Stop hass."""
|
||||||
|
# Ask integrations to shut down. It's messy but we can't
|
||||||
|
# do a clean stop without knowing what is broken
|
||||||
|
with contextlib.suppress(TimeoutError):
|
||||||
|
async with hass.timeout.async_timeout(10):
|
||||||
|
await hass.async_stop()
|
||||||
|
|
||||||
hass = await create_hass()
|
hass = await create_hass()
|
||||||
|
|
||||||
if runtime_config.skip_pip or runtime_config.skip_pip_packages:
|
if runtime_config.skip_pip or runtime_config.skip_pip_packages:
|
||||||
@@ -332,16 +338,13 @@ async def async_setup_hass(
|
|||||||
if not is_virtual_env():
|
if not is_virtual_env():
|
||||||
await async_mount_local_lib_path(runtime_config.config_dir)
|
await async_mount_local_lib_path(runtime_config.config_dir)
|
||||||
|
|
||||||
if hass.config.safe_mode:
|
|
||||||
_LOGGER.info("Starting in safe mode")
|
|
||||||
|
|
||||||
basic_setup_success = (
|
basic_setup_success = (
|
||||||
await async_from_config_dict(config_dict, hass) is not None
|
await async_from_config_dict(config_dict, hass) is not None
|
||||||
)
|
)
|
||||||
|
|
||||||
if config_dict is None:
|
if config_dict is None:
|
||||||
recovery_mode = True
|
recovery_mode = True
|
||||||
await hass.async_stop(force=True)
|
await stop_hass(hass)
|
||||||
hass = await create_hass()
|
hass = await create_hass()
|
||||||
|
|
||||||
elif not basic_setup_success:
|
elif not basic_setup_success:
|
||||||
@@ -349,7 +352,7 @@ async def async_setup_hass(
|
|||||||
"Unable to set up core integrations. Activating recovery mode"
|
"Unable to set up core integrations. Activating recovery mode"
|
||||||
)
|
)
|
||||||
recovery_mode = True
|
recovery_mode = True
|
||||||
await hass.async_stop(force=True)
|
await stop_hass(hass)
|
||||||
hass = await create_hass()
|
hass = await create_hass()
|
||||||
|
|
||||||
elif any(
|
elif any(
|
||||||
@@ -364,7 +367,7 @@ async def async_setup_hass(
|
|||||||
old_logging = hass.data.get(DATA_LOGGING)
|
old_logging = hass.data.get(DATA_LOGGING)
|
||||||
|
|
||||||
recovery_mode = True
|
recovery_mode = True
|
||||||
await hass.async_stop(force=True)
|
await stop_hass(hass)
|
||||||
hass = await create_hass()
|
hass = await create_hass()
|
||||||
|
|
||||||
if old_logging:
|
if old_logging:
|
||||||
@@ -387,6 +390,8 @@ async def async_setup_hass(
|
|||||||
{"recovery_mode": {}, "http": http_conf},
|
{"recovery_mode": {}, "http": http_conf},
|
||||||
hass,
|
hass,
|
||||||
)
|
)
|
||||||
|
elif hass.config.safe_mode:
|
||||||
|
_LOGGER.info("Starting in safe mode")
|
||||||
|
|
||||||
if runtime_config.open_ui:
|
if runtime_config.open_ui:
|
||||||
hass.add_job(open_hass_ui, hass)
|
hass.add_job(open_hass_ui, hass)
|
||||||
@@ -396,7 +401,7 @@ async def async_setup_hass(
|
|||||||
|
|
||||||
def open_hass_ui(hass: core.HomeAssistant) -> None:
|
def open_hass_ui(hass: core.HomeAssistant) -> None:
|
||||||
"""Open the UI."""
|
"""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:
|
if hass.config.api is None or "frontend" not in hass.config.components:
|
||||||
_LOGGER.warning("Cannot launch the UI because frontend not loaded")
|
_LOGGER.warning("Cannot launch the UI because frontend not loaded")
|
||||||
@@ -436,10 +441,9 @@ async def async_load_base_functionality(hass: core.HomeAssistant) -> None:
|
|||||||
if DATA_REGISTRIES_LOADED in hass.data:
|
if DATA_REGISTRIES_LOADED in hass.data:
|
||||||
return
|
return
|
||||||
hass.data[DATA_REGISTRIES_LOADED] = None
|
hass.data[DATA_REGISTRIES_LOADED] = None
|
||||||
entity.async_setup(hass)
|
|
||||||
frame.async_setup(hass)
|
|
||||||
template.async_setup(hass)
|
|
||||||
translation.async_setup(hass)
|
translation.async_setup(hass)
|
||||||
|
entity.async_setup(hass)
|
||||||
|
template.async_setup(hass)
|
||||||
await asyncio.gather(
|
await asyncio.gather(
|
||||||
create_eager_task(get_internal_store_manager(hass).async_initialize()),
|
create_eager_task(get_internal_store_manager(hass).async_initialize()),
|
||||||
create_eager_task(area_registry.async_load(hass)),
|
create_eager_task(area_registry.async_load(hass)),
|
||||||
@@ -454,8 +458,6 @@ async def async_load_base_functionality(hass: core.HomeAssistant) -> None:
|
|||||||
create_eager_task(restore_state.async_load(hass)),
|
create_eager_task(restore_state.async_load(hass)),
|
||||||
create_eager_task(hass.config_entries.async_initialize()),
|
create_eager_task(hass.config_entries.async_initialize()),
|
||||||
create_eager_task(async_get_system_info(hass)),
|
create_eager_task(async_get_system_info(hass)),
|
||||||
create_eager_task(condition.async_setup(hass)),
|
|
||||||
create_eager_task(trigger.async_setup(hass)),
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@@ -565,7 +567,8 @@ async def async_enable_logging(
|
|||||||
|
|
||||||
if not log_no_color:
|
if not log_no_color:
|
||||||
try:
|
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
|
# basicConfig must be called after importing colorlog in order to
|
||||||
# ensure that the handlers it sets up wraps the correct streams.
|
# ensure that the handlers it sets up wraps the correct streams.
|
||||||
@@ -616,34 +619,34 @@ async def async_enable_logging(
|
|||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
logger = logging.getLogger()
|
# Log errors to a file if we have write access to file or config dir
|
||||||
logger.setLevel(logging.INFO if verbose else logging.WARNING)
|
|
||||||
|
|
||||||
if log_file is None:
|
if log_file is None:
|
||||||
default_log_path = hass.config.path(ERROR_LOG_FILENAME)
|
err_log_path = hass.config.path(ERROR_LOG_FILENAME)
|
||||||
if "SUPERVISOR" in os.environ:
|
|
||||||
_LOGGER.info("Running in Supervisor, not logging to file")
|
|
||||||
# Rename the default log file if it exists, since previous versions created
|
|
||||||
# it even on Supervisor
|
|
||||||
if os.path.isfile(default_log_path):
|
|
||||||
with contextlib.suppress(OSError):
|
|
||||||
os.rename(default_log_path, f"{default_log_path}.old")
|
|
||||||
err_log_path = None
|
|
||||||
else:
|
|
||||||
err_log_path = default_log_path
|
|
||||||
else:
|
else:
|
||||||
err_log_path = os.path.abspath(log_file)
|
err_log_path = os.path.abspath(log_file)
|
||||||
|
|
||||||
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(
|
err_handler = await hass.async_add_executor_job(
|
||||||
_create_log_file, err_log_path, log_rotate_days
|
_create_log_file, err_log_path, log_rotate_days
|
||||||
)
|
)
|
||||||
|
|
||||||
err_handler.setFormatter(logging.Formatter(fmt, datefmt=FORMAT_DATETIME))
|
err_handler.setFormatter(logging.Formatter(fmt, datefmt=FORMAT_DATETIME))
|
||||||
|
|
||||||
|
logger = logging.getLogger()
|
||||||
logger.addHandler(err_handler)
|
logger.addHandler(err_handler)
|
||||||
|
logger.setLevel(logging.INFO if verbose else logging.WARNING)
|
||||||
|
|
||||||
# Save the log file location for access by other components.
|
# Save the log file location for access by other components.
|
||||||
hass.data[DATA_LOGGING] = err_log_path
|
hass.data[DATA_LOGGING] = err_log_path
|
||||||
|
else:
|
||||||
|
_LOGGER.error("Unable to set up error log %s (access denied)", err_log_path)
|
||||||
|
|
||||||
async_activate_log_queue_handler(hass)
|
async_activate_log_queue_handler(hass)
|
||||||
|
|
||||||
@@ -661,10 +664,11 @@ def _create_log_file(
|
|||||||
err_handler = _RotatingFileHandlerWithoutShouldRollOver(
|
err_handler = _RotatingFileHandlerWithoutShouldRollOver(
|
||||||
err_log_path, backupCount=1
|
err_log_path, backupCount=1
|
||||||
)
|
)
|
||||||
try:
|
|
||||||
err_handler.doRollover()
|
try:
|
||||||
except OSError as err:
|
err_handler.doRollover()
|
||||||
_LOGGER.error("Error rolling over log file: %s", err)
|
except OSError as err:
|
||||||
|
_LOGGER.error("Error rolling over log file: %s", err)
|
||||||
|
|
||||||
return err_handler
|
return err_handler
|
||||||
|
|
||||||
@@ -695,10 +699,10 @@ async def async_mount_local_lib_path(config_dir: str) -> str:
|
|||||||
|
|
||||||
def _get_domains(hass: core.HomeAssistant, config: dict[str, Any]) -> set[str]:
|
def _get_domains(hass: core.HomeAssistant, config: dict[str, Any]) -> set[str]:
|
||||||
"""Get domains of components to set up."""
|
"""Get domains of components to set up."""
|
||||||
# The common config section [homeassistant] could be filtered here,
|
# Filter out the repeating and common config section [homeassistant]
|
||||||
# but that is not necessary, since it corresponds to the core integration,
|
domains = {
|
||||||
# that is always unconditionally loaded.
|
domain for key in config if (domain := cv.domain_key(key)) != core.DOMAIN
|
||||||
domains = {cv.domain_key(key) for key in config}
|
}
|
||||||
|
|
||||||
# Add config entry and default domains
|
# Add config entry and default domains
|
||||||
if not hass.config.recovery_mode:
|
if not hass.config.recovery_mode:
|
||||||
@@ -714,106 +718,140 @@ def _get_domains(hass: core.HomeAssistant, config: dict[str, Any]) -> set[str]:
|
|||||||
return domains
|
return domains
|
||||||
|
|
||||||
|
|
||||||
async def _async_resolve_domains_and_preload(
|
async def _async_resolve_domains_to_setup(
|
||||||
hass: core.HomeAssistant, config: dict[str, Any]
|
hass: core.HomeAssistant, config: dict[str, Any]
|
||||||
) -> tuple[dict[str, Integration], dict[str, Integration]]:
|
) -> tuple[set[str], dict[str, loader.Integration]]:
|
||||||
"""Resolve all dependencies and return integrations to set up.
|
"""Resolve all dependencies and return list of domains to set up."""
|
||||||
|
|
||||||
The return value is a tuple of two dictionaries:
|
|
||||||
- The first dictionary contains integrations
|
|
||||||
specified by the configuration (including config entries).
|
|
||||||
- The second dictionary contains the same integrations as the first dictionary
|
|
||||||
together with all their dependencies.
|
|
||||||
"""
|
|
||||||
domains_to_setup = _get_domains(hass, config)
|
domains_to_setup = _get_domains(hass, config)
|
||||||
|
needed_requirements: set[str] = set()
|
||||||
# Also process all base platforms since we do not require the manifest
|
platform_integrations = conf_util.extract_platform_integrations(
|
||||||
# to list them as dependencies.
|
config, BASE_PLATFORMS
|
||||||
# We want to later avoid lock contention when multiple integrations try to load
|
)
|
||||||
# their manifests at once.
|
# Ensure base platforms that have platform integrations are added to
|
||||||
|
# to `domains_to_setup so they can be setup first instead of
|
||||||
|
# discovering them when later when a config entry setup task
|
||||||
|
# notices its 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
|
# For example if we have
|
||||||
# sensor:
|
# sensor:
|
||||||
# - platform: template
|
# - platform: template
|
||||||
#
|
#
|
||||||
# `template` has to be loaded to validate the config for sensor.
|
# `template` has to be loaded to validate the config for sensor
|
||||||
# The more platforms under `sensor:`, the longer
|
# 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
|
# it will take to finish setup for `sensor` because each of these
|
||||||
# platforms has to be imported before we can validate the config.
|
# platforms has to be imported before we can validate the config.
|
||||||
#
|
#
|
||||||
# Thankfully we are migrating away from the platform pattern
|
# Thankfully we are migrating away from the platform pattern
|
||||||
# so this will be less of a problem in the future.
|
# so this will be less of a problem in the future.
|
||||||
platform_integrations = conf_util.extract_platform_integrations(
|
domains_to_setup.update(platform_integrations)
|
||||||
config, BASE_PLATFORMS
|
|
||||||
)
|
# Load manifests for base platforms and platform based integrations
|
||||||
additional_domains_to_process = {
|
# that are defined under base platforms right away since we do not require
|
||||||
|
# the manifest to list them as dependencies and we want to avoid the lock
|
||||||
|
# contention when multiple integrations try to load them at once
|
||||||
|
additional_manifests_to_load = {
|
||||||
*BASE_PLATFORMS,
|
*BASE_PLATFORMS,
|
||||||
*chain.from_iterable(platform_integrations.values()),
|
*chain.from_iterable(platform_integrations.values()),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
translations_to_load = additional_manifests_to_load.copy()
|
||||||
|
|
||||||
# Resolve all dependencies so we know all integrations
|
# Resolve all dependencies so we know all integrations
|
||||||
# that will have to be loaded and start right-away
|
# that will have to be loaded and start right-away
|
||||||
integrations_or_excs = await loader.async_get_integrations(
|
integration_cache: dict[str, loader.Integration] = {}
|
||||||
hass, {*domains_to_setup, *additional_domains_to_process}
|
to_resolve: set[str] = domains_to_setup
|
||||||
)
|
while to_resolve or additional_manifests_to_load:
|
||||||
# Eliminate those missing or with invalid manifest
|
old_to_resolve: set[str] = to_resolve
|
||||||
integrations_to_process = {
|
to_resolve = set()
|
||||||
domain: itg
|
|
||||||
for domain, itg in integrations_or_excs.items()
|
|
||||||
if isinstance(itg, Integration)
|
|
||||||
}
|
|
||||||
integrations_dependencies = await loader.resolve_integrations_dependencies(
|
|
||||||
hass, integrations_to_process.values()
|
|
||||||
)
|
|
||||||
# Eliminate those without valid dependencies
|
|
||||||
integrations_to_process = {
|
|
||||||
domain: integrations_to_process[domain] for domain in integrations_dependencies
|
|
||||||
}
|
|
||||||
|
|
||||||
integrations_to_setup = {
|
if additional_manifests_to_load:
|
||||||
domain: itg
|
to_get = {*old_to_resolve, *additional_manifests_to_load}
|
||||||
for domain, itg in integrations_to_process.items()
|
additional_manifests_to_load.clear()
|
||||||
if domain in domains_to_setup
|
else:
|
||||||
}
|
to_get = old_to_resolve
|
||||||
all_integrations_to_setup = integrations_to_setup.copy()
|
|
||||||
all_integrations_to_setup.update(
|
|
||||||
(dep, loader.async_get_loaded_integration(hass, dep))
|
|
||||||
for domain in integrations_to_setup
|
|
||||||
for dep in integrations_dependencies[domain].difference(
|
|
||||||
all_integrations_to_setup
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
# Gather requirements for all integrations,
|
manifest_deps: set[str] = set()
|
||||||
# their dependencies and after dependencies.
|
resolve_dependencies_tasks: list[asyncio.Task[bool]] = []
|
||||||
# To gather all the requirements we must ignore exceptions here.
|
integrations_to_process: list[loader.Integration] = []
|
||||||
# The exceptions will be detected and handled later in the bootstrap process.
|
|
||||||
integrations_after_dependencies = (
|
for domain, itg in (await loader.async_get_integrations(hass, to_get)).items():
|
||||||
await loader.resolve_integrations_after_dependencies(
|
if not isinstance(itg, loader.Integration):
|
||||||
hass, integrations_to_process.values(), ignore_exceptions=True
|
continue
|
||||||
)
|
integration_cache[domain] = itg
|
||||||
)
|
needed_requirements.update(itg.requirements)
|
||||||
integrations_requirements = {
|
|
||||||
domain: itg.requirements for domain, itg in integrations_to_process.items()
|
# Make sure manifests for dependencies are loaded in the next
|
||||||
}
|
# loop to try to group as many as manifest loads in a single
|
||||||
integrations_requirements.update(
|
# call to avoid the creating one-off executor jobs later in
|
||||||
(dep, loader.async_get_loaded_integration(hass, dep).requirements)
|
# the setup process
|
||||||
for deps in integrations_after_dependencies.values()
|
additional_manifests_to_load.update(
|
||||||
for dep in deps.difference(integrations_requirements)
|
dep
|
||||||
)
|
for dep in chain(itg.dependencies, itg.after_dependencies)
|
||||||
all_requirements = set(chain.from_iterable(integrations_requirements.values()))
|
if dep not in integration_cache
|
||||||
|
)
|
||||||
|
|
||||||
|
if domain not in old_to_resolve:
|
||||||
|
continue
|
||||||
|
|
||||||
|
integrations_to_process.append(itg)
|
||||||
|
manifest_deps.update(itg.dependencies)
|
||||||
|
manifest_deps.update(itg.after_dependencies)
|
||||||
|
if not itg.all_dependencies_resolved:
|
||||||
|
resolve_dependencies_tasks.append(
|
||||||
|
create_eager_task(
|
||||||
|
itg.resolve_dependencies(),
|
||||||
|
name=f"resolve dependencies {domain}",
|
||||||
|
loop=hass.loop,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
if unseen_deps := manifest_deps - integration_cache.keys():
|
||||||
|
# If there are dependencies, try to preload all
|
||||||
|
# the integrations manifest at once and add them
|
||||||
|
# to the list of requirements we need to install
|
||||||
|
# so we can try to check if they are already installed
|
||||||
|
# in a single call below which avoids each integration
|
||||||
|
# having to wait for the lock to do it individually
|
||||||
|
deps = await loader.async_get_integrations(hass, unseen_deps)
|
||||||
|
for dependant_domain, dependant_itg in deps.items():
|
||||||
|
if isinstance(dependant_itg, loader.Integration):
|
||||||
|
integration_cache[dependant_domain] = dependant_itg
|
||||||
|
needed_requirements.update(dependant_itg.requirements)
|
||||||
|
|
||||||
|
if resolve_dependencies_tasks:
|
||||||
|
await asyncio.gather(*resolve_dependencies_tasks)
|
||||||
|
|
||||||
|
for itg in integrations_to_process:
|
||||||
|
try:
|
||||||
|
all_deps = itg.all_dependencies
|
||||||
|
except RuntimeError:
|
||||||
|
# Integration.all_dependencies raises RuntimeError if
|
||||||
|
# dependencies could not be resolved
|
||||||
|
continue
|
||||||
|
for dep in all_deps:
|
||||||
|
if dep in domains_to_setup:
|
||||||
|
continue
|
||||||
|
domains_to_setup.add(dep)
|
||||||
|
to_resolve.add(dep)
|
||||||
|
|
||||||
|
_LOGGER.info("Domains to be set up: %s", domains_to_setup)
|
||||||
|
|
||||||
# Optimistically check if requirements are already installed
|
# Optimistically check if requirements are already installed
|
||||||
# ahead of setting up the integrations so we can prime the cache
|
# ahead of setting up the integrations so we can prime the cache
|
||||||
# We do not wait for this since it's an optimization only
|
# We do not wait for this since its an optimization only
|
||||||
hass.async_create_background_task(
|
hass.async_create_background_task(
|
||||||
requirements.async_load_installed_versions(hass, all_requirements),
|
requirements.async_load_installed_versions(hass, needed_requirements),
|
||||||
"check installed requirements",
|
"check installed requirements",
|
||||||
eager_start=True,
|
eager_start=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
#
|
||||||
|
# Only add the domains_to_setup after we finish resolving
|
||||||
|
# as new domains are likely to added in the process
|
||||||
|
#
|
||||||
|
translations_to_load.update(domains_to_setup)
|
||||||
# Start loading translations for all integrations we are going to set up
|
# Start loading translations for all integrations we are going to set up
|
||||||
# in the background so they are ready when we need them. This avoids a
|
# in the background so they are ready when we need them. This avoids a
|
||||||
# lot of waiting for the translation load lock and a thundering herd of
|
# lot of waiting for the translation load lock and a thundering herd of
|
||||||
@@ -824,7 +862,6 @@ async def _async_resolve_domains_and_preload(
|
|||||||
# hold the translation load lock and if anything is fast enough to
|
# hold the translation load lock and if anything is fast enough to
|
||||||
# wait for the translation load lock, loading will be done by the
|
# wait for the translation load lock, loading will be done by the
|
||||||
# time it gets to it.
|
# time it gets to it.
|
||||||
translations_to_load = {*all_integrations_to_setup, *additional_domains_to_process}
|
|
||||||
hass.async_create_background_task(
|
hass.async_create_background_task(
|
||||||
translation.async_load_integrations(hass, translations_to_load),
|
translation.async_load_integrations(hass, translations_to_load),
|
||||||
"load translations",
|
"load translations",
|
||||||
@@ -836,13 +873,13 @@ async def _async_resolve_domains_and_preload(
|
|||||||
# in the setup process.
|
# in the setup process.
|
||||||
hass.async_create_background_task(
|
hass.async_create_background_task(
|
||||||
get_internal_store_manager(hass).async_preload(
|
get_internal_store_manager(hass).async_preload(
|
||||||
[*PRELOAD_STORAGE, *all_integrations_to_setup]
|
[*PRELOAD_STORAGE, *domains_to_setup]
|
||||||
),
|
),
|
||||||
"preload storage",
|
"preload storage",
|
||||||
eager_start=True,
|
eager_start=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
return integrations_to_setup, all_integrations_to_setup
|
return domains_to_setup, integration_cache
|
||||||
|
|
||||||
|
|
||||||
async def _async_set_up_integrations(
|
async def _async_set_up_integrations(
|
||||||
@@ -852,96 +889,76 @@ async def _async_set_up_integrations(
|
|||||||
watcher = _WatchPendingSetups(hass, _setup_started(hass))
|
watcher = _WatchPendingSetups(hass, _setup_started(hass))
|
||||||
watcher.async_start()
|
watcher.async_start()
|
||||||
|
|
||||||
integrations, all_integrations = await _async_resolve_domains_and_preload(
|
domains_to_setup, integration_cache = await _async_resolve_domains_to_setup(
|
||||||
hass, config
|
hass, config
|
||||||
)
|
)
|
||||||
# Detect all cycles
|
stage_2_domains = domains_to_setup.copy()
|
||||||
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
|
|
||||||
|
|
||||||
_LOGGER.info(
|
|
||||||
"Domains to be set up: %s\nDependencies: %s",
|
|
||||||
domains or "{}",
|
|
||||||
(all_domains - domains) or "{}",
|
|
||||||
)
|
|
||||||
|
|
||||||
async_set_domains_to_be_loaded(hass, all_domains)
|
|
||||||
|
|
||||||
# Initialize recorder
|
# Initialize recorder
|
||||||
if "recorder" in all_domains:
|
if "recorder" in domains_to_setup:
|
||||||
recorder.async_initialize_recorder(hass)
|
recorder.async_initialize_recorder(hass)
|
||||||
|
|
||||||
stages: list[tuple[str, set[str], int | None]] = [
|
# Initialize backup
|
||||||
|
if "backup" in domains_to_setup:
|
||||||
|
backup.async_initialize_backup(hass)
|
||||||
|
|
||||||
|
stage_0_and_1_domains: list[tuple[str, set[str], int | None]] = [
|
||||||
*(
|
*(
|
||||||
(name, domain_group, timeout)
|
(name, domain_group & domains_to_setup, timeout)
|
||||||
for name, domain_group, timeout in STAGE_0_INTEGRATIONS
|
for name, domain_group, timeout in STAGE_0_INTEGRATIONS
|
||||||
),
|
),
|
||||||
("1", STAGE_1_INTEGRATIONS, STAGE_1_TIMEOUT),
|
("stage 1", STAGE_1_INTEGRATIONS & domains_to_setup, STAGE_1_TIMEOUT),
|
||||||
("2", domains, STAGE_2_TIMEOUT),
|
|
||||||
]
|
]
|
||||||
|
|
||||||
_LOGGER.info("Setting up stage 0")
|
_LOGGER.info("Setting up stage 0 and 1")
|
||||||
for name, domain_group, timeout in stages:
|
for name, domain_group, timeout in stage_0_and_1_domains:
|
||||||
stage_domains_unfiltered = domain_group & all_domains
|
if not domain_group:
|
||||||
if not stage_domains_unfiltered:
|
|
||||||
_LOGGER.info("Nothing to set up in stage %s: %s", name, domain_group)
|
|
||||||
continue
|
continue
|
||||||
|
|
||||||
stage_domains = stage_domains_unfiltered - hass.config.components
|
_LOGGER.info("Setting up %s: %s", name, domain_group)
|
||||||
if not stage_domains:
|
to_be_loaded = domain_group.copy()
|
||||||
_LOGGER.info("Already set up stage %s: %s", name, stage_domains_unfiltered)
|
to_be_loaded.update(
|
||||||
continue
|
|
||||||
|
|
||||||
stage_dep_domains_unfiltered = {
|
|
||||||
dep
|
dep
|
||||||
for domain in stage_domains
|
for domain in domain_group
|
||||||
for dep in integrations_after_dependencies[domain]
|
if (integration := integration_cache.get(domain)) is not None
|
||||||
if dep not in stage_domains
|
for dep in integration.all_dependencies
|
||||||
}
|
|
||||||
stage_dep_domains = stage_dep_domains_unfiltered - hass.config.components
|
|
||||||
|
|
||||||
stage_all_domains = stage_domains | stage_dep_domains
|
|
||||||
|
|
||||||
_LOGGER.info(
|
|
||||||
"Setting up stage %s: %s; already set up: %s\n"
|
|
||||||
"Dependencies: %s; already set up: %s",
|
|
||||||
name,
|
|
||||||
stage_domains,
|
|
||||||
(stage_domains_unfiltered - stage_domains) or "{}",
|
|
||||||
stage_dep_domains or "{}",
|
|
||||||
(stage_dep_domains_unfiltered - stage_dep_domains) or "{}",
|
|
||||||
)
|
)
|
||||||
|
async_set_domains_to_be_loaded(hass, to_be_loaded)
|
||||||
|
stage_2_domains -= to_be_loaded
|
||||||
|
|
||||||
if timeout is None:
|
if timeout is None:
|
||||||
await _async_setup_multi_components(hass, stage_all_domains, config)
|
await _async_setup_multi_components(hass, domain_group, config)
|
||||||
continue
|
else:
|
||||||
|
try:
|
||||||
|
async with hass.timeout.async_timeout(timeout, cool_down=COOLDOWN_TIME):
|
||||||
|
await _async_setup_multi_components(hass, domain_group, config)
|
||||||
|
except TimeoutError:
|
||||||
|
_LOGGER.warning(
|
||||||
|
"Setup timed out for %s waiting on %s - moving forward",
|
||||||
|
name,
|
||||||
|
hass._active_tasks, # noqa: SLF001
|
||||||
|
)
|
||||||
|
|
||||||
|
# Add after dependencies when setting up stage 2 domains
|
||||||
|
async_set_domains_to_be_loaded(hass, stage_2_domains)
|
||||||
|
|
||||||
|
if stage_2_domains:
|
||||||
|
_LOGGER.info("Setting up stage 2: %s", stage_2_domains)
|
||||||
try:
|
try:
|
||||||
async with hass.timeout.async_timeout(
|
async with hass.timeout.async_timeout(
|
||||||
timeout,
|
STAGE_2_TIMEOUT, cool_down=COOLDOWN_TIME
|
||||||
cool_down=COOLDOWN_TIME,
|
|
||||||
cancel_message=f"Bootstrap stage {name} timeout",
|
|
||||||
):
|
):
|
||||||
await _async_setup_multi_components(hass, stage_all_domains, config)
|
await _async_setup_multi_components(hass, stage_2_domains, config)
|
||||||
except TimeoutError:
|
except TimeoutError:
|
||||||
_LOGGER.warning(
|
_LOGGER.warning(
|
||||||
"Setup timed out for stage %s waiting on %s - moving forward",
|
"Setup timed out for stage 2 waiting on %s - moving forward",
|
||||||
name,
|
|
||||||
hass._active_tasks, # noqa: SLF001
|
hass._active_tasks, # noqa: SLF001
|
||||||
)
|
)
|
||||||
|
|
||||||
# Wrap up startup
|
# Wrap up startup
|
||||||
_LOGGER.debug("Waiting for startup to wrap up")
|
_LOGGER.debug("Waiting for startup to wrap up")
|
||||||
try:
|
try:
|
||||||
async with hass.timeout.async_timeout(
|
async with hass.timeout.async_timeout(WRAP_UP_TIMEOUT, cool_down=COOLDOWN_TIME):
|
||||||
WRAP_UP_TIMEOUT,
|
|
||||||
cool_down=COOLDOWN_TIME,
|
|
||||||
cancel_message="Bootstrap startup wrap up timeout",
|
|
||||||
):
|
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
except TimeoutError:
|
except TimeoutError:
|
||||||
_LOGGER.warning(
|
_LOGGER.warning(
|
||||||
@@ -1036,6 +1053,8 @@ async def _async_setup_multi_components(
|
|||||||
config: dict[str, Any],
|
config: dict[str, Any],
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Set up multiple domains. Log on failure."""
|
"""Set up multiple domains. Log on failure."""
|
||||||
|
# Avoid creating tasks for domains that were setup in a previous stage
|
||||||
|
domains_not_yet_setup = domains - hass.config.components
|
||||||
# Create setup tasks for base platforms first since everything will have
|
# Create setup tasks for base platforms first since everything will have
|
||||||
# to wait to be imported, and the sooner we can get the base platforms
|
# to wait to be imported, and the sooner we can get the base platforms
|
||||||
# loaded the sooner we can start loading the rest of the integrations.
|
# loaded the sooner we can start loading the rest of the integrations.
|
||||||
@@ -1045,7 +1064,9 @@ async def _async_setup_multi_components(
|
|||||||
f"setup component {domain}",
|
f"setup component {domain}",
|
||||||
eager_start=True,
|
eager_start=True,
|
||||||
)
|
)
|
||||||
for domain in sorted(domains, key=SETUP_ORDER_SORT_KEY, reverse=True)
|
for domain in sorted(
|
||||||
|
domains_not_yet_setup, key=SETUP_ORDER_SORT_KEY, reverse=True
|
||||||
|
)
|
||||||
}
|
}
|
||||||
results = await asyncio.gather(*futures.values(), return_exceptions=True)
|
results = await asyncio.gather(*futures.values(), return_exceptions=True)
|
||||||
for idx, domain in enumerate(futures):
|
for idx, domain in enumerate(futures):
|
||||||
|
@@ -1,13 +1,5 @@
|
|||||||
{
|
{
|
||||||
"domain": "amazon",
|
"domain": "amazon",
|
||||||
"name": "Amazon",
|
"name": "Amazon",
|
||||||
"integrations": [
|
"integrations": ["alexa", "amazon_polly", "aws", "fire_tv", "route53"]
|
||||||
"alexa",
|
|
||||||
"alexa_devices",
|
|
||||||
"amazon_polly",
|
|
||||||
"aws",
|
|
||||||
"aws_s3",
|
|
||||||
"fire_tv",
|
|
||||||
"route53"
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
|
@@ -1,5 +0,0 @@
|
|||||||
{
|
|
||||||
"domain": "bosch",
|
|
||||||
"name": "Bosch",
|
|
||||||
"integrations": ["bosch_alarm", "bosch_shc", "home_connect"]
|
|
||||||
}
|
|
@@ -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",
|
"domain": "fritzbox",
|
||||||
"name": "FRITZ!",
|
"name": "FRITZ!Box",
|
||||||
"integrations": ["fritz", "fritzbox", "fritzbox_callmonitor"]
|
"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 +1,5 @@
|
|||||||
{
|
{
|
||||||
"domain": "motionblinds",
|
"domain": "motionblinds",
|
||||||
"name": "Motionblinds",
|
"name": "Motionblinds",
|
||||||
"integrations": ["motion_blinds", "motionblinds_ble"],
|
"integrations": ["motion_blinds", "motionblinds_ble"]
|
||||||
"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",
|
"domain": "sony",
|
||||||
"name": "Sony",
|
"name": "Sony",
|
||||||
"integrations": [
|
"integrations": ["braviatv", "ps4", "sony_projector", "songpal"]
|
||||||
"braviatv",
|
|
||||||
"ps4",
|
|
||||||
"sony_projector",
|
|
||||||
"songpal",
|
|
||||||
"playstation_network"
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
|
@@ -1,6 +1,5 @@
|
|||||||
{
|
{
|
||||||
"domain": "switchbot",
|
"domain": "switchbot",
|
||||||
"name": "SwitchBot",
|
"name": "SwitchBot",
|
||||||
"integrations": ["switchbot", "switchbot_cloud"],
|
"integrations": ["switchbot", "switchbot_cloud"]
|
||||||
"iot_standards": ["matter"]
|
|
||||||
}
|
}
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"domain": "third_reality",
|
"domain": "third_reality",
|
||||||
"name": "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",
|
"domain": "ubiquiti",
|
||||||
"name": "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 jaraco.abode.helpers.timeline import Groups as GROUPS
|
||||||
from requests.exceptions import ConnectTimeout, HTTPError
|
from requests.exceptions import ConnectTimeout, HTTPError
|
||||||
|
import voluptuous as vol
|
||||||
|
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import (
|
from homeassistant.const import (
|
||||||
ATTR_DATE,
|
ATTR_DATE,
|
||||||
ATTR_DEVICE_ID,
|
ATTR_DEVICE_ID,
|
||||||
|
ATTR_ENTITY_ID,
|
||||||
ATTR_TIME,
|
ATTR_TIME,
|
||||||
CONF_PASSWORD,
|
CONF_PASSWORD,
|
||||||
CONF_USERNAME,
|
CONF_USERNAME,
|
||||||
EVENT_HOMEASSISTANT_STOP,
|
EVENT_HOMEASSISTANT_STOP,
|
||||||
Platform,
|
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.exceptions import ConfigEntryAuthFailed, ConfigEntryNotReady
|
||||||
from homeassistant.helpers import config_validation as cv
|
from homeassistant.helpers import config_validation as cv
|
||||||
|
from homeassistant.helpers.dispatcher import dispatcher_send
|
||||||
from homeassistant.helpers.typing import ConfigType
|
from homeassistant.helpers.typing import ConfigType
|
||||||
|
|
||||||
from .const import CONF_POLLING, DOMAIN, LOGGER
|
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_NAME = "device_name"
|
||||||
ATTR_DEVICE_TYPE = "device_type"
|
ATTR_DEVICE_TYPE = "device_type"
|
||||||
@@ -39,12 +45,22 @@ ATTR_EVENT_CODE = "event_code"
|
|||||||
ATTR_EVENT_NAME = "event_name"
|
ATTR_EVENT_NAME = "event_name"
|
||||||
ATTR_EVENT_TYPE = "event_type"
|
ATTR_EVENT_TYPE = "event_type"
|
||||||
ATTR_EVENT_UTC = "event_utc"
|
ATTR_EVENT_UTC = "event_utc"
|
||||||
|
ATTR_SETTING = "setting"
|
||||||
ATTR_USER_NAME = "user_name"
|
ATTR_USER_NAME = "user_name"
|
||||||
ATTR_APP_TYPE = "app_type"
|
ATTR_APP_TYPE = "app_type"
|
||||||
ATTR_EVENT_BY = "event_by"
|
ATTR_EVENT_BY = "event_by"
|
||||||
|
ATTR_VALUE = "value"
|
||||||
|
|
||||||
CONFIG_SCHEMA = cv.removed(DOMAIN, raise_if_present=False)
|
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 = [
|
PLATFORMS = [
|
||||||
Platform.ALARM_CONTROL_PANEL,
|
Platform.ALARM_CONTROL_PANEL,
|
||||||
Platform.BINARY_SENSOR,
|
Platform.BINARY_SENSOR,
|
||||||
@@ -69,7 +85,7 @@ class AbodeSystem:
|
|||||||
|
|
||||||
async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
|
async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
|
||||||
"""Set up the Abode component."""
|
"""Set up the Abode component."""
|
||||||
async_setup_services(hass)
|
setup_hass_services(hass)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
@@ -122,6 +138,60 @@ async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||||||
return unload_ok
|
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:
|
async def setup_hass_events(hass: HomeAssistant) -> None:
|
||||||
"""Home Assistant start and stop callbacks."""
|
"""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.acaiascale import AcaiaScale
|
||||||
from aioacaia.exceptions import AcaiaDeviceNotFound, AcaiaError
|
from aioacaia.exceptions import AcaiaDeviceNotFound, AcaiaError
|
||||||
|
|
||||||
from homeassistant.components.bluetooth import async_get_scanner
|
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import CONF_ADDRESS
|
from homeassistant.const import CONF_ADDRESS
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
from homeassistant.helpers.debounce import Debouncer
|
|
||||||
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
||||||
|
|
||||||
from .const import CONF_IS_NEW_STYLE_SCALE
|
from .const import CONF_IS_NEW_STYLE_SCALE
|
||||||
|
|
||||||
SCAN_INTERVAL = timedelta(seconds=15)
|
SCAN_INTERVAL = timedelta(seconds=15)
|
||||||
UPDATE_DEBOUNCE_TIME = 0.2
|
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
@@ -40,20 +37,11 @@ class AcaiaCoordinator(DataUpdateCoordinator[None]):
|
|||||||
config_entry=entry,
|
config_entry=entry,
|
||||||
)
|
)
|
||||||
|
|
||||||
debouncer = Debouncer(
|
|
||||||
hass=hass,
|
|
||||||
logger=_LOGGER,
|
|
||||||
cooldown=UPDATE_DEBOUNCE_TIME,
|
|
||||||
immediate=True,
|
|
||||||
function=self.async_update_listeners,
|
|
||||||
)
|
|
||||||
|
|
||||||
self._scale = AcaiaScale(
|
self._scale = AcaiaScale(
|
||||||
address_or_ble_device=entry.data[CONF_ADDRESS],
|
address_or_ble_device=entry.data[CONF_ADDRESS],
|
||||||
name=entry.title,
|
name=entry.title,
|
||||||
is_new_style_scale=entry.data[CONF_IS_NEW_STYLE_SCALE],
|
is_new_style_scale=entry.data[CONF_IS_NEW_STYLE_SCALE],
|
||||||
notify_callback=debouncer.async_schedule_call,
|
notify_callback=self.async_update_listeners,
|
||||||
scanner=async_get_scanner(hass),
|
|
||||||
)
|
)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@@ -26,5 +26,5 @@
|
|||||||
"iot_class": "local_push",
|
"iot_class": "local_push",
|
||||||
"loggers": ["aioacaia"],
|
"loggers": ["aioacaia"],
|
||||||
"quality_scale": "platinum",
|
"quality_scale": "platinum",
|
||||||
"requirements": ["aioacaia==0.1.17"]
|
"requirements": ["aioacaia==0.1.14"]
|
||||||
}
|
}
|
||||||
|
@@ -2,23 +2,21 @@
|
|||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
import asyncio
|
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
from accuweather import AccuWeather
|
from accuweather import AccuWeather
|
||||||
|
|
||||||
from homeassistant.components.sensor import DOMAIN as SENSOR_PLATFORM
|
from homeassistant.components.sensor import DOMAIN as SENSOR_PLATFORM
|
||||||
from homeassistant.const import CONF_API_KEY, Platform
|
from homeassistant.const import CONF_API_KEY, CONF_NAME, Platform
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
from homeassistant.helpers import entity_registry as er
|
from homeassistant.helpers import entity_registry as er
|
||||||
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
||||||
|
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN, UPDATE_INTERVAL_DAILY_FORECAST, UPDATE_INTERVAL_OBSERVATION
|
||||||
from .coordinator import (
|
from .coordinator import (
|
||||||
AccuWeatherConfigEntry,
|
AccuWeatherConfigEntry,
|
||||||
AccuWeatherDailyForecastDataUpdateCoordinator,
|
AccuWeatherDailyForecastDataUpdateCoordinator,
|
||||||
AccuWeatherData,
|
AccuWeatherData,
|
||||||
AccuWeatherHourlyForecastDataUpdateCoordinator,
|
|
||||||
AccuWeatherObservationDataUpdateCoordinator,
|
AccuWeatherObservationDataUpdateCoordinator,
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -30,6 +28,7 @@ PLATFORMS = [Platform.SENSOR, Platform.WEATHER]
|
|||||||
async def async_setup_entry(hass: HomeAssistant, entry: AccuWeatherConfigEntry) -> bool:
|
async def async_setup_entry(hass: HomeAssistant, entry: AccuWeatherConfigEntry) -> bool:
|
||||||
"""Set up AccuWeather as config entry."""
|
"""Set up AccuWeather as config entry."""
|
||||||
api_key: str = entry.data[CONF_API_KEY]
|
api_key: str = entry.data[CONF_API_KEY]
|
||||||
|
name: str = entry.data[CONF_NAME]
|
||||||
|
|
||||||
location_key = entry.unique_id
|
location_key = entry.unique_id
|
||||||
|
|
||||||
@@ -42,28 +41,26 @@ async def async_setup_entry(hass: HomeAssistant, entry: AccuWeatherConfigEntry)
|
|||||||
hass,
|
hass,
|
||||||
entry,
|
entry,
|
||||||
accuweather,
|
accuweather,
|
||||||
|
name,
|
||||||
|
"observation",
|
||||||
|
UPDATE_INTERVAL_OBSERVATION,
|
||||||
)
|
)
|
||||||
|
|
||||||
coordinator_daily_forecast = AccuWeatherDailyForecastDataUpdateCoordinator(
|
coordinator_daily_forecast = AccuWeatherDailyForecastDataUpdateCoordinator(
|
||||||
hass,
|
hass,
|
||||||
entry,
|
entry,
|
||||||
accuweather,
|
accuweather,
|
||||||
)
|
name,
|
||||||
coordinator_hourly_forecast = AccuWeatherHourlyForecastDataUpdateCoordinator(
|
"daily forecast",
|
||||||
hass,
|
UPDATE_INTERVAL_DAILY_FORECAST,
|
||||||
entry,
|
|
||||||
accuweather,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
await asyncio.gather(
|
await coordinator_observation.async_config_entry_first_refresh()
|
||||||
coordinator_observation.async_config_entry_first_refresh(),
|
await coordinator_daily_forecast.async_config_entry_first_refresh()
|
||||||
coordinator_daily_forecast.async_config_entry_first_refresh(),
|
|
||||||
coordinator_hourly_forecast.async_config_entry_first_refresh(),
|
|
||||||
)
|
|
||||||
|
|
||||||
entry.runtime_data = AccuWeatherData(
|
entry.runtime_data = AccuWeatherData(
|
||||||
coordinator_observation=coordinator_observation,
|
coordinator_observation=coordinator_observation,
|
||||||
coordinator_daily_forecast=coordinator_daily_forecast,
|
coordinator_daily_forecast=coordinator_daily_forecast,
|
||||||
coordinator_hourly_forecast=coordinator_hourly_forecast,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
|
await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
|
||||||
|
@@ -3,7 +3,6 @@
|
|||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from asyncio import timeout
|
from asyncio import timeout
|
||||||
from collections.abc import Mapping
|
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
from accuweather import AccuWeather, ApiError, InvalidApiKeyError, RequestsExceededError
|
from accuweather import AccuWeather, ApiError, InvalidApiKeyError, RequestsExceededError
|
||||||
@@ -23,8 +22,6 @@ class AccuWeatherFlowHandler(ConfigFlow, domain=DOMAIN):
|
|||||||
"""Config flow for AccuWeather."""
|
"""Config flow for AccuWeather."""
|
||||||
|
|
||||||
VERSION = 1
|
VERSION = 1
|
||||||
_latitude: float | None = None
|
|
||||||
_longitude: float | None = None
|
|
||||||
|
|
||||||
async def async_step_user(
|
async def async_step_user(
|
||||||
self, user_input: dict[str, Any] | None = None
|
self, user_input: dict[str, Any] | None = None
|
||||||
@@ -53,7 +50,6 @@ class AccuWeatherFlowHandler(ConfigFlow, domain=DOMAIN):
|
|||||||
await self.async_set_unique_id(
|
await self.async_set_unique_id(
|
||||||
accuweather.location_key, raise_on_progress=False
|
accuweather.location_key, raise_on_progress=False
|
||||||
)
|
)
|
||||||
self._abort_if_unique_id_configured()
|
|
||||||
|
|
||||||
return self.async_create_entry(
|
return self.async_create_entry(
|
||||||
title=user_input[CONF_NAME], data=user_input
|
title=user_input[CONF_NAME], data=user_input
|
||||||
@@ -77,46 +73,3 @@ class AccuWeatherFlowHandler(ConfigFlow, domain=DOMAIN):
|
|||||||
),
|
),
|
||||||
errors=errors,
|
errors=errors,
|
||||||
)
|
)
|
||||||
|
|
||||||
async def async_step_reauth(
|
|
||||||
self, entry_data: Mapping[str, Any]
|
|
||||||
) -> ConfigFlowResult:
|
|
||||||
"""Handle configuration by re-auth."""
|
|
||||||
self._latitude = entry_data[CONF_LATITUDE]
|
|
||||||
self._longitude = entry_data[CONF_LONGITUDE]
|
|
||||||
|
|
||||||
return await self.async_step_reauth_confirm()
|
|
||||||
|
|
||||||
async def async_step_reauth_confirm(
|
|
||||||
self, user_input: dict[str, Any] | None = None
|
|
||||||
) -> ConfigFlowResult:
|
|
||||||
"""Dialog that informs the user that reauth is required."""
|
|
||||||
errors: dict[str, str] = {}
|
|
||||||
|
|
||||||
if user_input is not None:
|
|
||||||
websession = async_get_clientsession(self.hass)
|
|
||||||
try:
|
|
||||||
async with timeout(10):
|
|
||||||
accuweather = AccuWeather(
|
|
||||||
user_input[CONF_API_KEY],
|
|
||||||
websession,
|
|
||||||
latitude=self._latitude,
|
|
||||||
longitude=self._longitude,
|
|
||||||
)
|
|
||||||
await accuweather.async_get_location()
|
|
||||||
except (ApiError, ClientConnectorError, TimeoutError, ClientError):
|
|
||||||
errors["base"] = "cannot_connect"
|
|
||||||
except InvalidApiKeyError:
|
|
||||||
errors["base"] = "invalid_api_key"
|
|
||||||
except RequestsExceededError:
|
|
||||||
errors["base"] = "requests_exceeded"
|
|
||||||
else:
|
|
||||||
return self.async_update_reload_and_abort(
|
|
||||||
self._get_reauth_entry(), data_updates=user_input
|
|
||||||
)
|
|
||||||
|
|
||||||
return self.async_show_form(
|
|
||||||
step_id="reauth_confirm",
|
|
||||||
data_schema=vol.Schema({vol.Required(CONF_API_KEY): str}),
|
|
||||||
errors=errors,
|
|
||||||
)
|
|
||||||
|
@@ -24,7 +24,7 @@ from homeassistant.components.weather import (
|
|||||||
|
|
||||||
API_METRIC: Final = "Metric"
|
API_METRIC: Final = "Metric"
|
||||||
ATTRIBUTION: Final = "Data provided by AccuWeather"
|
ATTRIBUTION: Final = "Data provided by AccuWeather"
|
||||||
ATTR_CATEGORY_VALUE = "CategoryValue"
|
ATTR_CATEGORY: Final = "Category"
|
||||||
ATTR_DIRECTION: Final = "Direction"
|
ATTR_DIRECTION: Final = "Direction"
|
||||||
ATTR_ENGLISH: Final = "English"
|
ATTR_ENGLISH: Final = "English"
|
||||||
ATTR_LEVEL: Final = "level"
|
ATTR_LEVEL: Final = "level"
|
||||||
@@ -55,20 +55,5 @@ CONDITION_MAP = {
|
|||||||
for cond_ha, cond_codes in CONDITION_CLASSES.items()
|
for cond_ha, cond_codes in CONDITION_CLASSES.items()
|
||||||
for cond_code in cond_codes
|
for cond_code in cond_codes
|
||||||
}
|
}
|
||||||
AIR_QUALITY_CATEGORY_MAP = {
|
UPDATE_INTERVAL_OBSERVATION = timedelta(minutes=40)
|
||||||
1: "good",
|
|
||||||
2: "moderate",
|
|
||||||
3: "unhealthy",
|
|
||||||
4: "very_unhealthy",
|
|
||||||
5: "hazardous",
|
|
||||||
}
|
|
||||||
POLLEN_CATEGORY_MAP = {
|
|
||||||
1: "low",
|
|
||||||
2: "moderate",
|
|
||||||
3: "high",
|
|
||||||
4: "very_high",
|
|
||||||
5: "extreme",
|
|
||||||
}
|
|
||||||
UPDATE_INTERVAL_OBSERVATION = timedelta(minutes=10)
|
|
||||||
UPDATE_INTERVAL_DAILY_FORECAST = timedelta(hours=6)
|
UPDATE_INTERVAL_DAILY_FORECAST = timedelta(hours=6)
|
||||||
UPDATE_INTERVAL_HOURLY_FORECAST = timedelta(minutes=30)
|
|
||||||
|
@@ -3,7 +3,6 @@
|
|||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from asyncio import timeout
|
from asyncio import timeout
|
||||||
from collections.abc import Awaitable, Callable
|
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from datetime import timedelta
|
from datetime import timedelta
|
||||||
import logging
|
import logging
|
||||||
@@ -13,9 +12,7 @@ from accuweather import AccuWeather, ApiError, InvalidApiKeyError, RequestsExcee
|
|||||||
from aiohttp.client_exceptions import ClientConnectorError
|
from aiohttp.client_exceptions import ClientConnectorError
|
||||||
|
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import CONF_NAME
|
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
from homeassistant.exceptions import ConfigEntryAuthFailed
|
|
||||||
from homeassistant.helpers.device_registry import DeviceEntryType, DeviceInfo
|
from homeassistant.helpers.device_registry import DeviceEntryType, DeviceInfo
|
||||||
from homeassistant.helpers.update_coordinator import (
|
from homeassistant.helpers.update_coordinator import (
|
||||||
DataUpdateCoordinator,
|
DataUpdateCoordinator,
|
||||||
@@ -23,15 +20,9 @@ from homeassistant.helpers.update_coordinator import (
|
|||||||
UpdateFailed,
|
UpdateFailed,
|
||||||
)
|
)
|
||||||
|
|
||||||
from .const import (
|
from .const import DOMAIN, MANUFACTURER
|
||||||
DOMAIN,
|
|
||||||
MANUFACTURER,
|
|
||||||
UPDATE_INTERVAL_DAILY_FORECAST,
|
|
||||||
UPDATE_INTERVAL_HOURLY_FORECAST,
|
|
||||||
UPDATE_INTERVAL_OBSERVATION,
|
|
||||||
)
|
|
||||||
|
|
||||||
EXCEPTIONS = (ApiError, ClientConnectorError, RequestsExceededError)
|
EXCEPTIONS = (ApiError, ClientConnectorError, InvalidApiKeyError, RequestsExceededError)
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
@@ -42,7 +33,6 @@ class AccuWeatherData:
|
|||||||
|
|
||||||
coordinator_observation: AccuWeatherObservationDataUpdateCoordinator
|
coordinator_observation: AccuWeatherObservationDataUpdateCoordinator
|
||||||
coordinator_daily_forecast: AccuWeatherDailyForecastDataUpdateCoordinator
|
coordinator_daily_forecast: AccuWeatherDailyForecastDataUpdateCoordinator
|
||||||
coordinator_hourly_forecast: AccuWeatherHourlyForecastDataUpdateCoordinator
|
|
||||||
|
|
||||||
|
|
||||||
type AccuWeatherConfigEntry = ConfigEntry[AccuWeatherData]
|
type AccuWeatherConfigEntry = ConfigEntry[AccuWeatherData]
|
||||||
@@ -53,18 +43,18 @@ class AccuWeatherObservationDataUpdateCoordinator(
|
|||||||
):
|
):
|
||||||
"""Class to manage fetching AccuWeather data API."""
|
"""Class to manage fetching AccuWeather data API."""
|
||||||
|
|
||||||
config_entry: AccuWeatherConfigEntry
|
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
self,
|
self,
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
config_entry: AccuWeatherConfigEntry,
|
config_entry: AccuWeatherConfigEntry,
|
||||||
accuweather: AccuWeather,
|
accuweather: AccuWeather,
|
||||||
|
name: str,
|
||||||
|
coordinator_type: str,
|
||||||
|
update_interval: timedelta,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Initialize."""
|
"""Initialize."""
|
||||||
self.accuweather = accuweather
|
self.accuweather = accuweather
|
||||||
self.location_key = accuweather.location_key
|
self.location_key = accuweather.location_key
|
||||||
name = config_entry.data[CONF_NAME]
|
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
assert self.location_key is not None
|
assert self.location_key is not None
|
||||||
@@ -75,8 +65,8 @@ class AccuWeatherObservationDataUpdateCoordinator(
|
|||||||
hass,
|
hass,
|
||||||
_LOGGER,
|
_LOGGER,
|
||||||
config_entry=config_entry,
|
config_entry=config_entry,
|
||||||
name=f"{name} (observation)",
|
name=f"{name} ({coordinator_type})",
|
||||||
update_interval=UPDATE_INTERVAL_OBSERVATION,
|
update_interval=update_interval,
|
||||||
)
|
)
|
||||||
|
|
||||||
async def _async_update_data(self) -> dict[str, Any]:
|
async def _async_update_data(self) -> dict[str, Any]:
|
||||||
@@ -85,44 +75,30 @@ class AccuWeatherObservationDataUpdateCoordinator(
|
|||||||
async with timeout(10):
|
async with timeout(10):
|
||||||
result = await self.accuweather.async_get_current_conditions()
|
result = await self.accuweather.async_get_current_conditions()
|
||||||
except EXCEPTIONS as error:
|
except EXCEPTIONS as error:
|
||||||
raise UpdateFailed(
|
raise UpdateFailed(error) from error
|
||||||
translation_domain=DOMAIN,
|
|
||||||
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)
|
_LOGGER.debug("Requests remaining: %d", self.accuweather.requests_remaining)
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
|
||||||
class AccuWeatherForecastDataUpdateCoordinator(
|
class AccuWeatherDailyForecastDataUpdateCoordinator(
|
||||||
TimestampDataUpdateCoordinator[list[dict[str, Any]]]
|
TimestampDataUpdateCoordinator[list[dict[str, Any]]]
|
||||||
):
|
):
|
||||||
"""Base class for AccuWeather forecast."""
|
"""Class to manage fetching AccuWeather data API."""
|
||||||
|
|
||||||
config_entry: AccuWeatherConfigEntry
|
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
self,
|
self,
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
config_entry: AccuWeatherConfigEntry,
|
config_entry: AccuWeatherConfigEntry,
|
||||||
accuweather: AccuWeather,
|
accuweather: AccuWeather,
|
||||||
|
name: str,
|
||||||
coordinator_type: str,
|
coordinator_type: str,
|
||||||
update_interval: timedelta,
|
update_interval: timedelta,
|
||||||
fetch_method: Callable[..., Awaitable[list[dict[str, Any]]]],
|
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Initialize."""
|
"""Initialize."""
|
||||||
self.accuweather = accuweather
|
self.accuweather = accuweather
|
||||||
self.location_key = accuweather.location_key
|
self.location_key = accuweather.location_key
|
||||||
self._fetch_method = fetch_method
|
|
||||||
name = config_entry.data[CONF_NAME]
|
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
assert self.location_key is not None
|
assert self.location_key is not None
|
||||||
@@ -138,71 +114,18 @@ class AccuWeatherForecastDataUpdateCoordinator(
|
|||||||
)
|
)
|
||||||
|
|
||||||
async def _async_update_data(self) -> list[dict[str, Any]]:
|
async def _async_update_data(self) -> list[dict[str, Any]]:
|
||||||
"""Update forecast data via library."""
|
"""Update data via library."""
|
||||||
try:
|
try:
|
||||||
async with timeout(10):
|
async with timeout(10):
|
||||||
result = await self._fetch_method(language=self.hass.config.language)
|
result = await self.accuweather.async_get_daily_forecast()
|
||||||
except EXCEPTIONS as error:
|
except EXCEPTIONS as error:
|
||||||
raise UpdateFailed(
|
raise UpdateFailed(error) from error
|
||||||
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)
|
_LOGGER.debug("Requests remaining: %d", self.accuweather.requests_remaining)
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
|
||||||
|
|
||||||
class AccuWeatherDailyForecastDataUpdateCoordinator(
|
|
||||||
AccuWeatherForecastDataUpdateCoordinator
|
|
||||||
):
|
|
||||||
"""Coordinator for daily forecast."""
|
|
||||||
|
|
||||||
def __init__(
|
|
||||||
self,
|
|
||||||
hass: HomeAssistant,
|
|
||||||
config_entry: AccuWeatherConfigEntry,
|
|
||||||
accuweather: AccuWeather,
|
|
||||||
) -> None:
|
|
||||||
"""Initialize."""
|
|
||||||
super().__init__(
|
|
||||||
hass,
|
|
||||||
config_entry,
|
|
||||||
accuweather,
|
|
||||||
"daily forecast",
|
|
||||||
UPDATE_INTERVAL_DAILY_FORECAST,
|
|
||||||
fetch_method=accuweather.async_get_daily_forecast,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class AccuWeatherHourlyForecastDataUpdateCoordinator(
|
|
||||||
AccuWeatherForecastDataUpdateCoordinator
|
|
||||||
):
|
|
||||||
"""Coordinator for hourly forecast."""
|
|
||||||
|
|
||||||
def __init__(
|
|
||||||
self,
|
|
||||||
hass: HomeAssistant,
|
|
||||||
config_entry: AccuWeatherConfigEntry,
|
|
||||||
accuweather: AccuWeather,
|
|
||||||
) -> None:
|
|
||||||
"""Initialize."""
|
|
||||||
super().__init__(
|
|
||||||
hass,
|
|
||||||
config_entry,
|
|
||||||
accuweather,
|
|
||||||
"hourly forecast",
|
|
||||||
UPDATE_INTERVAL_HOURLY_FORECAST,
|
|
||||||
fetch_method=accuweather.async_get_hourly_forecast,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def _get_device_info(location_key: str, name: str) -> DeviceInfo:
|
def _get_device_info(location_key: str, name: str) -> DeviceInfo:
|
||||||
"""Get device info."""
|
"""Get device info."""
|
||||||
return DeviceInfo(
|
return DeviceInfo(
|
||||||
|
@@ -1,9 +1,6 @@
|
|||||||
{
|
{
|
||||||
"entity": {
|
"entity": {
|
||||||
"sensor": {
|
"sensor": {
|
||||||
"air_quality": {
|
|
||||||
"default": "mdi:air-filter"
|
|
||||||
},
|
|
||||||
"cloud_ceiling": {
|
"cloud_ceiling": {
|
||||||
"default": "mdi:weather-fog"
|
"default": "mdi:weather-fog"
|
||||||
},
|
},
|
||||||
@@ -37,6 +34,9 @@
|
|||||||
"thunderstorm_probability_night": {
|
"thunderstorm_probability_night": {
|
||||||
"default": "mdi:weather-lightning"
|
"default": "mdi:weather-lightning"
|
||||||
},
|
},
|
||||||
|
"translation_key": {
|
||||||
|
"default": "mdi:air-filter"
|
||||||
|
},
|
||||||
"tree_pollen": {
|
"tree_pollen": {
|
||||||
"default": "mdi:tree-outline"
|
"default": "mdi:tree-outline"
|
||||||
},
|
},
|
||||||
|
@@ -7,5 +7,6 @@
|
|||||||
"integration_type": "service",
|
"integration_type": "service",
|
||||||
"iot_class": "cloud_polling",
|
"iot_class": "cloud_polling",
|
||||||
"loggers": ["accuweather"],
|
"loggers": ["accuweather"],
|
||||||
"requirements": ["accuweather==4.2.2"]
|
"requirements": ["accuweather==4.1.0"],
|
||||||
|
"single_config_entry": true
|
||||||
}
|
}
|
||||||
|
@@ -29,9 +29,8 @@ from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
|
|||||||
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
||||||
|
|
||||||
from .const import (
|
from .const import (
|
||||||
AIR_QUALITY_CATEGORY_MAP,
|
|
||||||
API_METRIC,
|
API_METRIC,
|
||||||
ATTR_CATEGORY_VALUE,
|
ATTR_CATEGORY,
|
||||||
ATTR_DIRECTION,
|
ATTR_DIRECTION,
|
||||||
ATTR_ENGLISH,
|
ATTR_ENGLISH,
|
||||||
ATTR_LEVEL,
|
ATTR_LEVEL,
|
||||||
@@ -39,7 +38,6 @@ from .const import (
|
|||||||
ATTR_VALUE,
|
ATTR_VALUE,
|
||||||
ATTRIBUTION,
|
ATTRIBUTION,
|
||||||
MAX_FORECAST_DAYS,
|
MAX_FORECAST_DAYS,
|
||||||
POLLEN_CATEGORY_MAP,
|
|
||||||
)
|
)
|
||||||
from .coordinator import (
|
from .coordinator import (
|
||||||
AccuWeatherConfigEntry,
|
AccuWeatherConfigEntry,
|
||||||
@@ -61,9 +59,9 @@ class AccuWeatherSensorDescription(SensorEntityDescription):
|
|||||||
FORECAST_SENSOR_TYPES: tuple[AccuWeatherSensorDescription, ...] = (
|
FORECAST_SENSOR_TYPES: tuple[AccuWeatherSensorDescription, ...] = (
|
||||||
AccuWeatherSensorDescription(
|
AccuWeatherSensorDescription(
|
||||||
key="AirQuality",
|
key="AirQuality",
|
||||||
value_fn=lambda data: AIR_QUALITY_CATEGORY_MAP[data[ATTR_CATEGORY_VALUE]],
|
value_fn=lambda data: cast(str, data[ATTR_CATEGORY]),
|
||||||
device_class=SensorDeviceClass.ENUM,
|
device_class=SensorDeviceClass.ENUM,
|
||||||
options=list(AIR_QUALITY_CATEGORY_MAP.values()),
|
options=["good", "hazardous", "high", "low", "moderate", "unhealthy"],
|
||||||
translation_key="air_quality",
|
translation_key="air_quality",
|
||||||
),
|
),
|
||||||
AccuWeatherSensorDescription(
|
AccuWeatherSensorDescription(
|
||||||
@@ -85,9 +83,7 @@ FORECAST_SENSOR_TYPES: tuple[AccuWeatherSensorDescription, ...] = (
|
|||||||
entity_registry_enabled_default=False,
|
entity_registry_enabled_default=False,
|
||||||
native_unit_of_measurement=CONCENTRATION_PARTS_PER_CUBIC_METER,
|
native_unit_of_measurement=CONCENTRATION_PARTS_PER_CUBIC_METER,
|
||||||
value_fn=lambda data: cast(int, data[ATTR_VALUE]),
|
value_fn=lambda data: cast(int, data[ATTR_VALUE]),
|
||||||
attr_fn=lambda data: {
|
attr_fn=lambda data: {ATTR_LEVEL: data[ATTR_CATEGORY]},
|
||||||
ATTR_LEVEL: POLLEN_CATEGORY_MAP[data[ATTR_CATEGORY_VALUE]]
|
|
||||||
},
|
|
||||||
translation_key="grass_pollen",
|
translation_key="grass_pollen",
|
||||||
),
|
),
|
||||||
AccuWeatherSensorDescription(
|
AccuWeatherSensorDescription(
|
||||||
@@ -111,9 +107,7 @@ FORECAST_SENSOR_TYPES: tuple[AccuWeatherSensorDescription, ...] = (
|
|||||||
entity_registry_enabled_default=False,
|
entity_registry_enabled_default=False,
|
||||||
native_unit_of_measurement=CONCENTRATION_PARTS_PER_CUBIC_METER,
|
native_unit_of_measurement=CONCENTRATION_PARTS_PER_CUBIC_METER,
|
||||||
value_fn=lambda data: cast(int, data[ATTR_VALUE]),
|
value_fn=lambda data: cast(int, data[ATTR_VALUE]),
|
||||||
attr_fn=lambda data: {
|
attr_fn=lambda data: {ATTR_LEVEL: data[ATTR_CATEGORY]},
|
||||||
ATTR_LEVEL: POLLEN_CATEGORY_MAP[data[ATTR_CATEGORY_VALUE]]
|
|
||||||
},
|
|
||||||
translation_key="mold_pollen",
|
translation_key="mold_pollen",
|
||||||
),
|
),
|
||||||
AccuWeatherSensorDescription(
|
AccuWeatherSensorDescription(
|
||||||
@@ -121,9 +115,7 @@ FORECAST_SENSOR_TYPES: tuple[AccuWeatherSensorDescription, ...] = (
|
|||||||
native_unit_of_measurement=CONCENTRATION_PARTS_PER_CUBIC_METER,
|
native_unit_of_measurement=CONCENTRATION_PARTS_PER_CUBIC_METER,
|
||||||
entity_registry_enabled_default=False,
|
entity_registry_enabled_default=False,
|
||||||
value_fn=lambda data: cast(int, data[ATTR_VALUE]),
|
value_fn=lambda data: cast(int, data[ATTR_VALUE]),
|
||||||
attr_fn=lambda data: {
|
attr_fn=lambda data: {ATTR_LEVEL: data[ATTR_CATEGORY]},
|
||||||
ATTR_LEVEL: POLLEN_CATEGORY_MAP[data[ATTR_CATEGORY_VALUE]]
|
|
||||||
},
|
|
||||||
translation_key="ragweed_pollen",
|
translation_key="ragweed_pollen",
|
||||||
),
|
),
|
||||||
AccuWeatherSensorDescription(
|
AccuWeatherSensorDescription(
|
||||||
@@ -189,18 +181,14 @@ FORECAST_SENSOR_TYPES: tuple[AccuWeatherSensorDescription, ...] = (
|
|||||||
native_unit_of_measurement=CONCENTRATION_PARTS_PER_CUBIC_METER,
|
native_unit_of_measurement=CONCENTRATION_PARTS_PER_CUBIC_METER,
|
||||||
entity_registry_enabled_default=False,
|
entity_registry_enabled_default=False,
|
||||||
value_fn=lambda data: cast(int, data[ATTR_VALUE]),
|
value_fn=lambda data: cast(int, data[ATTR_VALUE]),
|
||||||
attr_fn=lambda data: {
|
attr_fn=lambda data: {ATTR_LEVEL: data[ATTR_CATEGORY]},
|
||||||
ATTR_LEVEL: POLLEN_CATEGORY_MAP[data[ATTR_CATEGORY_VALUE]]
|
|
||||||
},
|
|
||||||
translation_key="tree_pollen",
|
translation_key="tree_pollen",
|
||||||
),
|
),
|
||||||
AccuWeatherSensorDescription(
|
AccuWeatherSensorDescription(
|
||||||
key="UVIndex",
|
key="UVIndex",
|
||||||
native_unit_of_measurement=UV_INDEX,
|
native_unit_of_measurement=UV_INDEX,
|
||||||
value_fn=lambda data: cast(int, data[ATTR_VALUE]),
|
value_fn=lambda data: cast(int, data[ATTR_VALUE]),
|
||||||
attr_fn=lambda data: {
|
attr_fn=lambda data: {ATTR_LEVEL: data[ATTR_CATEGORY]},
|
||||||
ATTR_LEVEL: POLLEN_CATEGORY_MAP[data[ATTR_CATEGORY_VALUE]]
|
|
||||||
},
|
|
||||||
translation_key="uv_index_forecast",
|
translation_key="uv_index_forecast",
|
||||||
),
|
),
|
||||||
AccuWeatherSensorDescription(
|
AccuWeatherSensorDescription(
|
||||||
|
@@ -7,17 +7,6 @@
|
|||||||
"api_key": "[%key:common::config_flow::data::api_key%]",
|
"api_key": "[%key:common::config_flow::data::api_key%]",
|
||||||
"latitude": "[%key:common::config_flow::data::latitude%]",
|
"latitude": "[%key:common::config_flow::data::latitude%]",
|
||||||
"longitude": "[%key:common::config_flow::data::longitude%]"
|
"longitude": "[%key:common::config_flow::data::longitude%]"
|
||||||
},
|
|
||||||
"data_description": {
|
|
||||||
"api_key": "API key generated in the AccuWeather APIs portal."
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"reauth_confirm": {
|
|
||||||
"data": {
|
|
||||||
"api_key": "[%key:common::config_flow::data::api_key%]"
|
|
||||||
},
|
|
||||||
"data_description": {
|
|
||||||
"api_key": "[%key:component::accuweather::config::step::user::data_description::api_key%]"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -28,10 +17,6 @@
|
|||||||
"cannot_connect": "[%key:common::config_flow::error::cannot_connect%]",
|
"cannot_connect": "[%key:common::config_flow::error::cannot_connect%]",
|
||||||
"invalid_api_key": "[%key:common::config_flow::error::invalid_api_key%]",
|
"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."
|
"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": {
|
"entity": {
|
||||||
@@ -41,20 +26,10 @@
|
|||||||
"state": {
|
"state": {
|
||||||
"good": "Good",
|
"good": "Good",
|
||||||
"hazardous": "Hazardous",
|
"hazardous": "Hazardous",
|
||||||
|
"high": "High",
|
||||||
|
"low": "Low",
|
||||||
"moderate": "Moderate",
|
"moderate": "Moderate",
|
||||||
"unhealthy": "Unhealthy",
|
"unhealthy": "Unhealthy"
|
||||||
"very_unhealthy": "Very unhealthy"
|
|
||||||
},
|
|
||||||
"state_attributes": {
|
|
||||||
"options": {
|
|
||||||
"state": {
|
|
||||||
"good": "[%key:component::accuweather::entity::sensor::air_quality::state::good%]",
|
|
||||||
"hazardous": "[%key:component::accuweather::entity::sensor::air_quality::state::hazardous%]",
|
|
||||||
"moderate": "[%key:component::accuweather::entity::sensor::air_quality::state::moderate%]",
|
|
||||||
"unhealthy": "[%key:component::accuweather::entity::sensor::air_quality::state::unhealthy%]",
|
|
||||||
"very_unhealthy": "[%key:component::accuweather::entity::sensor::air_quality::state::very_unhealthy%]"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"apparent_temperature": {
|
"apparent_temperature": {
|
||||||
@@ -87,11 +62,12 @@
|
|||||||
"level": {
|
"level": {
|
||||||
"name": "Level",
|
"name": "Level",
|
||||||
"state": {
|
"state": {
|
||||||
"extreme": "Extreme",
|
"good": "[%key:component::accuweather::entity::sensor::air_quality::state::good%]",
|
||||||
"high": "[%key:common::state::high%]",
|
"hazardous": "[%key:component::accuweather::entity::sensor::air_quality::state::hazardous%]",
|
||||||
"low": "[%key:common::state::low%]",
|
"high": "[%key:component::accuweather::entity::sensor::air_quality::state::high%]",
|
||||||
"moderate": "Moderate",
|
"low": "[%key:component::accuweather::entity::sensor::air_quality::state::low%]",
|
||||||
"very_high": "[%key:common::state::very_high%]"
|
"moderate": "[%key:component::accuweather::entity::sensor::air_quality::state::moderate%]",
|
||||||
|
"unhealthy": "[%key:component::accuweather::entity::sensor::air_quality::state::unhealthy%]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -105,11 +81,12 @@
|
|||||||
"level": {
|
"level": {
|
||||||
"name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]",
|
"name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]",
|
||||||
"state": {
|
"state": {
|
||||||
"extreme": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::extreme%]",
|
"good": "[%key:component::accuweather::entity::sensor::air_quality::state::good%]",
|
||||||
"high": "[%key:common::state::high%]",
|
"hazardous": "[%key:component::accuweather::entity::sensor::air_quality::state::hazardous%]",
|
||||||
"low": "[%key:common::state::low%]",
|
"high": "[%key:component::accuweather::entity::sensor::air_quality::state::high%]",
|
||||||
"moderate": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::moderate%]",
|
"low": "[%key:component::accuweather::entity::sensor::air_quality::state::low%]",
|
||||||
"very_high": "[%key:common::state::very_high%]"
|
"moderate": "[%key:component::accuweather::entity::sensor::air_quality::state::moderate%]",
|
||||||
|
"unhealthy": "[%key:component::accuweather::entity::sensor::air_quality::state::unhealthy%]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -123,15 +100,6 @@
|
|||||||
"steady": "Steady",
|
"steady": "Steady",
|
||||||
"rising": "Rising",
|
"rising": "Rising",
|
||||||
"falling": "Falling"
|
"falling": "Falling"
|
||||||
},
|
|
||||||
"state_attributes": {
|
|
||||||
"options": {
|
|
||||||
"state": {
|
|
||||||
"falling": "[%key:component::accuweather::entity::sensor::pressure_tendency::state::falling%]",
|
|
||||||
"rising": "[%key:component::accuweather::entity::sensor::pressure_tendency::state::rising%]",
|
|
||||||
"steady": "[%key:component::accuweather::entity::sensor::pressure_tendency::state::steady%]"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"ragweed_pollen": {
|
"ragweed_pollen": {
|
||||||
@@ -140,11 +108,12 @@
|
|||||||
"level": {
|
"level": {
|
||||||
"name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]",
|
"name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]",
|
||||||
"state": {
|
"state": {
|
||||||
"extreme": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::extreme%]",
|
"good": "[%key:component::accuweather::entity::sensor::air_quality::state::good%]",
|
||||||
"high": "[%key:common::state::high%]",
|
"hazardous": "[%key:component::accuweather::entity::sensor::air_quality::state::hazardous%]",
|
||||||
"low": "[%key:common::state::low%]",
|
"high": "[%key:component::accuweather::entity::sensor::air_quality::state::high%]",
|
||||||
"moderate": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::moderate%]",
|
"low": "[%key:component::accuweather::entity::sensor::air_quality::state::low%]",
|
||||||
"very_high": "[%key:common::state::very_high%]"
|
"moderate": "[%key:component::accuweather::entity::sensor::air_quality::state::moderate%]",
|
||||||
|
"unhealthy": "[%key:component::accuweather::entity::sensor::air_quality::state::unhealthy%]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -185,11 +154,12 @@
|
|||||||
"level": {
|
"level": {
|
||||||
"name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]",
|
"name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]",
|
||||||
"state": {
|
"state": {
|
||||||
"extreme": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::extreme%]",
|
"good": "[%key:component::accuweather::entity::sensor::air_quality::state::good%]",
|
||||||
"high": "[%key:common::state::high%]",
|
"hazardous": "[%key:component::accuweather::entity::sensor::air_quality::state::hazardous%]",
|
||||||
"low": "[%key:common::state::low%]",
|
"high": "[%key:component::accuweather::entity::sensor::air_quality::state::high%]",
|
||||||
"moderate": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::moderate%]",
|
"low": "[%key:component::accuweather::entity::sensor::air_quality::state::low%]",
|
||||||
"very_high": "[%key:common::state::very_high%]"
|
"moderate": "[%key:component::accuweather::entity::sensor::air_quality::state::moderate%]",
|
||||||
|
"unhealthy": "[%key:component::accuweather::entity::sensor::air_quality::state::unhealthy%]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -200,11 +170,12 @@
|
|||||||
"level": {
|
"level": {
|
||||||
"name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]",
|
"name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]",
|
||||||
"state": {
|
"state": {
|
||||||
"extreme": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::extreme%]",
|
"good": "[%key:component::accuweather::entity::sensor::air_quality::state::good%]",
|
||||||
"high": "[%key:common::state::high%]",
|
"hazardous": "[%key:component::accuweather::entity::sensor::air_quality::state::hazardous%]",
|
||||||
"low": "[%key:common::state::low%]",
|
"high": "[%key:component::accuweather::entity::sensor::air_quality::state::high%]",
|
||||||
"moderate": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::moderate%]",
|
"low": "[%key:component::accuweather::entity::sensor::air_quality::state::low%]",
|
||||||
"very_high": "[%key:common::state::very_high%]"
|
"moderate": "[%key:component::accuweather::entity::sensor::air_quality::state::moderate%]",
|
||||||
|
"unhealthy": "[%key:component::accuweather::entity::sensor::air_quality::state::unhealthy%]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -215,11 +186,12 @@
|
|||||||
"level": {
|
"level": {
|
||||||
"name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]",
|
"name": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::name%]",
|
||||||
"state": {
|
"state": {
|
||||||
"extreme": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::extreme%]",
|
"good": "[%key:component::accuweather::entity::sensor::air_quality::state::good%]",
|
||||||
"high": "[%key:common::state::high%]",
|
"hazardous": "[%key:component::accuweather::entity::sensor::air_quality::state::hazardous%]",
|
||||||
"low": "[%key:common::state::low%]",
|
"high": "[%key:component::accuweather::entity::sensor::air_quality::state::high%]",
|
||||||
"moderate": "[%key:component::accuweather::entity::sensor::grass_pollen::state_attributes::level::state::moderate%]",
|
"low": "[%key:component::accuweather::entity::sensor::air_quality::state::low%]",
|
||||||
"very_high": "[%key:common::state::very_high%]"
|
"moderate": "[%key:component::accuweather::entity::sensor::air_quality::state::moderate%]",
|
||||||
|
"unhealthy": "[%key:component::accuweather::entity::sensor::air_quality::state::unhealthy%]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -250,17 +222,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}"
|
|
||||||
},
|
|
||||||
"forecast_update_error": {
|
|
||||||
"message": "An error occurred while retrieving weather forecast data from the AccuWeather API: {error}"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"system_health": {
|
"system_health": {
|
||||||
"info": {
|
"info": {
|
||||||
"can_reach_server": "Reach AccuWeather server",
|
"can_reach_server": "Reach AccuWeather server",
|
||||||
|
@@ -45,7 +45,6 @@ from .coordinator import (
|
|||||||
AccuWeatherConfigEntry,
|
AccuWeatherConfigEntry,
|
||||||
AccuWeatherDailyForecastDataUpdateCoordinator,
|
AccuWeatherDailyForecastDataUpdateCoordinator,
|
||||||
AccuWeatherData,
|
AccuWeatherData,
|
||||||
AccuWeatherHourlyForecastDataUpdateCoordinator,
|
|
||||||
AccuWeatherObservationDataUpdateCoordinator,
|
AccuWeatherObservationDataUpdateCoordinator,
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -65,7 +64,6 @@ class AccuWeatherEntity(
|
|||||||
CoordinatorWeatherEntity[
|
CoordinatorWeatherEntity[
|
||||||
AccuWeatherObservationDataUpdateCoordinator,
|
AccuWeatherObservationDataUpdateCoordinator,
|
||||||
AccuWeatherDailyForecastDataUpdateCoordinator,
|
AccuWeatherDailyForecastDataUpdateCoordinator,
|
||||||
AccuWeatherHourlyForecastDataUpdateCoordinator,
|
|
||||||
]
|
]
|
||||||
):
|
):
|
||||||
"""Define an AccuWeather entity."""
|
"""Define an AccuWeather entity."""
|
||||||
@@ -78,7 +76,6 @@ class AccuWeatherEntity(
|
|||||||
super().__init__(
|
super().__init__(
|
||||||
observation_coordinator=accuweather_data.coordinator_observation,
|
observation_coordinator=accuweather_data.coordinator_observation,
|
||||||
daily_coordinator=accuweather_data.coordinator_daily_forecast,
|
daily_coordinator=accuweather_data.coordinator_daily_forecast,
|
||||||
hourly_coordinator=accuweather_data.coordinator_hourly_forecast,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
self._attr_native_precipitation_unit = UnitOfPrecipitationDepth.MILLIMETERS
|
self._attr_native_precipitation_unit = UnitOfPrecipitationDepth.MILLIMETERS
|
||||||
@@ -89,13 +86,10 @@ class AccuWeatherEntity(
|
|||||||
self._attr_unique_id = accuweather_data.coordinator_observation.location_key
|
self._attr_unique_id = accuweather_data.coordinator_observation.location_key
|
||||||
self._attr_attribution = ATTRIBUTION
|
self._attr_attribution = ATTRIBUTION
|
||||||
self._attr_device_info = accuweather_data.coordinator_observation.device_info
|
self._attr_device_info = accuweather_data.coordinator_observation.device_info
|
||||||
self._attr_supported_features = (
|
self._attr_supported_features = WeatherEntityFeature.FORECAST_DAILY
|
||||||
WeatherEntityFeature.FORECAST_DAILY | WeatherEntityFeature.FORECAST_HOURLY
|
|
||||||
)
|
|
||||||
|
|
||||||
self.observation_coordinator = accuweather_data.coordinator_observation
|
self.observation_coordinator = accuweather_data.coordinator_observation
|
||||||
self.daily_coordinator = accuweather_data.coordinator_daily_forecast
|
self.daily_coordinator = accuweather_data.coordinator_daily_forecast
|
||||||
self.hourly_coordinator = accuweather_data.coordinator_hourly_forecast
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def condition(self) -> str | None:
|
def condition(self) -> str | None:
|
||||||
@@ -213,32 +207,3 @@ class AccuWeatherEntity(
|
|||||||
}
|
}
|
||||||
for item in self.daily_coordinator.data
|
for item in self.daily_coordinator.data
|
||||||
]
|
]
|
||||||
|
|
||||||
@callback
|
|
||||||
def _async_forecast_hourly(self) -> list[Forecast] | None:
|
|
||||||
"""Return the hourly forecast in native units."""
|
|
||||||
return [
|
|
||||||
{
|
|
||||||
ATTR_FORECAST_TIME: utc_from_timestamp(
|
|
||||||
item["EpochDateTime"]
|
|
||||||
).isoformat(),
|
|
||||||
ATTR_FORECAST_CLOUD_COVERAGE: item["CloudCover"],
|
|
||||||
ATTR_FORECAST_HUMIDITY: item["RelativeHumidity"],
|
|
||||||
ATTR_FORECAST_NATIVE_TEMP: item["Temperature"][ATTR_VALUE],
|
|
||||||
ATTR_FORECAST_NATIVE_APPARENT_TEMP: item["RealFeelTemperature"][
|
|
||||||
ATTR_VALUE
|
|
||||||
],
|
|
||||||
ATTR_FORECAST_NATIVE_PRECIPITATION: item["TotalLiquid"][ATTR_VALUE],
|
|
||||||
ATTR_FORECAST_PRECIPITATION_PROBABILITY: item[
|
|
||||||
"PrecipitationProbability"
|
|
||||||
],
|
|
||||||
ATTR_FORECAST_NATIVE_WIND_SPEED: item["Wind"][ATTR_SPEED][ATTR_VALUE],
|
|
||||||
ATTR_FORECAST_NATIVE_WIND_GUST_SPEED: item["WindGust"][ATTR_SPEED][
|
|
||||||
ATTR_VALUE
|
|
||||||
],
|
|
||||||
ATTR_FORECAST_UV_INDEX: item["UVIndex"],
|
|
||||||
ATTR_FORECAST_WIND_BEARING: item["Wind"][ATTR_DIRECTION]["Degrees"],
|
|
||||||
ATTR_FORECAST_CONDITION: CONDITION_MAP.get(item["WeatherIcon"]),
|
|
||||||
}
|
|
||||||
for item in self.hourly_coordinator.data
|
|
||||||
]
|
|
||||||
|
@@ -40,10 +40,9 @@ class AcmedaFlowHandler(ConfigFlow, domain=DOMAIN):
|
|||||||
entry.unique_id for entry in self._async_current_entries()
|
entry.unique_id for entry in self._async_current_entries()
|
||||||
}
|
}
|
||||||
|
|
||||||
hubs: list[aiopulse.Hub] = []
|
|
||||||
with suppress(TimeoutError):
|
with suppress(TimeoutError):
|
||||||
async with timeout(5):
|
async with timeout(5):
|
||||||
hubs = [
|
hubs: list[aiopulse.Hub] = [
|
||||||
hub
|
hub
|
||||||
async for hub in aiopulse.Hub.discover()
|
async for hub in aiopulse.Hub.discover()
|
||||||
if hub.id not in already_configured
|
if hub.id not in already_configured
|
||||||
|
@@ -1,57 +0,0 @@
|
|||||||
"""The Actron Air integration."""
|
|
||||||
|
|
||||||
from actron_neo_api import (
|
|
||||||
ActronAirNeoACSystem,
|
|
||||||
ActronNeoAPI,
|
|
||||||
ActronNeoAPIError,
|
|
||||||
ActronNeoAuthError,
|
|
||||||
)
|
|
||||||
|
|
||||||
from homeassistant.const import CONF_API_TOKEN, Platform
|
|
||||||
from homeassistant.core import HomeAssistant
|
|
||||||
|
|
||||||
from .const import _LOGGER
|
|
||||||
from .coordinator import (
|
|
||||||
ActronAirConfigEntry,
|
|
||||||
ActronAirRuntimeData,
|
|
||||||
ActronAirSystemCoordinator,
|
|
||||||
)
|
|
||||||
|
|
||||||
PLATFORM = [Platform.CLIMATE]
|
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass: HomeAssistant, entry: ActronAirConfigEntry) -> bool:
|
|
||||||
"""Set up Actron Air integration from a config entry."""
|
|
||||||
|
|
||||||
api = ActronNeoAPI(refresh_token=entry.data[CONF_API_TOKEN])
|
|
||||||
systems: list[ActronAirNeoACSystem] = []
|
|
||||||
|
|
||||||
try:
|
|
||||||
systems = await api.get_ac_systems()
|
|
||||||
await api.update_status()
|
|
||||||
except ActronNeoAuthError:
|
|
||||||
_LOGGER.error("Authentication error while setting up Actron Air integration")
|
|
||||||
raise
|
|
||||||
except ActronNeoAPIError as err:
|
|
||||||
_LOGGER.error("API error while setting up Actron Air integration: %s", err)
|
|
||||||
raise
|
|
||||||
|
|
||||||
system_coordinators: dict[str, ActronAirSystemCoordinator] = {}
|
|
||||||
for system in systems:
|
|
||||||
coordinator = ActronAirSystemCoordinator(hass, entry, api, system)
|
|
||||||
_LOGGER.debug("Setting up coordinator for system: %s", system["serial"])
|
|
||||||
await coordinator.async_config_entry_first_refresh()
|
|
||||||
system_coordinators[system["serial"]] = coordinator
|
|
||||||
|
|
||||||
entry.runtime_data = ActronAirRuntimeData(
|
|
||||||
api=api,
|
|
||||||
system_coordinators=system_coordinators,
|
|
||||||
)
|
|
||||||
|
|
||||||
await hass.config_entries.async_forward_entry_setups(entry, PLATFORM)
|
|
||||||
return True
|
|
||||||
|
|
||||||
|
|
||||||
async def async_unload_entry(hass: HomeAssistant, entry: ActronAirConfigEntry) -> bool:
|
|
||||||
"""Unload a config entry."""
|
|
||||||
return await hass.config_entries.async_unload_platforms(entry, PLATFORM)
|
|
@@ -1,259 +0,0 @@
|
|||||||
"""Climate platform for Actron Air integration."""
|
|
||||||
|
|
||||||
from typing import Any
|
|
||||||
|
|
||||||
from actron_neo_api import ActronAirNeoStatus, ActronAirNeoZone
|
|
||||||
|
|
||||||
from homeassistant.components.climate import (
|
|
||||||
FAN_AUTO,
|
|
||||||
FAN_HIGH,
|
|
||||||
FAN_LOW,
|
|
||||||
FAN_MEDIUM,
|
|
||||||
ClimateEntity,
|
|
||||||
ClimateEntityFeature,
|
|
||||||
HVACMode,
|
|
||||||
)
|
|
||||||
from homeassistant.const import ATTR_TEMPERATURE, UnitOfTemperature
|
|
||||||
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 .const import DOMAIN
|
|
||||||
from .coordinator import ActronAirConfigEntry, ActronAirSystemCoordinator
|
|
||||||
|
|
||||||
PARALLEL_UPDATES = 0
|
|
||||||
|
|
||||||
FAN_MODE_MAPPING_ACTRONAIR_TO_HA = {
|
|
||||||
"AUTO": FAN_AUTO,
|
|
||||||
"LOW": FAN_LOW,
|
|
||||||
"MED": FAN_MEDIUM,
|
|
||||||
"HIGH": FAN_HIGH,
|
|
||||||
}
|
|
||||||
FAN_MODE_MAPPING_HA_TO_ACTRONAIR = {
|
|
||||||
v: k for k, v in FAN_MODE_MAPPING_ACTRONAIR_TO_HA.items()
|
|
||||||
}
|
|
||||||
HVAC_MODE_MAPPING_ACTRONAIR_TO_HA = {
|
|
||||||
"COOL": HVACMode.COOL,
|
|
||||||
"HEAT": HVACMode.HEAT,
|
|
||||||
"FAN": HVACMode.FAN_ONLY,
|
|
||||||
"AUTO": HVACMode.AUTO,
|
|
||||||
"OFF": HVACMode.OFF,
|
|
||||||
}
|
|
||||||
HVAC_MODE_MAPPING_HA_TO_ACTRONAIR = {
|
|
||||||
v: k for k, v in HVAC_MODE_MAPPING_ACTRONAIR_TO_HA.items()
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(
|
|
||||||
hass: HomeAssistant,
|
|
||||||
entry: ActronAirConfigEntry,
|
|
||||||
async_add_entities: AddConfigEntryEntitiesCallback,
|
|
||||||
) -> None:
|
|
||||||
"""Set up Actron Air climate entities."""
|
|
||||||
system_coordinators = entry.runtime_data.system_coordinators
|
|
||||||
entities: list[ClimateEntity] = []
|
|
||||||
|
|
||||||
for coordinator in system_coordinators.values():
|
|
||||||
status = coordinator.data
|
|
||||||
name = status.ac_system.system_name
|
|
||||||
entities.append(ActronSystemClimate(coordinator, name))
|
|
||||||
|
|
||||||
entities.extend(
|
|
||||||
ActronZoneClimate(coordinator, zone)
|
|
||||||
for zone in status.remote_zone_info
|
|
||||||
if zone.exists
|
|
||||||
)
|
|
||||||
|
|
||||||
async_add_entities(entities)
|
|
||||||
|
|
||||||
|
|
||||||
class BaseClimateEntity(CoordinatorEntity[ActronAirSystemCoordinator], ClimateEntity):
|
|
||||||
"""Base class for Actron Air climate entities."""
|
|
||||||
|
|
||||||
_attr_has_entity_name = True
|
|
||||||
_attr_temperature_unit = UnitOfTemperature.CELSIUS
|
|
||||||
_attr_supported_features = (
|
|
||||||
ClimateEntityFeature.TARGET_TEMPERATURE
|
|
||||||
| ClimateEntityFeature.FAN_MODE
|
|
||||||
| ClimateEntityFeature.TURN_ON
|
|
||||||
| ClimateEntityFeature.TURN_OFF
|
|
||||||
)
|
|
||||||
_attr_name = None
|
|
||||||
_attr_fan_modes = list(FAN_MODE_MAPPING_ACTRONAIR_TO_HA.values())
|
|
||||||
_attr_hvac_modes = list(HVAC_MODE_MAPPING_ACTRONAIR_TO_HA.values())
|
|
||||||
|
|
||||||
def __init__(
|
|
||||||
self,
|
|
||||||
coordinator: ActronAirSystemCoordinator,
|
|
||||||
name: str,
|
|
||||||
) -> None:
|
|
||||||
"""Initialize an Actron Air unit."""
|
|
||||||
super().__init__(coordinator)
|
|
||||||
self._serial_number = coordinator.serial_number
|
|
||||||
|
|
||||||
|
|
||||||
class ActronSystemClimate(BaseClimateEntity):
|
|
||||||
"""Representation of the Actron Air system."""
|
|
||||||
|
|
||||||
_attr_supported_features = (
|
|
||||||
ClimateEntityFeature.TARGET_TEMPERATURE
|
|
||||||
| ClimateEntityFeature.FAN_MODE
|
|
||||||
| ClimateEntityFeature.TURN_ON
|
|
||||||
| ClimateEntityFeature.TURN_OFF
|
|
||||||
)
|
|
||||||
|
|
||||||
def __init__(
|
|
||||||
self,
|
|
||||||
coordinator: ActronAirSystemCoordinator,
|
|
||||||
name: str,
|
|
||||||
) -> None:
|
|
||||||
"""Initialize an Actron Air unit."""
|
|
||||||
super().__init__(coordinator, name)
|
|
||||||
serial_number = coordinator.serial_number
|
|
||||||
self._attr_unique_id = serial_number
|
|
||||||
self._attr_device_info = DeviceInfo(
|
|
||||||
identifiers={(DOMAIN, serial_number)},
|
|
||||||
name=self._status.ac_system.system_name,
|
|
||||||
manufacturer="Actron Air",
|
|
||||||
model_id=self._status.ac_system.master_wc_model,
|
|
||||||
sw_version=self._status.ac_system.master_wc_firmware_version,
|
|
||||||
serial_number=serial_number,
|
|
||||||
)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def min_temp(self) -> float:
|
|
||||||
"""Return the minimum temperature that can be set."""
|
|
||||||
return self._status.min_temp
|
|
||||||
|
|
||||||
@property
|
|
||||||
def max_temp(self) -> float:
|
|
||||||
"""Return the maximum temperature that can be set."""
|
|
||||||
return self._status.max_temp
|
|
||||||
|
|
||||||
@property
|
|
||||||
def _status(self) -> ActronAirNeoStatus:
|
|
||||||
"""Get the current status from the coordinator."""
|
|
||||||
return self.coordinator.data
|
|
||||||
|
|
||||||
@property
|
|
||||||
def hvac_mode(self) -> HVACMode | None:
|
|
||||||
"""Return the current HVAC mode."""
|
|
||||||
if not self._status.user_aircon_settings.is_on:
|
|
||||||
return HVACMode.OFF
|
|
||||||
|
|
||||||
mode = self._status.user_aircon_settings.mode
|
|
||||||
return HVAC_MODE_MAPPING_ACTRONAIR_TO_HA.get(mode)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def fan_mode(self) -> str | None:
|
|
||||||
"""Return the current fan mode."""
|
|
||||||
fan_mode = self._status.user_aircon_settings.fan_mode
|
|
||||||
return FAN_MODE_MAPPING_ACTRONAIR_TO_HA.get(fan_mode)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def current_humidity(self) -> float:
|
|
||||||
"""Return the current humidity."""
|
|
||||||
return self._status.master_info.live_humidity_pc
|
|
||||||
|
|
||||||
@property
|
|
||||||
def current_temperature(self) -> float:
|
|
||||||
"""Return the current temperature."""
|
|
||||||
return self._status.master_info.live_temp_c
|
|
||||||
|
|
||||||
@property
|
|
||||||
def target_temperature(self) -> float:
|
|
||||||
"""Return the target temperature."""
|
|
||||||
return self._status.user_aircon_settings.temperature_setpoint_cool_c
|
|
||||||
|
|
||||||
async def async_set_fan_mode(self, fan_mode: str) -> None:
|
|
||||||
"""Set a new fan mode."""
|
|
||||||
api_fan_mode = FAN_MODE_MAPPING_HA_TO_ACTRONAIR.get(fan_mode.lower())
|
|
||||||
await self._status.user_aircon_settings.set_fan_mode(api_fan_mode)
|
|
||||||
|
|
||||||
async def async_set_hvac_mode(self, hvac_mode: HVACMode) -> None:
|
|
||||||
"""Set the HVAC mode."""
|
|
||||||
ac_mode = HVAC_MODE_MAPPING_HA_TO_ACTRONAIR.get(hvac_mode)
|
|
||||||
await self._status.ac_system.set_system_mode(ac_mode)
|
|
||||||
|
|
||||||
async def async_set_temperature(self, **kwargs: Any) -> None:
|
|
||||||
"""Set the temperature."""
|
|
||||||
temp = kwargs.get(ATTR_TEMPERATURE)
|
|
||||||
await self._status.user_aircon_settings.set_temperature(temperature=temp)
|
|
||||||
|
|
||||||
|
|
||||||
class ActronZoneClimate(BaseClimateEntity):
|
|
||||||
"""Representation of a zone within the Actron Air system."""
|
|
||||||
|
|
||||||
_attr_supported_features = (
|
|
||||||
ClimateEntityFeature.TARGET_TEMPERATURE
|
|
||||||
| ClimateEntityFeature.TURN_ON
|
|
||||||
| ClimateEntityFeature.TURN_OFF
|
|
||||||
)
|
|
||||||
|
|
||||||
def __init__(
|
|
||||||
self,
|
|
||||||
coordinator: ActronAirSystemCoordinator,
|
|
||||||
zone: ActronAirNeoZone,
|
|
||||||
) -> None:
|
|
||||||
"""Initialize an Actron Air unit."""
|
|
||||||
super().__init__(coordinator, zone.title)
|
|
||||||
serial_number = coordinator.serial_number
|
|
||||||
self._zone_id: int = zone.zone_id
|
|
||||||
self._attr_unique_id: str = f"{serial_number}_zone_{zone.zone_id}"
|
|
||||||
self._attr_device_info: DeviceInfo = DeviceInfo(
|
|
||||||
identifiers={(DOMAIN, self._attr_unique_id)},
|
|
||||||
name=zone.title,
|
|
||||||
manufacturer="Actron Air",
|
|
||||||
model="Zone",
|
|
||||||
suggested_area=zone.title,
|
|
||||||
via_device=(DOMAIN, serial_number),
|
|
||||||
)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def min_temp(self) -> float:
|
|
||||||
"""Return the minimum temperature that can be set."""
|
|
||||||
return self._zone.min_temp
|
|
||||||
|
|
||||||
@property
|
|
||||||
def max_temp(self) -> float:
|
|
||||||
"""Return the maximum temperature that can be set."""
|
|
||||||
return self._zone.max_temp
|
|
||||||
|
|
||||||
@property
|
|
||||||
def _zone(self) -> ActronAirNeoZone:
|
|
||||||
"""Get the current zone data from the coordinator."""
|
|
||||||
status = self.coordinator.data
|
|
||||||
return status.zones[self._zone_id]
|
|
||||||
|
|
||||||
@property
|
|
||||||
def hvac_mode(self) -> HVACMode | None:
|
|
||||||
"""Return the current HVAC mode."""
|
|
||||||
if self._zone.is_active:
|
|
||||||
mode = self._zone.hvac_mode
|
|
||||||
return HVAC_MODE_MAPPING_ACTRONAIR_TO_HA.get(mode)
|
|
||||||
return HVACMode.OFF
|
|
||||||
|
|
||||||
@property
|
|
||||||
def current_humidity(self) -> float | None:
|
|
||||||
"""Return the current humidity."""
|
|
||||||
return self._zone.humidity
|
|
||||||
|
|
||||||
@property
|
|
||||||
def current_temperature(self) -> float | None:
|
|
||||||
"""Return the current temperature."""
|
|
||||||
return self._zone.live_temp_c
|
|
||||||
|
|
||||||
@property
|
|
||||||
def target_temperature(self) -> float | None:
|
|
||||||
"""Return the target temperature."""
|
|
||||||
return self._zone.temperature_setpoint_cool_c
|
|
||||||
|
|
||||||
async def async_set_hvac_mode(self, hvac_mode: HVACMode) -> None:
|
|
||||||
"""Set the HVAC mode."""
|
|
||||||
is_enabled = hvac_mode != HVACMode.OFF
|
|
||||||
await self._zone.enable(is_enabled)
|
|
||||||
|
|
||||||
async def async_set_temperature(self, **kwargs: Any) -> None:
|
|
||||||
"""Set the temperature."""
|
|
||||||
await self._zone.set_temperature(temperature=kwargs["temperature"])
|
|
@@ -1,132 +0,0 @@
|
|||||||
"""Setup config flow for Actron Air integration."""
|
|
||||||
|
|
||||||
import asyncio
|
|
||||||
from typing import Any
|
|
||||||
|
|
||||||
from actron_neo_api import ActronNeoAPI, ActronNeoAuthError
|
|
||||||
|
|
||||||
from homeassistant.config_entries import ConfigFlow, ConfigFlowResult
|
|
||||||
from homeassistant.const import CONF_API_TOKEN
|
|
||||||
from homeassistant.exceptions import HomeAssistantError
|
|
||||||
|
|
||||||
from .const import _LOGGER, DOMAIN
|
|
||||||
|
|
||||||
|
|
||||||
class ActronAirConfigFlow(ConfigFlow, domain=DOMAIN):
|
|
||||||
"""Handle a config flow for Actron Air."""
|
|
||||||
|
|
||||||
def __init__(self) -> None:
|
|
||||||
"""Initialize the config flow."""
|
|
||||||
self._api: ActronNeoAPI | None = None
|
|
||||||
self._device_code: str | None = None
|
|
||||||
self._user_code: str = ""
|
|
||||||
self._verification_uri: str = ""
|
|
||||||
self._expires_minutes: str = "30"
|
|
||||||
self.login_task: asyncio.Task | None = None
|
|
||||||
|
|
||||||
async def async_step_user(
|
|
||||||
self, user_input: dict[str, Any] | None = None
|
|
||||||
) -> ConfigFlowResult:
|
|
||||||
"""Handle the initial step."""
|
|
||||||
if self._api is None:
|
|
||||||
_LOGGER.debug("Initiating device authorization")
|
|
||||||
self._api = ActronNeoAPI()
|
|
||||||
try:
|
|
||||||
device_code_response = await self._api.request_device_code()
|
|
||||||
except ActronNeoAuthError as err:
|
|
||||||
_LOGGER.error("OAuth2 flow failed: %s", err)
|
|
||||||
return self.async_abort(reason="oauth2_error")
|
|
||||||
|
|
||||||
self._device_code = device_code_response["device_code"]
|
|
||||||
self._user_code = device_code_response["user_code"]
|
|
||||||
self._verification_uri = device_code_response["verification_uri_complete"]
|
|
||||||
self._expires_minutes = str(device_code_response["expires_in"] // 60)
|
|
||||||
|
|
||||||
async def _wait_for_authorization() -> None:
|
|
||||||
"""Wait for the user to authorize the device."""
|
|
||||||
assert self._api is not None
|
|
||||||
assert self._device_code is not None
|
|
||||||
_LOGGER.debug("Waiting for device authorization")
|
|
||||||
try:
|
|
||||||
await self._api.poll_for_token(self._device_code)
|
|
||||||
_LOGGER.debug("Authorization successful")
|
|
||||||
except ActronNeoAuthError as ex:
|
|
||||||
_LOGGER.exception("Error while waiting for device authorization")
|
|
||||||
raise CannotConnect from ex
|
|
||||||
|
|
||||||
_LOGGER.debug("Checking login task")
|
|
||||||
if self.login_task is None:
|
|
||||||
_LOGGER.debug("Creating task for device authorization")
|
|
||||||
self.login_task = self.hass.async_create_task(_wait_for_authorization())
|
|
||||||
|
|
||||||
if self.login_task.done():
|
|
||||||
_LOGGER.debug("Login task is done, checking results")
|
|
||||||
if exception := self.login_task.exception():
|
|
||||||
if isinstance(exception, CannotConnect):
|
|
||||||
return self.async_show_progress_done(
|
|
||||||
next_step_id="connection_error"
|
|
||||||
)
|
|
||||||
return self.async_show_progress_done(next_step_id="timeout")
|
|
||||||
return self.async_show_progress_done(next_step_id="finish_login")
|
|
||||||
|
|
||||||
return self.async_show_progress(
|
|
||||||
step_id="user",
|
|
||||||
progress_action="wait_for_authorization",
|
|
||||||
description_placeholders={
|
|
||||||
"user_code": self._user_code,
|
|
||||||
"verification_uri": self._verification_uri,
|
|
||||||
"expires_minutes": self._expires_minutes,
|
|
||||||
},
|
|
||||||
progress_task=self.login_task,
|
|
||||||
)
|
|
||||||
|
|
||||||
async def async_step_finish_login(
|
|
||||||
self, user_input: dict[str, Any] | None = None
|
|
||||||
) -> ConfigFlowResult:
|
|
||||||
"""Handle the finalization of login."""
|
|
||||||
_LOGGER.debug("Finalizing authorization")
|
|
||||||
assert self._api is not None
|
|
||||||
|
|
||||||
try:
|
|
||||||
user_data = await self._api.get_user_info()
|
|
||||||
except ActronNeoAuthError as err:
|
|
||||||
_LOGGER.error("Error getting user info: %s", err)
|
|
||||||
return self.async_abort(reason="oauth2_error")
|
|
||||||
|
|
||||||
unique_id = str(user_data["id"])
|
|
||||||
await self.async_set_unique_id(unique_id)
|
|
||||||
self._abort_if_unique_id_configured()
|
|
||||||
|
|
||||||
return self.async_create_entry(
|
|
||||||
title=user_data["email"],
|
|
||||||
data={CONF_API_TOKEN: self._api.refresh_token_value},
|
|
||||||
)
|
|
||||||
|
|
||||||
async def async_step_timeout(
|
|
||||||
self,
|
|
||||||
user_input: dict[str, Any] | None = None,
|
|
||||||
) -> ConfigFlowResult:
|
|
||||||
"""Handle issues that need transition await from progress step."""
|
|
||||||
if user_input is None:
|
|
||||||
return self.async_show_form(
|
|
||||||
step_id="timeout",
|
|
||||||
)
|
|
||||||
del self.login_task
|
|
||||||
return await self.async_step_user()
|
|
||||||
|
|
||||||
async def async_step_connection_error(
|
|
||||||
self, user_input: dict[str, Any] | None = None
|
|
||||||
) -> ConfigFlowResult:
|
|
||||||
"""Handle connection error from progress step."""
|
|
||||||
if user_input is None:
|
|
||||||
return self.async_show_form(step_id="connection_error")
|
|
||||||
|
|
||||||
# Reset state and try again
|
|
||||||
self._api = None
|
|
||||||
self._device_code = None
|
|
||||||
self.login_task = None
|
|
||||||
return await self.async_step_user()
|
|
||||||
|
|
||||||
|
|
||||||
class CannotConnect(HomeAssistantError):
|
|
||||||
"""Error to indicate we cannot connect."""
|
|
@@ -1,6 +0,0 @@
|
|||||||
"""Constants used by Actron Air integration."""
|
|
||||||
|
|
||||||
import logging
|
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__package__)
|
|
||||||
DOMAIN = "actron_air"
|
|
@@ -1,69 +0,0 @@
|
|||||||
"""Coordinator for Actron Air integration."""
|
|
||||||
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
from dataclasses import dataclass
|
|
||||||
from datetime import timedelta
|
|
||||||
|
|
||||||
from actron_neo_api import ActronAirNeoACSystem, ActronAirNeoStatus, ActronNeoAPI
|
|
||||||
|
|
||||||
from homeassistant.config_entries import ConfigEntry
|
|
||||||
from homeassistant.core import HomeAssistant
|
|
||||||
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
|
||||||
from homeassistant.util import dt as dt_util
|
|
||||||
|
|
||||||
from .const import _LOGGER
|
|
||||||
|
|
||||||
STALE_DEVICE_TIMEOUT = timedelta(hours=24)
|
|
||||||
ERROR_NO_SYSTEMS_FOUND = "no_systems_found"
|
|
||||||
ERROR_UNKNOWN = "unknown_error"
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
|
||||||
class ActronAirRuntimeData:
|
|
||||||
"""Runtime data for the Actron Air integration."""
|
|
||||||
|
|
||||||
api: ActronNeoAPI
|
|
||||||
system_coordinators: dict[str, ActronAirSystemCoordinator]
|
|
||||||
|
|
||||||
|
|
||||||
type ActronAirConfigEntry = ConfigEntry[ActronAirRuntimeData]
|
|
||||||
|
|
||||||
AUTH_ERROR_THRESHOLD = 3
|
|
||||||
SCAN_INTERVAL = timedelta(seconds=30)
|
|
||||||
|
|
||||||
|
|
||||||
class ActronAirSystemCoordinator(DataUpdateCoordinator[ActronAirNeoACSystem]):
|
|
||||||
"""System coordinator for Actron Air integration."""
|
|
||||||
|
|
||||||
def __init__(
|
|
||||||
self,
|
|
||||||
hass: HomeAssistant,
|
|
||||||
entry: ActronAirConfigEntry,
|
|
||||||
api: ActronNeoAPI,
|
|
||||||
system: ActronAirNeoACSystem,
|
|
||||||
) -> None:
|
|
||||||
"""Initialize the coordinator."""
|
|
||||||
super().__init__(
|
|
||||||
hass,
|
|
||||||
_LOGGER,
|
|
||||||
name="Actron Air Status",
|
|
||||||
update_interval=SCAN_INTERVAL,
|
|
||||||
config_entry=entry,
|
|
||||||
)
|
|
||||||
self.system = system
|
|
||||||
self.serial_number = system["serial"]
|
|
||||||
self.api = api
|
|
||||||
self.status = self.api.state_manager.get_status(self.serial_number)
|
|
||||||
self.last_seen = dt_util.utcnow()
|
|
||||||
|
|
||||||
async def _async_update_data(self) -> ActronAirNeoStatus:
|
|
||||||
"""Fetch updates and merge incremental changes into the full state."""
|
|
||||||
await self.api.update_status()
|
|
||||||
self.status = self.api.state_manager.get_status(self.serial_number)
|
|
||||||
self.last_seen = dt_util.utcnow()
|
|
||||||
return self.status
|
|
||||||
|
|
||||||
def is_device_stale(self) -> bool:
|
|
||||||
"""Check if a device is stale (not seen for a while)."""
|
|
||||||
return (dt_util.utcnow() - self.last_seen) > STALE_DEVICE_TIMEOUT
|
|
@@ -1,16 +0,0 @@
|
|||||||
{
|
|
||||||
"domain": "actron_air",
|
|
||||||
"name": "Actron Air",
|
|
||||||
"codeowners": ["@kclif9", "@JagadishDhanamjayam"],
|
|
||||||
"config_flow": true,
|
|
||||||
"dhcp": [
|
|
||||||
{
|
|
||||||
"hostname": "neo-*",
|
|
||||||
"macaddress": "FC0FE7*"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"documentation": "https://www.home-assistant.io/integrations/actron_air",
|
|
||||||
"iot_class": "cloud_polling",
|
|
||||||
"quality_scale": "bronze",
|
|
||||||
"requirements": ["actron-neo-api==0.1.84"]
|
|
||||||
}
|
|
@@ -1,78 +0,0 @@
|
|||||||
rules:
|
|
||||||
# Bronze
|
|
||||||
action-setup:
|
|
||||||
status: exempt
|
|
||||||
comment: This integration does not have custom service actions.
|
|
||||||
appropriate-polling: done
|
|
||||||
brands: done
|
|
||||||
common-modules: done
|
|
||||||
config-flow-test-coverage: done
|
|
||||||
config-flow: done
|
|
||||||
dependency-transparency: done
|
|
||||||
docs-actions:
|
|
||||||
status: exempt
|
|
||||||
comment: This integration does not have custom service actions.
|
|
||||||
docs-high-level-description: done
|
|
||||||
docs-installation-instructions: done
|
|
||||||
docs-removal-instructions: done
|
|
||||||
entity-event-setup:
|
|
||||||
status: exempt
|
|
||||||
comment: This integration does not subscribe to external events.
|
|
||||||
entity-unique-id: done
|
|
||||||
has-entity-name: done
|
|
||||||
runtime-data: done
|
|
||||||
test-before-configure: done
|
|
||||||
test-before-setup: done
|
|
||||||
unique-config-entry: done
|
|
||||||
|
|
||||||
# Silver
|
|
||||||
action-exceptions: todo
|
|
||||||
config-entry-unloading: done
|
|
||||||
docs-configuration-parameters:
|
|
||||||
status: exempt
|
|
||||||
comment: No options flow
|
|
||||||
docs-installation-parameters: done
|
|
||||||
entity-unavailable: done
|
|
||||||
integration-owner: done
|
|
||||||
log-when-unavailable: done
|
|
||||||
parallel-updates: done
|
|
||||||
reauthentication-flow: todo
|
|
||||||
test-coverage: todo
|
|
||||||
|
|
||||||
# Gold
|
|
||||||
devices: done
|
|
||||||
diagnostics: todo
|
|
||||||
discovery-update-info:
|
|
||||||
status: exempt
|
|
||||||
comment: This integration uses DHCP discovery, however is cloud polling. Therefore there is no information to update.
|
|
||||||
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
|
|
||||||
dynamic-devices: todo
|
|
||||||
entity-category:
|
|
||||||
status: exempt
|
|
||||||
comment: This integration does not use entity categories.
|
|
||||||
entity-device-class:
|
|
||||||
status: exempt
|
|
||||||
comment: This integration does not use entity device classes.
|
|
||||||
entity-disabled-by-default:
|
|
||||||
status: exempt
|
|
||||||
comment: Not required for this integration at this stage.
|
|
||||||
entity-translations: todo
|
|
||||||
exception-translations: todo
|
|
||||||
icon-translations: todo
|
|
||||||
reconfiguration-flow: todo
|
|
||||||
repair-issues:
|
|
||||||
status: exempt
|
|
||||||
comment: This integration does not have any known issues that require repair.
|
|
||||||
stale-devices: todo
|
|
||||||
|
|
||||||
# Platinum
|
|
||||||
async-dependency: done
|
|
||||||
inject-websession: todo
|
|
||||||
strict-typing: todo
|
|
@@ -1,29 +0,0 @@
|
|||||||
{
|
|
||||||
"config": {
|
|
||||||
"step": {
|
|
||||||
"user": {
|
|
||||||
"title": "Actron Air OAuth2 Authorization"
|
|
||||||
},
|
|
||||||
"timeout": {
|
|
||||||
"title": "Authorization timeout",
|
|
||||||
"description": "The authorization process timed out. Please try again.",
|
|
||||||
"data": {}
|
|
||||||
},
|
|
||||||
"connection_error": {
|
|
||||||
"title": "Connection error",
|
|
||||||
"description": "Failed to connect to Actron Air. Please check your internet connection and try again.",
|
|
||||||
"data": {}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"progress": {
|
|
||||||
"wait_for_authorization": "To authenticate, open the following URL and login at Actron Air:\n{verification_uri}\nIf the code is not automatically copied, paste the following code to authorize the integration:\n\n```{user_code}```\n\n\nThe login attempt will time out after {expires_minutes} minutes."
|
|
||||||
},
|
|
||||||
"error": {
|
|
||||||
"oauth2_error": "Failed to start OAuth2 flow. Please try again later."
|
|
||||||
},
|
|
||||||
"abort": {
|
|
||||||
"oauth2_error": "Failed to start OAuth2 flow",
|
|
||||||
"already_configured": "[%key:common::config_flow::abort::already_configured_account%]"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -2,38 +2,25 @@
|
|||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import Platform
|
from homeassistant.const import Platform
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
|
|
||||||
from .const import CONNECTION_TYPE, LOCAL
|
PLATFORMS = [Platform.CLIMATE]
|
||||||
from .coordinator import AdaxCloudCoordinator, AdaxConfigEntry, AdaxLocalCoordinator
|
|
||||||
|
|
||||||
PLATFORMS = [Platform.CLIMATE, Platform.SENSOR]
|
|
||||||
|
|
||||||
|
|
||||||
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."""
|
"""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)
|
await hass.config_entries.async_forward_entry_setups(entry, PLATFORMS)
|
||||||
return True
|
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."""
|
"""Unload a config entry."""
|
||||||
return await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
return await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
||||||
|
|
||||||
|
|
||||||
async def async_migrate_entry(
|
async def async_migrate_entry(hass: HomeAssistant, config_entry: ConfigEntry) -> bool:
|
||||||
hass: HomeAssistant, config_entry: AdaxConfigEntry
|
|
||||||
) -> bool:
|
|
||||||
"""Migrate old entry."""
|
"""Migrate old entry."""
|
||||||
# convert title and unique_id to string
|
# convert title and unique_id to string
|
||||||
if config_entry.version == 1:
|
if config_entry.version == 1:
|
||||||
|
@@ -12,42 +12,57 @@ from homeassistant.components.climate import (
|
|||||||
ClimateEntityFeature,
|
ClimateEntityFeature,
|
||||||
HVACMode,
|
HVACMode,
|
||||||
)
|
)
|
||||||
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import (
|
from homeassistant.const import (
|
||||||
ATTR_TEMPERATURE,
|
ATTR_TEMPERATURE,
|
||||||
|
CONF_IP_ADDRESS,
|
||||||
|
CONF_PASSWORD,
|
||||||
|
CONF_TOKEN,
|
||||||
CONF_UNIQUE_ID,
|
CONF_UNIQUE_ID,
|
||||||
PRECISION_WHOLE,
|
PRECISION_WHOLE,
|
||||||
UnitOfTemperature,
|
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.device_registry import DeviceInfo
|
||||||
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
|
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
|
||||||
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
|
||||||
|
|
||||||
from . import AdaxConfigEntry
|
from .const import ACCOUNT_ID, CONNECTION_TYPE, DOMAIN, LOCAL
|
||||||
from .const import CONNECTION_TYPE, DOMAIN, LOCAL
|
|
||||||
from .coordinator import AdaxCloudCoordinator, AdaxLocalCoordinator
|
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(
|
async def async_setup_entry(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
entry: AdaxConfigEntry,
|
entry: ConfigEntry,
|
||||||
async_add_entities: AddConfigEntryEntitiesCallback,
|
async_add_entities: AddConfigEntryEntitiesCallback,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Set up the Adax thermostat with config flow."""
|
"""Set up the Adax thermostat with config flow."""
|
||||||
if entry.data.get(CONNECTION_TYPE) == LOCAL:
|
if entry.data.get(CONNECTION_TYPE) == LOCAL:
|
||||||
local_coordinator = cast(AdaxLocalCoordinator, entry.runtime_data)
|
adax_data_handler = AdaxLocal(
|
||||||
async_add_entities(
|
entry.data[CONF_IP_ADDRESS],
|
||||||
[LocalAdaxDevice(local_coordinator, entry.data[CONF_UNIQUE_ID])],
|
entry.data[CONF_TOKEN],
|
||||||
|
websession=async_get_clientsession(hass, verify_ssl=False),
|
||||||
)
|
)
|
||||||
else:
|
|
||||||
cloud_coordinator = cast(AdaxCloudCoordinator, entry.runtime_data)
|
|
||||||
async_add_entities(
|
async_add_entities(
|
||||||
AdaxDevice(cloud_coordinator, device_id)
|
[LocalAdaxDevice(adax_data_handler, entry.data[CONF_UNIQUE_ID])], True
|
||||||
for device_id in cloud_coordinator.data
|
|
||||||
)
|
)
|
||||||
|
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."""
|
"""Representation of a heater."""
|
||||||
|
|
||||||
_attr_hvac_modes = [HVACMode.HEAT, HVACMode.OFF]
|
_attr_hvac_modes = [HVACMode.HEAT, HVACMode.OFF]
|
||||||
@@ -61,37 +76,20 @@ class AdaxDevice(CoordinatorEntity[AdaxCloudCoordinator], ClimateEntity):
|
|||||||
_attr_target_temperature_step = PRECISION_WHOLE
|
_attr_target_temperature_step = PRECISION_WHOLE
|
||||||
_attr_temperature_unit = UnitOfTemperature.CELSIUS
|
_attr_temperature_unit = UnitOfTemperature.CELSIUS
|
||||||
|
|
||||||
def __init__(
|
def __init__(self, heater_data: dict[str, Any], adax_data_handler: Adax) -> None:
|
||||||
self,
|
|
||||||
coordinator: AdaxCloudCoordinator,
|
|
||||||
device_id: str,
|
|
||||||
) -> None:
|
|
||||||
"""Initialize the heater."""
|
"""Initialize the heater."""
|
||||||
super().__init__(coordinator)
|
self._device_id = heater_data["id"]
|
||||||
self._adax_data_handler: Adax = coordinator.adax_data_handler
|
self._adax_data_handler = adax_data_handler
|
||||||
self._device_id = device_id
|
|
||||||
|
|
||||||
self._attr_name = self.room["name"]
|
self._attr_unique_id = f"{heater_data['homeId']}_{heater_data['id']}"
|
||||||
self._attr_unique_id = f"{self.room['homeId']}_{self._device_id}"
|
|
||||||
self._attr_device_info = DeviceInfo(
|
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
|
# Instead of setting the device name to the entity name, adax
|
||||||
# should be updated to set has_entity_name = True, and set the entity
|
# should be updated to set has_entity_name = True, and set the entity
|
||||||
# name to None
|
# name to None
|
||||||
name=cast(str | None, self.name),
|
name=cast(str | None, self.name),
|
||||||
manufacturer="Adax",
|
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:
|
async def async_set_hvac_mode(self, hvac_mode: HVACMode) -> None:
|
||||||
"""Set hvac mode."""
|
"""Set hvac mode."""
|
||||||
@@ -106,9 +104,7 @@ class AdaxDevice(CoordinatorEntity[AdaxCloudCoordinator], ClimateEntity):
|
|||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
return
|
return
|
||||||
|
await self._adax_data_handler.update()
|
||||||
# Request data refresh from source to verify that update was successful
|
|
||||||
await self.coordinator.async_request_refresh()
|
|
||||||
|
|
||||||
async def async_set_temperature(self, **kwargs: Any) -> None:
|
async def async_set_temperature(self, **kwargs: Any) -> None:
|
||||||
"""Set new target temperature."""
|
"""Set new target temperature."""
|
||||||
@@ -118,31 +114,28 @@ class AdaxDevice(CoordinatorEntity[AdaxCloudCoordinator], ClimateEntity):
|
|||||||
self._device_id, temperature, True
|
self._device_id, temperature, True
|
||||||
)
|
)
|
||||||
|
|
||||||
@callback
|
async def async_update(self) -> None:
|
||||||
def _handle_coordinator_update(self) -> None:
|
"""Get the latest data."""
|
||||||
"""Handle updated data from the coordinator."""
|
for room in await self._adax_data_handler.get_rooms():
|
||||||
if room := self.room:
|
if room["id"] != self._device_id:
|
||||||
self._apply_data(room)
|
continue
|
||||||
super()._handle_coordinator_update()
|
self._attr_name = room["name"]
|
||||||
|
self._attr_current_temperature = room.get("temperature")
|
||||||
def _apply_data(self, room: dict[str, Any]) -> None:
|
self._attr_target_temperature = room.get("targetTemperature")
|
||||||
"""Update the appropriate attributues based on received data."""
|
if room["heatingEnabled"]:
|
||||||
self._attr_current_temperature = room.get("temperature")
|
self._attr_hvac_mode = HVACMode.HEAT
|
||||||
self._attr_target_temperature = room.get("targetTemperature")
|
self._attr_icon = "mdi:radiator"
|
||||||
if room["heatingEnabled"]:
|
else:
|
||||||
self._attr_hvac_mode = HVACMode.HEAT
|
self._attr_hvac_mode = HVACMode.OFF
|
||||||
self._attr_icon = "mdi:radiator"
|
self._attr_icon = "mdi:radiator-off"
|
||||||
else:
|
return
|
||||||
self._attr_hvac_mode = HVACMode.OFF
|
|
||||||
self._attr_icon = "mdi:radiator-off"
|
|
||||||
|
|
||||||
|
|
||||||
class LocalAdaxDevice(CoordinatorEntity[AdaxLocalCoordinator], ClimateEntity):
|
class LocalAdaxDevice(ClimateEntity):
|
||||||
"""Representation of a heater."""
|
"""Representation of a heater."""
|
||||||
|
|
||||||
_attr_hvac_modes = [HVACMode.HEAT, HVACMode.OFF]
|
_attr_hvac_modes = [HVACMode.HEAT, HVACMode.OFF]
|
||||||
_attr_hvac_mode = HVACMode.OFF
|
_attr_hvac_mode = HVACMode.HEAT
|
||||||
_attr_icon = "mdi:radiator-off"
|
|
||||||
_attr_max_temp = 35
|
_attr_max_temp = 35
|
||||||
_attr_min_temp = 5
|
_attr_min_temp = 5
|
||||||
_attr_supported_features = (
|
_attr_supported_features = (
|
||||||
@@ -153,10 +146,9 @@ class LocalAdaxDevice(CoordinatorEntity[AdaxLocalCoordinator], ClimateEntity):
|
|||||||
_attr_target_temperature_step = PRECISION_WHOLE
|
_attr_target_temperature_step = PRECISION_WHOLE
|
||||||
_attr_temperature_unit = UnitOfTemperature.CELSIUS
|
_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."""
|
"""Initialize the heater."""
|
||||||
super().__init__(coordinator)
|
self._adax_data_handler = adax_data_handler
|
||||||
self._adax_data_handler: AdaxLocal = coordinator.adax_data_handler
|
|
||||||
self._attr_unique_id = unique_id
|
self._attr_unique_id = unique_id
|
||||||
self._attr_device_info = DeviceInfo(
|
self._attr_device_info = DeviceInfo(
|
||||||
identifiers={(DOMAIN, unique_id)},
|
identifiers={(DOMAIN, unique_id)},
|
||||||
@@ -177,20 +169,17 @@ class LocalAdaxDevice(CoordinatorEntity[AdaxLocalCoordinator], ClimateEntity):
|
|||||||
return
|
return
|
||||||
await self._adax_data_handler.set_target_temperature(temperature)
|
await self._adax_data_handler.set_target_temperature(temperature)
|
||||||
|
|
||||||
@callback
|
async def async_update(self) -> None:
|
||||||
def _handle_coordinator_update(self) -> None:
|
"""Get the latest data."""
|
||||||
"""Handle updated data from the coordinator."""
|
data = await self._adax_data_handler.get_status()
|
||||||
if data := self.coordinator.data:
|
self._attr_current_temperature = data["current_temperature"]
|
||||||
self._attr_current_temperature = data["current_temperature"]
|
self._attr_available = self._attr_current_temperature is not None
|
||||||
self._attr_available = self._attr_current_temperature is not None
|
if (target_temp := data["target_temperature"]) == 0:
|
||||||
if (target_temp := data["target_temperature"]) == 0:
|
self._attr_hvac_mode = HVACMode.OFF
|
||||||
self._attr_hvac_mode = HVACMode.OFF
|
self._attr_icon = "mdi:radiator-off"
|
||||||
self._attr_icon = "mdi:radiator-off"
|
if target_temp == 0:
|
||||||
if target_temp == 0:
|
self._attr_target_temperature = self._attr_min_temp
|
||||||
self._attr_target_temperature = self._attr_min_temp
|
else:
|
||||||
else:
|
self._attr_hvac_mode = HVACMode.HEAT
|
||||||
self._attr_hvac_mode = HVACMode.HEAT
|
self._attr_icon = "mdi:radiator"
|
||||||
self._attr_icon = "mdi:radiator"
|
self._attr_target_temperature = target_temp
|
||||||
self._attr_target_temperature = target_temp
|
|
||||||
|
|
||||||
super()._handle_coordinator_update()
|
|
||||||
|
@@ -1,6 +1,5 @@
|
|||||||
"""Constants for the Adax integration."""
|
"""Constants for the Adax integration."""
|
||||||
|
|
||||||
import datetime
|
|
||||||
from typing import Final
|
from typing import Final
|
||||||
|
|
||||||
ACCOUNT_ID: Final = "account_id"
|
ACCOUNT_ID: Final = "account_id"
|
||||||
@@ -10,5 +9,3 @@ DOMAIN: Final = "adax"
|
|||||||
LOCAL = "Local"
|
LOCAL = "Local"
|
||||||
WIFI_SSID = "wifi_ssid"
|
WIFI_SSID = "wifi_ssid"
|
||||||
WIFI_PSWD = "wifi_pswd"
|
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",
|
"domain": "adax",
|
||||||
"name": "Adax",
|
"name": "Adax",
|
||||||
"codeowners": ["@danielhiversen", "@lazytarget"],
|
"codeowners": ["@danielhiversen"],
|
||||||
"config_flow": true,
|
"config_flow": true,
|
||||||
"documentation": "https://www.home-assistant.io/integrations/adax",
|
"documentation": "https://www.home-assistant.io/integrations/adax",
|
||||||
"iot_class": "local_polling",
|
"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"])
|
|
@@ -5,14 +5,14 @@
|
|||||||
"data": {
|
"data": {
|
||||||
"connection_type": "Select connection type"
|
"connection_type": "Select connection type"
|
||||||
},
|
},
|
||||||
"description": "Select connection type. Local requires heaters with Bluetooth"
|
"description": "Select connection type. Local requires heaters with bluetooth"
|
||||||
},
|
},
|
||||||
"local": {
|
"local": {
|
||||||
"data": {
|
"data": {
|
||||||
"wifi_ssid": "Wi-Fi SSID",
|
"wifi_ssid": "Wi-Fi SSID",
|
||||||
"wifi_pswd": "Wi-Fi password"
|
"wifi_pswd": "Wi-Fi Password"
|
||||||
},
|
},
|
||||||
"description": "Reset the heater by pressing + and OK until display shows 'Reset'. Then press and hold OK button on the heater until the blue LED starts blinking before pressing Submit. Configuring heater might take some minutes."
|
"description": "Reset the heater by pressing + and OK until display shows 'Reset'. Then press and hold OK button on the heater until the blue led starts blinking before pressing Submit. Configuring heater might take some minutes."
|
||||||
},
|
},
|
||||||
"cloud": {
|
"cloud": {
|
||||||
"data": {
|
"data": {
|
||||||
|
@@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from decimal import Decimal
|
|
||||||
import logging
|
import logging
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
@@ -15,7 +14,6 @@ from homeassistant.components.climate import (
|
|||||||
FAN_MEDIUM,
|
FAN_MEDIUM,
|
||||||
ClimateEntity,
|
ClimateEntity,
|
||||||
ClimateEntityFeature,
|
ClimateEntityFeature,
|
||||||
HVACAction,
|
|
||||||
HVACMode,
|
HVACMode,
|
||||||
)
|
)
|
||||||
from homeassistant.const import ATTR_TEMPERATURE, PRECISION_WHOLE, UnitOfTemperature
|
from homeassistant.const import ATTR_TEMPERATURE, PRECISION_WHOLE, UnitOfTemperature
|
||||||
@@ -51,14 +49,6 @@ ADVANTAGE_AIR_MYTEMP_ENABLED = "climateControlModeEnabled"
|
|||||||
ADVANTAGE_AIR_HEAT_TARGET = "myAutoHeatTargetTemp"
|
ADVANTAGE_AIR_HEAT_TARGET = "myAutoHeatTargetTemp"
|
||||||
ADVANTAGE_AIR_COOL_TARGET = "myAutoCoolTargetTemp"
|
ADVANTAGE_AIR_COOL_TARGET = "myAutoCoolTargetTemp"
|
||||||
ADVANTAGE_AIR_MYFAN = "autoAA"
|
ADVANTAGE_AIR_MYFAN = "autoAA"
|
||||||
ADVANTAGE_AIR_MYAUTO_MODE_SET = "myAutoModeCurrentSetMode"
|
|
||||||
|
|
||||||
HVAC_ACTIONS = {
|
|
||||||
"cool": HVACAction.COOLING,
|
|
||||||
"heat": HVACAction.HEATING,
|
|
||||||
"vent": HVACAction.FAN,
|
|
||||||
"dry": HVACAction.DRYING,
|
|
||||||
}
|
|
||||||
|
|
||||||
HVAC_MODES = [
|
HVAC_MODES = [
|
||||||
HVACMode.OFF,
|
HVACMode.OFF,
|
||||||
@@ -185,17 +175,6 @@ class AdvantageAirAC(AdvantageAirAcEntity, ClimateEntity):
|
|||||||
return ADVANTAGE_AIR_HVAC_MODES.get(self._ac["mode"])
|
return ADVANTAGE_AIR_HVAC_MODES.get(self._ac["mode"])
|
||||||
return HVACMode.OFF
|
return HVACMode.OFF
|
||||||
|
|
||||||
@property
|
|
||||||
def hvac_action(self) -> HVACAction | None:
|
|
||||||
"""Return the current running HVAC action."""
|
|
||||||
if self._ac["state"] == ADVANTAGE_AIR_STATE_OFF:
|
|
||||||
return HVACAction.OFF
|
|
||||||
if self._ac["mode"] == "myauto":
|
|
||||||
return HVAC_ACTIONS.get(
|
|
||||||
self._ac.get(ADVANTAGE_AIR_MYAUTO_MODE_SET, HVACAction.OFF)
|
|
||||||
)
|
|
||||||
return HVAC_ACTIONS.get(self._ac["mode"])
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def fan_mode(self) -> str | None:
|
def fan_mode(self) -> str | None:
|
||||||
"""Return the current fan modes."""
|
"""Return the current fan modes."""
|
||||||
@@ -294,22 +273,6 @@ class AdvantageAirZone(AdvantageAirZoneEntity, ClimateEntity):
|
|||||||
return HVACMode.HEAT_COOL
|
return HVACMode.HEAT_COOL
|
||||||
return HVACMode.OFF
|
return HVACMode.OFF
|
||||||
|
|
||||||
@property
|
|
||||||
def hvac_action(self) -> HVACAction | None:
|
|
||||||
"""Return the HVAC action, inheriting from master AC if zone is open but idle if air is <= 5%."""
|
|
||||||
if self._ac["state"] == ADVANTAGE_AIR_STATE_OFF:
|
|
||||||
return HVACAction.OFF
|
|
||||||
master_action = HVAC_ACTIONS.get(self._ac["mode"], HVACAction.OFF)
|
|
||||||
if self._ac["mode"] == "myauto":
|
|
||||||
master_action = HVAC_ACTIONS.get(
|
|
||||||
str(self._ac.get(ADVANTAGE_AIR_MYAUTO_MODE_SET)), HVACAction.OFF
|
|
||||||
)
|
|
||||||
if self._zone["state"] == ADVANTAGE_AIR_STATE_OPEN:
|
|
||||||
if self._zone["value"] <= Decimal(5):
|
|
||||||
return HVACAction.IDLE
|
|
||||||
return master_action
|
|
||||||
return HVACAction.OFF
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def current_temperature(self) -> float | None:
|
def current_temperature(self) -> float | None:
|
||||||
"""Return the current temperature."""
|
"""Return the current temperature."""
|
||||||
|
@@ -7,4 +7,3 @@ ADVANTAGE_AIR_STATE_CLOSE = "close"
|
|||||||
ADVANTAGE_AIR_STATE_ON = "on"
|
ADVANTAGE_AIR_STATE_ON = "on"
|
||||||
ADVANTAGE_AIR_STATE_OFF = "off"
|
ADVANTAGE_AIR_STATE_OFF = "off"
|
||||||
ADVANTAGE_AIR_AUTOFAN_ENABLED = "aaAutoFanModeEnabled"
|
ADVANTAGE_AIR_AUTOFAN_ENABLED = "aaAutoFanModeEnabled"
|
||||||
ADVANTAGE_AIR_NIGHT_MODE_ENABLED = "quietNightModeEnabled"
|
|
||||||
|
@@ -41,7 +41,7 @@ async def async_setup_entry(
|
|||||||
entities.append(
|
entities.append(
|
||||||
AdvantageAirThingCover(instance, thing, CoverDeviceClass.BLIND)
|
AdvantageAirThingCover(instance, thing, CoverDeviceClass.BLIND)
|
||||||
)
|
)
|
||||||
elif thing["channelDipState"] in [3, 10]: # 3 & 10 = "Garage door"
|
elif thing["channelDipState"] == 3: # 3 = "Garage door"
|
||||||
entities.append(
|
entities.append(
|
||||||
AdvantageAirThingCover(instance, thing, CoverDeviceClass.GARAGE)
|
AdvantageAirThingCover(instance, thing, CoverDeviceClass.GARAGE)
|
||||||
)
|
)
|
||||||
|
@@ -8,7 +8,7 @@ from homeassistant.helpers.device_registry import DeviceInfo
|
|||||||
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
|
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
|
||||||
|
|
||||||
from . import AdvantageAirDataConfigEntry
|
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 .entity import AdvantageAirEntity, AdvantageAirThingEntity
|
||||||
from .models import AdvantageAirData
|
from .models import AdvantageAirData
|
||||||
|
|
||||||
@@ -52,8 +52,8 @@ class AdvantageAirLight(AdvantageAirEntity, LightEntity):
|
|||||||
self._id: str = light["id"]
|
self._id: str = light["id"]
|
||||||
self._attr_unique_id += f"-{self._id}"
|
self._attr_unique_id += f"-{self._id}"
|
||||||
self._attr_device_info = DeviceInfo(
|
self._attr_device_info = DeviceInfo(
|
||||||
identifiers={(DOMAIN, self._attr_unique_id)},
|
identifiers={(ADVANTAGE_AIR_DOMAIN, self._attr_unique_id)},
|
||||||
via_device=(DOMAIN, self.coordinator.data["system"]["rid"]),
|
via_device=(ADVANTAGE_AIR_DOMAIN, self.coordinator.data["system"]["rid"]),
|
||||||
manufacturer="Advantage Air",
|
manufacturer="Advantage Air",
|
||||||
model=light.get("moduleType"),
|
model=light.get("moduleType"),
|
||||||
name=light["name"],
|
name=light["name"],
|
||||||
|
@@ -9,7 +9,6 @@ from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
|
|||||||
from . import AdvantageAirDataConfigEntry
|
from . import AdvantageAirDataConfigEntry
|
||||||
from .const import (
|
from .const import (
|
||||||
ADVANTAGE_AIR_AUTOFAN_ENABLED,
|
ADVANTAGE_AIR_AUTOFAN_ENABLED,
|
||||||
ADVANTAGE_AIR_NIGHT_MODE_ENABLED,
|
|
||||||
ADVANTAGE_AIR_STATE_OFF,
|
ADVANTAGE_AIR_STATE_OFF,
|
||||||
ADVANTAGE_AIR_STATE_ON,
|
ADVANTAGE_AIR_STATE_ON,
|
||||||
)
|
)
|
||||||
@@ -33,8 +32,6 @@ async def async_setup_entry(
|
|||||||
entities.append(AdvantageAirFreshAir(instance, ac_key))
|
entities.append(AdvantageAirFreshAir(instance, ac_key))
|
||||||
if ADVANTAGE_AIR_AUTOFAN_ENABLED in ac_device["info"]:
|
if ADVANTAGE_AIR_AUTOFAN_ENABLED in ac_device["info"]:
|
||||||
entities.append(AdvantageAirMyFan(instance, ac_key))
|
entities.append(AdvantageAirMyFan(instance, ac_key))
|
||||||
if ADVANTAGE_AIR_NIGHT_MODE_ENABLED in ac_device["info"]:
|
|
||||||
entities.append(AdvantageAirNightMode(instance, ac_key))
|
|
||||||
if things := instance.coordinator.data.get("myThings"):
|
if things := instance.coordinator.data.get("myThings"):
|
||||||
entities.extend(
|
entities.extend(
|
||||||
AdvantageAirRelay(instance, thing)
|
AdvantageAirRelay(instance, thing)
|
||||||
@@ -96,32 +93,6 @@ class AdvantageAirMyFan(AdvantageAirAcEntity, SwitchEntity):
|
|||||||
await self.async_update_ac({ADVANTAGE_AIR_AUTOFAN_ENABLED: False})
|
await self.async_update_ac({ADVANTAGE_AIR_AUTOFAN_ENABLED: False})
|
||||||
|
|
||||||
|
|
||||||
class AdvantageAirNightMode(AdvantageAirAcEntity, SwitchEntity):
|
|
||||||
"""Representation of Advantage 'MySleep$aver' Mode control."""
|
|
||||||
|
|
||||||
_attr_icon = "mdi:weather-night"
|
|
||||||
_attr_name = "MySleep$aver"
|
|
||||||
_attr_device_class = SwitchDeviceClass.SWITCH
|
|
||||||
|
|
||||||
def __init__(self, instance: AdvantageAirData, ac_key: str) -> None:
|
|
||||||
"""Initialize an Advantage Air Night Mode control."""
|
|
||||||
super().__init__(instance, ac_key)
|
|
||||||
self._attr_unique_id += "-nightmode"
|
|
||||||
|
|
||||||
@property
|
|
||||||
def is_on(self) -> bool:
|
|
||||||
"""Return the Night Mode status."""
|
|
||||||
return self._ac[ADVANTAGE_AIR_NIGHT_MODE_ENABLED]
|
|
||||||
|
|
||||||
async def async_turn_on(self, **kwargs: Any) -> None:
|
|
||||||
"""Turn Night Mode on."""
|
|
||||||
await self.async_update_ac({ADVANTAGE_AIR_NIGHT_MODE_ENABLED: True})
|
|
||||||
|
|
||||||
async def async_turn_off(self, **kwargs: Any) -> None:
|
|
||||||
"""Turn Night Mode off."""
|
|
||||||
await self.async_update_ac({ADVANTAGE_AIR_NIGHT_MODE_ENABLED: False})
|
|
||||||
|
|
||||||
|
|
||||||
class AdvantageAirRelay(AdvantageAirThingEntity, SwitchEntity):
|
class AdvantageAirRelay(AdvantageAirThingEntity, SwitchEntity):
|
||||||
"""Representation of Advantage Air Thing."""
|
"""Representation of Advantage Air Thing."""
|
||||||
|
|
||||||
|
@@ -6,7 +6,7 @@ from homeassistant.helpers.device_registry import DeviceInfo
|
|||||||
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
|
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
|
||||||
|
|
||||||
from . import AdvantageAirDataConfigEntry
|
from . import AdvantageAirDataConfigEntry
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN as ADVANTAGE_AIR_DOMAIN
|
||||||
from .entity import AdvantageAirEntity
|
from .entity import AdvantageAirEntity
|
||||||
from .models import AdvantageAirData
|
from .models import AdvantageAirData
|
||||||
|
|
||||||
@@ -32,7 +32,9 @@ class AdvantageAirApp(AdvantageAirEntity, UpdateEntity):
|
|||||||
"""Initialize the Advantage Air App."""
|
"""Initialize the Advantage Air App."""
|
||||||
super().__init__(instance)
|
super().__init__(instance)
|
||||||
self._attr_device_info = DeviceInfo(
|
self._attr_device_info = DeviceInfo(
|
||||||
identifiers={(DOMAIN, self.coordinator.data["system"]["rid"])},
|
identifiers={
|
||||||
|
(ADVANTAGE_AIR_DOMAIN, self.coordinator.data["system"]["rid"])
|
||||||
|
},
|
||||||
manufacturer="Advantage Air",
|
manufacturer="Advantage Air",
|
||||||
model=self.coordinator.data["system"]["sysType"],
|
model=self.coordinator.data["system"]["sysType"],
|
||||||
name=self.coordinator.data["system"]["name"],
|
name=self.coordinator.data["system"]["name"],
|
||||||
|
@@ -71,14 +71,7 @@ class AemetConfigFlow(ConfigFlow, domain=DOMAIN):
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
return self.async_show_form(
|
return self.async_show_form(step_id="user", data_schema=schema, errors=errors)
|
||||||
step_id="user",
|
|
||||||
data_schema=schema,
|
|
||||||
errors=errors,
|
|
||||||
description_placeholders={
|
|
||||||
"api_key_url": "https://opendata.aemet.es/centrodedescargas/altaUsuario"
|
|
||||||
},
|
|
||||||
)
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
@callback
|
@callback
|
||||||
|
@@ -185,7 +185,6 @@ FORECAST_SENSORS: Final[tuple[AemetSensorEntityDescription, ...]] = (
|
|||||||
keys=[AOD_TOWN, AOD_FORECAST_DAILY, AOD_FORECAST_CURRENT, AOD_WIND_DIRECTION],
|
keys=[AOD_TOWN, AOD_FORECAST_DAILY, AOD_FORECAST_CURRENT, AOD_WIND_DIRECTION],
|
||||||
name="Daily forecast wind bearing",
|
name="Daily forecast wind bearing",
|
||||||
native_unit_of_measurement=DEGREE,
|
native_unit_of_measurement=DEGREE,
|
||||||
device_class=SensorDeviceClass.WIND_DIRECTION,
|
|
||||||
),
|
),
|
||||||
AemetSensorEntityDescription(
|
AemetSensorEntityDescription(
|
||||||
entity_registry_enabled_default=False,
|
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],
|
keys=[AOD_TOWN, AOD_FORECAST_HOURLY, AOD_FORECAST_CURRENT, AOD_WIND_DIRECTION],
|
||||||
name="Hourly forecast wind bearing",
|
name="Hourly forecast wind bearing",
|
||||||
native_unit_of_measurement=DEGREE,
|
native_unit_of_measurement=DEGREE,
|
||||||
device_class=SensorDeviceClass.WIND_DIRECTION,
|
|
||||||
),
|
),
|
||||||
AemetSensorEntityDescription(
|
AemetSensorEntityDescription(
|
||||||
entity_registry_enabled_default=False,
|
entity_registry_enabled_default=False,
|
||||||
@@ -336,8 +334,7 @@ WEATHER_SENSORS: Final[tuple[AemetSensorEntityDescription, ...]] = (
|
|||||||
keys=[AOD_WEATHER, AOD_WIND_DIRECTION],
|
keys=[AOD_WEATHER, AOD_WIND_DIRECTION],
|
||||||
name="Wind bearing",
|
name="Wind bearing",
|
||||||
native_unit_of_measurement=DEGREE,
|
native_unit_of_measurement=DEGREE,
|
||||||
state_class=SensorStateClass.MEASUREMENT_ANGLE,
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
device_class=SensorDeviceClass.WIND_DIRECTION,
|
|
||||||
),
|
),
|
||||||
AemetSensorEntityDescription(
|
AemetSensorEntityDescription(
|
||||||
key=ATTR_API_WIND_MAX_SPEED,
|
key=ATTR_API_WIND_MAX_SPEED,
|
||||||
|
@@ -14,7 +14,7 @@
|
|||||||
"longitude": "[%key:common::config_flow::data::longitude%]",
|
"longitude": "[%key:common::config_flow::data::longitude%]",
|
||||||
"name": "Name of the integration"
|
"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"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@@ -51,7 +51,7 @@
|
|||||||
"issues": {
|
"issues": {
|
||||||
"deprecated_yaml_import_issue_cannot_connect": {
|
"deprecated_yaml_import_issue_cannot_connect": {
|
||||||
"title": "The {integration_title} YAML configuration import failed",
|
"title": "The {integration_title} YAML configuration import failed",
|
||||||
"description": "Configuring {integration_title} using YAML is being removed but there was a connection error importing your YAML configuration.\n\nEnsure connection to {integration_title} works and restart Home Assistant to try again or remove the {integration_title} YAML configuration from your configuration.yaml file and continue to [set up the integration]({url}) manually."
|
"description": "Configuring {integration_title} using YAML is being removed but there was an connection error importing your YAML configuration.\n\nEnsure connection to {integration_title} works and restart Home Assistant to try again or remove the {integration_title} YAML configuration from your configuration.yaml file and continue to [set up the integration]({url}) manually."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -10,7 +10,7 @@ from homeassistant.exceptions import ConfigEntryNotReady
|
|||||||
from homeassistant.helpers import device_registry as dr
|
from homeassistant.helpers import device_registry as dr
|
||||||
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
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"
|
ATTRIBUTION = "ispyconnect.com"
|
||||||
DEFAULT_BRAND = "Agent DVR by ispyconnect.com"
|
DEFAULT_BRAND = "Agent DVR by ispyconnect.com"
|
||||||
@@ -46,7 +46,7 @@ async def async_setup_entry(
|
|||||||
|
|
||||||
device_registry.async_get_or_create(
|
device_registry.async_get_or_create(
|
||||||
config_entry_id=config_entry.entry_id,
|
config_entry_id=config_entry.entry_id,
|
||||||
identifiers={(DOMAIN, agent_client.unique)},
|
identifiers={(AGENT_DOMAIN, agent_client.unique)},
|
||||||
manufacturer="iSpyConnect",
|
manufacturer="iSpyConnect",
|
||||||
name=f"Agent {agent_client.name}",
|
name=f"Agent {agent_client.name}",
|
||||||
model="Agent DVR",
|
model="Agent DVR",
|
||||||
|
@@ -12,7 +12,7 @@ from homeassistant.helpers.device_registry import DeviceInfo
|
|||||||
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
|
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
|
||||||
|
|
||||||
from . import AgentDVRConfigEntry
|
from . import AgentDVRConfigEntry
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN as AGENT_DOMAIN
|
||||||
|
|
||||||
CONF_HOME_MODE_NAME = "home"
|
CONF_HOME_MODE_NAME = "home"
|
||||||
CONF_AWAY_MODE_NAME = "away"
|
CONF_AWAY_MODE_NAME = "away"
|
||||||
@@ -47,7 +47,7 @@ class AgentBaseStation(AlarmControlPanelEntity):
|
|||||||
self._client = client
|
self._client = client
|
||||||
self._attr_unique_id = f"{client.unique}_CP"
|
self._attr_unique_id = f"{client.unique}_CP"
|
||||||
self._attr_device_info = DeviceInfo(
|
self._attr_device_info = DeviceInfo(
|
||||||
identifiers={(DOMAIN, client.unique)},
|
identifiers={(AGENT_DOMAIN, client.unique)},
|
||||||
name=f"{client.name} {CONST_ALARM_CONTROL_PANEL_NAME}",
|
name=f"{client.name} {CONST_ALARM_CONTROL_PANEL_NAME}",
|
||||||
manufacturer="Agent",
|
manufacturer="Agent",
|
||||||
model=CONST_ALARM_CONTROL_PANEL_NAME,
|
model=CONST_ALARM_CONTROL_PANEL_NAME,
|
||||||
|
@@ -15,7 +15,7 @@ from homeassistant.helpers.entity_platform import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
from . import AgentDVRConfigEntry
|
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)
|
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
|
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(
|
self._attr_device_info = DeviceInfo(
|
||||||
identifiers={(DOMAIN, self.unique_id)},
|
identifiers={(AGENT_DOMAIN, self.unique_id)},
|
||||||
manufacturer="Agent",
|
manufacturer="Agent",
|
||||||
model="Camera",
|
model="Camera",
|
||||||
name=f"{device.client.name} {device.name}",
|
name=f"{device.client.name} {device.name}",
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user