mirror of
https://github.com/esphome/esphome.git
synced 2025-11-10 20:05:48 +00:00
Compare commits
1 Commits
2023.6.2
...
jesserockz
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b6e765daaa |
@@ -4,61 +4,53 @@
|
|||||||
"postCreateCommand": [
|
"postCreateCommand": [
|
||||||
"script/devcontainer-post-create"
|
"script/devcontainer-post-create"
|
||||||
],
|
],
|
||||||
"containerEnv": {
|
|
||||||
"DEVCONTAINER": "1"
|
|
||||||
},
|
|
||||||
"runArgs": [
|
"runArgs": [
|
||||||
"--privileged",
|
"--privileged",
|
||||||
"-e",
|
"-e",
|
||||||
"ESPHOME_DASHBOARD_USE_PING=1"
|
"ESPHOME_DASHBOARD_USE_PING=1"
|
||||||
],
|
],
|
||||||
"appPort": 6052,
|
"appPort": 6052,
|
||||||
"customizations": {
|
"extensions": [
|
||||||
"vscode": {
|
// python
|
||||||
"extensions": [
|
"ms-python.python",
|
||||||
// python
|
"visualstudioexptteam.vscodeintellicode",
|
||||||
"ms-python.python",
|
// yaml
|
||||||
"visualstudioexptteam.vscodeintellicode",
|
"redhat.vscode-yaml",
|
||||||
// yaml
|
// cpp
|
||||||
"redhat.vscode-yaml",
|
"ms-vscode.cpptools",
|
||||||
// cpp
|
// editorconfig
|
||||||
"ms-vscode.cpptools",
|
"editorconfig.editorconfig",
|
||||||
// editorconfig
|
],
|
||||||
"editorconfig.editorconfig",
|
"settings": {
|
||||||
],
|
"python.languageServer": "Pylance",
|
||||||
"settings": {
|
"python.pythonPath": "/usr/bin/python3",
|
||||||
"python.languageServer": "Pylance",
|
"python.linting.pylintEnabled": true,
|
||||||
"python.pythonPath": "/usr/bin/python3",
|
"python.linting.enabled": true,
|
||||||
"python.linting.pylintEnabled": true,
|
"python.formatting.provider": "black",
|
||||||
"python.linting.enabled": true,
|
"editor.formatOnPaste": false,
|
||||||
"python.formatting.provider": "black",
|
"editor.formatOnSave": true,
|
||||||
"editor.formatOnPaste": false,
|
"editor.formatOnType": true,
|
||||||
"editor.formatOnSave": true,
|
"files.trimTrailingWhitespace": true,
|
||||||
"editor.formatOnType": true,
|
"terminal.integrated.defaultProfile.linux": "bash",
|
||||||
"files.trimTrailingWhitespace": true,
|
"yaml.customTags": [
|
||||||
"terminal.integrated.defaultProfile.linux": "bash",
|
"!secret scalar",
|
||||||
"yaml.customTags": [
|
"!lambda scalar",
|
||||||
"!secret scalar",
|
"!include_dir_named scalar",
|
||||||
"!lambda scalar",
|
"!include_dir_list scalar",
|
||||||
"!extend scalar",
|
"!include_dir_merge_list scalar",
|
||||||
"!include_dir_named scalar",
|
"!include_dir_merge_named scalar"
|
||||||
"!include_dir_list scalar",
|
],
|
||||||
"!include_dir_merge_list scalar",
|
"files.exclude": {
|
||||||
"!include_dir_merge_named scalar"
|
"**/.git": true,
|
||||||
],
|
"**/.DS_Store": true,
|
||||||
"files.exclude": {
|
"**/*.pyc": {
|
||||||
"**/.git": true,
|
"when": "$(basename).py"
|
||||||
"**/.DS_Store": true,
|
},
|
||||||
"**/*.pyc": {
|
"**/__pycache__": true
|
||||||
"when": "$(basename).py"
|
},
|
||||||
},
|
"files.associations": {
|
||||||
"**/__pycache__": true
|
"**/.vscode/*.json": "jsonc"
|
||||||
},
|
},
|
||||||
"files.associations": {
|
"C_Cpp.clang_format_path": "/usr/bin/clang-format-13",
|
||||||
"**/.vscode/*.json": "jsonc"
|
|
||||||
},
|
|
||||||
"C_Cpp.clang_format_path": "/usr/bin/clang-format-13"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
1
.gitattributes
vendored
1
.gitattributes
vendored
@@ -1,3 +1,2 @@
|
|||||||
# Normalize line endings to LF in the repository
|
# Normalize line endings to LF in the repository
|
||||||
* text eol=lf
|
* text eol=lf
|
||||||
*.png binary
|
|
||||||
|
|||||||
2
.github/workflows/ci-docker.yml
vendored
2
.github/workflows/ci-docker.yml
vendored
@@ -11,7 +11,6 @@ on:
|
|||||||
- ".github/workflows/**"
|
- ".github/workflows/**"
|
||||||
- "requirements*.txt"
|
- "requirements*.txt"
|
||||||
- "platformio.ini"
|
- "platformio.ini"
|
||||||
- "script/platformio_install_deps.py"
|
|
||||||
|
|
||||||
pull_request:
|
pull_request:
|
||||||
paths:
|
paths:
|
||||||
@@ -19,7 +18,6 @@ on:
|
|||||||
- ".github/workflows/**"
|
- ".github/workflows/**"
|
||||||
- "requirements*.txt"
|
- "requirements*.txt"
|
||||||
- "platformio.ini"
|
- "platformio.ini"
|
||||||
- "script/platformio_install_deps.py"
|
|
||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
contents: read
|
contents: read
|
||||||
|
|||||||
399
.github/workflows/ci.yml
vendored
399
.github/workflows/ci.yml
vendored
@@ -12,266 +12,59 @@ on:
|
|||||||
permissions:
|
permissions:
|
||||||
contents: read
|
contents: read
|
||||||
|
|
||||||
env:
|
|
||||||
DEFAULT_PYTHON: "3.9"
|
|
||||||
PYUPGRADE_TARGET: "--py39-plus"
|
|
||||||
CLANG_FORMAT_VERSION: "13.0.1"
|
|
||||||
|
|
||||||
concurrency:
|
concurrency:
|
||||||
# yamllint disable-line rule:line-length
|
# yamllint disable-line rule:line-length
|
||||||
group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
|
group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
|
||||||
cancel-in-progress: true
|
cancel-in-progress: true
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
common:
|
ci:
|
||||||
name: Create common environment
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v3.5.2
|
|
||||||
- name: Set up Python ${{ env.DEFAULT_PYTHON }}
|
|
||||||
uses: actions/setup-python@v4.6.0
|
|
||||||
with:
|
|
||||||
python-version: ${{ env.DEFAULT_PYTHON }}
|
|
||||||
- name: Restore Python virtual environment
|
|
||||||
id: cache-venv
|
|
||||||
uses: actions/cache@v3.3.1
|
|
||||||
with:
|
|
||||||
path: venv
|
|
||||||
# yamllint disable-line rule:line-length
|
|
||||||
key: ${{ runner.os }}-${{ env.DEFAULT_PYTHON }}-venv-${{ hashFiles('requirements.txt', 'requirements_optional.txt', 'requirements_test.txt') }}
|
|
||||||
- name: Create Python virtual environment
|
|
||||||
if: steps.cache-venv.outputs.cache-hit != 'true'
|
|
||||||
run: |
|
|
||||||
python -m venv venv
|
|
||||||
. venv/bin/activate
|
|
||||||
python --version
|
|
||||||
pip install -r requirements.txt -r requirements_optional.txt -r requirements_test.txt
|
|
||||||
pip install -e .
|
|
||||||
|
|
||||||
yamllint:
|
|
||||||
name: yamllint
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v3.5.2
|
|
||||||
- name: Run yamllint
|
|
||||||
uses: frenck/action-yamllint@v1.4.1
|
|
||||||
|
|
||||||
black:
|
|
||||||
name: Check black
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs:
|
|
||||||
- common
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v3.5.2
|
|
||||||
- name: Restore Python virtual environment
|
|
||||||
uses: actions/cache/restore@v3.3.1
|
|
||||||
with:
|
|
||||||
path: venv
|
|
||||||
# yamllint disable-line rule:line-length
|
|
||||||
key: ${{ runner.os }}-${{ env.DEFAULT_PYTHON }}-venv-${{ hashFiles('requirements.txt', 'requirements_optional.txt', 'requirements_test.txt') }}
|
|
||||||
- name: Run black
|
|
||||||
run: |
|
|
||||||
. venv/bin/activate
|
|
||||||
black --verbose esphome tests
|
|
||||||
- name: Suggested changes
|
|
||||||
run: script/ci-suggest-changes
|
|
||||||
if: always()
|
|
||||||
|
|
||||||
flake8:
|
|
||||||
name: Check flake8
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs:
|
|
||||||
- common
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v3.5.2
|
|
||||||
- name: Restore Python virtual environment
|
|
||||||
uses: actions/cache/restore@v3.3.1
|
|
||||||
with:
|
|
||||||
path: venv
|
|
||||||
# yamllint disable-line rule:line-length
|
|
||||||
key: ${{ runner.os }}-${{ env.DEFAULT_PYTHON }}-venv-${{ hashFiles('requirements.txt', 'requirements_optional.txt', 'requirements_test.txt') }}
|
|
||||||
- name: Run flake8
|
|
||||||
run: |
|
|
||||||
. venv/bin/activate
|
|
||||||
flake8 esphome
|
|
||||||
- name: Suggested changes
|
|
||||||
run: script/ci-suggest-changes
|
|
||||||
if: always()
|
|
||||||
|
|
||||||
pylint:
|
|
||||||
name: Check pylint
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs:
|
|
||||||
- common
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v3.5.2
|
|
||||||
- name: Restore Python virtual environment
|
|
||||||
uses: actions/cache/restore@v3.3.1
|
|
||||||
with:
|
|
||||||
path: venv
|
|
||||||
# yamllint disable-line rule:line-length
|
|
||||||
key: ${{ runner.os }}-${{ env.DEFAULT_PYTHON }}-venv-${{ hashFiles('requirements.txt', 'requirements_optional.txt', 'requirements_test.txt') }}
|
|
||||||
- name: Run pylint
|
|
||||||
run: |
|
|
||||||
. venv/bin/activate
|
|
||||||
pylint -f parseable --persistent=n esphome
|
|
||||||
- name: Suggested changes
|
|
||||||
run: script/ci-suggest-changes
|
|
||||||
if: always()
|
|
||||||
|
|
||||||
pyupgrade:
|
|
||||||
name: Check pyupgrade
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs:
|
|
||||||
- common
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v3.5.2
|
|
||||||
- name: Restore Python virtual environment
|
|
||||||
uses: actions/cache/restore@v3.3.1
|
|
||||||
with:
|
|
||||||
path: venv
|
|
||||||
# yamllint disable-line rule:line-length
|
|
||||||
key: ${{ runner.os }}-${{ env.DEFAULT_PYTHON }}-venv-${{ hashFiles('requirements.txt', 'requirements_optional.txt', 'requirements_test.txt') }}
|
|
||||||
- name: Run pyupgrade
|
|
||||||
run: |
|
|
||||||
. venv/bin/activate
|
|
||||||
pyupgrade ${{ env.PYUPGRADE_TARGET }} `find esphome -name "*.py" -type f`
|
|
||||||
- name: Suggested changes
|
|
||||||
run: script/ci-suggest-changes
|
|
||||||
if: always()
|
|
||||||
|
|
||||||
ci-custom:
|
|
||||||
name: Run script/ci-custom
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs:
|
|
||||||
- common
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v3.5.2
|
|
||||||
- name: Restore Python virtual environment
|
|
||||||
uses: actions/cache/restore@v3.3.1
|
|
||||||
with:
|
|
||||||
path: venv
|
|
||||||
# yamllint disable-line rule:line-length
|
|
||||||
key: ${{ runner.os }}-${{ env.DEFAULT_PYTHON }}-venv-${{ hashFiles('requirements.txt', 'requirements_optional.txt', 'requirements_test.txt') }}
|
|
||||||
- name: Register matcher
|
|
||||||
run: echo "::add-matcher::.github/workflows/matchers/ci-custom.json"
|
|
||||||
- name: Run script/ci-custom
|
|
||||||
run: |
|
|
||||||
. venv/bin/activate
|
|
||||||
script/ci-custom.py
|
|
||||||
script/build_codeowners.py --check
|
|
||||||
|
|
||||||
pytest:
|
|
||||||
name: Run pytest
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs:
|
|
||||||
- common
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v3.5.2
|
|
||||||
- name: Restore Python virtual environment
|
|
||||||
uses: actions/cache/restore@v3.3.1
|
|
||||||
with:
|
|
||||||
path: venv
|
|
||||||
# yamllint disable-line rule:line-length
|
|
||||||
key: ${{ runner.os }}-${{ env.DEFAULT_PYTHON }}-venv-${{ hashFiles('requirements.txt', 'requirements_optional.txt', 'requirements_test.txt') }}
|
|
||||||
- name: Register matcher
|
|
||||||
run: echo "::add-matcher::.github/workflows/matchers/pytest.json"
|
|
||||||
- name: Run pytest
|
|
||||||
run: |
|
|
||||||
. venv/bin/activate
|
|
||||||
pytest -vv --tb=native tests
|
|
||||||
|
|
||||||
clang-format:
|
|
||||||
name: Check clang-format
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs:
|
|
||||||
- common
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v3.5.2
|
|
||||||
- name: Restore Python virtual environment
|
|
||||||
uses: actions/cache/restore@v3.3.1
|
|
||||||
with:
|
|
||||||
path: venv
|
|
||||||
# yamllint disable-line rule:line-length
|
|
||||||
key: ${{ runner.os }}-${{ env.DEFAULT_PYTHON }}-venv-${{ hashFiles('requirements.txt', 'requirements_optional.txt', 'requirements_test.txt') }}
|
|
||||||
- name: Install clang-format
|
|
||||||
run: |
|
|
||||||
. venv/bin/activate
|
|
||||||
pip install clang-format==${{ env.CLANG_FORMAT_VERSION }}
|
|
||||||
- name: Run clang-format
|
|
||||||
run: |
|
|
||||||
. venv/bin/activate
|
|
||||||
script/clang-format -i
|
|
||||||
git diff-index --quiet HEAD --
|
|
||||||
- name: Suggested changes
|
|
||||||
run: script/ci-suggest-changes
|
|
||||||
if: always()
|
|
||||||
|
|
||||||
compile-tests:
|
|
||||||
name: Run YAML test ${{ matrix.file }}
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs:
|
|
||||||
- common
|
|
||||||
- black
|
|
||||||
- ci-custom
|
|
||||||
- clang-format
|
|
||||||
- flake8
|
|
||||||
- pylint
|
|
||||||
- pytest
|
|
||||||
- pyupgrade
|
|
||||||
- yamllint
|
|
||||||
strategy:
|
|
||||||
fail-fast: false
|
|
||||||
max-parallel: 2
|
|
||||||
matrix:
|
|
||||||
file: [1, 2, 3, 3.1, 4, 5, 6, 7, 8]
|
|
||||||
steps:
|
|
||||||
- name: Check out code from GitHub
|
|
||||||
uses: actions/checkout@v3.5.2
|
|
||||||
- name: Restore Python virtual environment
|
|
||||||
uses: actions/cache/restore@v3.3.1
|
|
||||||
with:
|
|
||||||
path: venv
|
|
||||||
# yamllint disable-line rule:line-length
|
|
||||||
key: ${{ runner.os }}-${{ env.DEFAULT_PYTHON }}-venv-${{ hashFiles('requirements.txt', 'requirements_optional.txt', 'requirements_test.txt') }}
|
|
||||||
- name: Cache platformio
|
|
||||||
uses: actions/cache@v3.3.1
|
|
||||||
with:
|
|
||||||
path: ~/.platformio
|
|
||||||
# yamllint disable-line rule:line-length
|
|
||||||
key: platformio-test${{ matrix.file }}-${{ hashFiles('platformio.ini') }}
|
|
||||||
- name: Run esphome compile tests/test${{ matrix.file }}.yaml
|
|
||||||
run: |
|
|
||||||
. venv/bin/activate
|
|
||||||
esphome compile tests/test${{ matrix.file }}.yaml
|
|
||||||
|
|
||||||
clang-tidy:
|
|
||||||
name: ${{ matrix.name }}
|
name: ${{ matrix.name }}
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs:
|
|
||||||
- common
|
|
||||||
- black
|
|
||||||
- ci-custom
|
|
||||||
- clang-format
|
|
||||||
- flake8
|
|
||||||
- pylint
|
|
||||||
- pytest
|
|
||||||
- pyupgrade
|
|
||||||
- yamllint
|
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
max-parallel: 2
|
|
||||||
matrix:
|
matrix:
|
||||||
include:
|
include:
|
||||||
|
- id: ci-custom
|
||||||
|
name: Run script/ci-custom
|
||||||
|
- id: lint-python
|
||||||
|
name: Run script/lint-python
|
||||||
|
- id: test
|
||||||
|
file: tests/test1.yaml
|
||||||
|
name: Test tests/test1.yaml
|
||||||
|
pio_cache_key: test1
|
||||||
|
- id: test
|
||||||
|
file: tests/test2.yaml
|
||||||
|
name: Test tests/test2.yaml
|
||||||
|
pio_cache_key: test2
|
||||||
|
- id: test
|
||||||
|
file: tests/test3.yaml
|
||||||
|
name: Test tests/test3.yaml
|
||||||
|
pio_cache_key: test3
|
||||||
|
- id: test
|
||||||
|
file: tests/test3.1.yaml
|
||||||
|
name: Test tests/test3.1.yaml
|
||||||
|
pio_cache_key: test3.1
|
||||||
|
- id: test
|
||||||
|
file: tests/test4.yaml
|
||||||
|
name: Test tests/test4.yaml
|
||||||
|
pio_cache_key: test4
|
||||||
|
- id: test
|
||||||
|
file: tests/test5.yaml
|
||||||
|
name: Test tests/test5.yaml
|
||||||
|
pio_cache_key: test5
|
||||||
|
- id: test
|
||||||
|
file: tests/test6.yaml
|
||||||
|
name: Test tests/test6.yaml
|
||||||
|
pio_cache_key: test6
|
||||||
|
- id: test
|
||||||
|
file: tests/test7.yaml
|
||||||
|
name: Test tests/test7.yaml
|
||||||
|
pio_cache_key: test7
|
||||||
|
- id: pytest
|
||||||
|
name: Run pytest
|
||||||
|
- id: clang-format
|
||||||
|
name: Run script/clang-format
|
||||||
- id: clang-tidy
|
- id: clang-tidy
|
||||||
name: Run script/clang-tidy for ESP8266
|
name: Run script/clang-tidy for ESP8266
|
||||||
options: --environment esp8266-arduino-tidy --grep USE_ESP8266
|
options: --environment esp8266-arduino-tidy --grep USE_ESP8266
|
||||||
@@ -296,65 +89,119 @@ jobs:
|
|||||||
name: Run script/clang-tidy for ESP32 IDF
|
name: Run script/clang-tidy for ESP32 IDF
|
||||||
options: --environment esp32-idf-tidy --grep USE_ESP_IDF
|
options: --environment esp32-idf-tidy --grep USE_ESP_IDF
|
||||||
pio_cache_key: tidyesp32-idf
|
pio_cache_key: tidyesp32-idf
|
||||||
|
- id: yamllint
|
||||||
|
name: Run yamllint
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- uses: actions/checkout@v3
|
||||||
uses: actions/checkout@v3.5.2
|
- name: Set up Python
|
||||||
- name: Restore Python virtual environment
|
uses: actions/setup-python@v4
|
||||||
uses: actions/cache/restore@v3.3.1
|
id: python
|
||||||
with:
|
with:
|
||||||
path: venv
|
python-version: "3.9"
|
||||||
|
|
||||||
|
- name: Cache virtualenv
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: .venv
|
||||||
# yamllint disable-line rule:line-length
|
# yamllint disable-line rule:line-length
|
||||||
key: ${{ runner.os }}-${{ env.DEFAULT_PYTHON }}-venv-${{ hashFiles('requirements.txt', 'requirements_optional.txt', 'requirements_test.txt') }}
|
key: venv-${{ steps.python.outputs.python-version }}-${{ hashFiles('requirements*.txt') }}
|
||||||
# Use per check platformio cache because checks use different parts
|
restore-keys: |
|
||||||
|
venv-${{ steps.python.outputs.python-version }}-
|
||||||
|
|
||||||
|
- name: Set up virtualenv
|
||||||
|
# yamllint disable rule:line-length
|
||||||
|
run: |
|
||||||
|
python -m venv .venv
|
||||||
|
source .venv/bin/activate
|
||||||
|
pip install -U pip
|
||||||
|
pip install -r requirements.txt -r requirements_optional.txt -r requirements_test.txt
|
||||||
|
pip install -e .
|
||||||
|
echo "$GITHUB_WORKSPACE/.venv/bin" >> $GITHUB_PATH
|
||||||
|
echo "VIRTUAL_ENV=$GITHUB_WORKSPACE/.venv" >> $GITHUB_ENV
|
||||||
|
# yamllint enable rule:line-length
|
||||||
|
|
||||||
|
# Use per check platformio cache because checks use different parts
|
||||||
- name: Cache platformio
|
- name: Cache platformio
|
||||||
uses: actions/cache@v3.3.1
|
uses: actions/cache@v3
|
||||||
with:
|
with:
|
||||||
path: ~/.platformio
|
path: ~/.platformio
|
||||||
# yamllint disable-line rule:line-length
|
# yamllint disable-line rule:line-length
|
||||||
key: platformio-${{ matrix.pio_cache_key }}-${{ hashFiles('platformio.ini') }}
|
key: platformio-${{ matrix.pio_cache_key }}-${{ hashFiles('platformio.ini') }}
|
||||||
|
if: matrix.id == 'test' || matrix.id == 'clang-tidy'
|
||||||
|
|
||||||
- name: Install clang-tidy
|
- name: Install clang tools
|
||||||
run: sudo apt-get install clang-tidy-11
|
run: |
|
||||||
|
sudo apt-get install \
|
||||||
|
clang-format-13 \
|
||||||
|
clang-tidy-11
|
||||||
|
if: matrix.id == 'clang-tidy' || matrix.id == 'clang-format'
|
||||||
|
|
||||||
- name: Register problem matchers
|
- name: Register problem matchers
|
||||||
run: |
|
run: |
|
||||||
|
echo "::add-matcher::.github/workflows/matchers/ci-custom.json"
|
||||||
|
echo "::add-matcher::.github/workflows/matchers/lint-python.json"
|
||||||
|
echo "::add-matcher::.github/workflows/matchers/python.json"
|
||||||
|
echo "::add-matcher::.github/workflows/matchers/pytest.json"
|
||||||
echo "::add-matcher::.github/workflows/matchers/gcc.json"
|
echo "::add-matcher::.github/workflows/matchers/gcc.json"
|
||||||
echo "::add-matcher::.github/workflows/matchers/clang-tidy.json"
|
echo "::add-matcher::.github/workflows/matchers/clang-tidy.json"
|
||||||
|
|
||||||
- name: Run clang-tidy
|
- name: Lint Custom
|
||||||
run: |
|
run: |
|
||||||
. venv/bin/activate
|
script/ci-custom.py
|
||||||
script/clang-tidy --all-headers --fix ${{ matrix.options }}
|
script/build_codeowners.py --check
|
||||||
|
if: matrix.id == 'ci-custom'
|
||||||
|
|
||||||
|
- name: Lint Python
|
||||||
|
run: script/lint-python -a
|
||||||
|
if: matrix.id == 'lint-python'
|
||||||
|
|
||||||
|
- run: esphome compile ${{ matrix.file }}
|
||||||
|
if: matrix.id == 'test'
|
||||||
env:
|
env:
|
||||||
# Also cache libdeps, store them in a ~/.platformio subfolder
|
# Also cache libdeps, store them in a ~/.platformio subfolder
|
||||||
PLATFORMIO_LIBDEPS_DIR: ~/.platformio/libdeps
|
PLATFORMIO_LIBDEPS_DIR: ~/.platformio/libdeps
|
||||||
|
|
||||||
|
- name: Run pytest
|
||||||
|
run: |
|
||||||
|
pytest -vv --tb=native tests
|
||||||
|
if: matrix.id == 'pytest'
|
||||||
|
|
||||||
|
# Also run git-diff-index so that the step is marked as failed on
|
||||||
|
# formatting errors, since clang-format doesn't do anything but
|
||||||
|
# change files if -i is passed.
|
||||||
|
- name: Run clang-format
|
||||||
|
run: |
|
||||||
|
script/clang-format -i
|
||||||
|
git diff-index --quiet HEAD --
|
||||||
|
if: matrix.id == 'clang-format'
|
||||||
|
|
||||||
|
- name: Run clang-tidy
|
||||||
|
run: |
|
||||||
|
script/clang-tidy --all-headers --fix ${{ matrix.options }}
|
||||||
|
if: matrix.id == 'clang-tidy'
|
||||||
|
env:
|
||||||
|
# Also cache libdeps, store them in a ~/.platformio subfolder
|
||||||
|
PLATFORMIO_LIBDEPS_DIR: ~/.platformio/libdeps
|
||||||
|
|
||||||
|
- name: Run yamllint
|
||||||
|
if: matrix.id == 'yamllint'
|
||||||
|
uses: frenck/action-yamllint@v1.4.0
|
||||||
|
|
||||||
- name: Suggested changes
|
- name: Suggested changes
|
||||||
run: script/ci-suggest-changes
|
run: script/ci-suggest-changes
|
||||||
# yamllint disable-line rule:line-length
|
# yamllint disable-line rule:line-length
|
||||||
if: always()
|
if: always() && (matrix.id == 'clang-tidy' || matrix.id == 'clang-format' || matrix.id == 'lint-python')
|
||||||
|
|
||||||
ci-status:
|
ci-status:
|
||||||
name: CI Status
|
name: CI Status
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs:
|
needs: [ci]
|
||||||
- common
|
|
||||||
- black
|
|
||||||
- ci-custom
|
|
||||||
- clang-format
|
|
||||||
- flake8
|
|
||||||
- pylint
|
|
||||||
- pytest
|
|
||||||
- pyupgrade
|
|
||||||
- yamllint
|
|
||||||
- compile-tests
|
|
||||||
- clang-tidy
|
|
||||||
if: always()
|
if: always()
|
||||||
steps:
|
steps:
|
||||||
- name: Success
|
- name: Successful deploy
|
||||||
if: ${{ !(contains(needs.*.result, 'failure')) }}
|
if: ${{ !(contains(needs.*.result, 'failure')) }}
|
||||||
run: exit 0
|
run: exit 0
|
||||||
- name: Failure
|
- name: Failing deploy
|
||||||
if: ${{ contains(needs.*.result, 'failure') }}
|
if: ${{ contains(needs.*.result, 'failure') }}
|
||||||
run: exit 1
|
run: exit 1
|
||||||
|
|||||||
6
.github/workflows/release.yml
vendored
6
.github/workflows/release.yml
vendored
@@ -49,11 +49,9 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
python-version: "3.x"
|
python-version: "3.x"
|
||||||
- name: Set up python environment
|
- name: Set up python environment
|
||||||
env:
|
|
||||||
ESPHOME_NO_VENV: 1
|
|
||||||
run: |
|
run: |
|
||||||
script/setup
|
script/setup
|
||||||
pip install twine
|
pip install setuptools wheel twine
|
||||||
- name: Build
|
- name: Build
|
||||||
run: python setup.py sdist bdist_wheel
|
run: python setup.py sdist bdist_wheel
|
||||||
- name: Upload
|
- name: Upload
|
||||||
@@ -119,7 +117,7 @@ jobs:
|
|||||||
--suffix "${{ matrix.image.suffix }}"
|
--suffix "${{ matrix.image.suffix }}"
|
||||||
|
|
||||||
- name: Build and push
|
- name: Build and push
|
||||||
uses: docker/build-push-action@v4
|
uses: docker/build-push-action@v3
|
||||||
with:
|
with:
|
||||||
context: .
|
context: .
|
||||||
file: ./docker/Dockerfile
|
file: ./docker/Dockerfile
|
||||||
|
|||||||
6
.github/workflows/stale.yml
vendored
6
.github/workflows/stale.yml
vendored
@@ -18,7 +18,7 @@ jobs:
|
|||||||
stale:
|
stale:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/stale@v8
|
- uses: actions/stale@v7
|
||||||
with:
|
with:
|
||||||
days-before-pr-stale: 90
|
days-before-pr-stale: 90
|
||||||
days-before-pr-close: 7
|
days-before-pr-close: 7
|
||||||
@@ -26,7 +26,7 @@ jobs:
|
|||||||
days-before-issue-close: -1
|
days-before-issue-close: -1
|
||||||
remove-stale-when-updated: true
|
remove-stale-when-updated: true
|
||||||
stale-pr-label: "stale"
|
stale-pr-label: "stale"
|
||||||
exempt-pr-labels: "not-stale"
|
exempt-pr-labels: "no-stale"
|
||||||
stale-pr-message: >
|
stale-pr-message: >
|
||||||
There hasn't been any activity on this pull request recently. This
|
There hasn't been any activity on this pull request recently. This
|
||||||
pull request has been automatically marked as stale because of that
|
pull request has been automatically marked as stale because of that
|
||||||
@@ -38,7 +38,7 @@ jobs:
|
|||||||
close-issues:
|
close-issues:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/stale@v8
|
- uses: actions/stale@v7
|
||||||
with:
|
with:
|
||||||
days-before-pr-stale: -1
|
days-before-pr-stale: -1
|
||||||
days-before-pr-close: -1
|
days-before-pr-close: -1
|
||||||
|
|||||||
60
.github/workflows/sync-device-classes.yml
vendored
60
.github/workflows/sync-device-classes.yml
vendored
@@ -1,60 +0,0 @@
|
|||||||
---
|
|
||||||
name: Synchronise Device Classes from Home Assistant
|
|
||||||
|
|
||||||
on:
|
|
||||||
workflow_dispatch:
|
|
||||||
schedule:
|
|
||||||
- cron: '45 6 * * *'
|
|
||||||
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
pull-requests: write
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
sync:
|
|
||||||
name: Sync Device Classes
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v3
|
|
||||||
|
|
||||||
- name: Checkout Home Assistant
|
|
||||||
uses: actions/checkout@v3
|
|
||||||
with:
|
|
||||||
repository: home-assistant/core
|
|
||||||
path: lib/home-assistant
|
|
||||||
|
|
||||||
- name: Setup Python
|
|
||||||
uses: actions/setup-python@v4
|
|
||||||
with:
|
|
||||||
python-version: 3.11
|
|
||||||
|
|
||||||
- name: Install Home Assistant
|
|
||||||
run: |
|
|
||||||
python -m pip install --upgrade pip
|
|
||||||
pip install -e lib/home-assistant
|
|
||||||
|
|
||||||
- name: Sync
|
|
||||||
run: |
|
|
||||||
python ./script/sync-device_class.py
|
|
||||||
|
|
||||||
- name: Get PR template
|
|
||||||
id: pr-template-body
|
|
||||||
run: |
|
|
||||||
body=$(cat .github/PULL_REQUEST_TEMPLATE.md)
|
|
||||||
delimiter="$(openssl rand -hex 8)"
|
|
||||||
echo "body<<$delimiter" >> $GITHUB_OUTPUT
|
|
||||||
echo "$body" >> $GITHUB_OUTPUT
|
|
||||||
echo "$delimiter" >> $GITHUB_OUTPUT
|
|
||||||
|
|
||||||
- name: Commit changes
|
|
||||||
uses: peter-evans/create-pull-request@v5
|
|
||||||
with:
|
|
||||||
commit-message: "Synchronise Device Classes from Home Assistant"
|
|
||||||
committer: esphomebot <esphome@nabucasa.com>
|
|
||||||
author: esphomebot <esphome@nabucasa.com>
|
|
||||||
branch: sync/device-classes
|
|
||||||
delete-branch: true
|
|
||||||
title: "Synchronise Device Classes from Home Assistant"
|
|
||||||
body: ${{ steps.pr-template-body.outputs.body }}
|
|
||||||
token: ${{ secrets.DEVICE_CLASS_SYNC_TOKEN }}
|
|
||||||
@@ -2,8 +2,8 @@
|
|||||||
# See https://pre-commit.com for more information
|
# See https://pre-commit.com for more information
|
||||||
# See https://pre-commit.com/hooks.html for more hooks
|
# See https://pre-commit.com/hooks.html for more hooks
|
||||||
repos:
|
repos:
|
||||||
- repo: https://github.com/psf/black
|
- repo: https://github.com/ambv/black
|
||||||
rev: 23.3.0
|
rev: 23.1.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: black
|
- id: black
|
||||||
args:
|
args:
|
||||||
@@ -27,7 +27,7 @@ repos:
|
|||||||
- --branch=release
|
- --branch=release
|
||||||
- --branch=beta
|
- --branch=beta
|
||||||
- repo: https://github.com/asottile/pyupgrade
|
- repo: https://github.com/asottile/pyupgrade
|
||||||
rev: v3.4.0
|
rev: v3.3.1
|
||||||
hooks:
|
hooks:
|
||||||
- id: pyupgrade
|
- id: pyupgrade
|
||||||
args: [--py39-plus]
|
args: [--py39-plus]
|
||||||
|
|||||||
33
.vscode/tasks.json
vendored
33
.vscode/tasks.json
vendored
@@ -2,24 +2,15 @@
|
|||||||
"version": "2.0.0",
|
"version": "2.0.0",
|
||||||
"tasks": [
|
"tasks": [
|
||||||
{
|
{
|
||||||
"label": "Run Dashboard",
|
"label": "run",
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
"command": "${command:python.interpreterPath}",
|
"command": "python3 -m esphome dashboard config/",
|
||||||
"args": [
|
|
||||||
"-m",
|
|
||||||
"esphome",
|
|
||||||
"dashboard",
|
|
||||||
"config/"
|
|
||||||
],
|
|
||||||
"problemMatcher": []
|
"problemMatcher": []
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"label": "clang-tidy",
|
"label": "clang-tidy",
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
"command": "${command:python.interpreterPath}",
|
"command": "./script/clang-tidy",
|
||||||
"args": [
|
|
||||||
"./script/clang-tidy"
|
|
||||||
],
|
|
||||||
"problemMatcher": [
|
"problemMatcher": [
|
||||||
{
|
{
|
||||||
"owner": "clang-tidy",
|
"owner": "clang-tidy",
|
||||||
@@ -36,24 +27,6 @@
|
|||||||
]
|
]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "Generate proto files",
|
|
||||||
"type": "shell",
|
|
||||||
"command": "${command:python.interpreterPath}",
|
|
||||||
"args": [
|
|
||||||
"./script/api_protobuf/api_protobuf.py"
|
|
||||||
],
|
|
||||||
"group": {
|
|
||||||
"kind": "build",
|
|
||||||
"isDefault": true
|
|
||||||
},
|
|
||||||
"presentation": {
|
|
||||||
"reveal": "never",
|
|
||||||
"close": true,
|
|
||||||
"panel": "new"
|
|
||||||
},
|
|
||||||
"problemMatcher": []
|
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
24
CODEOWNERS
24
CODEOWNERS
@@ -17,13 +17,10 @@ esphome/components/adc/* @esphome/core
|
|||||||
esphome/components/adc128s102/* @DeerMaximum
|
esphome/components/adc128s102/* @DeerMaximum
|
||||||
esphome/components/addressable_light/* @justfalter
|
esphome/components/addressable_light/* @justfalter
|
||||||
esphome/components/airthings_ble/* @jeromelaban
|
esphome/components/airthings_ble/* @jeromelaban
|
||||||
esphome/components/airthings_wave_base/* @jeromelaban @ncareau
|
|
||||||
esphome/components/airthings_wave_mini/* @ncareau
|
esphome/components/airthings_wave_mini/* @ncareau
|
||||||
esphome/components/airthings_wave_plus/* @jeromelaban
|
esphome/components/airthings_wave_plus/* @jeromelaban
|
||||||
esphome/components/alarm_control_panel/* @grahambrown11
|
|
||||||
esphome/components/am43/* @buxtronix
|
esphome/components/am43/* @buxtronix
|
||||||
esphome/components/am43/cover/* @buxtronix
|
esphome/components/am43/cover/* @buxtronix
|
||||||
esphome/components/am43/sensor/* @buxtronix
|
|
||||||
esphome/components/analog_threshold/* @ianchi
|
esphome/components/analog_threshold/* @ianchi
|
||||||
esphome/components/animation/* @syndlex
|
esphome/components/animation/* @syndlex
|
||||||
esphome/components/anova/* @buxtronix
|
esphome/components/anova/* @buxtronix
|
||||||
@@ -86,7 +83,6 @@ esphome/components/esp32_ble_server/* @jesserockz
|
|||||||
esphome/components/esp32_camera_web_server/* @ayufan
|
esphome/components/esp32_camera_web_server/* @ayufan
|
||||||
esphome/components/esp32_can/* @Sympatron
|
esphome/components/esp32_can/* @Sympatron
|
||||||
esphome/components/esp32_improv/* @jesserockz
|
esphome/components/esp32_improv/* @jesserockz
|
||||||
esphome/components/esp32_rmt_led_strip/* @jesserockz
|
|
||||||
esphome/components/esp8266/* @esphome/core
|
esphome/components/esp8266/* @esphome/core
|
||||||
esphome/components/ethernet_info/* @gtjadsonsantos
|
esphome/components/ethernet_info/* @gtjadsonsantos
|
||||||
esphome/components/exposure_notifications/* @OttoWinter
|
esphome/components/exposure_notifications/* @OttoWinter
|
||||||
@@ -98,7 +94,6 @@ esphome/components/feedback/* @ianchi
|
|||||||
esphome/components/fingerprint_grow/* @OnFreund @loongyh
|
esphome/components/fingerprint_grow/* @OnFreund @loongyh
|
||||||
esphome/components/fs3000/* @kahrendt
|
esphome/components/fs3000/* @kahrendt
|
||||||
esphome/components/globals/* @esphome/core
|
esphome/components/globals/* @esphome/core
|
||||||
esphome/components/gp8403/* @jesserockz
|
|
||||||
esphome/components/gpio/* @esphome/core
|
esphome/components/gpio/* @esphome/core
|
||||||
esphome/components/gps/* @coogle
|
esphome/components/gps/* @coogle
|
||||||
esphome/components/graph/* @synco
|
esphome/components/graph/* @synco
|
||||||
@@ -109,19 +104,13 @@ esphome/components/hbridge/fan/* @WeekendWarrior
|
|||||||
esphome/components/hbridge/light/* @DotNetDann
|
esphome/components/hbridge/light/* @DotNetDann
|
||||||
esphome/components/heatpumpir/* @rob-deutsch
|
esphome/components/heatpumpir/* @rob-deutsch
|
||||||
esphome/components/hitachi_ac424/* @sourabhjaiswal
|
esphome/components/hitachi_ac424/* @sourabhjaiswal
|
||||||
esphome/components/hm3301/* @freekode
|
|
||||||
esphome/components/homeassistant/* @OttoWinter
|
esphome/components/homeassistant/* @OttoWinter
|
||||||
esphome/components/honeywellabp/* @RubyBailey
|
esphome/components/honeywellabp/* @RubyBailey
|
||||||
esphome/components/host/* @esphome/core
|
|
||||||
esphome/components/hrxl_maxsonar_wr/* @netmikey
|
esphome/components/hrxl_maxsonar_wr/* @netmikey
|
||||||
esphome/components/hte501/* @Stock-M
|
esphome/components/hte501/* @Stock-M
|
||||||
esphome/components/hydreon_rgxx/* @functionpointer
|
esphome/components/hydreon_rgxx/* @functionpointer
|
||||||
esphome/components/hyt271/* @Philippe12
|
|
||||||
esphome/components/i2c/* @esphome/core
|
esphome/components/i2c/* @esphome/core
|
||||||
esphome/components/i2s_audio/* @jesserockz
|
esphome/components/i2s_audio/* @jesserockz
|
||||||
esphome/components/i2s_audio/media_player/* @jesserockz
|
|
||||||
esphome/components/i2s_audio/microphone/* @jesserockz
|
|
||||||
esphome/components/i2s_audio/speaker/* @jesserockz
|
|
||||||
esphome/components/ili9xxx/* @nielsnl68
|
esphome/components/ili9xxx/* @nielsnl68
|
||||||
esphome/components/improv_base/* @esphome/core
|
esphome/components/improv_base/* @esphome/core
|
||||||
esphome/components/improv_serial/* @esphome/core
|
esphome/components/improv_serial/* @esphome/core
|
||||||
@@ -147,7 +136,6 @@ esphome/components/ltr390/* @sjtrny
|
|||||||
esphome/components/matrix_keypad/* @ssieb
|
esphome/components/matrix_keypad/* @ssieb
|
||||||
esphome/components/max31865/* @DAVe3283
|
esphome/components/max31865/* @DAVe3283
|
||||||
esphome/components/max44009/* @berfenger
|
esphome/components/max44009/* @berfenger
|
||||||
esphome/components/max6956/* @looping40
|
|
||||||
esphome/components/max7219digit/* @rspaargaren
|
esphome/components/max7219digit/* @rspaargaren
|
||||||
esphome/components/max9611/* @mckaymatthew
|
esphome/components/max9611/* @mckaymatthew
|
||||||
esphome/components/mcp23008/* @jesserockz
|
esphome/components/mcp23008/* @jesserockz
|
||||||
@@ -166,14 +154,11 @@ esphome/components/mcp9808/* @k7hpn
|
|||||||
esphome/components/md5/* @esphome/core
|
esphome/components/md5/* @esphome/core
|
||||||
esphome/components/mdns/* @esphome/core
|
esphome/components/mdns/* @esphome/core
|
||||||
esphome/components/media_player/* @jesserockz
|
esphome/components/media_player/* @jesserockz
|
||||||
esphome/components/microphone/* @jesserockz
|
|
||||||
esphome/components/mics_4514/* @jesserockz
|
esphome/components/mics_4514/* @jesserockz
|
||||||
esphome/components/midea/* @dudanov
|
esphome/components/midea/* @dudanov
|
||||||
esphome/components/midea_ir/* @dudanov
|
esphome/components/midea_ir/* @dudanov
|
||||||
esphome/components/mitsubishi/* @RubyBailey
|
esphome/components/mitsubishi/* @RubyBailey
|
||||||
esphome/components/mlx90393/* @functionpointer
|
esphome/components/mlx90393/* @functionpointer
|
||||||
esphome/components/mlx90614/* @jesserockz
|
|
||||||
esphome/components/mmc5603/* @benhoff
|
|
||||||
esphome/components/modbus_controller/* @martgras
|
esphome/components/modbus_controller/* @martgras
|
||||||
esphome/components/modbus_controller/binary_sensor/* @martgras
|
esphome/components/modbus_controller/binary_sensor/* @martgras
|
||||||
esphome/components/modbus_controller/number/* @martgras
|
esphome/components/modbus_controller/number/* @martgras
|
||||||
@@ -197,7 +182,6 @@ esphome/components/nfc/* @jesserockz
|
|||||||
esphome/components/number/* @esphome/core
|
esphome/components/number/* @esphome/core
|
||||||
esphome/components/ota/* @esphome/core
|
esphome/components/ota/* @esphome/core
|
||||||
esphome/components/output/* @esphome/core
|
esphome/components/output/* @esphome/core
|
||||||
esphome/components/pca6416a/* @Mat931
|
|
||||||
esphome/components/pca9554/* @hwstar
|
esphome/components/pca9554/* @hwstar
|
||||||
esphome/components/pcf85063/* @brogon
|
esphome/components/pcf85063/* @brogon
|
||||||
esphome/components/pid/* @OttoWinter
|
esphome/components/pid/* @OttoWinter
|
||||||
@@ -223,7 +207,6 @@ esphome/components/restart/* @esphome/core
|
|||||||
esphome/components/rf_bridge/* @jesserockz
|
esphome/components/rf_bridge/* @jesserockz
|
||||||
esphome/components/rgbct/* @jesserockz
|
esphome/components/rgbct/* @jesserockz
|
||||||
esphome/components/rp2040/* @jesserockz
|
esphome/components/rp2040/* @jesserockz
|
||||||
esphome/components/rp2040_pio_led_strip/* @Papa-DMan
|
|
||||||
esphome/components/rp2040_pwm/* @jesserockz
|
esphome/components/rp2040_pwm/* @jesserockz
|
||||||
esphome/components/rtttl/* @glmnet
|
esphome/components/rtttl/* @glmnet
|
||||||
esphome/components/safe_mode/* @jsuanet @paulmonigatti
|
esphome/components/safe_mode/* @jsuanet @paulmonigatti
|
||||||
@@ -245,7 +228,7 @@ esphome/components/shutdown/* @esphome/core @jsuanet
|
|||||||
esphome/components/sigma_delta_output/* @Cat-Ion
|
esphome/components/sigma_delta_output/* @Cat-Ion
|
||||||
esphome/components/sim800l/* @glmnet
|
esphome/components/sim800l/* @glmnet
|
||||||
esphome/components/sm10bit_base/* @Cossid
|
esphome/components/sm10bit_base/* @Cossid
|
||||||
esphome/components/sm2135/* @BoukeHaarsma23 @dd32 @matika77
|
esphome/components/sm2135/* @BoukeHaarsma23
|
||||||
esphome/components/sm2235/* @Cossid
|
esphome/components/sm2235/* @Cossid
|
||||||
esphome/components/sm2335/* @Cossid
|
esphome/components/sm2335/* @Cossid
|
||||||
esphome/components/sml/* @alengwenus
|
esphome/components/sml/* @alengwenus
|
||||||
@@ -253,7 +236,6 @@ esphome/components/smt100/* @piechade
|
|||||||
esphome/components/sn74hc165/* @jesserockz
|
esphome/components/sn74hc165/* @jesserockz
|
||||||
esphome/components/socket/* @esphome/core
|
esphome/components/socket/* @esphome/core
|
||||||
esphome/components/sonoff_d1/* @anatoly-savchenkov
|
esphome/components/sonoff_d1/* @anatoly-savchenkov
|
||||||
esphome/components/speaker/* @jesserockz
|
|
||||||
esphome/components/spi/* @esphome/core
|
esphome/components/spi/* @esphome/core
|
||||||
esphome/components/sprinkler/* @kbx81
|
esphome/components/sprinkler/* @kbx81
|
||||||
esphome/components/sps30/* @martgras
|
esphome/components/sps30/* @martgras
|
||||||
@@ -279,16 +261,13 @@ esphome/components/tca9548a/* @andreashergert1984
|
|||||||
esphome/components/tcl112/* @glmnet
|
esphome/components/tcl112/* @glmnet
|
||||||
esphome/components/tee501/* @Stock-M
|
esphome/components/tee501/* @Stock-M
|
||||||
esphome/components/teleinfo/* @0hax
|
esphome/components/teleinfo/* @0hax
|
||||||
esphome/components/template/alarm_control_panel/* @grahambrown11
|
|
||||||
esphome/components/thermostat/* @kbx81
|
esphome/components/thermostat/* @kbx81
|
||||||
esphome/components/time/* @OttoWinter
|
esphome/components/time/* @OttoWinter
|
||||||
esphome/components/tlc5947/* @rnauber
|
esphome/components/tlc5947/* @rnauber
|
||||||
esphome/components/tm1621/* @Philippe12
|
esphome/components/tm1621/* @Philippe12
|
||||||
esphome/components/tm1637/* @glmnet
|
esphome/components/tm1637/* @glmnet
|
||||||
esphome/components/tm1638/* @skykingjwc
|
esphome/components/tm1638/* @skykingjwc
|
||||||
esphome/components/tm1651/* @freekode
|
|
||||||
esphome/components/tmp102/* @timsavage
|
esphome/components/tmp102/* @timsavage
|
||||||
esphome/components/tmp1075/* @sybrenstuvel
|
|
||||||
esphome/components/tmp117/* @Azimath
|
esphome/components/tmp117/* @Azimath
|
||||||
esphome/components/tof10120/* @wstrzalka
|
esphome/components/tof10120/* @wstrzalka
|
||||||
esphome/components/toshiba/* @kbx81
|
esphome/components/toshiba/* @kbx81
|
||||||
@@ -307,7 +286,6 @@ esphome/components/ufire_ise/* @pvizeli
|
|||||||
esphome/components/ultrasonic/* @OttoWinter
|
esphome/components/ultrasonic/* @OttoWinter
|
||||||
esphome/components/vbus/* @ssieb
|
esphome/components/vbus/* @ssieb
|
||||||
esphome/components/version/* @esphome/core
|
esphome/components/version/* @esphome/core
|
||||||
esphome/components/voice_assistant/* @jesserockz
|
|
||||||
esphome/components/wake_on_lan/* @willwill2will54
|
esphome/components/wake_on_lan/* @willwill2will54
|
||||||
esphome/components/web_server_base/* @OttoWinter
|
esphome/components/web_server_base/* @OttoWinter
|
||||||
esphome/components/whirlpool/* @glmnet
|
esphome/components/whirlpool/* @glmnet
|
||||||
|
|||||||
@@ -24,13 +24,10 @@ RUN \
|
|||||||
python3-setuptools=52.0.0-4 \
|
python3-setuptools=52.0.0-4 \
|
||||||
python3-pil=8.1.2+dfsg-0.3+deb11u1 \
|
python3-pil=8.1.2+dfsg-0.3+deb11u1 \
|
||||||
python3-cryptography=3.3.2-1 \
|
python3-cryptography=3.3.2-1 \
|
||||||
python3-venv=3.9.2-3 \
|
|
||||||
iputils-ping=3:20210202-1 \
|
iputils-ping=3:20210202-1 \
|
||||||
git=1:2.30.2-1+deb11u2 \
|
git=1:2.30.2-1 \
|
||||||
curl=7.74.0-1.3+deb11u7 \
|
curl=7.74.0-1.3+deb11u7 \
|
||||||
openssh-client=1:8.4p1-5+deb11u1 \
|
openssh-client=1:8.4p1-5+deb11u1 \
|
||||||
libcairo2=1.16.0-5 \
|
|
||||||
python3-cffi=1.14.5-1 \
|
|
||||||
&& rm -rf \
|
&& rm -rf \
|
||||||
/tmp/* \
|
/tmp/* \
|
||||||
/var/{cache,log}/* \
|
/var/{cache,log}/* \
|
||||||
@@ -54,7 +51,7 @@ RUN \
|
|||||||
# Ubuntu python3-pip is missing wheel
|
# Ubuntu python3-pip is missing wheel
|
||||||
pip3 install --no-cache-dir \
|
pip3 install --no-cache-dir \
|
||||||
wheel==0.37.1 \
|
wheel==0.37.1 \
|
||||||
platformio==6.1.7 \
|
platformio==6.1.6 \
|
||||||
# Change some platformio settings
|
# Change some platformio settings
|
||||||
&& platformio settings set enable_telemetry No \
|
&& platformio settings set enable_telemetry No \
|
||||||
&& platformio settings set check_platformio_interval 1000000 \
|
&& platformio settings set check_platformio_interval 1000000 \
|
||||||
@@ -62,10 +59,10 @@ RUN \
|
|||||||
|
|
||||||
|
|
||||||
# First install requirements to leverage caching when requirements don't change
|
# First install requirements to leverage caching when requirements don't change
|
||||||
COPY requirements.txt requirements_optional.txt script/platformio_install_deps.py platformio.ini /
|
COPY requirements.txt requirements_optional.txt docker/platformio_install_deps.py platformio.ini /
|
||||||
RUN \
|
RUN \
|
||||||
pip3 install --no-cache-dir -r /requirements.txt -r /requirements_optional.txt \
|
pip3 install --no-cache-dir -r /requirements.txt -r /requirements_optional.txt \
|
||||||
&& /platformio_install_deps.py /platformio.ini --libraries
|
&& /platformio_install_deps.py /platformio.ini
|
||||||
|
|
||||||
|
|
||||||
# ======================= docker-type image =======================
|
# ======================= docker-type image =======================
|
||||||
|
|||||||
30
docker/platformio_install_deps.py
Executable file
30
docker/platformio_install_deps.py
Executable file
@@ -0,0 +1,30 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
# This script is used in the docker containers to preinstall
|
||||||
|
# all platformio libraries in the global storage
|
||||||
|
|
||||||
|
import configparser
|
||||||
|
import subprocess
|
||||||
|
import sys
|
||||||
|
|
||||||
|
config = configparser.ConfigParser(inline_comment_prefixes=(';', ))
|
||||||
|
config.read(sys.argv[1])
|
||||||
|
|
||||||
|
libs = []
|
||||||
|
# Extract from every lib_deps key in all sections
|
||||||
|
for section in config.sections():
|
||||||
|
conf = config[section]
|
||||||
|
if "lib_deps" not in conf:
|
||||||
|
continue
|
||||||
|
for lib_dep in conf["lib_deps"].splitlines():
|
||||||
|
if not lib_dep:
|
||||||
|
# Empty line or comment
|
||||||
|
continue
|
||||||
|
if lib_dep.startswith("${"):
|
||||||
|
# Extending from another section
|
||||||
|
continue
|
||||||
|
if "@" not in lib_dep:
|
||||||
|
# No version pinned, this is an internal lib
|
||||||
|
continue
|
||||||
|
libs.append(lib_dep)
|
||||||
|
|
||||||
|
subprocess.check_call(['platformio', 'lib', '-g', 'install', *libs])
|
||||||
@@ -18,9 +18,6 @@ from esphome.const import (
|
|||||||
CONF_LOGGER,
|
CONF_LOGGER,
|
||||||
CONF_NAME,
|
CONF_NAME,
|
||||||
CONF_OTA,
|
CONF_OTA,
|
||||||
CONF_MQTT,
|
|
||||||
CONF_MDNS,
|
|
||||||
CONF_DISABLED,
|
|
||||||
CONF_PASSWORD,
|
CONF_PASSWORD,
|
||||||
CONF_PORT,
|
CONF_PORT,
|
||||||
CONF_ESPHOME,
|
CONF_ESPHOME,
|
||||||
@@ -45,7 +42,7 @@ from esphome.log import color, setup_log, Fore
|
|||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
def choose_prompt(options, purpose: str = None):
|
def choose_prompt(options):
|
||||||
if not options:
|
if not options:
|
||||||
raise EsphomeError(
|
raise EsphomeError(
|
||||||
"Found no valid options for upload/logging, please make sure relevant "
|
"Found no valid options for upload/logging, please make sure relevant "
|
||||||
@@ -56,9 +53,7 @@ def choose_prompt(options, purpose: str = None):
|
|||||||
if len(options) == 1:
|
if len(options) == 1:
|
||||||
return options[0][1]
|
return options[0][1]
|
||||||
|
|
||||||
safe_print(
|
safe_print("Found multiple options, please choose one:")
|
||||||
f'Found multiple options{f" for {purpose}" if purpose else ""}, please choose one:'
|
|
||||||
)
|
|
||||||
for i, (desc, _) in enumerate(options):
|
for i, (desc, _) in enumerate(options):
|
||||||
safe_print(f" [{i+1}] {desc}")
|
safe_print(f" [{i+1}] {desc}")
|
||||||
|
|
||||||
@@ -77,9 +72,7 @@ def choose_prompt(options, purpose: str = None):
|
|||||||
return options[opt - 1][1]
|
return options[opt - 1][1]
|
||||||
|
|
||||||
|
|
||||||
def choose_upload_log_host(
|
def choose_upload_log_host(default, check_default, show_ota, show_mqtt, show_api):
|
||||||
default, check_default, show_ota, show_mqtt, show_api, purpose: str = None
|
|
||||||
):
|
|
||||||
options = []
|
options = []
|
||||||
for port in get_serial_ports():
|
for port in get_serial_ports():
|
||||||
options.append((f"{port.path} ({port.description})", port.path))
|
options.append((f"{port.path} ({port.description})", port.path))
|
||||||
@@ -87,7 +80,7 @@ def choose_upload_log_host(
|
|||||||
options.append((f"Over The Air ({CORE.address})", CORE.address))
|
options.append((f"Over The Air ({CORE.address})", CORE.address))
|
||||||
if default == "OTA":
|
if default == "OTA":
|
||||||
return CORE.address
|
return CORE.address
|
||||||
if show_mqtt and CONF_MQTT in CORE.config:
|
if show_mqtt and "mqtt" in CORE.config:
|
||||||
options.append((f"MQTT ({CORE.config['mqtt'][CONF_BROKER]})", "MQTT"))
|
options.append((f"MQTT ({CORE.config['mqtt'][CONF_BROKER]})", "MQTT"))
|
||||||
if default == "OTA":
|
if default == "OTA":
|
||||||
return "MQTT"
|
return "MQTT"
|
||||||
@@ -95,7 +88,7 @@ def choose_upload_log_host(
|
|||||||
return default
|
return default
|
||||||
if check_default is not None and check_default in [opt[1] for opt in options]:
|
if check_default is not None and check_default in [opt[1] for opt in options]:
|
||||||
return check_default
|
return check_default
|
||||||
return choose_prompt(options, purpose=purpose)
|
return choose_prompt(options)
|
||||||
|
|
||||||
|
|
||||||
def get_port_type(port):
|
def get_port_type(port):
|
||||||
@@ -159,8 +152,6 @@ def run_miniterm(config, port):
|
|||||||
_LOGGER.error("Could not connect to serial port %s", port)
|
_LOGGER.error("Could not connect to serial port %s", port)
|
||||||
return 1
|
return 1
|
||||||
|
|
||||||
return 0
|
|
||||||
|
|
||||||
|
|
||||||
def wrap_to_code(name, comp):
|
def wrap_to_code(name, comp):
|
||||||
coro = coroutine(comp.to_code)
|
coro = coroutine(comp.to_code)
|
||||||
@@ -295,30 +286,19 @@ def upload_program(config, args, host):
|
|||||||
|
|
||||||
return 1 # Unknown target platform
|
return 1 # Unknown target platform
|
||||||
|
|
||||||
|
from esphome import espota2
|
||||||
|
|
||||||
if CONF_OTA not in config:
|
if CONF_OTA not in config:
|
||||||
raise EsphomeError(
|
raise EsphomeError(
|
||||||
"Cannot upload Over the Air as the config does not include the ota: "
|
"Cannot upload Over the Air as the config does not include the ota: "
|
||||||
"component"
|
"component"
|
||||||
)
|
)
|
||||||
|
|
||||||
from esphome import espota2
|
|
||||||
|
|
||||||
ota_conf = config[CONF_OTA]
|
ota_conf = config[CONF_OTA]
|
||||||
remote_port = ota_conf[CONF_PORT]
|
remote_port = ota_conf[CONF_PORT]
|
||||||
password = ota_conf.get(CONF_PASSWORD, "")
|
password = ota_conf.get(CONF_PASSWORD, "")
|
||||||
|
|
||||||
if (
|
|
||||||
get_port_type(host) == "MQTT" or config[CONF_MDNS][CONF_DISABLED]
|
|
||||||
) and CONF_MQTT in config:
|
|
||||||
from esphome import mqtt
|
|
||||||
|
|
||||||
host = mqtt.get_esphome_device_ip(
|
|
||||||
config, args.username, args.password, args.client_id
|
|
||||||
)
|
|
||||||
|
|
||||||
if getattr(args, "file", None) is not None:
|
if getattr(args, "file", None) is not None:
|
||||||
return espota2.run_ota(host, remote_port, password, args.file)
|
return espota2.run_ota(host, remote_port, password, args.file)
|
||||||
|
|
||||||
return espota2.run_ota(host, remote_port, password, CORE.firmware_bin)
|
return espota2.run_ota(host, remote_port, password, CORE.firmware_bin)
|
||||||
|
|
||||||
|
|
||||||
@@ -328,13 +308,6 @@ def show_logs(config, args, port):
|
|||||||
if get_port_type(port) == "SERIAL":
|
if get_port_type(port) == "SERIAL":
|
||||||
return run_miniterm(config, port)
|
return run_miniterm(config, port)
|
||||||
if get_port_type(port) == "NETWORK" and "api" in config:
|
if get_port_type(port) == "NETWORK" and "api" in config:
|
||||||
if config[CONF_MDNS][CONF_DISABLED] and CONF_MQTT in config:
|
|
||||||
from esphome import mqtt
|
|
||||||
|
|
||||||
port = mqtt.get_esphome_device_ip(
|
|
||||||
config, args.username, args.password, args.client_id
|
|
||||||
)
|
|
||||||
|
|
||||||
from esphome.components.api.client import run_logs
|
from esphome.components.api.client import run_logs
|
||||||
|
|
||||||
return run_logs(config, port)
|
return run_logs(config, port)
|
||||||
@@ -399,7 +372,6 @@ def command_upload(args, config):
|
|||||||
show_ota=True,
|
show_ota=True,
|
||||||
show_mqtt=False,
|
show_mqtt=False,
|
||||||
show_api=False,
|
show_api=False,
|
||||||
purpose="uploading",
|
|
||||||
)
|
)
|
||||||
exit_code = upload_program(config, args, port)
|
exit_code = upload_program(config, args, port)
|
||||||
if exit_code != 0:
|
if exit_code != 0:
|
||||||
@@ -408,15 +380,6 @@ def command_upload(args, config):
|
|||||||
return 0
|
return 0
|
||||||
|
|
||||||
|
|
||||||
def command_discover(args, config):
|
|
||||||
if "mqtt" in config:
|
|
||||||
from esphome import mqtt
|
|
||||||
|
|
||||||
return mqtt.show_discover(config, args.username, args.password, args.client_id)
|
|
||||||
|
|
||||||
raise EsphomeError("No discover method configured (mqtt)")
|
|
||||||
|
|
||||||
|
|
||||||
def command_logs(args, config):
|
def command_logs(args, config):
|
||||||
port = choose_upload_log_host(
|
port = choose_upload_log_host(
|
||||||
default=args.device,
|
default=args.device,
|
||||||
@@ -424,7 +387,6 @@ def command_logs(args, config):
|
|||||||
show_ota=False,
|
show_ota=False,
|
||||||
show_mqtt=True,
|
show_mqtt=True,
|
||||||
show_api=True,
|
show_api=True,
|
||||||
purpose="logging",
|
|
||||||
)
|
)
|
||||||
return show_logs(config, args, port)
|
return show_logs(config, args, port)
|
||||||
|
|
||||||
@@ -443,7 +405,6 @@ def command_run(args, config):
|
|||||||
show_ota=True,
|
show_ota=True,
|
||||||
show_mqtt=False,
|
show_mqtt=False,
|
||||||
show_api=True,
|
show_api=True,
|
||||||
purpose="uploading",
|
|
||||||
)
|
)
|
||||||
exit_code = upload_program(config, args, port)
|
exit_code = upload_program(config, args, port)
|
||||||
if exit_code != 0:
|
if exit_code != 0:
|
||||||
@@ -457,7 +418,6 @@ def command_run(args, config):
|
|||||||
show_ota=False,
|
show_ota=False,
|
||||||
show_mqtt=True,
|
show_mqtt=True,
|
||||||
show_api=True,
|
show_api=True,
|
||||||
purpose="logging",
|
|
||||||
)
|
)
|
||||||
return show_logs(config, args, port)
|
return show_logs(config, args, port)
|
||||||
|
|
||||||
@@ -661,7 +621,6 @@ POST_CONFIG_ACTIONS = {
|
|||||||
"clean": command_clean,
|
"clean": command_clean,
|
||||||
"idedata": command_idedata,
|
"idedata": command_idedata,
|
||||||
"rename": command_rename,
|
"rename": command_rename,
|
||||||
"discover": command_discover,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -750,15 +709,6 @@ def parse_args(argv):
|
|||||||
help="Manually specify the serial port/address to use, for example /dev/ttyUSB0.",
|
help="Manually specify the serial port/address to use, for example /dev/ttyUSB0.",
|
||||||
)
|
)
|
||||||
|
|
||||||
parser_discover = subparsers.add_parser(
|
|
||||||
"discover",
|
|
||||||
help="Validate the configuration and show all discovered devices.",
|
|
||||||
parents=[mqtt_options],
|
|
||||||
)
|
|
||||||
parser_discover.add_argument(
|
|
||||||
"configuration", help="Your YAML configuration file.", nargs=1
|
|
||||||
)
|
|
||||||
|
|
||||||
parser_run = subparsers.add_parser(
|
parser_run = subparsers.add_parser(
|
||||||
"run",
|
"run",
|
||||||
help="Validate the configuration, create a binary, upload it, and start logs.",
|
help="Validate the configuration, create a binary, upload it, and start logs.",
|
||||||
@@ -980,8 +930,6 @@ def run_esphome(argv):
|
|||||||
_LOGGER.error(e, exc_info=args.verbose)
|
_LOGGER.error(e, exc_info=args.verbose)
|
||||||
return 1
|
return 1
|
||||||
|
|
||||||
_LOGGER.info("ESPHome %s", const.__version__)
|
|
||||||
|
|
||||||
for conf_path in args.configuration:
|
for conf_path in args.configuration:
|
||||||
if any(os.path.basename(conf_path) == x for x in SECRETS_FILES):
|
if any(os.path.basename(conf_path) == x for x in SECRETS_FILES):
|
||||||
_LOGGER.warning("Skipping secrets file %s", conf_path)
|
_LOGGER.warning("Skipping secrets file %s", conf_path)
|
||||||
|
|||||||
@@ -1,118 +1 @@
|
|||||||
import esphome.codegen as cg
|
|
||||||
import esphome.config_validation as cv
|
|
||||||
from esphome import pins
|
|
||||||
from esphome.const import CONF_INPUT
|
|
||||||
|
|
||||||
from esphome.core import CORE
|
|
||||||
from esphome.components.esp32 import get_esp32_variant
|
|
||||||
from esphome.components.esp32.const import (
|
|
||||||
VARIANT_ESP32,
|
|
||||||
VARIANT_ESP32C3,
|
|
||||||
VARIANT_ESP32H2,
|
|
||||||
VARIANT_ESP32S2,
|
|
||||||
VARIANT_ESP32S3,
|
|
||||||
)
|
|
||||||
|
|
||||||
CODEOWNERS = ["@esphome/core"]
|
CODEOWNERS = ["@esphome/core"]
|
||||||
|
|
||||||
ATTENUATION_MODES = {
|
|
||||||
"0db": cg.global_ns.ADC_ATTEN_DB_0,
|
|
||||||
"2.5db": cg.global_ns.ADC_ATTEN_DB_2_5,
|
|
||||||
"6db": cg.global_ns.ADC_ATTEN_DB_6,
|
|
||||||
"11db": cg.global_ns.ADC_ATTEN_DB_11,
|
|
||||||
"auto": "auto",
|
|
||||||
}
|
|
||||||
|
|
||||||
adc1_channel_t = cg.global_ns.enum("adc1_channel_t")
|
|
||||||
|
|
||||||
# From https://github.com/espressif/esp-idf/blob/master/components/driver/include/driver/adc_common.h
|
|
||||||
# pin to adc1 channel mapping
|
|
||||||
ESP32_VARIANT_ADC1_PIN_TO_CHANNEL = {
|
|
||||||
VARIANT_ESP32: {
|
|
||||||
36: adc1_channel_t.ADC1_CHANNEL_0,
|
|
||||||
37: adc1_channel_t.ADC1_CHANNEL_1,
|
|
||||||
38: adc1_channel_t.ADC1_CHANNEL_2,
|
|
||||||
39: adc1_channel_t.ADC1_CHANNEL_3,
|
|
||||||
32: adc1_channel_t.ADC1_CHANNEL_4,
|
|
||||||
33: adc1_channel_t.ADC1_CHANNEL_5,
|
|
||||||
34: adc1_channel_t.ADC1_CHANNEL_6,
|
|
||||||
35: adc1_channel_t.ADC1_CHANNEL_7,
|
|
||||||
},
|
|
||||||
VARIANT_ESP32S2: {
|
|
||||||
1: adc1_channel_t.ADC1_CHANNEL_0,
|
|
||||||
2: adc1_channel_t.ADC1_CHANNEL_1,
|
|
||||||
3: adc1_channel_t.ADC1_CHANNEL_2,
|
|
||||||
4: adc1_channel_t.ADC1_CHANNEL_3,
|
|
||||||
5: adc1_channel_t.ADC1_CHANNEL_4,
|
|
||||||
6: adc1_channel_t.ADC1_CHANNEL_5,
|
|
||||||
7: adc1_channel_t.ADC1_CHANNEL_6,
|
|
||||||
8: adc1_channel_t.ADC1_CHANNEL_7,
|
|
||||||
9: adc1_channel_t.ADC1_CHANNEL_8,
|
|
||||||
10: adc1_channel_t.ADC1_CHANNEL_9,
|
|
||||||
},
|
|
||||||
VARIANT_ESP32S3: {
|
|
||||||
1: adc1_channel_t.ADC1_CHANNEL_0,
|
|
||||||
2: adc1_channel_t.ADC1_CHANNEL_1,
|
|
||||||
3: adc1_channel_t.ADC1_CHANNEL_2,
|
|
||||||
4: adc1_channel_t.ADC1_CHANNEL_3,
|
|
||||||
5: adc1_channel_t.ADC1_CHANNEL_4,
|
|
||||||
6: adc1_channel_t.ADC1_CHANNEL_5,
|
|
||||||
7: adc1_channel_t.ADC1_CHANNEL_6,
|
|
||||||
8: adc1_channel_t.ADC1_CHANNEL_7,
|
|
||||||
9: adc1_channel_t.ADC1_CHANNEL_8,
|
|
||||||
10: adc1_channel_t.ADC1_CHANNEL_9,
|
|
||||||
},
|
|
||||||
VARIANT_ESP32C3: {
|
|
||||||
0: adc1_channel_t.ADC1_CHANNEL_0,
|
|
||||||
1: adc1_channel_t.ADC1_CHANNEL_1,
|
|
||||||
2: adc1_channel_t.ADC1_CHANNEL_2,
|
|
||||||
3: adc1_channel_t.ADC1_CHANNEL_3,
|
|
||||||
4: adc1_channel_t.ADC1_CHANNEL_4,
|
|
||||||
},
|
|
||||||
VARIANT_ESP32H2: {
|
|
||||||
0: adc1_channel_t.ADC1_CHANNEL_0,
|
|
||||||
1: adc1_channel_t.ADC1_CHANNEL_1,
|
|
||||||
2: adc1_channel_t.ADC1_CHANNEL_2,
|
|
||||||
3: adc1_channel_t.ADC1_CHANNEL_3,
|
|
||||||
4: adc1_channel_t.ADC1_CHANNEL_4,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def validate_adc_pin(value):
|
|
||||||
if str(value).upper() == "VCC":
|
|
||||||
return cv.only_on_esp8266("VCC")
|
|
||||||
|
|
||||||
if str(value).upper() == "TEMPERATURE":
|
|
||||||
return cv.only_on_rp2040("TEMPERATURE")
|
|
||||||
|
|
||||||
if CORE.is_esp32:
|
|
||||||
value = pins.internal_gpio_input_pin_number(value)
|
|
||||||
variant = get_esp32_variant()
|
|
||||||
if variant not in ESP32_VARIANT_ADC1_PIN_TO_CHANNEL:
|
|
||||||
raise cv.Invalid(f"This ESP32 variant ({variant}) is not supported")
|
|
||||||
|
|
||||||
if value not in ESP32_VARIANT_ADC1_PIN_TO_CHANNEL[variant]:
|
|
||||||
raise cv.Invalid(f"{variant} doesn't support ADC on this pin")
|
|
||||||
return pins.internal_gpio_input_pin_schema(value)
|
|
||||||
|
|
||||||
if CORE.is_esp8266:
|
|
||||||
from esphome.components.esp8266.gpio import CONF_ANALOG
|
|
||||||
|
|
||||||
value = pins.internal_gpio_pin_number({CONF_ANALOG: True, CONF_INPUT: True})(
|
|
||||||
value
|
|
||||||
)
|
|
||||||
|
|
||||||
if value != 17: # A0
|
|
||||||
raise cv.Invalid("ESP8266: Only pin A0 (GPIO17) supports ADC.")
|
|
||||||
return pins.gpio_pin_schema(
|
|
||||||
{CONF_ANALOG: True, CONF_INPUT: True}, internal=True
|
|
||||||
)(value)
|
|
||||||
|
|
||||||
if CORE.is_rp2040:
|
|
||||||
value = pins.internal_gpio_input_pin_number(value)
|
|
||||||
if value not in (26, 27, 28, 29):
|
|
||||||
raise cv.Invalid("RP2040: Only pins 26, 27, 28 and 29 support ADC.")
|
|
||||||
return pins.internal_gpio_input_pin_schema(value)
|
|
||||||
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|||||||
@@ -1,27 +1,133 @@
|
|||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
|
from esphome import pins
|
||||||
from esphome.components import sensor, voltage_sampler
|
from esphome.components import sensor, voltage_sampler
|
||||||
from esphome.components.esp32 import get_esp32_variant
|
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
CONF_ATTENUATION,
|
CONF_ATTENUATION,
|
||||||
|
CONF_RAW,
|
||||||
CONF_ID,
|
CONF_ID,
|
||||||
|
CONF_INPUT,
|
||||||
CONF_NUMBER,
|
CONF_NUMBER,
|
||||||
CONF_PIN,
|
CONF_PIN,
|
||||||
CONF_RAW,
|
|
||||||
DEVICE_CLASS_VOLTAGE,
|
DEVICE_CLASS_VOLTAGE,
|
||||||
STATE_CLASS_MEASUREMENT,
|
STATE_CLASS_MEASUREMENT,
|
||||||
UNIT_VOLT,
|
UNIT_VOLT,
|
||||||
)
|
)
|
||||||
from esphome.core import CORE
|
from esphome.core import CORE
|
||||||
|
from esphome.components.esp32 import get_esp32_variant
|
||||||
from . import (
|
from esphome.components.esp32.const import (
|
||||||
ATTENUATION_MODES,
|
VARIANT_ESP32,
|
||||||
ESP32_VARIANT_ADC1_PIN_TO_CHANNEL,
|
VARIANT_ESP32C3,
|
||||||
validate_adc_pin,
|
VARIANT_ESP32H2,
|
||||||
|
VARIANT_ESP32S2,
|
||||||
|
VARIANT_ESP32S3,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
AUTO_LOAD = ["voltage_sampler"]
|
AUTO_LOAD = ["voltage_sampler"]
|
||||||
|
|
||||||
|
ATTENUATION_MODES = {
|
||||||
|
"0db": cg.global_ns.ADC_ATTEN_DB_0,
|
||||||
|
"2.5db": cg.global_ns.ADC_ATTEN_DB_2_5,
|
||||||
|
"6db": cg.global_ns.ADC_ATTEN_DB_6,
|
||||||
|
"11db": cg.global_ns.ADC_ATTEN_DB_11,
|
||||||
|
"auto": "auto",
|
||||||
|
}
|
||||||
|
|
||||||
|
adc1_channel_t = cg.global_ns.enum("adc1_channel_t")
|
||||||
|
|
||||||
|
# From https://github.com/espressif/esp-idf/blob/master/components/driver/include/driver/adc_common.h
|
||||||
|
# pin to adc1 channel mapping
|
||||||
|
ESP32_VARIANT_ADC1_PIN_TO_CHANNEL = {
|
||||||
|
VARIANT_ESP32: {
|
||||||
|
36: adc1_channel_t.ADC1_CHANNEL_0,
|
||||||
|
37: adc1_channel_t.ADC1_CHANNEL_1,
|
||||||
|
38: adc1_channel_t.ADC1_CHANNEL_2,
|
||||||
|
39: adc1_channel_t.ADC1_CHANNEL_3,
|
||||||
|
32: adc1_channel_t.ADC1_CHANNEL_4,
|
||||||
|
33: adc1_channel_t.ADC1_CHANNEL_5,
|
||||||
|
34: adc1_channel_t.ADC1_CHANNEL_6,
|
||||||
|
35: adc1_channel_t.ADC1_CHANNEL_7,
|
||||||
|
},
|
||||||
|
VARIANT_ESP32S2: {
|
||||||
|
1: adc1_channel_t.ADC1_CHANNEL_0,
|
||||||
|
2: adc1_channel_t.ADC1_CHANNEL_1,
|
||||||
|
3: adc1_channel_t.ADC1_CHANNEL_2,
|
||||||
|
4: adc1_channel_t.ADC1_CHANNEL_3,
|
||||||
|
5: adc1_channel_t.ADC1_CHANNEL_4,
|
||||||
|
6: adc1_channel_t.ADC1_CHANNEL_5,
|
||||||
|
7: adc1_channel_t.ADC1_CHANNEL_6,
|
||||||
|
8: adc1_channel_t.ADC1_CHANNEL_7,
|
||||||
|
9: adc1_channel_t.ADC1_CHANNEL_8,
|
||||||
|
10: adc1_channel_t.ADC1_CHANNEL_9,
|
||||||
|
},
|
||||||
|
VARIANT_ESP32S3: {
|
||||||
|
1: adc1_channel_t.ADC1_CHANNEL_0,
|
||||||
|
2: adc1_channel_t.ADC1_CHANNEL_1,
|
||||||
|
3: adc1_channel_t.ADC1_CHANNEL_2,
|
||||||
|
4: adc1_channel_t.ADC1_CHANNEL_3,
|
||||||
|
5: adc1_channel_t.ADC1_CHANNEL_4,
|
||||||
|
6: adc1_channel_t.ADC1_CHANNEL_5,
|
||||||
|
7: adc1_channel_t.ADC1_CHANNEL_6,
|
||||||
|
8: adc1_channel_t.ADC1_CHANNEL_7,
|
||||||
|
9: adc1_channel_t.ADC1_CHANNEL_8,
|
||||||
|
10: adc1_channel_t.ADC1_CHANNEL_9,
|
||||||
|
},
|
||||||
|
VARIANT_ESP32C3: {
|
||||||
|
0: adc1_channel_t.ADC1_CHANNEL_0,
|
||||||
|
1: adc1_channel_t.ADC1_CHANNEL_1,
|
||||||
|
2: adc1_channel_t.ADC1_CHANNEL_2,
|
||||||
|
3: adc1_channel_t.ADC1_CHANNEL_3,
|
||||||
|
4: adc1_channel_t.ADC1_CHANNEL_4,
|
||||||
|
},
|
||||||
|
VARIANT_ESP32H2: {
|
||||||
|
0: adc1_channel_t.ADC1_CHANNEL_0,
|
||||||
|
1: adc1_channel_t.ADC1_CHANNEL_1,
|
||||||
|
2: adc1_channel_t.ADC1_CHANNEL_2,
|
||||||
|
3: adc1_channel_t.ADC1_CHANNEL_3,
|
||||||
|
4: adc1_channel_t.ADC1_CHANNEL_4,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def validate_adc_pin(value):
|
||||||
|
if str(value).upper() == "VCC":
|
||||||
|
return cv.only_on_esp8266("VCC")
|
||||||
|
|
||||||
|
if str(value).upper() == "TEMPERATURE":
|
||||||
|
return cv.only_on_rp2040("TEMPERATURE")
|
||||||
|
|
||||||
|
if CORE.is_esp32:
|
||||||
|
value = pins.internal_gpio_input_pin_number(value)
|
||||||
|
variant = get_esp32_variant()
|
||||||
|
if variant not in ESP32_VARIANT_ADC1_PIN_TO_CHANNEL:
|
||||||
|
raise cv.Invalid(f"This ESP32 variant ({variant}) is not supported")
|
||||||
|
|
||||||
|
if value not in ESP32_VARIANT_ADC1_PIN_TO_CHANNEL[variant]:
|
||||||
|
raise cv.Invalid(f"{variant} doesn't support ADC on this pin")
|
||||||
|
return pins.internal_gpio_input_pin_schema(value)
|
||||||
|
|
||||||
|
if CORE.is_esp8266:
|
||||||
|
from esphome.components.esp8266.gpio import CONF_ANALOG
|
||||||
|
|
||||||
|
value = pins.internal_gpio_pin_number({CONF_ANALOG: True, CONF_INPUT: True})(
|
||||||
|
value
|
||||||
|
)
|
||||||
|
|
||||||
|
if value != 17: # A0
|
||||||
|
raise cv.Invalid("ESP8266: Only pin A0 (GPIO17) supports ADC.")
|
||||||
|
return pins.gpio_pin_schema(
|
||||||
|
{CONF_ANALOG: True, CONF_INPUT: True}, internal=True
|
||||||
|
)(value)
|
||||||
|
|
||||||
|
if CORE.is_rp2040:
|
||||||
|
value = pins.internal_gpio_input_pin_number(value)
|
||||||
|
if value not in (26, 27, 28, 29):
|
||||||
|
raise cv.Invalid("RP2040: Only pins 26, 27, 28 and 29 support ADC.")
|
||||||
|
return pins.internal_gpio_input_pin_schema(value)
|
||||||
|
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
def validate_config(config):
|
def validate_config(config):
|
||||||
if config[CONF_RAW] and config.get(CONF_ATTENUATION, None) == "auto":
|
if config[CONF_RAW] and config.get(CONF_ATTENUATION, None) == "auto":
|
||||||
|
|||||||
@@ -1,83 +0,0 @@
|
|||||||
import esphome.codegen as cg
|
|
||||||
import esphome.config_validation as cv
|
|
||||||
from esphome.components import sensor, ble_client
|
|
||||||
|
|
||||||
from esphome.const import (
|
|
||||||
DEVICE_CLASS_HUMIDITY,
|
|
||||||
DEVICE_CLASS_TEMPERATURE,
|
|
||||||
DEVICE_CLASS_PRESSURE,
|
|
||||||
STATE_CLASS_MEASUREMENT,
|
|
||||||
UNIT_PERCENT,
|
|
||||||
UNIT_CELSIUS,
|
|
||||||
UNIT_HECTOPASCAL,
|
|
||||||
CONF_HUMIDITY,
|
|
||||||
CONF_TVOC,
|
|
||||||
CONF_PRESSURE,
|
|
||||||
CONF_TEMPERATURE,
|
|
||||||
DEVICE_CLASS_VOLATILE_ORGANIC_COMPOUNDS_PARTS,
|
|
||||||
UNIT_PARTS_PER_BILLION,
|
|
||||||
ICON_RADIATOR,
|
|
||||||
)
|
|
||||||
|
|
||||||
CODEOWNERS = ["@ncareau", "@jeromelaban"]
|
|
||||||
|
|
||||||
DEPENDENCIES = ["ble_client"]
|
|
||||||
|
|
||||||
airthings_wave_base_ns = cg.esphome_ns.namespace("airthings_wave_base")
|
|
||||||
AirthingsWaveBase = airthings_wave_base_ns.class_(
|
|
||||||
"AirthingsWaveBase", cg.PollingComponent, ble_client.BLEClientNode
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
BASE_SCHEMA = (
|
|
||||||
sensor.SENSOR_SCHEMA.extend(
|
|
||||||
{
|
|
||||||
cv.Optional(CONF_HUMIDITY): sensor.sensor_schema(
|
|
||||||
unit_of_measurement=UNIT_PERCENT,
|
|
||||||
device_class=DEVICE_CLASS_HUMIDITY,
|
|
||||||
state_class=STATE_CLASS_MEASUREMENT,
|
|
||||||
accuracy_decimals=0,
|
|
||||||
),
|
|
||||||
cv.Optional(CONF_TEMPERATURE): sensor.sensor_schema(
|
|
||||||
unit_of_measurement=UNIT_CELSIUS,
|
|
||||||
accuracy_decimals=2,
|
|
||||||
device_class=DEVICE_CLASS_TEMPERATURE,
|
|
||||||
state_class=STATE_CLASS_MEASUREMENT,
|
|
||||||
),
|
|
||||||
cv.Optional(CONF_PRESSURE): sensor.sensor_schema(
|
|
||||||
unit_of_measurement=UNIT_HECTOPASCAL,
|
|
||||||
accuracy_decimals=1,
|
|
||||||
device_class=DEVICE_CLASS_PRESSURE,
|
|
||||||
state_class=STATE_CLASS_MEASUREMENT,
|
|
||||||
),
|
|
||||||
cv.Optional(CONF_TVOC): sensor.sensor_schema(
|
|
||||||
unit_of_measurement=UNIT_PARTS_PER_BILLION,
|
|
||||||
icon=ICON_RADIATOR,
|
|
||||||
accuracy_decimals=0,
|
|
||||||
device_class=DEVICE_CLASS_VOLATILE_ORGANIC_COMPOUNDS_PARTS,
|
|
||||||
state_class=STATE_CLASS_MEASUREMENT,
|
|
||||||
),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
.extend(cv.polling_component_schema("5min"))
|
|
||||||
.extend(ble_client.BLE_CLIENT_SCHEMA)
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
async def wave_base_to_code(var, config):
|
|
||||||
await cg.register_component(var, config)
|
|
||||||
|
|
||||||
await ble_client.register_ble_node(var, config)
|
|
||||||
|
|
||||||
if CONF_HUMIDITY in config:
|
|
||||||
sens = await sensor.new_sensor(config[CONF_HUMIDITY])
|
|
||||||
cg.add(var.set_humidity(sens))
|
|
||||||
if CONF_TEMPERATURE in config:
|
|
||||||
sens = await sensor.new_sensor(config[CONF_TEMPERATURE])
|
|
||||||
cg.add(var.set_temperature(sens))
|
|
||||||
if CONF_PRESSURE in config:
|
|
||||||
sens = await sensor.new_sensor(config[CONF_PRESSURE])
|
|
||||||
cg.add(var.set_pressure(sens))
|
|
||||||
if CONF_TVOC in config:
|
|
||||||
sens = await sensor.new_sensor(config[CONF_TVOC])
|
|
||||||
cg.add(var.set_tvoc(sens))
|
|
||||||
@@ -1,83 +0,0 @@
|
|||||||
#include "airthings_wave_base.h"
|
|
||||||
|
|
||||||
#ifdef USE_ESP32
|
|
||||||
|
|
||||||
namespace esphome {
|
|
||||||
namespace airthings_wave_base {
|
|
||||||
|
|
||||||
static const char *const TAG = "airthings_wave_base";
|
|
||||||
|
|
||||||
void AirthingsWaveBase::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
|
||||||
esp_ble_gattc_cb_param_t *param) {
|
|
||||||
switch (event) {
|
|
||||||
case ESP_GATTC_OPEN_EVT: {
|
|
||||||
if (param->open.status == ESP_GATT_OK) {
|
|
||||||
ESP_LOGI(TAG, "Connected successfully!");
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case ESP_GATTC_DISCONNECT_EVT: {
|
|
||||||
ESP_LOGW(TAG, "Disconnected!");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case ESP_GATTC_SEARCH_CMPL_EVT: {
|
|
||||||
this->handle_ = 0;
|
|
||||||
auto *chr = this->parent()->get_characteristic(this->service_uuid_, this->sensors_data_characteristic_uuid_);
|
|
||||||
if (chr == nullptr) {
|
|
||||||
ESP_LOGW(TAG, "No sensor characteristic found at service %s char %s", this->service_uuid_.to_string().c_str(),
|
|
||||||
this->sensors_data_characteristic_uuid_.to_string().c_str());
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
this->handle_ = chr->handle;
|
|
||||||
this->node_state = esp32_ble_tracker::ClientState::ESTABLISHED;
|
|
||||||
|
|
||||||
this->request_read_values_();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case ESP_GATTC_READ_CHAR_EVT: {
|
|
||||||
if (param->read.conn_id != this->parent()->get_conn_id())
|
|
||||||
break;
|
|
||||||
if (param->read.status != ESP_GATT_OK) {
|
|
||||||
ESP_LOGW(TAG, "Error reading char at handle %d, status=%d", param->read.handle, param->read.status);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (param->read.handle == this->handle_) {
|
|
||||||
this->read_sensors(param->read.value, param->read.value_len);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool AirthingsWaveBase::is_valid_voc_value_(uint16_t voc) { return 0 <= voc && voc <= 16383; }
|
|
||||||
|
|
||||||
void AirthingsWaveBase::update() {
|
|
||||||
if (this->node_state != esp32_ble_tracker::ClientState::ESTABLISHED) {
|
|
||||||
if (!this->parent()->enabled) {
|
|
||||||
ESP_LOGW(TAG, "Reconnecting to device");
|
|
||||||
this->parent()->set_enabled(true);
|
|
||||||
this->parent()->connect();
|
|
||||||
} else {
|
|
||||||
ESP_LOGW(TAG, "Connection in progress");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void AirthingsWaveBase::request_read_values_() {
|
|
||||||
auto status = esp_ble_gattc_read_char(this->parent()->get_gattc_if(), this->parent()->get_conn_id(), this->handle_,
|
|
||||||
ESP_GATT_AUTH_REQ_NONE);
|
|
||||||
if (status) {
|
|
||||||
ESP_LOGW(TAG, "Error sending read request for sensor, status=%d", status);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace airthings_wave_base
|
|
||||||
} // namespace esphome
|
|
||||||
|
|
||||||
#endif // USE_ESP32
|
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#ifdef USE_ESP32
|
|
||||||
|
|
||||||
#include <esp_gattc_api.h>
|
|
||||||
#include <algorithm>
|
|
||||||
#include <iterator>
|
|
||||||
#include "esphome/components/ble_client/ble_client.h"
|
|
||||||
#include "esphome/components/esp32_ble_tracker/esp32_ble_tracker.h"
|
|
||||||
#include "esphome/components/sensor/sensor.h"
|
|
||||||
#include "esphome/core/component.h"
|
|
||||||
#include "esphome/core/log.h"
|
|
||||||
|
|
||||||
namespace esphome {
|
|
||||||
namespace airthings_wave_base {
|
|
||||||
|
|
||||||
class AirthingsWaveBase : public PollingComponent, public ble_client::BLEClientNode {
|
|
||||||
public:
|
|
||||||
AirthingsWaveBase() = default;
|
|
||||||
|
|
||||||
void update() override;
|
|
||||||
|
|
||||||
void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
|
||||||
esp_ble_gattc_cb_param_t *param) override;
|
|
||||||
|
|
||||||
void set_temperature(sensor::Sensor *temperature) { temperature_sensor_ = temperature; }
|
|
||||||
void set_humidity(sensor::Sensor *humidity) { humidity_sensor_ = humidity; }
|
|
||||||
void set_pressure(sensor::Sensor *pressure) { pressure_sensor_ = pressure; }
|
|
||||||
void set_tvoc(sensor::Sensor *tvoc) { tvoc_sensor_ = tvoc; }
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool is_valid_voc_value_(uint16_t voc);
|
|
||||||
|
|
||||||
virtual void read_sensors(uint8_t *value, uint16_t value_len) = 0;
|
|
||||||
void request_read_values_();
|
|
||||||
|
|
||||||
sensor::Sensor *temperature_sensor_{nullptr};
|
|
||||||
sensor::Sensor *humidity_sensor_{nullptr};
|
|
||||||
sensor::Sensor *pressure_sensor_{nullptr};
|
|
||||||
sensor::Sensor *tvoc_sensor_{nullptr};
|
|
||||||
|
|
||||||
uint16_t handle_;
|
|
||||||
esp32_ble_tracker::ESPBTUUID service_uuid_;
|
|
||||||
esp32_ble_tracker::ESPBTUUID sensors_data_characteristic_uuid_;
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace airthings_wave_base
|
|
||||||
} // namespace esphome
|
|
||||||
|
|
||||||
#endif // USE_ESP32
|
|
||||||
@@ -7,47 +7,105 @@ namespace airthings_wave_mini {
|
|||||||
|
|
||||||
static const char *const TAG = "airthings_wave_mini";
|
static const char *const TAG = "airthings_wave_mini";
|
||||||
|
|
||||||
void AirthingsWaveMini::read_sensors(uint8_t *raw_value, uint16_t value_len) {
|
void AirthingsWaveMini::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
||||||
|
esp_ble_gattc_cb_param_t *param) {
|
||||||
|
switch (event) {
|
||||||
|
case ESP_GATTC_OPEN_EVT: {
|
||||||
|
if (param->open.status == ESP_GATT_OK) {
|
||||||
|
ESP_LOGI(TAG, "Connected successfully!");
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case ESP_GATTC_DISCONNECT_EVT: {
|
||||||
|
ESP_LOGW(TAG, "Disconnected!");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case ESP_GATTC_SEARCH_CMPL_EVT: {
|
||||||
|
this->handle_ = 0;
|
||||||
|
auto *chr = this->parent()->get_characteristic(service_uuid_, sensors_data_characteristic_uuid_);
|
||||||
|
if (chr == nullptr) {
|
||||||
|
ESP_LOGW(TAG, "No sensor characteristic found at service %s char %s", service_uuid_.to_string().c_str(),
|
||||||
|
sensors_data_characteristic_uuid_.to_string().c_str());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
this->handle_ = chr->handle;
|
||||||
|
this->node_state = esp32_ble_tracker::ClientState::ESTABLISHED;
|
||||||
|
|
||||||
|
request_read_values_();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case ESP_GATTC_READ_CHAR_EVT: {
|
||||||
|
if (param->read.conn_id != this->parent()->get_conn_id())
|
||||||
|
break;
|
||||||
|
if (param->read.status != ESP_GATT_OK) {
|
||||||
|
ESP_LOGW(TAG, "Error reading char at handle %d, status=%d", param->read.handle, param->read.status);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (param->read.handle == this->handle_) {
|
||||||
|
read_sensors_(param->read.value, param->read.value_len);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AirthingsWaveMini::read_sensors_(uint8_t *raw_value, uint16_t value_len) {
|
||||||
auto *value = (WaveMiniReadings *) raw_value;
|
auto *value = (WaveMiniReadings *) raw_value;
|
||||||
|
|
||||||
if (sizeof(WaveMiniReadings) <= value_len) {
|
if (sizeof(WaveMiniReadings) <= value_len) {
|
||||||
if (this->humidity_sensor_ != nullptr) {
|
this->humidity_sensor_->publish_state(value->humidity / 100.0f);
|
||||||
this->humidity_sensor_->publish_state(value->humidity / 100.0f);
|
this->pressure_sensor_->publish_state(value->pressure / 50.0f);
|
||||||
}
|
this->temperature_sensor_->publish_state(value->temperature / 100.0f - 273.15f);
|
||||||
|
if (is_valid_voc_value_(value->voc)) {
|
||||||
if (this->pressure_sensor_ != nullptr) {
|
|
||||||
this->pressure_sensor_->publish_state(value->pressure / 50.0f);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this->temperature_sensor_ != nullptr) {
|
|
||||||
this->temperature_sensor_->publish_state(value->temperature / 100.0f - 273.15f);
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((this->tvoc_sensor_ != nullptr) && this->is_valid_voc_value_(value->voc)) {
|
|
||||||
this->tvoc_sensor_->publish_state(value->voc);
|
this->tvoc_sensor_->publish_state(value->voc);
|
||||||
}
|
}
|
||||||
|
|
||||||
// This instance must not stay connected
|
// This instance must not stay connected
|
||||||
// so other clients can connect to it (e.g. the
|
// so other clients can connect to it (e.g. the
|
||||||
// mobile app).
|
// mobile app).
|
||||||
this->parent()->set_enabled(false);
|
parent()->set_enabled(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool AirthingsWaveMini::is_valid_voc_value_(uint16_t voc) { return 0 <= voc && voc <= 16383; }
|
||||||
|
|
||||||
|
void AirthingsWaveMini::update() {
|
||||||
|
if (this->node_state != esp32_ble_tracker::ClientState::ESTABLISHED) {
|
||||||
|
if (!parent()->enabled) {
|
||||||
|
ESP_LOGW(TAG, "Reconnecting to device");
|
||||||
|
parent()->set_enabled(true);
|
||||||
|
parent()->connect();
|
||||||
|
} else {
|
||||||
|
ESP_LOGW(TAG, "Connection in progress");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AirthingsWaveMini::request_read_values_() {
|
||||||
|
auto status = esp_ble_gattc_read_char(this->parent()->get_gattc_if(), this->parent()->get_conn_id(), this->handle_,
|
||||||
|
ESP_GATT_AUTH_REQ_NONE);
|
||||||
|
if (status) {
|
||||||
|
ESP_LOGW(TAG, "Error sending read request for sensor, status=%d", status);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void AirthingsWaveMini::dump_config() {
|
void AirthingsWaveMini::dump_config() {
|
||||||
// these really don't belong here, but there doesn't seem to be a
|
|
||||||
// practical way to have the base class use LOG_SENSOR and include
|
|
||||||
// the TAG from this component
|
|
||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
LOG_SENSOR(" ", "Pressure", this->pressure_sensor_);
|
LOG_SENSOR(" ", "Pressure", this->pressure_sensor_);
|
||||||
LOG_SENSOR(" ", "TVOC", this->tvoc_sensor_);
|
LOG_SENSOR(" ", "TVOC", this->tvoc_sensor_);
|
||||||
}
|
}
|
||||||
|
|
||||||
AirthingsWaveMini::AirthingsWaveMini() {
|
AirthingsWaveMini::AirthingsWaveMini()
|
||||||
this->service_uuid_ = esp32_ble_tracker::ESPBTUUID::from_raw(SERVICE_UUID);
|
: PollingComponent(10000),
|
||||||
this->sensors_data_characteristic_uuid_ = esp32_ble_tracker::ESPBTUUID::from_raw(CHARACTERISTIC_UUID);
|
service_uuid_(esp32_ble_tracker::ESPBTUUID::from_raw(SERVICE_UUID)),
|
||||||
}
|
sensors_data_characteristic_uuid_(esp32_ble_tracker::ESPBTUUID::from_raw(CHARACTERISTIC_UUID)) {}
|
||||||
|
|
||||||
} // namespace airthings_wave_mini
|
} // namespace airthings_wave_mini
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -2,7 +2,14 @@
|
|||||||
|
|
||||||
#ifdef USE_ESP32
|
#ifdef USE_ESP32
|
||||||
|
|
||||||
#include "esphome/components/airthings_wave_base/airthings_wave_base.h"
|
#include <esp_gattc_api.h>
|
||||||
|
#include <algorithm>
|
||||||
|
#include <iterator>
|
||||||
|
#include "esphome/components/ble_client/ble_client.h"
|
||||||
|
#include "esphome/components/esp32_ble_tracker/esp32_ble_tracker.h"
|
||||||
|
#include "esphome/components/sensor/sensor.h"
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace airthings_wave_mini {
|
namespace airthings_wave_mini {
|
||||||
@@ -10,14 +17,35 @@ namespace airthings_wave_mini {
|
|||||||
static const char *const SERVICE_UUID = "b42e3882-ade7-11e4-89d3-123b93f75cba";
|
static const char *const SERVICE_UUID = "b42e3882-ade7-11e4-89d3-123b93f75cba";
|
||||||
static const char *const CHARACTERISTIC_UUID = "b42e3b98-ade7-11e4-89d3-123b93f75cba";
|
static const char *const CHARACTERISTIC_UUID = "b42e3b98-ade7-11e4-89d3-123b93f75cba";
|
||||||
|
|
||||||
class AirthingsWaveMini : public airthings_wave_base::AirthingsWaveBase {
|
class AirthingsWaveMini : public PollingComponent, public ble_client::BLEClientNode {
|
||||||
public:
|
public:
|
||||||
AirthingsWaveMini();
|
AirthingsWaveMini();
|
||||||
|
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
void update() override;
|
||||||
|
|
||||||
|
void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
||||||
|
esp_ble_gattc_cb_param_t *param) override;
|
||||||
|
|
||||||
|
void set_temperature(sensor::Sensor *temperature) { temperature_sensor_ = temperature; }
|
||||||
|
void set_humidity(sensor::Sensor *humidity) { humidity_sensor_ = humidity; }
|
||||||
|
void set_pressure(sensor::Sensor *pressure) { pressure_sensor_ = pressure; }
|
||||||
|
void set_tvoc(sensor::Sensor *tvoc) { tvoc_sensor_ = tvoc; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void read_sensors(uint8_t *value, uint16_t value_len) override;
|
bool is_valid_voc_value_(uint16_t voc);
|
||||||
|
|
||||||
|
void read_sensors_(uint8_t *value, uint16_t value_len);
|
||||||
|
void request_read_values_();
|
||||||
|
|
||||||
|
sensor::Sensor *temperature_sensor_{nullptr};
|
||||||
|
sensor::Sensor *humidity_sensor_{nullptr};
|
||||||
|
sensor::Sensor *pressure_sensor_{nullptr};
|
||||||
|
sensor::Sensor *tvoc_sensor_{nullptr};
|
||||||
|
|
||||||
|
uint16_t handle_;
|
||||||
|
esp32_ble_tracker::ESPBTUUID service_uuid_;
|
||||||
|
esp32_ble_tracker::ESPBTUUID sensors_data_characteristic_uuid_;
|
||||||
|
|
||||||
struct WaveMiniReadings {
|
struct WaveMiniReadings {
|
||||||
uint16_t unused01;
|
uint16_t unused01;
|
||||||
|
|||||||
@@ -1,28 +1,82 @@
|
|||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
from esphome.components import airthings_wave_base
|
from esphome.components import sensor, ble_client
|
||||||
|
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
|
DEVICE_CLASS_HUMIDITY,
|
||||||
|
DEVICE_CLASS_TEMPERATURE,
|
||||||
|
DEVICE_CLASS_PRESSURE,
|
||||||
|
STATE_CLASS_MEASUREMENT,
|
||||||
|
UNIT_PERCENT,
|
||||||
|
UNIT_CELSIUS,
|
||||||
|
UNIT_HECTOPASCAL,
|
||||||
CONF_ID,
|
CONF_ID,
|
||||||
|
CONF_HUMIDITY,
|
||||||
|
CONF_TVOC,
|
||||||
|
CONF_PRESSURE,
|
||||||
|
CONF_TEMPERATURE,
|
||||||
|
UNIT_PARTS_PER_BILLION,
|
||||||
|
ICON_RADIATOR,
|
||||||
)
|
)
|
||||||
|
|
||||||
DEPENDENCIES = airthings_wave_base.DEPENDENCIES
|
DEPENDENCIES = ["ble_client"]
|
||||||
|
|
||||||
AUTO_LOAD = ["airthings_wave_base"]
|
|
||||||
|
|
||||||
airthings_wave_mini_ns = cg.esphome_ns.namespace("airthings_wave_mini")
|
airthings_wave_mini_ns = cg.esphome_ns.namespace("airthings_wave_mini")
|
||||||
AirthingsWaveMini = airthings_wave_mini_ns.class_(
|
AirthingsWaveMini = airthings_wave_mini_ns.class_(
|
||||||
"AirthingsWaveMini", airthings_wave_base.AirthingsWaveBase
|
"AirthingsWaveMini", cg.PollingComponent, ble_client.BLEClientNode
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
CONFIG_SCHEMA = airthings_wave_base.BASE_SCHEMA.extend(
|
CONFIG_SCHEMA = cv.All(
|
||||||
{
|
cv.Schema(
|
||||||
cv.GenerateID(): cv.declare_id(AirthingsWaveMini),
|
{
|
||||||
}
|
cv.GenerateID(): cv.declare_id(AirthingsWaveMini),
|
||||||
|
cv.Optional(CONF_HUMIDITY): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_PERCENT,
|
||||||
|
device_class=DEVICE_CLASS_HUMIDITY,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
accuracy_decimals=2,
|
||||||
|
),
|
||||||
|
cv.Optional(CONF_TEMPERATURE): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_CELSIUS,
|
||||||
|
accuracy_decimals=2,
|
||||||
|
device_class=DEVICE_CLASS_TEMPERATURE,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
cv.Optional(CONF_PRESSURE): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_HECTOPASCAL,
|
||||||
|
accuracy_decimals=2,
|
||||||
|
device_class=DEVICE_CLASS_PRESSURE,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
cv.Optional(CONF_TVOC): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_PARTS_PER_BILLION,
|
||||||
|
icon=ICON_RADIATOR,
|
||||||
|
accuracy_decimals=0,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.extend(cv.polling_component_schema("5min"))
|
||||||
|
.extend(ble_client.BLE_CLIENT_SCHEMA),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
var = cg.new_Pvariable(config[CONF_ID])
|
var = cg.new_Pvariable(config[CONF_ID])
|
||||||
await airthings_wave_base.wave_base_to_code(var, config)
|
await cg.register_component(var, config)
|
||||||
|
|
||||||
|
await ble_client.register_ble_node(var, config)
|
||||||
|
|
||||||
|
if CONF_HUMIDITY in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_HUMIDITY])
|
||||||
|
cg.add(var.set_humidity(sens))
|
||||||
|
if CONF_TEMPERATURE in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_TEMPERATURE])
|
||||||
|
cg.add(var.set_temperature(sens))
|
||||||
|
if CONF_PRESSURE in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_PRESSURE])
|
||||||
|
cg.add(var.set_pressure(sens))
|
||||||
|
if CONF_TVOC in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_TVOC])
|
||||||
|
cg.add(var.set_tvoc(sens))
|
||||||
|
|||||||
@@ -7,7 +7,55 @@ namespace airthings_wave_plus {
|
|||||||
|
|
||||||
static const char *const TAG = "airthings_wave_plus";
|
static const char *const TAG = "airthings_wave_plus";
|
||||||
|
|
||||||
void AirthingsWavePlus::read_sensors(uint8_t *raw_value, uint16_t value_len) {
|
void AirthingsWavePlus::gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
||||||
|
esp_ble_gattc_cb_param_t *param) {
|
||||||
|
switch (event) {
|
||||||
|
case ESP_GATTC_OPEN_EVT: {
|
||||||
|
if (param->open.status == ESP_GATT_OK) {
|
||||||
|
ESP_LOGI(TAG, "Connected successfully!");
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case ESP_GATTC_DISCONNECT_EVT: {
|
||||||
|
ESP_LOGW(TAG, "Disconnected!");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case ESP_GATTC_SEARCH_CMPL_EVT: {
|
||||||
|
this->handle_ = 0;
|
||||||
|
auto *chr = this->parent()->get_characteristic(service_uuid_, sensors_data_characteristic_uuid_);
|
||||||
|
if (chr == nullptr) {
|
||||||
|
ESP_LOGW(TAG, "No sensor characteristic found at service %s char %s", service_uuid_.to_string().c_str(),
|
||||||
|
sensors_data_characteristic_uuid_.to_string().c_str());
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
this->handle_ = chr->handle;
|
||||||
|
this->node_state = esp32_ble_tracker::ClientState::ESTABLISHED;
|
||||||
|
|
||||||
|
request_read_values_();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case ESP_GATTC_READ_CHAR_EVT: {
|
||||||
|
if (param->read.conn_id != this->parent()->get_conn_id())
|
||||||
|
break;
|
||||||
|
if (param->read.status != ESP_GATT_OK) {
|
||||||
|
ESP_LOGW(TAG, "Error reading char at handle %d, status=%d", param->read.handle, param->read.status);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (param->read.handle == this->handle_) {
|
||||||
|
read_sensors_(param->read.value, param->read.value_len);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AirthingsWavePlus::read_sensors_(uint8_t *raw_value, uint16_t value_len) {
|
||||||
auto *value = (WavePlusReadings *) raw_value;
|
auto *value = (WavePlusReadings *) raw_value;
|
||||||
|
|
||||||
if (sizeof(WavePlusReadings) <= value_len) {
|
if (sizeof(WavePlusReadings) <= value_len) {
|
||||||
@@ -16,38 +64,26 @@ void AirthingsWavePlus::read_sensors(uint8_t *raw_value, uint16_t value_len) {
|
|||||||
if (value->version == 1) {
|
if (value->version == 1) {
|
||||||
ESP_LOGD(TAG, "ambient light = %d", value->ambientLight);
|
ESP_LOGD(TAG, "ambient light = %d", value->ambientLight);
|
||||||
|
|
||||||
if (this->humidity_sensor_ != nullptr) {
|
this->humidity_sensor_->publish_state(value->humidity / 2.0f);
|
||||||
this->humidity_sensor_->publish_state(value->humidity / 2.0f);
|
if (is_valid_radon_value_(value->radon)) {
|
||||||
}
|
|
||||||
|
|
||||||
if ((this->radon_sensor_ != nullptr) && this->is_valid_radon_value_(value->radon)) {
|
|
||||||
this->radon_sensor_->publish_state(value->radon);
|
this->radon_sensor_->publish_state(value->radon);
|
||||||
}
|
}
|
||||||
|
if (is_valid_radon_value_(value->radon_lt)) {
|
||||||
if ((this->radon_long_term_sensor_ != nullptr) && this->is_valid_radon_value_(value->radon_lt)) {
|
|
||||||
this->radon_long_term_sensor_->publish_state(value->radon_lt);
|
this->radon_long_term_sensor_->publish_state(value->radon_lt);
|
||||||
}
|
}
|
||||||
|
this->temperature_sensor_->publish_state(value->temperature / 100.0f);
|
||||||
if (this->temperature_sensor_ != nullptr) {
|
this->pressure_sensor_->publish_state(value->pressure / 50.0f);
|
||||||
this->temperature_sensor_->publish_state(value->temperature / 100.0f);
|
if (is_valid_co2_value_(value->co2)) {
|
||||||
}
|
|
||||||
|
|
||||||
if (this->pressure_sensor_ != nullptr) {
|
|
||||||
this->pressure_sensor_->publish_state(value->pressure / 50.0f);
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((this->co2_sensor_ != nullptr) && this->is_valid_co2_value_(value->co2)) {
|
|
||||||
this->co2_sensor_->publish_state(value->co2);
|
this->co2_sensor_->publish_state(value->co2);
|
||||||
}
|
}
|
||||||
|
if (is_valid_voc_value_(value->voc)) {
|
||||||
if ((this->tvoc_sensor_ != nullptr) && this->is_valid_voc_value_(value->voc)) {
|
|
||||||
this->tvoc_sensor_->publish_state(value->voc);
|
this->tvoc_sensor_->publish_state(value->voc);
|
||||||
}
|
}
|
||||||
|
|
||||||
// This instance must not stay connected
|
// This instance must not stay connected
|
||||||
// so other clients can connect to it (e.g. the
|
// so other clients can connect to it (e.g. the
|
||||||
// mobile app).
|
// mobile app).
|
||||||
this->parent()->set_enabled(false);
|
parent()->set_enabled(false);
|
||||||
} else {
|
} else {
|
||||||
ESP_LOGE(TAG, "Invalid payload version (%d != 1, newer version or not a Wave Plus?)", value->version);
|
ESP_LOGE(TAG, "Invalid payload version (%d != 1, newer version or not a Wave Plus?)", value->version);
|
||||||
}
|
}
|
||||||
@@ -56,26 +92,44 @@ void AirthingsWavePlus::read_sensors(uint8_t *raw_value, uint16_t value_len) {
|
|||||||
|
|
||||||
bool AirthingsWavePlus::is_valid_radon_value_(uint16_t radon) { return 0 <= radon && radon <= 16383; }
|
bool AirthingsWavePlus::is_valid_radon_value_(uint16_t radon) { return 0 <= radon && radon <= 16383; }
|
||||||
|
|
||||||
|
bool AirthingsWavePlus::is_valid_voc_value_(uint16_t voc) { return 0 <= voc && voc <= 16383; }
|
||||||
|
|
||||||
bool AirthingsWavePlus::is_valid_co2_value_(uint16_t co2) { return 0 <= co2 && co2 <= 16383; }
|
bool AirthingsWavePlus::is_valid_co2_value_(uint16_t co2) { return 0 <= co2 && co2 <= 16383; }
|
||||||
|
|
||||||
void AirthingsWavePlus::dump_config() {
|
void AirthingsWavePlus::update() {
|
||||||
// these really don't belong here, but there doesn't seem to be a
|
if (this->node_state != esp32_ble_tracker::ClientState::ESTABLISHED) {
|
||||||
// practical way to have the base class use LOG_SENSOR and include
|
if (!parent()->enabled) {
|
||||||
// the TAG from this component
|
ESP_LOGW(TAG, "Reconnecting to device");
|
||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
parent()->set_enabled(true);
|
||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
parent()->connect();
|
||||||
LOG_SENSOR(" ", "Pressure", this->pressure_sensor_);
|
} else {
|
||||||
LOG_SENSOR(" ", "TVOC", this->tvoc_sensor_);
|
ESP_LOGW(TAG, "Connection in progress");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AirthingsWavePlus::request_read_values_() {
|
||||||
|
auto status = esp_ble_gattc_read_char(this->parent()->get_gattc_if(), this->parent()->get_conn_id(), this->handle_,
|
||||||
|
ESP_GATT_AUTH_REQ_NONE);
|
||||||
|
if (status) {
|
||||||
|
ESP_LOGW(TAG, "Error sending read request for sensor, status=%d", status);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void AirthingsWavePlus::dump_config() {
|
||||||
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
LOG_SENSOR(" ", "Radon", this->radon_sensor_);
|
LOG_SENSOR(" ", "Radon", this->radon_sensor_);
|
||||||
LOG_SENSOR(" ", "Radon Long Term", this->radon_long_term_sensor_);
|
LOG_SENSOR(" ", "Radon Long Term", this->radon_long_term_sensor_);
|
||||||
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
|
LOG_SENSOR(" ", "Pressure", this->pressure_sensor_);
|
||||||
LOG_SENSOR(" ", "CO2", this->co2_sensor_);
|
LOG_SENSOR(" ", "CO2", this->co2_sensor_);
|
||||||
|
LOG_SENSOR(" ", "TVOC", this->tvoc_sensor_);
|
||||||
}
|
}
|
||||||
|
|
||||||
AirthingsWavePlus::AirthingsWavePlus() {
|
AirthingsWavePlus::AirthingsWavePlus()
|
||||||
this->service_uuid_ = esp32_ble_tracker::ESPBTUUID::from_raw(SERVICE_UUID);
|
: PollingComponent(10000),
|
||||||
this->sensors_data_characteristic_uuid_ = esp32_ble_tracker::ESPBTUUID::from_raw(CHARACTERISTIC_UUID);
|
service_uuid_(esp32_ble_tracker::ESPBTUUID::from_raw(SERVICE_UUID)),
|
||||||
}
|
sensors_data_characteristic_uuid_(esp32_ble_tracker::ESPBTUUID::from_raw(CHARACTERISTIC_UUID)) {}
|
||||||
|
|
||||||
} // namespace airthings_wave_plus
|
} // namespace airthings_wave_plus
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -2,7 +2,14 @@
|
|||||||
|
|
||||||
#ifdef USE_ESP32
|
#ifdef USE_ESP32
|
||||||
|
|
||||||
#include "esphome/components/airthings_wave_base/airthings_wave_base.h"
|
#include <esp_gattc_api.h>
|
||||||
|
#include <algorithm>
|
||||||
|
#include <iterator>
|
||||||
|
#include "esphome/components/ble_client/ble_client.h"
|
||||||
|
#include "esphome/components/esp32_ble_tracker/esp32_ble_tracker.h"
|
||||||
|
#include "esphome/components/sensor/sensor.h"
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace airthings_wave_plus {
|
namespace airthings_wave_plus {
|
||||||
@@ -10,25 +17,43 @@ namespace airthings_wave_plus {
|
|||||||
static const char *const SERVICE_UUID = "b42e1c08-ade7-11e4-89d3-123b93f75cba";
|
static const char *const SERVICE_UUID = "b42e1c08-ade7-11e4-89d3-123b93f75cba";
|
||||||
static const char *const CHARACTERISTIC_UUID = "b42e2a68-ade7-11e4-89d3-123b93f75cba";
|
static const char *const CHARACTERISTIC_UUID = "b42e2a68-ade7-11e4-89d3-123b93f75cba";
|
||||||
|
|
||||||
class AirthingsWavePlus : public airthings_wave_base::AirthingsWaveBase {
|
class AirthingsWavePlus : public PollingComponent, public ble_client::BLEClientNode {
|
||||||
public:
|
public:
|
||||||
AirthingsWavePlus();
|
AirthingsWavePlus();
|
||||||
|
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
void update() override;
|
||||||
|
|
||||||
|
void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
||||||
|
esp_ble_gattc_cb_param_t *param) override;
|
||||||
|
|
||||||
|
void set_temperature(sensor::Sensor *temperature) { temperature_sensor_ = temperature; }
|
||||||
void set_radon(sensor::Sensor *radon) { radon_sensor_ = radon; }
|
void set_radon(sensor::Sensor *radon) { radon_sensor_ = radon; }
|
||||||
void set_radon_long_term(sensor::Sensor *radon_long_term) { radon_long_term_sensor_ = radon_long_term; }
|
void set_radon_long_term(sensor::Sensor *radon_long_term) { radon_long_term_sensor_ = radon_long_term; }
|
||||||
|
void set_humidity(sensor::Sensor *humidity) { humidity_sensor_ = humidity; }
|
||||||
|
void set_pressure(sensor::Sensor *pressure) { pressure_sensor_ = pressure; }
|
||||||
void set_co2(sensor::Sensor *co2) { co2_sensor_ = co2; }
|
void set_co2(sensor::Sensor *co2) { co2_sensor_ = co2; }
|
||||||
|
void set_tvoc(sensor::Sensor *tvoc) { tvoc_sensor_ = tvoc; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
bool is_valid_radon_value_(uint16_t radon);
|
bool is_valid_radon_value_(uint16_t radon);
|
||||||
|
bool is_valid_voc_value_(uint16_t voc);
|
||||||
bool is_valid_co2_value_(uint16_t co2);
|
bool is_valid_co2_value_(uint16_t co2);
|
||||||
|
|
||||||
void read_sensors(uint8_t *value, uint16_t value_len) override;
|
void read_sensors_(uint8_t *value, uint16_t value_len);
|
||||||
|
void request_read_values_();
|
||||||
|
|
||||||
|
sensor::Sensor *temperature_sensor_{nullptr};
|
||||||
sensor::Sensor *radon_sensor_{nullptr};
|
sensor::Sensor *radon_sensor_{nullptr};
|
||||||
sensor::Sensor *radon_long_term_sensor_{nullptr};
|
sensor::Sensor *radon_long_term_sensor_{nullptr};
|
||||||
|
sensor::Sensor *humidity_sensor_{nullptr};
|
||||||
|
sensor::Sensor *pressure_sensor_{nullptr};
|
||||||
sensor::Sensor *co2_sensor_{nullptr};
|
sensor::Sensor *co2_sensor_{nullptr};
|
||||||
|
sensor::Sensor *tvoc_sensor_{nullptr};
|
||||||
|
|
||||||
|
uint16_t handle_;
|
||||||
|
esp32_ble_tracker::ESPBTUUID service_uuid_;
|
||||||
|
esp32_ble_tracker::ESPBTUUID sensors_data_characteristic_uuid_;
|
||||||
|
|
||||||
struct WavePlusReadings {
|
struct WavePlusReadings {
|
||||||
uint8_t version;
|
uint8_t version;
|
||||||
|
|||||||
@@ -1,64 +1,116 @@
|
|||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
from esphome.components import sensor, airthings_wave_base
|
from esphome.components import sensor, ble_client
|
||||||
|
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
DEVICE_CLASS_CARBON_DIOXIDE,
|
DEVICE_CLASS_CARBON_DIOXIDE,
|
||||||
|
DEVICE_CLASS_HUMIDITY,
|
||||||
|
DEVICE_CLASS_TEMPERATURE,
|
||||||
|
DEVICE_CLASS_PRESSURE,
|
||||||
STATE_CLASS_MEASUREMENT,
|
STATE_CLASS_MEASUREMENT,
|
||||||
|
UNIT_PERCENT,
|
||||||
|
UNIT_CELSIUS,
|
||||||
|
UNIT_HECTOPASCAL,
|
||||||
ICON_RADIOACTIVE,
|
ICON_RADIOACTIVE,
|
||||||
CONF_ID,
|
CONF_ID,
|
||||||
CONF_RADON,
|
CONF_RADON,
|
||||||
CONF_RADON_LONG_TERM,
|
CONF_RADON_LONG_TERM,
|
||||||
|
CONF_HUMIDITY,
|
||||||
|
CONF_TVOC,
|
||||||
CONF_CO2,
|
CONF_CO2,
|
||||||
|
CONF_PRESSURE,
|
||||||
|
CONF_TEMPERATURE,
|
||||||
UNIT_BECQUEREL_PER_CUBIC_METER,
|
UNIT_BECQUEREL_PER_CUBIC_METER,
|
||||||
UNIT_PARTS_PER_MILLION,
|
UNIT_PARTS_PER_MILLION,
|
||||||
|
UNIT_PARTS_PER_BILLION,
|
||||||
|
ICON_RADIATOR,
|
||||||
)
|
)
|
||||||
|
|
||||||
DEPENDENCIES = airthings_wave_base.DEPENDENCIES
|
DEPENDENCIES = ["ble_client"]
|
||||||
|
|
||||||
AUTO_LOAD = ["airthings_wave_base"]
|
|
||||||
|
|
||||||
airthings_wave_plus_ns = cg.esphome_ns.namespace("airthings_wave_plus")
|
airthings_wave_plus_ns = cg.esphome_ns.namespace("airthings_wave_plus")
|
||||||
AirthingsWavePlus = airthings_wave_plus_ns.class_(
|
AirthingsWavePlus = airthings_wave_plus_ns.class_(
|
||||||
"AirthingsWavePlus", airthings_wave_base.AirthingsWaveBase
|
"AirthingsWavePlus", cg.PollingComponent, ble_client.BLEClientNode
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
CONFIG_SCHEMA = airthings_wave_base.BASE_SCHEMA.extend(
|
CONFIG_SCHEMA = cv.All(
|
||||||
{
|
cv.Schema(
|
||||||
cv.GenerateID(): cv.declare_id(AirthingsWavePlus),
|
{
|
||||||
cv.Optional(CONF_RADON): sensor.sensor_schema(
|
cv.GenerateID(): cv.declare_id(AirthingsWavePlus),
|
||||||
unit_of_measurement=UNIT_BECQUEREL_PER_CUBIC_METER,
|
cv.Optional(CONF_HUMIDITY): sensor.sensor_schema(
|
||||||
icon=ICON_RADIOACTIVE,
|
unit_of_measurement=UNIT_PERCENT,
|
||||||
accuracy_decimals=0,
|
device_class=DEVICE_CLASS_HUMIDITY,
|
||||||
state_class=STATE_CLASS_MEASUREMENT,
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
),
|
accuracy_decimals=0,
|
||||||
cv.Optional(CONF_RADON_LONG_TERM): sensor.sensor_schema(
|
),
|
||||||
unit_of_measurement=UNIT_BECQUEREL_PER_CUBIC_METER,
|
cv.Optional(CONF_RADON): sensor.sensor_schema(
|
||||||
icon=ICON_RADIOACTIVE,
|
unit_of_measurement=UNIT_BECQUEREL_PER_CUBIC_METER,
|
||||||
accuracy_decimals=0,
|
icon=ICON_RADIOACTIVE,
|
||||||
state_class=STATE_CLASS_MEASUREMENT,
|
accuracy_decimals=0,
|
||||||
),
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
cv.Optional(CONF_CO2): sensor.sensor_schema(
|
),
|
||||||
unit_of_measurement=UNIT_PARTS_PER_MILLION,
|
cv.Optional(CONF_RADON_LONG_TERM): sensor.sensor_schema(
|
||||||
accuracy_decimals=0,
|
unit_of_measurement=UNIT_BECQUEREL_PER_CUBIC_METER,
|
||||||
device_class=DEVICE_CLASS_CARBON_DIOXIDE,
|
icon=ICON_RADIOACTIVE,
|
||||||
state_class=STATE_CLASS_MEASUREMENT,
|
accuracy_decimals=0,
|
||||||
),
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
}
|
),
|
||||||
|
cv.Optional(CONF_TEMPERATURE): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_CELSIUS,
|
||||||
|
accuracy_decimals=2,
|
||||||
|
device_class=DEVICE_CLASS_TEMPERATURE,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
cv.Optional(CONF_PRESSURE): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_HECTOPASCAL,
|
||||||
|
accuracy_decimals=1,
|
||||||
|
device_class=DEVICE_CLASS_PRESSURE,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
cv.Optional(CONF_CO2): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_PARTS_PER_MILLION,
|
||||||
|
accuracy_decimals=0,
|
||||||
|
device_class=DEVICE_CLASS_CARBON_DIOXIDE,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
cv.Optional(CONF_TVOC): sensor.sensor_schema(
|
||||||
|
unit_of_measurement=UNIT_PARTS_PER_BILLION,
|
||||||
|
icon=ICON_RADIATOR,
|
||||||
|
accuracy_decimals=0,
|
||||||
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
|
),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.extend(cv.polling_component_schema("5min"))
|
||||||
|
.extend(ble_client.BLE_CLIENT_SCHEMA),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
var = cg.new_Pvariable(config[CONF_ID])
|
var = cg.new_Pvariable(config[CONF_ID])
|
||||||
await airthings_wave_base.wave_base_to_code(var, config)
|
await cg.register_component(var, config)
|
||||||
|
|
||||||
|
await ble_client.register_ble_node(var, config)
|
||||||
|
|
||||||
|
if CONF_HUMIDITY in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_HUMIDITY])
|
||||||
|
cg.add(var.set_humidity(sens))
|
||||||
if CONF_RADON in config:
|
if CONF_RADON in config:
|
||||||
sens = await sensor.new_sensor(config[CONF_RADON])
|
sens = await sensor.new_sensor(config[CONF_RADON])
|
||||||
cg.add(var.set_radon(sens))
|
cg.add(var.set_radon(sens))
|
||||||
if CONF_RADON_LONG_TERM in config:
|
if CONF_RADON_LONG_TERM in config:
|
||||||
sens = await sensor.new_sensor(config[CONF_RADON_LONG_TERM])
|
sens = await sensor.new_sensor(config[CONF_RADON_LONG_TERM])
|
||||||
cg.add(var.set_radon_long_term(sens))
|
cg.add(var.set_radon_long_term(sens))
|
||||||
|
if CONF_TEMPERATURE in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_TEMPERATURE])
|
||||||
|
cg.add(var.set_temperature(sens))
|
||||||
|
if CONF_PRESSURE in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_PRESSURE])
|
||||||
|
cg.add(var.set_pressure(sens))
|
||||||
if CONF_CO2 in config:
|
if CONF_CO2 in config:
|
||||||
sens = await sensor.new_sensor(config[CONF_CO2])
|
sens = await sensor.new_sensor(config[CONF_CO2])
|
||||||
cg.add(var.set_co2(sens))
|
cg.add(var.set_co2(sens))
|
||||||
|
if CONF_TVOC in config:
|
||||||
|
sens = await sensor.new_sensor(config[CONF_TVOC])
|
||||||
|
cg.add(var.set_tvoc(sens))
|
||||||
|
|||||||
@@ -1,165 +0,0 @@
|
|||||||
import esphome.codegen as cg
|
|
||||||
import esphome.config_validation as cv
|
|
||||||
from esphome import automation
|
|
||||||
from esphome.automation import maybe_simple_id
|
|
||||||
from esphome.core import CORE, coroutine_with_priority
|
|
||||||
from esphome.const import (
|
|
||||||
CONF_ID,
|
|
||||||
CONF_ON_STATE,
|
|
||||||
CONF_TRIGGER_ID,
|
|
||||||
CONF_CODE,
|
|
||||||
)
|
|
||||||
from esphome.cpp_helpers import setup_entity
|
|
||||||
|
|
||||||
CODEOWNERS = ["@grahambrown11"]
|
|
||||||
IS_PLATFORM_COMPONENT = True
|
|
||||||
|
|
||||||
CONF_ON_TRIGGERED = "on_triggered"
|
|
||||||
CONF_ON_CLEARED = "on_cleared"
|
|
||||||
|
|
||||||
alarm_control_panel_ns = cg.esphome_ns.namespace("alarm_control_panel")
|
|
||||||
AlarmControlPanel = alarm_control_panel_ns.class_("AlarmControlPanel", cg.EntityBase)
|
|
||||||
|
|
||||||
StateTrigger = alarm_control_panel_ns.class_(
|
|
||||||
"StateTrigger", automation.Trigger.template()
|
|
||||||
)
|
|
||||||
TriggeredTrigger = alarm_control_panel_ns.class_(
|
|
||||||
"TriggeredTrigger", automation.Trigger.template()
|
|
||||||
)
|
|
||||||
ClearedTrigger = alarm_control_panel_ns.class_(
|
|
||||||
"ClearedTrigger", automation.Trigger.template()
|
|
||||||
)
|
|
||||||
ArmAwayAction = alarm_control_panel_ns.class_("ArmAwayAction", automation.Action)
|
|
||||||
ArmHomeAction = alarm_control_panel_ns.class_("ArmHomeAction", automation.Action)
|
|
||||||
DisarmAction = alarm_control_panel_ns.class_("DisarmAction", automation.Action)
|
|
||||||
PendingAction = alarm_control_panel_ns.class_("PendingAction", automation.Action)
|
|
||||||
TriggeredAction = alarm_control_panel_ns.class_("TriggeredAction", automation.Action)
|
|
||||||
AlarmControlPanelCondition = alarm_control_panel_ns.class_(
|
|
||||||
"AlarmControlPanelCondition", automation.Condition
|
|
||||||
)
|
|
||||||
|
|
||||||
ALARM_CONTROL_PANEL_SCHEMA = cv.ENTITY_BASE_SCHEMA.extend(
|
|
||||||
{
|
|
||||||
cv.GenerateID(): cv.declare_id(AlarmControlPanel),
|
|
||||||
cv.Optional(CONF_ON_STATE): automation.validate_automation(
|
|
||||||
{
|
|
||||||
cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(StateTrigger),
|
|
||||||
}
|
|
||||||
),
|
|
||||||
cv.Optional(CONF_ON_TRIGGERED): automation.validate_automation(
|
|
||||||
{
|
|
||||||
cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(TriggeredTrigger),
|
|
||||||
}
|
|
||||||
),
|
|
||||||
cv.Optional(CONF_ON_CLEARED): automation.validate_automation(
|
|
||||||
{
|
|
||||||
cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(ClearedTrigger),
|
|
||||||
}
|
|
||||||
),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
ALARM_CONTROL_PANEL_ACTION_SCHEMA = maybe_simple_id(
|
|
||||||
{
|
|
||||||
cv.GenerateID(): cv.use_id(AlarmControlPanel),
|
|
||||||
cv.Optional(CONF_CODE): cv.templatable(cv.string),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
ALARM_CONTROL_PANEL_CONDITION_SCHEMA = maybe_simple_id(
|
|
||||||
{
|
|
||||||
cv.GenerateID(): cv.use_id(AlarmControlPanel),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
async def setup_alarm_control_panel_core_(var, config):
|
|
||||||
await setup_entity(var, config)
|
|
||||||
for conf in config.get(CONF_ON_STATE, []):
|
|
||||||
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
|
||||||
await automation.build_automation(trigger, [], conf)
|
|
||||||
for conf in config.get(CONF_ON_TRIGGERED, []):
|
|
||||||
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
|
||||||
await automation.build_automation(trigger, [], conf)
|
|
||||||
for conf in config.get(CONF_ON_CLEARED, []):
|
|
||||||
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
|
||||||
await automation.build_automation(trigger, [], conf)
|
|
||||||
|
|
||||||
|
|
||||||
async def register_alarm_control_panel(var, config):
|
|
||||||
if not CORE.has_id(config[CONF_ID]):
|
|
||||||
var = cg.Pvariable(config[CONF_ID], var)
|
|
||||||
cg.add(cg.App.register_alarm_control_panel(var))
|
|
||||||
await setup_alarm_control_panel_core_(var, config)
|
|
||||||
|
|
||||||
|
|
||||||
@automation.register_action(
|
|
||||||
"alarm_control_panel.arm_away", ArmAwayAction, ALARM_CONTROL_PANEL_ACTION_SCHEMA
|
|
||||||
)
|
|
||||||
async def alarm_action_arm_away_to_code(config, action_id, template_arg, args):
|
|
||||||
paren = await cg.get_variable(config[CONF_ID])
|
|
||||||
var = cg.new_Pvariable(action_id, template_arg, paren)
|
|
||||||
if CONF_CODE in config:
|
|
||||||
templatable_ = await cg.templatable(config[CONF_CODE], args, cg.std_string)
|
|
||||||
cg.add(var.set_code(templatable_))
|
|
||||||
return var
|
|
||||||
|
|
||||||
|
|
||||||
@automation.register_action(
|
|
||||||
"alarm_control_panel.arm_home", ArmHomeAction, ALARM_CONTROL_PANEL_ACTION_SCHEMA
|
|
||||||
)
|
|
||||||
async def alarm_action_arm_home_to_code(config, action_id, template_arg, args):
|
|
||||||
paren = await cg.get_variable(config[CONF_ID])
|
|
||||||
var = cg.new_Pvariable(action_id, template_arg, paren)
|
|
||||||
if CONF_CODE in config:
|
|
||||||
templatable_ = await cg.templatable(config[CONF_CODE], args, cg.std_string)
|
|
||||||
cg.add(var.set_code(templatable_))
|
|
||||||
return var
|
|
||||||
|
|
||||||
|
|
||||||
@automation.register_action(
|
|
||||||
"alarm_control_panel.disarm", DisarmAction, ALARM_CONTROL_PANEL_ACTION_SCHEMA
|
|
||||||
)
|
|
||||||
async def alarm_action_disarm_to_code(config, action_id, template_arg, args):
|
|
||||||
paren = await cg.get_variable(config[CONF_ID])
|
|
||||||
var = cg.new_Pvariable(action_id, template_arg, paren)
|
|
||||||
if CONF_CODE in config:
|
|
||||||
templatable_ = await cg.templatable(config[CONF_CODE], args, cg.std_string)
|
|
||||||
cg.add(var.set_code(templatable_))
|
|
||||||
return var
|
|
||||||
|
|
||||||
|
|
||||||
@automation.register_action(
|
|
||||||
"alarm_control_panel.pending", PendingAction, ALARM_CONTROL_PANEL_ACTION_SCHEMA
|
|
||||||
)
|
|
||||||
async def alarm_action_pending_to_code(config, action_id, template_arg, args):
|
|
||||||
paren = await cg.get_variable(config[CONF_ID])
|
|
||||||
var = cg.new_Pvariable(action_id, template_arg, paren)
|
|
||||||
return var
|
|
||||||
|
|
||||||
|
|
||||||
@automation.register_action(
|
|
||||||
"alarm_control_panel.triggered", TriggeredAction, ALARM_CONTROL_PANEL_ACTION_SCHEMA
|
|
||||||
)
|
|
||||||
async def alarm_action_trigger_to_code(config, action_id, template_arg, args):
|
|
||||||
paren = await cg.get_variable(config[CONF_ID])
|
|
||||||
var = cg.new_Pvariable(action_id, template_arg, paren)
|
|
||||||
return var
|
|
||||||
|
|
||||||
|
|
||||||
@automation.register_condition(
|
|
||||||
"alarm_control_panel.is_armed",
|
|
||||||
AlarmControlPanelCondition,
|
|
||||||
ALARM_CONTROL_PANEL_CONDITION_SCHEMA,
|
|
||||||
)
|
|
||||||
async def alarm_control_panel_is_armed_to_code(
|
|
||||||
config, condition_id, template_arg, args
|
|
||||||
):
|
|
||||||
paren = await cg.get_variable(config[CONF_ID])
|
|
||||||
return cg.new_Pvariable(condition_id, template_arg, paren)
|
|
||||||
|
|
||||||
|
|
||||||
@coroutine_with_priority(100.0)
|
|
||||||
async def to_code(config):
|
|
||||||
cg.add_global(alarm_control_panel_ns.using)
|
|
||||||
cg.add_define("USE_ALARM_CONTROL_PANEL")
|
|
||||||
@@ -1,111 +0,0 @@
|
|||||||
#include <utility>
|
|
||||||
|
|
||||||
#include "alarm_control_panel.h"
|
|
||||||
|
|
||||||
#include "esphome/core/application.h"
|
|
||||||
#include "esphome/core/helpers.h"
|
|
||||||
#include "esphome/core/log.h"
|
|
||||||
|
|
||||||
namespace esphome {
|
|
||||||
namespace alarm_control_panel {
|
|
||||||
|
|
||||||
static const char *const TAG = "alarm_control_panel";
|
|
||||||
|
|
||||||
AlarmControlPanelCall AlarmControlPanel::make_call() { return AlarmControlPanelCall(this); }
|
|
||||||
|
|
||||||
bool AlarmControlPanel::is_state_armed(AlarmControlPanelState state) {
|
|
||||||
switch (state) {
|
|
||||||
case ACP_STATE_ARMED_AWAY:
|
|
||||||
case ACP_STATE_ARMED_HOME:
|
|
||||||
case ACP_STATE_ARMED_NIGHT:
|
|
||||||
case ACP_STATE_ARMED_VACATION:
|
|
||||||
case ACP_STATE_ARMED_CUSTOM_BYPASS:
|
|
||||||
return true;
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
void AlarmControlPanel::publish_state(AlarmControlPanelState state) {
|
|
||||||
this->last_update_ = millis();
|
|
||||||
if (state != this->current_state_) {
|
|
||||||
auto prev_state = this->current_state_;
|
|
||||||
ESP_LOGD(TAG, "Set state to: %s, previous: %s", LOG_STR_ARG(alarm_control_panel_state_to_string(state)),
|
|
||||||
LOG_STR_ARG(alarm_control_panel_state_to_string(prev_state)));
|
|
||||||
this->current_state_ = state;
|
|
||||||
this->state_callback_.call();
|
|
||||||
if (state == ACP_STATE_TRIGGERED) {
|
|
||||||
this->triggered_callback_.call();
|
|
||||||
}
|
|
||||||
if (prev_state == ACP_STATE_TRIGGERED) {
|
|
||||||
this->cleared_callback_.call();
|
|
||||||
}
|
|
||||||
if (state == this->desired_state_) {
|
|
||||||
// only store when in the desired state
|
|
||||||
this->pref_.save(&state);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void AlarmControlPanel::add_on_state_callback(std::function<void()> &&callback) {
|
|
||||||
this->state_callback_.add(std::move(callback));
|
|
||||||
}
|
|
||||||
|
|
||||||
void AlarmControlPanel::add_on_triggered_callback(std::function<void()> &&callback) {
|
|
||||||
this->triggered_callback_.add(std::move(callback));
|
|
||||||
}
|
|
||||||
|
|
||||||
void AlarmControlPanel::add_on_cleared_callback(std::function<void()> &&callback) {
|
|
||||||
this->cleared_callback_.add(std::move(callback));
|
|
||||||
}
|
|
||||||
|
|
||||||
void AlarmControlPanel::arm_away(optional<std::string> code) {
|
|
||||||
auto call = this->make_call();
|
|
||||||
call.arm_away();
|
|
||||||
if (code.has_value())
|
|
||||||
call.set_code(code.value());
|
|
||||||
call.perform();
|
|
||||||
}
|
|
||||||
|
|
||||||
void AlarmControlPanel::arm_home(optional<std::string> code) {
|
|
||||||
auto call = this->make_call();
|
|
||||||
call.arm_home();
|
|
||||||
if (code.has_value())
|
|
||||||
call.set_code(code.value());
|
|
||||||
call.perform();
|
|
||||||
}
|
|
||||||
|
|
||||||
void AlarmControlPanel::arm_night(optional<std::string> code) {
|
|
||||||
auto call = this->make_call();
|
|
||||||
call.arm_night();
|
|
||||||
if (code.has_value())
|
|
||||||
call.set_code(code.value());
|
|
||||||
call.perform();
|
|
||||||
}
|
|
||||||
|
|
||||||
void AlarmControlPanel::arm_vacation(optional<std::string> code) {
|
|
||||||
auto call = this->make_call();
|
|
||||||
call.arm_vacation();
|
|
||||||
if (code.has_value())
|
|
||||||
call.set_code(code.value());
|
|
||||||
call.perform();
|
|
||||||
}
|
|
||||||
|
|
||||||
void AlarmControlPanel::arm_custom_bypass(optional<std::string> code) {
|
|
||||||
auto call = this->make_call();
|
|
||||||
call.arm_custom_bypass();
|
|
||||||
if (code.has_value())
|
|
||||||
call.set_code(code.value());
|
|
||||||
call.perform();
|
|
||||||
}
|
|
||||||
|
|
||||||
void AlarmControlPanel::disarm(optional<std::string> code) {
|
|
||||||
auto call = this->make_call();
|
|
||||||
call.disarm();
|
|
||||||
if (code.has_value())
|
|
||||||
call.set_code(code.value());
|
|
||||||
call.perform();
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace alarm_control_panel
|
|
||||||
} // namespace esphome
|
|
||||||
@@ -1,136 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <map>
|
|
||||||
|
|
||||||
#include "alarm_control_panel_call.h"
|
|
||||||
#include "alarm_control_panel_state.h"
|
|
||||||
|
|
||||||
#include "esphome/core/automation.h"
|
|
||||||
#include "esphome/core/entity_base.h"
|
|
||||||
#include "esphome/core/log.h"
|
|
||||||
|
|
||||||
namespace esphome {
|
|
||||||
namespace alarm_control_panel {
|
|
||||||
|
|
||||||
enum AlarmControlPanelFeature : uint8_t {
|
|
||||||
// Matches Home Assistant values
|
|
||||||
ACP_FEAT_ARM_HOME = 1 << 0,
|
|
||||||
ACP_FEAT_ARM_AWAY = 1 << 1,
|
|
||||||
ACP_FEAT_ARM_NIGHT = 1 << 2,
|
|
||||||
ACP_FEAT_TRIGGER = 1 << 3,
|
|
||||||
ACP_FEAT_ARM_CUSTOM_BYPASS = 1 << 4,
|
|
||||||
ACP_FEAT_ARM_VACATION = 1 << 5,
|
|
||||||
};
|
|
||||||
|
|
||||||
class AlarmControlPanel : public EntityBase {
|
|
||||||
public:
|
|
||||||
/** Make a AlarmControlPanelCall
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
AlarmControlPanelCall make_call();
|
|
||||||
|
|
||||||
/** Set the state of the alarm_control_panel.
|
|
||||||
*
|
|
||||||
* @param state The AlarmControlPanelState.
|
|
||||||
*/
|
|
||||||
void publish_state(AlarmControlPanelState state);
|
|
||||||
|
|
||||||
/** Add a callback for when the state of the alarm_control_panel changes
|
|
||||||
*
|
|
||||||
* @param callback The callback function
|
|
||||||
*/
|
|
||||||
void add_on_state_callback(std::function<void()> &&callback);
|
|
||||||
|
|
||||||
/** Add a callback for when the state of the alarm_control_panel chanes to triggered
|
|
||||||
*
|
|
||||||
* @param callback The callback function
|
|
||||||
*/
|
|
||||||
void add_on_triggered_callback(std::function<void()> &&callback);
|
|
||||||
|
|
||||||
/** Add a callback for when the state of the alarm_control_panel clears from triggered
|
|
||||||
*
|
|
||||||
* @param callback The callback function
|
|
||||||
*/
|
|
||||||
void add_on_cleared_callback(std::function<void()> &&callback);
|
|
||||||
|
|
||||||
/** A numeric representation of the supported features as per HomeAssistant
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
virtual uint32_t get_supported_features() const = 0;
|
|
||||||
|
|
||||||
/** Returns if the alarm_control_panel has a code
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
virtual bool get_requires_code() const = 0;
|
|
||||||
|
|
||||||
/** Returns if the alarm_control_panel requires a code to arm
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
virtual bool get_requires_code_to_arm() const = 0;
|
|
||||||
|
|
||||||
/** arm the alarm in away mode
|
|
||||||
*
|
|
||||||
* @param code The code
|
|
||||||
*/
|
|
||||||
void arm_away(optional<std::string> code = nullopt);
|
|
||||||
|
|
||||||
/** arm the alarm in home mode
|
|
||||||
*
|
|
||||||
* @param code The code
|
|
||||||
*/
|
|
||||||
void arm_home(optional<std::string> code = nullopt);
|
|
||||||
|
|
||||||
/** arm the alarm in night mode
|
|
||||||
*
|
|
||||||
* @param code The code
|
|
||||||
*/
|
|
||||||
void arm_night(optional<std::string> code = nullopt);
|
|
||||||
|
|
||||||
/** arm the alarm in vacation mode
|
|
||||||
*
|
|
||||||
* @param code The code
|
|
||||||
*/
|
|
||||||
void arm_vacation(optional<std::string> code = nullopt);
|
|
||||||
|
|
||||||
/** arm the alarm in custom bypass mode
|
|
||||||
*
|
|
||||||
* @param code The code
|
|
||||||
*/
|
|
||||||
void arm_custom_bypass(optional<std::string> code = nullopt);
|
|
||||||
|
|
||||||
/** disarm the alarm
|
|
||||||
*
|
|
||||||
* @param code The code
|
|
||||||
*/
|
|
||||||
void disarm(optional<std::string> code = nullopt);
|
|
||||||
|
|
||||||
/** Get the state
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
AlarmControlPanelState get_state() const { return this->current_state_; }
|
|
||||||
|
|
||||||
// is the state one of the armed states
|
|
||||||
bool is_state_armed(AlarmControlPanelState state);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
friend AlarmControlPanelCall;
|
|
||||||
// in order to store last panel state in flash
|
|
||||||
ESPPreferenceObject pref_;
|
|
||||||
// current state
|
|
||||||
AlarmControlPanelState current_state_;
|
|
||||||
// the desired (or previous) state
|
|
||||||
AlarmControlPanelState desired_state_;
|
|
||||||
// last time the state was updated
|
|
||||||
uint32_t last_update_;
|
|
||||||
// the call control function
|
|
||||||
virtual void control(const AlarmControlPanelCall &call) = 0;
|
|
||||||
// state callback
|
|
||||||
CallbackManager<void()> state_callback_{};
|
|
||||||
// trigger callback
|
|
||||||
CallbackManager<void()> triggered_callback_{};
|
|
||||||
// clear callback
|
|
||||||
CallbackManager<void()> cleared_callback_{};
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace alarm_control_panel
|
|
||||||
} // namespace esphome
|
|
||||||
@@ -1,99 +0,0 @@
|
|||||||
#include "alarm_control_panel_call.h"
|
|
||||||
|
|
||||||
#include "alarm_control_panel.h"
|
|
||||||
|
|
||||||
#include "esphome/core/log.h"
|
|
||||||
|
|
||||||
namespace esphome {
|
|
||||||
namespace alarm_control_panel {
|
|
||||||
|
|
||||||
static const char *const TAG = "alarm_control_panel";
|
|
||||||
|
|
||||||
AlarmControlPanelCall::AlarmControlPanelCall(AlarmControlPanel *parent) : parent_(parent) {}
|
|
||||||
|
|
||||||
AlarmControlPanelCall &AlarmControlPanelCall::set_code(const std::string &code) {
|
|
||||||
this->code_ = code;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
AlarmControlPanelCall &AlarmControlPanelCall::arm_away() {
|
|
||||||
this->state_ = ACP_STATE_ARMED_AWAY;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
AlarmControlPanelCall &AlarmControlPanelCall::arm_home() {
|
|
||||||
this->state_ = ACP_STATE_ARMED_HOME;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
AlarmControlPanelCall &AlarmControlPanelCall::arm_night() {
|
|
||||||
this->state_ = ACP_STATE_ARMED_NIGHT;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
AlarmControlPanelCall &AlarmControlPanelCall::arm_vacation() {
|
|
||||||
this->state_ = ACP_STATE_ARMED_VACATION;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
AlarmControlPanelCall &AlarmControlPanelCall::arm_custom_bypass() {
|
|
||||||
this->state_ = ACP_STATE_ARMED_CUSTOM_BYPASS;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
AlarmControlPanelCall &AlarmControlPanelCall::disarm() {
|
|
||||||
this->state_ = ACP_STATE_DISARMED;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
AlarmControlPanelCall &AlarmControlPanelCall::pending() {
|
|
||||||
this->state_ = ACP_STATE_PENDING;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
AlarmControlPanelCall &AlarmControlPanelCall::triggered() {
|
|
||||||
this->state_ = ACP_STATE_TRIGGERED;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
const optional<AlarmControlPanelState> &AlarmControlPanelCall::get_state() const { return this->state_; }
|
|
||||||
const optional<std::string> &AlarmControlPanelCall::get_code() const { return this->code_; }
|
|
||||||
|
|
||||||
void AlarmControlPanelCall::validate_() {
|
|
||||||
if (this->state_.has_value()) {
|
|
||||||
auto state = *this->state_;
|
|
||||||
if (this->parent_->is_state_armed(state) && this->parent_->get_state() != ACP_STATE_DISARMED) {
|
|
||||||
ESP_LOGW(TAG, "Cannot arm when not disarmed");
|
|
||||||
this->state_.reset();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (state == ACP_STATE_PENDING && this->parent_->get_state() == ACP_STATE_DISARMED) {
|
|
||||||
ESP_LOGW(TAG, "Cannot trip alarm when disarmed");
|
|
||||||
this->state_.reset();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (state == ACP_STATE_DISARMED &&
|
|
||||||
!(this->parent_->is_state_armed(this->parent_->get_state()) ||
|
|
||||||
this->parent_->get_state() == ACP_STATE_PENDING || this->parent_->get_state() == ACP_STATE_ARMING ||
|
|
||||||
this->parent_->get_state() == ACP_STATE_TRIGGERED)) {
|
|
||||||
ESP_LOGW(TAG, "Cannot disarm when not armed");
|
|
||||||
this->state_.reset();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (state == ACP_STATE_ARMED_HOME && (this->parent_->get_supported_features() & ACP_FEAT_ARM_HOME) == 0) {
|
|
||||||
ESP_LOGW(TAG, "Cannot arm home when not supported");
|
|
||||||
this->state_.reset();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void AlarmControlPanelCall::perform() {
|
|
||||||
this->validate_();
|
|
||||||
if (this->state_) {
|
|
||||||
this->parent_->control(*this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace alarm_control_panel
|
|
||||||
} // namespace esphome
|
|
||||||
@@ -1,40 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
#include "alarm_control_panel_state.h"
|
|
||||||
|
|
||||||
#include "esphome/core/helpers.h"
|
|
||||||
|
|
||||||
namespace esphome {
|
|
||||||
namespace alarm_control_panel {
|
|
||||||
|
|
||||||
class AlarmControlPanel;
|
|
||||||
|
|
||||||
class AlarmControlPanelCall {
|
|
||||||
public:
|
|
||||||
AlarmControlPanelCall(AlarmControlPanel *parent);
|
|
||||||
|
|
||||||
AlarmControlPanelCall &set_code(const std::string &code);
|
|
||||||
AlarmControlPanelCall &arm_away();
|
|
||||||
AlarmControlPanelCall &arm_home();
|
|
||||||
AlarmControlPanelCall &arm_night();
|
|
||||||
AlarmControlPanelCall &arm_vacation();
|
|
||||||
AlarmControlPanelCall &arm_custom_bypass();
|
|
||||||
AlarmControlPanelCall &disarm();
|
|
||||||
AlarmControlPanelCall &pending();
|
|
||||||
AlarmControlPanelCall &triggered();
|
|
||||||
|
|
||||||
void perform();
|
|
||||||
const optional<AlarmControlPanelState> &get_state() const;
|
|
||||||
const optional<std::string> &get_code() const;
|
|
||||||
|
|
||||||
protected:
|
|
||||||
AlarmControlPanel *parent_;
|
|
||||||
optional<std::string> code_{};
|
|
||||||
optional<AlarmControlPanelState> state_{};
|
|
||||||
void validate_();
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace alarm_control_panel
|
|
||||||
} // namespace esphome
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
#include "alarm_control_panel_state.h"
|
|
||||||
|
|
||||||
namespace esphome {
|
|
||||||
namespace alarm_control_panel {
|
|
||||||
|
|
||||||
const LogString *alarm_control_panel_state_to_string(AlarmControlPanelState state) {
|
|
||||||
switch (state) {
|
|
||||||
case ACP_STATE_DISARMED:
|
|
||||||
return LOG_STR("DISARMED");
|
|
||||||
case ACP_STATE_ARMED_HOME:
|
|
||||||
return LOG_STR("ARMED_HOME");
|
|
||||||
case ACP_STATE_ARMED_AWAY:
|
|
||||||
return LOG_STR("ARMED_AWAY");
|
|
||||||
case ACP_STATE_ARMED_NIGHT:
|
|
||||||
return LOG_STR("NIGHT");
|
|
||||||
case ACP_STATE_ARMED_VACATION:
|
|
||||||
return LOG_STR("ARMED_VACATION");
|
|
||||||
case ACP_STATE_ARMED_CUSTOM_BYPASS:
|
|
||||||
return LOG_STR("ARMED_CUSTOM_BYPASS");
|
|
||||||
case ACP_STATE_PENDING:
|
|
||||||
return LOG_STR("PENDING");
|
|
||||||
case ACP_STATE_ARMING:
|
|
||||||
return LOG_STR("ARMING");
|
|
||||||
case ACP_STATE_DISARMING:
|
|
||||||
return LOG_STR("DISARMING");
|
|
||||||
case ACP_STATE_TRIGGERED:
|
|
||||||
return LOG_STR("TRIGGERED");
|
|
||||||
default:
|
|
||||||
return LOG_STR("UNKNOWN");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace alarm_control_panel
|
|
||||||
} // namespace esphome
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#include <cstdint>
|
|
||||||
#include "esphome/core/log.h"
|
|
||||||
|
|
||||||
namespace esphome {
|
|
||||||
namespace alarm_control_panel {
|
|
||||||
|
|
||||||
enum AlarmControlPanelState : uint8_t {
|
|
||||||
ACP_STATE_DISARMED = 0,
|
|
||||||
ACP_STATE_ARMED_HOME = 1,
|
|
||||||
ACP_STATE_ARMED_AWAY = 2,
|
|
||||||
ACP_STATE_ARMED_NIGHT = 3,
|
|
||||||
ACP_STATE_ARMED_VACATION = 4,
|
|
||||||
ACP_STATE_ARMED_CUSTOM_BYPASS = 5,
|
|
||||||
ACP_STATE_PENDING = 6,
|
|
||||||
ACP_STATE_ARMING = 7,
|
|
||||||
ACP_STATE_DISARMING = 8,
|
|
||||||
ACP_STATE_TRIGGERED = 9
|
|
||||||
};
|
|
||||||
|
|
||||||
/** Returns a string representation of the state.
|
|
||||||
*
|
|
||||||
* @param state The AlarmControlPanelState.
|
|
||||||
*/
|
|
||||||
const LogString *alarm_control_panel_state_to_string(AlarmControlPanelState state);
|
|
||||||
|
|
||||||
} // namespace alarm_control_panel
|
|
||||||
} // namespace esphome
|
|
||||||
@@ -1,115 +0,0 @@
|
|||||||
|
|
||||||
#pragma once
|
|
||||||
#include "esphome/core/automation.h"
|
|
||||||
#include "alarm_control_panel.h"
|
|
||||||
|
|
||||||
namespace esphome {
|
|
||||||
namespace alarm_control_panel {
|
|
||||||
|
|
||||||
class StateTrigger : public Trigger<> {
|
|
||||||
public:
|
|
||||||
explicit StateTrigger(AlarmControlPanel *alarm_control_panel) {
|
|
||||||
alarm_control_panel->add_on_state_callback([this]() { this->trigger(); });
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
class TriggeredTrigger : public Trigger<> {
|
|
||||||
public:
|
|
||||||
explicit TriggeredTrigger(AlarmControlPanel *alarm_control_panel) {
|
|
||||||
alarm_control_panel->add_on_triggered_callback([this]() { this->trigger(); });
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
class ClearedTrigger : public Trigger<> {
|
|
||||||
public:
|
|
||||||
explicit ClearedTrigger(AlarmControlPanel *alarm_control_panel) {
|
|
||||||
alarm_control_panel->add_on_cleared_callback([this]() { this->trigger(); });
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
template<typename... Ts> class ArmAwayAction : public Action<Ts...> {
|
|
||||||
public:
|
|
||||||
explicit ArmAwayAction(AlarmControlPanel *alarm_control_panel) : alarm_control_panel_(alarm_control_panel) {}
|
|
||||||
|
|
||||||
TEMPLATABLE_VALUE(std::string, code)
|
|
||||||
|
|
||||||
void play(Ts... x) override {
|
|
||||||
auto call = this->alarm_control_panel_->make_call();
|
|
||||||
auto code = this->code_.optional_value(x...);
|
|
||||||
if (code.has_value()) {
|
|
||||||
call.set_code(code.value());
|
|
||||||
}
|
|
||||||
call.arm_away();
|
|
||||||
call.perform();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
AlarmControlPanel *alarm_control_panel_;
|
|
||||||
};
|
|
||||||
|
|
||||||
template<typename... Ts> class ArmHomeAction : public Action<Ts...> {
|
|
||||||
public:
|
|
||||||
explicit ArmHomeAction(AlarmControlPanel *alarm_control_panel) : alarm_control_panel_(alarm_control_panel) {}
|
|
||||||
|
|
||||||
TEMPLATABLE_VALUE(std::string, code)
|
|
||||||
|
|
||||||
void play(Ts... x) override {
|
|
||||||
auto call = this->alarm_control_panel_->make_call();
|
|
||||||
auto code = this->code_.optional_value(x...);
|
|
||||||
if (code.has_value()) {
|
|
||||||
call.set_code(code.value());
|
|
||||||
}
|
|
||||||
call.arm_home();
|
|
||||||
call.perform();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
AlarmControlPanel *alarm_control_panel_;
|
|
||||||
};
|
|
||||||
|
|
||||||
template<typename... Ts> class DisarmAction : public Action<Ts...> {
|
|
||||||
public:
|
|
||||||
explicit DisarmAction(AlarmControlPanel *alarm_control_panel) : alarm_control_panel_(alarm_control_panel) {}
|
|
||||||
|
|
||||||
TEMPLATABLE_VALUE(std::string, code)
|
|
||||||
|
|
||||||
void play(Ts... x) override { this->alarm_control_panel_->disarm(this->code_.optional_value(x...)); }
|
|
||||||
|
|
||||||
protected:
|
|
||||||
AlarmControlPanel *alarm_control_panel_;
|
|
||||||
};
|
|
||||||
|
|
||||||
template<typename... Ts> class PendingAction : public Action<Ts...> {
|
|
||||||
public:
|
|
||||||
explicit PendingAction(AlarmControlPanel *alarm_control_panel) : alarm_control_panel_(alarm_control_panel) {}
|
|
||||||
|
|
||||||
void play(Ts... x) override { this->alarm_control_panel_->make_call().pending().perform(); }
|
|
||||||
|
|
||||||
protected:
|
|
||||||
AlarmControlPanel *alarm_control_panel_;
|
|
||||||
};
|
|
||||||
|
|
||||||
template<typename... Ts> class TriggeredAction : public Action<Ts...> {
|
|
||||||
public:
|
|
||||||
explicit TriggeredAction(AlarmControlPanel *alarm_control_panel) : alarm_control_panel_(alarm_control_panel) {}
|
|
||||||
|
|
||||||
void play(Ts... x) override { this->alarm_control_panel_->make_call().triggered().perform(); }
|
|
||||||
|
|
||||||
protected:
|
|
||||||
AlarmControlPanel *alarm_control_panel_;
|
|
||||||
};
|
|
||||||
|
|
||||||
template<typename... Ts> class AlarmControlPanelCondition : public Condition<Ts...> {
|
|
||||||
public:
|
|
||||||
AlarmControlPanelCondition(AlarmControlPanel *parent) : parent_(parent) {}
|
|
||||||
bool check(Ts... x) override {
|
|
||||||
return this->parent_->is_state_armed(this->parent_->get_state()) ||
|
|
||||||
this->parent_->get_state() == ACP_STATE_PENDING || this->parent_->get_state() == ACP_STATE_TRIGGERED;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
|
||||||
AlarmControlPanel *parent_;
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace alarm_control_panel
|
|
||||||
} // namespace esphome
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
CODEOWNERS = ["@buxtronix"]
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
#include "am43_sensor.h"
|
#include "am43.h"
|
||||||
#include "esphome/core/hal.h"
|
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
#include "esphome/core/hal.h"
|
||||||
|
|
||||||
#ifdef USE_ESP32
|
#ifdef USE_ESP32
|
||||||
|
|
||||||
@@ -5,7 +5,7 @@ from esphome.const import CONF_ID, CONF_PIN
|
|||||||
|
|
||||||
CODEOWNERS = ["@buxtronix"]
|
CODEOWNERS = ["@buxtronix"]
|
||||||
DEPENDENCIES = ["ble_client"]
|
DEPENDENCIES = ["ble_client"]
|
||||||
AUTO_LOAD = ["am43"]
|
AUTO_LOAD = ["am43", "sensor"]
|
||||||
|
|
||||||
CONF_INVERT_POSITION = "invert_position"
|
CONF_INVERT_POSITION = "invert_position"
|
||||||
|
|
||||||
@@ -27,10 +27,10 @@ CONFIG_SCHEMA = (
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
async def to_code(config):
|
def to_code(config):
|
||||||
var = cg.new_Pvariable(config[CONF_ID])
|
var = cg.new_Pvariable(config[CONF_ID])
|
||||||
cg.add(var.set_pin(config[CONF_PIN]))
|
cg.add(var.set_pin(config[CONF_PIN]))
|
||||||
cg.add(var.set_invert_position(config[CONF_INVERT_POSITION]))
|
cg.add(var.set_invert_position(config[CONF_INVERT_POSITION]))
|
||||||
await cg.register_component(var, config)
|
yield cg.register_component(var, config)
|
||||||
await cover.register_cover(var, config)
|
yield cover.register_cover(var, config)
|
||||||
await ble_client.register_ble_node(var, config)
|
yield ble_client.register_ble_node(var, config)
|
||||||
|
|||||||
@@ -40,7 +40,6 @@ void Am43Component::loop() {
|
|||||||
|
|
||||||
CoverTraits Am43Component::get_traits() {
|
CoverTraits Am43Component::get_traits() {
|
||||||
auto traits = CoverTraits();
|
auto traits = CoverTraits();
|
||||||
traits.set_supports_stop(true);
|
|
||||||
traits.set_supports_position(true);
|
traits.set_supports_position(true);
|
||||||
traits.set_supports_tilt(false);
|
traits.set_supports_tilt(false);
|
||||||
traits.set_is_assumed_state(false);
|
traits.set_is_assumed_state(false);
|
||||||
|
|||||||
@@ -11,7 +11,6 @@ from esphome.const import (
|
|||||||
UNIT_PERCENT,
|
UNIT_PERCENT,
|
||||||
)
|
)
|
||||||
|
|
||||||
AUTO_LOAD = ["am43"]
|
|
||||||
CODEOWNERS = ["@buxtronix"]
|
CODEOWNERS = ["@buxtronix"]
|
||||||
|
|
||||||
am43_ns = cg.esphome_ns.namespace("am43")
|
am43_ns = cg.esphome_ns.namespace("am43")
|
||||||
@@ -39,15 +38,15 @@ CONFIG_SCHEMA = (
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
async def to_code(config):
|
def to_code(config):
|
||||||
var = cg.new_Pvariable(config[CONF_ID])
|
var = cg.new_Pvariable(config[CONF_ID])
|
||||||
await cg.register_component(var, config)
|
yield cg.register_component(var, config)
|
||||||
await ble_client.register_ble_node(var, config)
|
yield ble_client.register_ble_node(var, config)
|
||||||
|
|
||||||
if CONF_BATTERY_LEVEL in config:
|
if CONF_BATTERY_LEVEL in config:
|
||||||
sens = await sensor.new_sensor(config[CONF_BATTERY_LEVEL])
|
sens = yield sensor.new_sensor(config[CONF_BATTERY_LEVEL])
|
||||||
cg.add(var.set_battery(sens))
|
cg.add(var.set_battery(sens))
|
||||||
|
|
||||||
if CONF_ILLUMINANCE in config:
|
if CONF_ILLUMINANCE in config:
|
||||||
sens = await sensor.new_sensor(config[CONF_ILLUMINANCE])
|
sens = yield sensor.new_sensor(config[CONF_ILLUMINANCE])
|
||||||
cg.add(var.set_illuminance(sens))
|
cg.add(var.set_illuminance(sens))
|
||||||
@@ -3,17 +3,9 @@ import logging
|
|||||||
from esphome import core
|
from esphome import core
|
||||||
from esphome.components import display, font
|
from esphome.components import display, font
|
||||||
import esphome.components.image as espImage
|
import esphome.components.image as espImage
|
||||||
from esphome.components.image import CONF_USE_TRANSPARENCY
|
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
from esphome.const import (
|
from esphome.const import CONF_FILE, CONF_ID, CONF_RAW_DATA_ID, CONF_RESIZE, CONF_TYPE
|
||||||
CONF_FILE,
|
|
||||||
CONF_ID,
|
|
||||||
CONF_RAW_DATA_ID,
|
|
||||||
CONF_REPEAT,
|
|
||||||
CONF_RESIZE,
|
|
||||||
CONF_TYPE,
|
|
||||||
)
|
|
||||||
from esphome.core import CORE, HexInt
|
from esphome.core import CORE, HexInt
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
@@ -21,55 +13,18 @@ _LOGGER = logging.getLogger(__name__)
|
|||||||
DEPENDENCIES = ["display"]
|
DEPENDENCIES = ["display"]
|
||||||
MULTI_CONF = True
|
MULTI_CONF = True
|
||||||
|
|
||||||
CONF_LOOP = "loop"
|
|
||||||
CONF_START_FRAME = "start_frame"
|
|
||||||
CONF_END_FRAME = "end_frame"
|
|
||||||
|
|
||||||
Animation_ = display.display_ns.class_("Animation", espImage.Image_)
|
Animation_ = display.display_ns.class_("Animation", espImage.Image_)
|
||||||
|
|
||||||
|
|
||||||
def validate_cross_dependencies(config):
|
|
||||||
"""
|
|
||||||
Validate fields whose possible values depend on other fields.
|
|
||||||
For example, validate that explicitly transparent image types
|
|
||||||
have "use_transparency" set to True.
|
|
||||||
Also set the default value for those kind of dependent fields.
|
|
||||||
"""
|
|
||||||
image_type = config[CONF_TYPE]
|
|
||||||
is_transparent_type = image_type in ["TRANSPARENT_BINARY", "RGBA"]
|
|
||||||
# If the use_transparency option was not specified, set the default depending on the image type
|
|
||||||
if CONF_USE_TRANSPARENCY not in config:
|
|
||||||
config[CONF_USE_TRANSPARENCY] = is_transparent_type
|
|
||||||
|
|
||||||
if is_transparent_type and not config[CONF_USE_TRANSPARENCY]:
|
|
||||||
raise cv.Invalid(f"Image type {image_type} must always be transparent.")
|
|
||||||
|
|
||||||
return config
|
|
||||||
|
|
||||||
|
|
||||||
ANIMATION_SCHEMA = cv.Schema(
|
ANIMATION_SCHEMA = cv.Schema(
|
||||||
cv.All(
|
{
|
||||||
{
|
cv.Required(CONF_ID): cv.declare_id(Animation_),
|
||||||
cv.Required(CONF_ID): cv.declare_id(Animation_),
|
cv.Required(CONF_FILE): cv.file_,
|
||||||
cv.Required(CONF_FILE): cv.file_,
|
cv.Optional(CONF_RESIZE): cv.dimensions,
|
||||||
cv.Optional(CONF_RESIZE): cv.dimensions,
|
cv.Optional(CONF_TYPE, default="BINARY"): cv.enum(
|
||||||
cv.Optional(CONF_TYPE, default="BINARY"): cv.enum(
|
espImage.IMAGE_TYPE, upper=True
|
||||||
espImage.IMAGE_TYPE, upper=True
|
),
|
||||||
),
|
cv.GenerateID(CONF_RAW_DATA_ID): cv.declare_id(cg.uint8),
|
||||||
# Not setting default here on purpose; the default depends on the image type,
|
}
|
||||||
# and thus will be set in the "validate_cross_dependencies" validator.
|
|
||||||
cv.Optional(CONF_USE_TRANSPARENCY): cv.boolean,
|
|
||||||
cv.Optional(CONF_LOOP): cv.All(
|
|
||||||
{
|
|
||||||
cv.Optional(CONF_START_FRAME, default=0): cv.positive_int,
|
|
||||||
cv.Optional(CONF_END_FRAME): cv.positive_int,
|
|
||||||
cv.Optional(CONF_REPEAT): cv.positive_int,
|
|
||||||
}
|
|
||||||
),
|
|
||||||
cv.GenerateID(CONF_RAW_DATA_ID): cv.declare_id(cg.uint8),
|
|
||||||
},
|
|
||||||
validate_cross_dependencies,
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
CONFIG_SCHEMA = cv.All(font.validate_pillow_installed, ANIMATION_SCHEMA)
|
CONFIG_SCHEMA = cv.All(font.validate_pillow_installed, ANIMATION_SCHEMA)
|
||||||
@@ -95,19 +50,16 @@ async def to_code(config):
|
|||||||
else:
|
else:
|
||||||
if width > 500 or height > 500:
|
if width > 500 or height > 500:
|
||||||
_LOGGER.warning(
|
_LOGGER.warning(
|
||||||
'The image "%s" you requested is very big. Please consider'
|
"The image you requested is very big. Please consider using"
|
||||||
" using the resize parameter.",
|
" the resize parameter."
|
||||||
path,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
transparent = config[CONF_USE_TRANSPARENCY]
|
|
||||||
|
|
||||||
if config[CONF_TYPE] == "GRAYSCALE":
|
if config[CONF_TYPE] == "GRAYSCALE":
|
||||||
data = [0 for _ in range(height * width * frames)]
|
data = [0 for _ in range(height * width * frames)]
|
||||||
pos = 0
|
pos = 0
|
||||||
for frameIndex in range(frames):
|
for frameIndex in range(frames):
|
||||||
image.seek(frameIndex)
|
image.seek(frameIndex)
|
||||||
frame = image.convert("LA", dither=Image.NONE)
|
frame = image.convert("L", dither=Image.NONE)
|
||||||
if CONF_RESIZE in config:
|
if CONF_RESIZE in config:
|
||||||
frame = frame.resize([width, height])
|
frame = frame.resize([width, height])
|
||||||
pixels = list(frame.getdata())
|
pixels = list(frame.getdata())
|
||||||
@@ -115,22 +67,18 @@ async def to_code(config):
|
|||||||
raise core.EsphomeError(
|
raise core.EsphomeError(
|
||||||
f"Unexpected number of pixels in {path} frame {frameIndex}: ({len(pixels)} != {height*width})"
|
f"Unexpected number of pixels in {path} frame {frameIndex}: ({len(pixels)} != {height*width})"
|
||||||
)
|
)
|
||||||
for pix, a in pixels:
|
for pix in pixels:
|
||||||
if transparent:
|
|
||||||
if pix == 1:
|
|
||||||
pix = 0
|
|
||||||
if a < 0x80:
|
|
||||||
pix = 1
|
|
||||||
|
|
||||||
data[pos] = pix
|
data[pos] = pix
|
||||||
pos += 1
|
pos += 1
|
||||||
|
|
||||||
elif config[CONF_TYPE] == "RGBA":
|
elif config[CONF_TYPE] == "RGB24":
|
||||||
data = [0 for _ in range(height * width * 4 * frames)]
|
data = [0 for _ in range(height * width * 3 * frames)]
|
||||||
pos = 0
|
pos = 0
|
||||||
for frameIndex in range(frames):
|
for frameIndex in range(frames):
|
||||||
image.seek(frameIndex)
|
image.seek(frameIndex)
|
||||||
frame = image.convert("RGBA")
|
if CONF_RESIZE in config:
|
||||||
|
image.thumbnail(config[CONF_RESIZE])
|
||||||
|
frame = image.convert("RGB")
|
||||||
if CONF_RESIZE in config:
|
if CONF_RESIZE in config:
|
||||||
frame = frame.resize([width, height])
|
frame = frame.resize([width, height])
|
||||||
pixels = list(frame.getdata())
|
pixels = list(frame.getdata())
|
||||||
@@ -145,44 +93,13 @@ async def to_code(config):
|
|||||||
pos += 1
|
pos += 1
|
||||||
data[pos] = pix[2]
|
data[pos] = pix[2]
|
||||||
pos += 1
|
pos += 1
|
||||||
data[pos] = pix[3]
|
|
||||||
pos += 1
|
|
||||||
|
|
||||||
elif config[CONF_TYPE] == "RGB24":
|
elif config[CONF_TYPE] == "RGB565":
|
||||||
data = [0 for _ in range(height * width * 3 * frames)]
|
|
||||||
pos = 0
|
|
||||||
for frameIndex in range(frames):
|
|
||||||
image.seek(frameIndex)
|
|
||||||
frame = image.convert("RGBA")
|
|
||||||
if CONF_RESIZE in config:
|
|
||||||
frame = frame.resize([width, height])
|
|
||||||
pixels = list(frame.getdata())
|
|
||||||
if len(pixels) != height * width:
|
|
||||||
raise core.EsphomeError(
|
|
||||||
f"Unexpected number of pixels in {path} frame {frameIndex}: ({len(pixels)} != {height*width})"
|
|
||||||
)
|
|
||||||
for r, g, b, a in pixels:
|
|
||||||
if transparent:
|
|
||||||
if r == 0 and g == 0 and b == 1:
|
|
||||||
b = 0
|
|
||||||
if a < 0x80:
|
|
||||||
r = 0
|
|
||||||
g = 0
|
|
||||||
b = 1
|
|
||||||
|
|
||||||
data[pos] = r
|
|
||||||
pos += 1
|
|
||||||
data[pos] = g
|
|
||||||
pos += 1
|
|
||||||
data[pos] = b
|
|
||||||
pos += 1
|
|
||||||
|
|
||||||
elif config[CONF_TYPE] in ["RGB565", "TRANSPARENT_IMAGE"]:
|
|
||||||
data = [0 for _ in range(height * width * 2 * frames)]
|
data = [0 for _ in range(height * width * 2 * frames)]
|
||||||
pos = 0
|
pos = 0
|
||||||
for frameIndex in range(frames):
|
for frameIndex in range(frames):
|
||||||
image.seek(frameIndex)
|
image.seek(frameIndex)
|
||||||
frame = image.convert("RGBA")
|
frame = image.convert("RGB")
|
||||||
if CONF_RESIZE in config:
|
if CONF_RESIZE in config:
|
||||||
frame = frame.resize([width, height])
|
frame = frame.resize([width, height])
|
||||||
pixels = list(frame.getdata())
|
pixels = list(frame.getdata())
|
||||||
@@ -190,21 +107,14 @@ async def to_code(config):
|
|||||||
raise core.EsphomeError(
|
raise core.EsphomeError(
|
||||||
f"Unexpected number of pixels in {path} frame {frameIndex}: ({len(pixels)} != {height*width})"
|
f"Unexpected number of pixels in {path} frame {frameIndex}: ({len(pixels)} != {height*width})"
|
||||||
)
|
)
|
||||||
for r, g, b, a in pixels:
|
for pix in pixels:
|
||||||
R = r >> 3
|
R = pix[0] >> 3
|
||||||
G = g >> 2
|
G = pix[1] >> 2
|
||||||
B = b >> 3
|
B = pix[2] >> 3
|
||||||
rgb = (R << 11) | (G << 5) | B
|
rgb = (R << 11) | (G << 5) | B
|
||||||
|
|
||||||
if transparent:
|
|
||||||
if rgb == 0x0020:
|
|
||||||
rgb = 0
|
|
||||||
if a < 0x80:
|
|
||||||
rgb = 0x0020
|
|
||||||
|
|
||||||
data[pos] = rgb >> 8
|
data[pos] = rgb >> 8
|
||||||
pos += 1
|
pos += 1
|
||||||
data[pos] = rgb & 0xFF
|
data[pos] = rgb & 255
|
||||||
pos += 1
|
pos += 1
|
||||||
|
|
||||||
elif config[CONF_TYPE] in ["BINARY", "TRANSPARENT_BINARY"]:
|
elif config[CONF_TYPE] in ["BINARY", "TRANSPARENT_BINARY"]:
|
||||||
@@ -212,31 +122,19 @@ async def to_code(config):
|
|||||||
data = [0 for _ in range((height * width8 // 8) * frames)]
|
data = [0 for _ in range((height * width8 // 8) * frames)]
|
||||||
for frameIndex in range(frames):
|
for frameIndex in range(frames):
|
||||||
image.seek(frameIndex)
|
image.seek(frameIndex)
|
||||||
if transparent:
|
|
||||||
alpha = image.split()[-1]
|
|
||||||
has_alpha = alpha.getextrema()[0] < 0xFF
|
|
||||||
frame = image.convert("1", dither=Image.NONE)
|
frame = image.convert("1", dither=Image.NONE)
|
||||||
if CONF_RESIZE in config:
|
if CONF_RESIZE in config:
|
||||||
frame = frame.resize([width, height])
|
frame = frame.resize([width, height])
|
||||||
if transparent:
|
for y in range(height):
|
||||||
alpha = alpha.resize([width, height])
|
for x in range(width):
|
||||||
for x, y in [(i, j) for i in range(width) for j in range(height)]:
|
if frame.getpixel((x, y)):
|
||||||
if transparent and has_alpha:
|
|
||||||
if not alpha.getpixel((x, y)):
|
|
||||||
continue
|
continue
|
||||||
elif frame.getpixel((x, y)):
|
pos = x + y * width8 + (height * width8 * frameIndex)
|
||||||
continue
|
data[pos // 8] |= 0x80 >> (pos % 8)
|
||||||
|
|
||||||
pos = x + y * width8 + (height * width8 * frameIndex)
|
|
||||||
data[pos // 8] |= 0x80 >> (pos % 8)
|
|
||||||
else:
|
|
||||||
raise core.EsphomeError(
|
|
||||||
f"Animation f{config[CONF_ID]} has not supported type {config[CONF_TYPE]}."
|
|
||||||
)
|
|
||||||
|
|
||||||
rhs = [HexInt(x) for x in data]
|
rhs = [HexInt(x) for x in data]
|
||||||
prog_arr = cg.progmem_array(config[CONF_RAW_DATA_ID], rhs)
|
prog_arr = cg.progmem_array(config[CONF_RAW_DATA_ID], rhs)
|
||||||
var = cg.new_Pvariable(
|
cg.new_Pvariable(
|
||||||
config[CONF_ID],
|
config[CONF_ID],
|
||||||
prog_arr,
|
prog_arr,
|
||||||
width,
|
width,
|
||||||
@@ -244,9 +142,3 @@ async def to_code(config):
|
|||||||
frames,
|
frames,
|
||||||
espImage.IMAGE_TYPE[config[CONF_TYPE]],
|
espImage.IMAGE_TYPE[config[CONF_TYPE]],
|
||||||
)
|
)
|
||||||
cg.add(var.set_transparency(transparent))
|
|
||||||
if CONF_LOOP in config:
|
|
||||||
start = config[CONF_LOOP][CONF_START_FRAME]
|
|
||||||
end = config[CONF_LOOP].get(CONF_END_FRAME, frames)
|
|
||||||
count = config[CONF_LOOP].get(CONF_REPEAT, -1)
|
|
||||||
cg.add(var.set_loop(start, end, count))
|
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ from esphome.components import i2c
|
|||||||
from esphome.const import CONF_ID
|
from esphome.const import CONF_ID
|
||||||
|
|
||||||
DEPENDENCIES = ["i2c"]
|
DEPENDENCIES = ["i2c"]
|
||||||
|
AUTO_LOAD = ["sensor", "binary_sensor"]
|
||||||
MULTI_CONF = True
|
MULTI_CONF = True
|
||||||
|
|
||||||
CONF_APDS9960_ID = "apds9960_id"
|
CONF_APDS9960_ID = "apds9960_id"
|
||||||
|
|||||||
@@ -116,12 +116,8 @@ void APDS9960::setup() {
|
|||||||
APDS9960_WRITE_BYTE(0x80, val);
|
APDS9960_WRITE_BYTE(0x80, val);
|
||||||
}
|
}
|
||||||
bool APDS9960::is_color_enabled_() const {
|
bool APDS9960::is_color_enabled_() const {
|
||||||
#ifdef USE_SENSOR
|
return this->red_channel_ != nullptr || this->green_channel_ != nullptr || this->blue_channel_ != nullptr ||
|
||||||
return this->red_sensor_ != nullptr || this->green_sensor_ != nullptr || this->blue_sensor_ != nullptr ||
|
this->clear_channel_ != nullptr;
|
||||||
this->clear_sensor_ != nullptr;
|
|
||||||
#else
|
|
||||||
return false;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void APDS9960::dump_config() {
|
void APDS9960::dump_config() {
|
||||||
@@ -129,15 +125,6 @@ void APDS9960::dump_config() {
|
|||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
|
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
|
|
||||||
#ifdef USE_SENSOR
|
|
||||||
LOG_SENSOR(" ", "Red channel", this->red_sensor_);
|
|
||||||
LOG_SENSOR(" ", "Green channel", this->green_sensor_);
|
|
||||||
LOG_SENSOR(" ", "Blue channel", this->blue_sensor_);
|
|
||||||
LOG_SENSOR(" ", "Clear channel", this->clear_sensor_);
|
|
||||||
LOG_SENSOR(" ", "Proximity", this->proximity_sensor_);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
switch (this->error_code_) {
|
switch (this->error_code_) {
|
||||||
case COMMUNICATION_FAILED:
|
case COMMUNICATION_FAILED:
|
||||||
@@ -194,22 +181,17 @@ void APDS9960::read_color_data_(uint8_t status) {
|
|||||||
float blue_perc = (uint_blue / float(UINT16_MAX)) * 100.0f;
|
float blue_perc = (uint_blue / float(UINT16_MAX)) * 100.0f;
|
||||||
|
|
||||||
ESP_LOGD(TAG, "Got clear=%.1f%% red=%.1f%% green=%.1f%% blue=%.1f%%", clear_perc, red_perc, green_perc, blue_perc);
|
ESP_LOGD(TAG, "Got clear=%.1f%% red=%.1f%% green=%.1f%% blue=%.1f%%", clear_perc, red_perc, green_perc, blue_perc);
|
||||||
#ifdef USE_SENSOR
|
if (this->clear_channel_ != nullptr)
|
||||||
if (this->clear_sensor_ != nullptr)
|
this->clear_channel_->publish_state(clear_perc);
|
||||||
this->clear_sensor_->publish_state(clear_perc);
|
if (this->red_channel_ != nullptr)
|
||||||
if (this->red_sensor_ != nullptr)
|
this->red_channel_->publish_state(red_perc);
|
||||||
this->red_sensor_->publish_state(red_perc);
|
if (this->green_channel_ != nullptr)
|
||||||
if (this->green_sensor_ != nullptr)
|
this->green_channel_->publish_state(green_perc);
|
||||||
this->green_sensor_->publish_state(green_perc);
|
if (this->blue_channel_ != nullptr)
|
||||||
if (this->blue_sensor_ != nullptr)
|
this->blue_channel_->publish_state(blue_perc);
|
||||||
this->blue_sensor_->publish_state(blue_perc);
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
void APDS9960::read_proximity_data_(uint8_t status) {
|
void APDS9960::read_proximity_data_(uint8_t status) {
|
||||||
#ifndef USE_SENSOR
|
if (this->proximity_ == nullptr)
|
||||||
return;
|
|
||||||
#else
|
|
||||||
if (this->proximity_sensor_ == nullptr)
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if ((status & 0b10) == 0x00) {
|
if ((status & 0b10) == 0x00) {
|
||||||
@@ -222,8 +204,7 @@ void APDS9960::read_proximity_data_(uint8_t status) {
|
|||||||
|
|
||||||
float prox_perc = (prox / float(UINT8_MAX)) * 100.0f;
|
float prox_perc = (prox / float(UINT8_MAX)) * 100.0f;
|
||||||
ESP_LOGD(TAG, "Got proximity=%.1f%%", prox_perc);
|
ESP_LOGD(TAG, "Got proximity=%.1f%%", prox_perc);
|
||||||
this->proximity_sensor_->publish_state(prox_perc);
|
this->proximity_->publish_state(prox_perc);
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
void APDS9960::read_gesture_data_() {
|
void APDS9960::read_gesture_data_() {
|
||||||
if (!this->is_gesture_enabled_())
|
if (!this->is_gesture_enabled_())
|
||||||
@@ -275,29 +256,28 @@ void APDS9960::read_gesture_data_() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
void APDS9960::report_gesture_(int gesture) {
|
void APDS9960::report_gesture_(int gesture) {
|
||||||
#ifdef USE_BINARY_SENSOR
|
|
||||||
binary_sensor::BinarySensor *bin;
|
binary_sensor::BinarySensor *bin;
|
||||||
switch (gesture) {
|
switch (gesture) {
|
||||||
case 1:
|
case 1:
|
||||||
bin = this->up_direction_binary_sensor_;
|
bin = this->up_direction_;
|
||||||
this->gesture_up_started_ = false;
|
this->gesture_up_started_ = false;
|
||||||
this->gesture_down_started_ = false;
|
this->gesture_down_started_ = false;
|
||||||
ESP_LOGD(TAG, "Got gesture UP");
|
ESP_LOGD(TAG, "Got gesture UP");
|
||||||
break;
|
break;
|
||||||
case 2:
|
case 2:
|
||||||
bin = this->down_direction_binary_sensor_;
|
bin = this->down_direction_;
|
||||||
this->gesture_up_started_ = false;
|
this->gesture_up_started_ = false;
|
||||||
this->gesture_down_started_ = false;
|
this->gesture_down_started_ = false;
|
||||||
ESP_LOGD(TAG, "Got gesture DOWN");
|
ESP_LOGD(TAG, "Got gesture DOWN");
|
||||||
break;
|
break;
|
||||||
case 3:
|
case 3:
|
||||||
bin = this->left_direction_binary_sensor_;
|
bin = this->left_direction_;
|
||||||
this->gesture_left_started_ = false;
|
this->gesture_left_started_ = false;
|
||||||
this->gesture_right_started_ = false;
|
this->gesture_right_started_ = false;
|
||||||
ESP_LOGD(TAG, "Got gesture LEFT");
|
ESP_LOGD(TAG, "Got gesture LEFT");
|
||||||
break;
|
break;
|
||||||
case 4:
|
case 4:
|
||||||
bin = this->right_direction_binary_sensor_;
|
bin = this->right_direction_;
|
||||||
this->gesture_left_started_ = false;
|
this->gesture_left_started_ = false;
|
||||||
this->gesture_right_started_ = false;
|
this->gesture_right_started_ = false;
|
||||||
ESP_LOGD(TAG, "Got gesture RIGHT");
|
ESP_LOGD(TAG, "Got gesture RIGHT");
|
||||||
@@ -310,7 +290,6 @@ void APDS9960::report_gesture_(int gesture) {
|
|||||||
bin->publish_state(true);
|
bin->publish_state(true);
|
||||||
bin->publish_state(false);
|
bin->publish_state(false);
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
void APDS9960::process_dataset_(int up, int down, int left, int right) {
|
void APDS9960::process_dataset_(int up, int down, int left, int right) {
|
||||||
/* Algorithm: (see Figure 11 in datasheet)
|
/* Algorithm: (see Figure 11 in datasheet)
|
||||||
@@ -386,22 +365,10 @@ void APDS9960::process_dataset_(int up, int down, int left, int right) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
float APDS9960::get_setup_priority() const { return setup_priority::DATA; }
|
float APDS9960::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
bool APDS9960::is_proximity_enabled_() const {
|
bool APDS9960::is_proximity_enabled_() const { return this->proximity_ != nullptr || this->is_gesture_enabled_(); }
|
||||||
return
|
|
||||||
#ifdef USE_SENSOR
|
|
||||||
this->proximity_sensor_ != nullptr
|
|
||||||
#else
|
|
||||||
false
|
|
||||||
#endif
|
|
||||||
|| this->is_gesture_enabled_();
|
|
||||||
}
|
|
||||||
bool APDS9960::is_gesture_enabled_() const {
|
bool APDS9960::is_gesture_enabled_() const {
|
||||||
#ifdef USE_BINARY_SENSOR
|
return this->up_direction_ != nullptr || this->left_direction_ != nullptr || this->down_direction_ != nullptr ||
|
||||||
return this->up_direction_binary_sensor_ != nullptr || this->left_direction_binary_sensor_ != nullptr ||
|
this->right_direction_ != nullptr;
|
||||||
this->down_direction_binary_sensor_ != nullptr || this->right_direction_binary_sensor_ != nullptr;
|
|
||||||
#else
|
|
||||||
return false;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace apds9960
|
} // namespace apds9960
|
||||||
|
|||||||
@@ -1,34 +1,14 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "esphome/components/i2c/i2c.h"
|
|
||||||
#include "esphome/core/component.h"
|
#include "esphome/core/component.h"
|
||||||
#include "esphome/core/defines.h"
|
#include "esphome/components/i2c/i2c.h"
|
||||||
#ifdef USE_SENSOR
|
|
||||||
#include "esphome/components/sensor/sensor.h"
|
#include "esphome/components/sensor/sensor.h"
|
||||||
#endif
|
|
||||||
#ifdef USE_BINARY_SENSOR
|
|
||||||
#include "esphome/components/binary_sensor/binary_sensor.h"
|
#include "esphome/components/binary_sensor/binary_sensor.h"
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace apds9960 {
|
namespace apds9960 {
|
||||||
|
|
||||||
class APDS9960 : public PollingComponent, public i2c::I2CDevice {
|
class APDS9960 : public PollingComponent, public i2c::I2CDevice {
|
||||||
#ifdef USE_SENSOR
|
|
||||||
SUB_SENSOR(red)
|
|
||||||
SUB_SENSOR(green)
|
|
||||||
SUB_SENSOR(blue)
|
|
||||||
SUB_SENSOR(clear)
|
|
||||||
SUB_SENSOR(proximity)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef USE_BINARY_SENSOR
|
|
||||||
SUB_BINARY_SENSOR(up_direction)
|
|
||||||
SUB_BINARY_SENSOR(right_direction)
|
|
||||||
SUB_BINARY_SENSOR(down_direction)
|
|
||||||
SUB_BINARY_SENSOR(left_direction)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
@@ -43,6 +23,16 @@ class APDS9960 : public PollingComponent, public i2c::I2CDevice {
|
|||||||
void set_gesture_gain(uint8_t gain) { this->gesture_gain_ = gain; }
|
void set_gesture_gain(uint8_t gain) { this->gesture_gain_ = gain; }
|
||||||
void set_gesture_wait_time(uint8_t wait_time) { this->gesture_wait_time_ = wait_time; }
|
void set_gesture_wait_time(uint8_t wait_time) { this->gesture_wait_time_ = wait_time; }
|
||||||
|
|
||||||
|
void set_red_channel(sensor::Sensor *red_channel) { red_channel_ = red_channel; }
|
||||||
|
void set_green_channel(sensor::Sensor *green_channel) { green_channel_ = green_channel; }
|
||||||
|
void set_blue_channel(sensor::Sensor *blue_channel) { blue_channel_ = blue_channel; }
|
||||||
|
void set_clear_channel(sensor::Sensor *clear_channel) { clear_channel_ = clear_channel; }
|
||||||
|
void set_up_direction(binary_sensor::BinarySensor *up_direction) { up_direction_ = up_direction; }
|
||||||
|
void set_right_direction(binary_sensor::BinarySensor *right_direction) { right_direction_ = right_direction; }
|
||||||
|
void set_down_direction(binary_sensor::BinarySensor *down_direction) { down_direction_ = down_direction; }
|
||||||
|
void set_left_direction(binary_sensor::BinarySensor *left_direction) { left_direction_ = left_direction; }
|
||||||
|
void set_proximity(sensor::Sensor *proximity) { proximity_ = proximity; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
bool is_color_enabled_() const;
|
bool is_color_enabled_() const;
|
||||||
bool is_proximity_enabled_() const;
|
bool is_proximity_enabled_() const;
|
||||||
@@ -60,6 +50,15 @@ class APDS9960 : public PollingComponent, public i2c::I2CDevice {
|
|||||||
uint8_t gesture_gain_;
|
uint8_t gesture_gain_;
|
||||||
uint8_t gesture_wait_time_;
|
uint8_t gesture_wait_time_;
|
||||||
|
|
||||||
|
sensor::Sensor *red_channel_{nullptr};
|
||||||
|
sensor::Sensor *green_channel_{nullptr};
|
||||||
|
sensor::Sensor *blue_channel_{nullptr};
|
||||||
|
sensor::Sensor *clear_channel_{nullptr};
|
||||||
|
binary_sensor::BinarySensor *up_direction_{nullptr};
|
||||||
|
binary_sensor::BinarySensor *right_direction_{nullptr};
|
||||||
|
binary_sensor::BinarySensor *down_direction_{nullptr};
|
||||||
|
binary_sensor::BinarySensor *left_direction_{nullptr};
|
||||||
|
sensor::Sensor *proximity_{nullptr};
|
||||||
enum ErrorCode {
|
enum ErrorCode {
|
||||||
NONE = 0,
|
NONE = 0,
|
||||||
COMMUNICATION_FAILED,
|
COMMUNICATION_FAILED,
|
||||||
|
|||||||
@@ -6,14 +6,19 @@ from . import APDS9960, CONF_APDS9960_ID
|
|||||||
|
|
||||||
DEPENDENCIES = ["apds9960"]
|
DEPENDENCIES = ["apds9960"]
|
||||||
|
|
||||||
DIRECTIONS = ["up", "down", "left", "right"]
|
DIRECTIONS = {
|
||||||
|
"UP": "set_up_direction",
|
||||||
|
"DOWN": "set_down_direction",
|
||||||
|
"LEFT": "set_left_direction",
|
||||||
|
"RIGHT": "set_right_direction",
|
||||||
|
}
|
||||||
|
|
||||||
CONFIG_SCHEMA = binary_sensor.binary_sensor_schema(
|
CONFIG_SCHEMA = binary_sensor.binary_sensor_schema(
|
||||||
device_class=DEVICE_CLASS_MOVING
|
device_class=DEVICE_CLASS_MOVING
|
||||||
).extend(
|
).extend(
|
||||||
{
|
{
|
||||||
cv.GenerateID(CONF_APDS9960_ID): cv.use_id(APDS9960),
|
cv.GenerateID(CONF_APDS9960_ID): cv.use_id(APDS9960),
|
||||||
cv.Required(CONF_DIRECTION): cv.one_of(*DIRECTIONS, lower=True),
|
cv.Required(CONF_DIRECTION): cv.one_of(*DIRECTIONS, upper=True),
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -21,5 +26,5 @@ CONFIG_SCHEMA = binary_sensor.binary_sensor_schema(
|
|||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
hub = await cg.get_variable(config[CONF_APDS9960_ID])
|
hub = await cg.get_variable(config[CONF_APDS9960_ID])
|
||||||
var = await binary_sensor.new_binary_sensor(config)
|
var = await binary_sensor.new_binary_sensor(config)
|
||||||
func = getattr(hub, f"set_{config[CONF_DIRECTION]}_direction_binary_sensor")
|
func = getattr(hub, DIRECTIONS[config[CONF_DIRECTION]])
|
||||||
cg.add(func(var))
|
cg.add(func(var))
|
||||||
|
|||||||
@@ -11,7 +11,13 @@ from . import APDS9960, CONF_APDS9960_ID
|
|||||||
|
|
||||||
DEPENDENCIES = ["apds9960"]
|
DEPENDENCIES = ["apds9960"]
|
||||||
|
|
||||||
TYPES = ["clear", "red", "green", "blue", "proximity"]
|
TYPES = {
|
||||||
|
"CLEAR": "set_clear_channel",
|
||||||
|
"RED": "set_red_channel",
|
||||||
|
"GREEN": "set_green_channel",
|
||||||
|
"BLUE": "set_blue_channel",
|
||||||
|
"PROXIMITY": "set_proximity",
|
||||||
|
}
|
||||||
|
|
||||||
CONFIG_SCHEMA = sensor.sensor_schema(
|
CONFIG_SCHEMA = sensor.sensor_schema(
|
||||||
unit_of_measurement=UNIT_PERCENT,
|
unit_of_measurement=UNIT_PERCENT,
|
||||||
@@ -20,7 +26,7 @@ CONFIG_SCHEMA = sensor.sensor_schema(
|
|||||||
state_class=STATE_CLASS_MEASUREMENT,
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
).extend(
|
).extend(
|
||||||
{
|
{
|
||||||
cv.Required(CONF_TYPE): cv.one_of(*TYPES, lower=True),
|
cv.Required(CONF_TYPE): cv.one_of(*TYPES, upper=True),
|
||||||
cv.GenerateID(CONF_APDS9960_ID): cv.use_id(APDS9960),
|
cv.GenerateID(CONF_APDS9960_ID): cv.use_id(APDS9960),
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
@@ -29,5 +35,5 @@ CONFIG_SCHEMA = sensor.sensor_schema(
|
|||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
hub = await cg.get_variable(config[CONF_APDS9960_ID])
|
hub = await cg.get_variable(config[CONF_APDS9960_ID])
|
||||||
var = await sensor.new_sensor(config)
|
var = await sensor.new_sensor(config)
|
||||||
func = getattr(hub, f"set_{config[CONF_TYPE]}_sensor")
|
func = getattr(hub, TYPES[config[CONF_TYPE]])
|
||||||
cg.add(func(var))
|
cg.add(func(var))
|
||||||
|
|||||||
@@ -53,11 +53,6 @@ service APIConnection {
|
|||||||
rpc bluetooth_gatt_write_descriptor(BluetoothGATTWriteDescriptorRequest) returns (void) {}
|
rpc bluetooth_gatt_write_descriptor(BluetoothGATTWriteDescriptorRequest) returns (void) {}
|
||||||
rpc bluetooth_gatt_notify(BluetoothGATTNotifyRequest) returns (void) {}
|
rpc bluetooth_gatt_notify(BluetoothGATTNotifyRequest) returns (void) {}
|
||||||
rpc subscribe_bluetooth_connections_free(SubscribeBluetoothConnectionsFreeRequest) returns (BluetoothConnectionsFreeResponse) {}
|
rpc subscribe_bluetooth_connections_free(SubscribeBluetoothConnectionsFreeRequest) returns (BluetoothConnectionsFreeResponse) {}
|
||||||
rpc unsubscribe_bluetooth_le_advertisements(UnsubscribeBluetoothLEAdvertisementsRequest) returns (void) {}
|
|
||||||
|
|
||||||
rpc subscribe_voice_assistant(SubscribeVoiceAssistantRequest) returns (void) {}
|
|
||||||
|
|
||||||
rpc alarm_control_panel_command (AlarmControlPanelCommandRequest) returns (void) {}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -208,14 +203,11 @@ message DeviceInfoResponse {
|
|||||||
|
|
||||||
uint32 webserver_port = 10;
|
uint32 webserver_port = 10;
|
||||||
|
|
||||||
uint32 legacy_bluetooth_proxy_version = 11;
|
uint32 bluetooth_proxy_version = 11;
|
||||||
uint32 bluetooth_proxy_feature_flags = 15;
|
|
||||||
|
|
||||||
string manufacturer = 12;
|
string manufacturer = 12;
|
||||||
|
|
||||||
string friendly_name = 13;
|
string friendly_name = 13;
|
||||||
|
|
||||||
uint32 voice_assistant_version = 14;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message ListEntitiesRequest {
|
message ListEntitiesRequest {
|
||||||
@@ -291,7 +283,6 @@ message ListEntitiesCoverResponse {
|
|||||||
bool disabled_by_default = 9;
|
bool disabled_by_default = 9;
|
||||||
string icon = 10;
|
string icon = 10;
|
||||||
EntityCategory entity_category = 11;
|
EntityCategory entity_category = 11;
|
||||||
bool supports_stop = 12;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
enum LegacyCoverState {
|
enum LegacyCoverState {
|
||||||
@@ -865,7 +856,8 @@ message ClimateStateResponse {
|
|||||||
float target_temperature = 4;
|
float target_temperature = 4;
|
||||||
float target_temperature_low = 5;
|
float target_temperature_low = 5;
|
||||||
float target_temperature_high = 6;
|
float target_temperature_high = 6;
|
||||||
bool unused_legacy_away = 7;
|
// For older peers, equal to preset == CLIMATE_PRESET_AWAY
|
||||||
|
bool legacy_away = 7;
|
||||||
ClimateAction action = 8;
|
ClimateAction action = 8;
|
||||||
ClimateFanMode fan_mode = 9;
|
ClimateFanMode fan_mode = 9;
|
||||||
ClimateSwingMode swing_mode = 10;
|
ClimateSwingMode swing_mode = 10;
|
||||||
@@ -888,8 +880,9 @@ message ClimateCommandRequest {
|
|||||||
float target_temperature_low = 7;
|
float target_temperature_low = 7;
|
||||||
bool has_target_temperature_high = 8;
|
bool has_target_temperature_high = 8;
|
||||||
float target_temperature_high = 9;
|
float target_temperature_high = 9;
|
||||||
bool unused_has_legacy_away = 10;
|
// legacy, for older peers, newer ones should use CLIMATE_PRESET_AWAY in preset
|
||||||
bool unused_legacy_away = 11;
|
bool has_legacy_away = 10;
|
||||||
|
bool legacy_away = 11;
|
||||||
bool has_fan_mode = 12;
|
bool has_fan_mode = 12;
|
||||||
ClimateFanMode fan_mode = 13;
|
ClimateFanMode fan_mode = 13;
|
||||||
bool has_swing_mode = 14;
|
bool has_swing_mode = 14;
|
||||||
@@ -1132,9 +1125,6 @@ message MediaPlayerCommandRequest {
|
|||||||
message SubscribeBluetoothLEAdvertisementsRequest {
|
message SubscribeBluetoothLEAdvertisementsRequest {
|
||||||
option (id) = 66;
|
option (id) = 66;
|
||||||
option (source) = SOURCE_CLIENT;
|
option (source) = SOURCE_CLIENT;
|
||||||
option (ifdef) = "USE_BLUETOOTH_PROXY";
|
|
||||||
|
|
||||||
uint32 flags = 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message BluetoothServiceData {
|
message BluetoothServiceData {
|
||||||
@@ -1159,23 +1149,6 @@ message BluetoothLEAdvertisementResponse {
|
|||||||
uint32 address_type = 7;
|
uint32 address_type = 7;
|
||||||
}
|
}
|
||||||
|
|
||||||
message BluetoothLERawAdvertisement {
|
|
||||||
uint64 address = 1;
|
|
||||||
sint32 rssi = 2;
|
|
||||||
uint32 address_type = 3;
|
|
||||||
|
|
||||||
bytes data = 4;
|
|
||||||
}
|
|
||||||
|
|
||||||
message BluetoothLERawAdvertisementsResponse {
|
|
||||||
option (id) = 93;
|
|
||||||
option (source) = SOURCE_SERVER;
|
|
||||||
option (ifdef) = "USE_BLUETOOTH_PROXY";
|
|
||||||
option (no_delay) = true;
|
|
||||||
|
|
||||||
repeated BluetoothLERawAdvertisement advertisements = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
enum BluetoothDeviceRequestType {
|
enum BluetoothDeviceRequestType {
|
||||||
BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT = 0;
|
BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT = 0;
|
||||||
BLUETOOTH_DEVICE_REQUEST_TYPE_DISCONNECT = 1;
|
BLUETOOTH_DEVICE_REQUEST_TYPE_DISCONNECT = 1;
|
||||||
@@ -1183,7 +1156,6 @@ enum BluetoothDeviceRequestType {
|
|||||||
BLUETOOTH_DEVICE_REQUEST_TYPE_UNPAIR = 3;
|
BLUETOOTH_DEVICE_REQUEST_TYPE_UNPAIR = 3;
|
||||||
BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITH_CACHE = 4;
|
BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITH_CACHE = 4;
|
||||||
BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITHOUT_CACHE = 5;
|
BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITHOUT_CACHE = 5;
|
||||||
BLUETOOTH_DEVICE_REQUEST_TYPE_CLEAR_CACHE = 6;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
message BluetoothDeviceRequest {
|
message BluetoothDeviceRequest {
|
||||||
@@ -1387,132 +1359,3 @@ message BluetoothDeviceUnpairingResponse {
|
|||||||
bool success = 2;
|
bool success = 2;
|
||||||
int32 error = 3;
|
int32 error = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
message UnsubscribeBluetoothLEAdvertisementsRequest {
|
|
||||||
option (id) = 87;
|
|
||||||
option (source) = SOURCE_CLIENT;
|
|
||||||
option (ifdef) = "USE_BLUETOOTH_PROXY";
|
|
||||||
}
|
|
||||||
|
|
||||||
message BluetoothDeviceClearCacheResponse {
|
|
||||||
option (id) = 88;
|
|
||||||
option (source) = SOURCE_SERVER;
|
|
||||||
option (ifdef) = "USE_BLUETOOTH_PROXY";
|
|
||||||
|
|
||||||
uint64 address = 1;
|
|
||||||
bool success = 2;
|
|
||||||
int32 error = 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
// ==================== PUSH TO TALK ====================
|
|
||||||
message SubscribeVoiceAssistantRequest {
|
|
||||||
option (id) = 89;
|
|
||||||
option (source) = SOURCE_CLIENT;
|
|
||||||
option (ifdef) = "USE_VOICE_ASSISTANT";
|
|
||||||
|
|
||||||
bool subscribe = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
message VoiceAssistantRequest {
|
|
||||||
option (id) = 90;
|
|
||||||
option (source) = SOURCE_SERVER;
|
|
||||||
option (ifdef) = "USE_VOICE_ASSISTANT";
|
|
||||||
|
|
||||||
bool start = 1;
|
|
||||||
string conversation_id = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message VoiceAssistantResponse {
|
|
||||||
option (id) = 91;
|
|
||||||
option (source) = SOURCE_CLIENT;
|
|
||||||
option (ifdef) = "USE_VOICE_ASSISTANT";
|
|
||||||
|
|
||||||
uint32 port = 1;
|
|
||||||
bool error = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
enum VoiceAssistantEvent {
|
|
||||||
VOICE_ASSISTANT_ERROR = 0;
|
|
||||||
VOICE_ASSISTANT_RUN_START = 1;
|
|
||||||
VOICE_ASSISTANT_RUN_END = 2;
|
|
||||||
VOICE_ASSISTANT_STT_START = 3;
|
|
||||||
VOICE_ASSISTANT_STT_END = 4;
|
|
||||||
VOICE_ASSISTANT_INTENT_START = 5;
|
|
||||||
VOICE_ASSISTANT_INTENT_END = 6;
|
|
||||||
VOICE_ASSISTANT_TTS_START = 7;
|
|
||||||
VOICE_ASSISTANT_TTS_END = 8;
|
|
||||||
}
|
|
||||||
|
|
||||||
message VoiceAssistantEventData {
|
|
||||||
string name = 1;
|
|
||||||
string value = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message VoiceAssistantEventResponse {
|
|
||||||
option (id) = 92;
|
|
||||||
option (source) = SOURCE_CLIENT;
|
|
||||||
option (ifdef) = "USE_VOICE_ASSISTANT";
|
|
||||||
|
|
||||||
VoiceAssistantEvent event_type = 1;
|
|
||||||
repeated VoiceAssistantEventData data = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
// ==================== ALARM CONTROL PANEL ====================
|
|
||||||
enum AlarmControlPanelState {
|
|
||||||
ALARM_STATE_DISARMED = 0;
|
|
||||||
ALARM_STATE_ARMED_HOME = 1;
|
|
||||||
ALARM_STATE_ARMED_AWAY = 2;
|
|
||||||
ALARM_STATE_ARMED_NIGHT = 3;
|
|
||||||
ALARM_STATE_ARMED_VACATION = 4;
|
|
||||||
ALARM_STATE_ARMED_CUSTOM_BYPASS = 5;
|
|
||||||
ALARM_STATE_PENDING = 6;
|
|
||||||
ALARM_STATE_ARMING = 7;
|
|
||||||
ALARM_STATE_DISARMING = 8;
|
|
||||||
ALARM_STATE_TRIGGERED = 9;
|
|
||||||
}
|
|
||||||
|
|
||||||
enum AlarmControlPanelStateCommand {
|
|
||||||
ALARM_CONTROL_PANEL_DISARM = 0;
|
|
||||||
ALARM_CONTROL_PANEL_ARM_AWAY = 1;
|
|
||||||
ALARM_CONTROL_PANEL_ARM_HOME = 2;
|
|
||||||
ALARM_CONTROL_PANEL_ARM_NIGHT = 3;
|
|
||||||
ALARM_CONTROL_PANEL_ARM_VACATION = 4;
|
|
||||||
ALARM_CONTROL_PANEL_ARM_CUSTOM_BYPASS = 5;
|
|
||||||
ALARM_CONTROL_PANEL_TRIGGER = 6;
|
|
||||||
}
|
|
||||||
|
|
||||||
message ListEntitiesAlarmControlPanelResponse {
|
|
||||||
option (id) = 94;
|
|
||||||
option (source) = SOURCE_SERVER;
|
|
||||||
option (ifdef) = "USE_ALARM_CONTROL_PANEL";
|
|
||||||
|
|
||||||
string object_id = 1;
|
|
||||||
fixed32 key = 2;
|
|
||||||
string name = 3;
|
|
||||||
string unique_id = 4;
|
|
||||||
string icon = 5;
|
|
||||||
bool disabled_by_default = 6;
|
|
||||||
EntityCategory entity_category = 7;
|
|
||||||
uint32 supported_features = 8;
|
|
||||||
bool requires_code = 9;
|
|
||||||
bool requires_code_to_arm = 10;
|
|
||||||
}
|
|
||||||
|
|
||||||
message AlarmControlPanelStateResponse {
|
|
||||||
option (id) = 95;
|
|
||||||
option (source) = SOURCE_SERVER;
|
|
||||||
option (ifdef) = "USE_ALARM_CONTROL_PANEL";
|
|
||||||
option (no_delay) = true;
|
|
||||||
fixed32 key = 1;
|
|
||||||
AlarmControlPanelState state = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
message AlarmControlPanelCommandRequest {
|
|
||||||
option (id) = 96;
|
|
||||||
option (source) = SOURCE_CLIENT;
|
|
||||||
option (ifdef) = "USE_ALARM_CONTROL_PANEL";
|
|
||||||
option (no_delay) = true;
|
|
||||||
fixed32 key = 1;
|
|
||||||
AlarmControlPanelStateCommand command = 2;
|
|
||||||
string code = 3;
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
#include "api_connection.h"
|
#include "api_connection.h"
|
||||||
#include <cerrno>
|
#include <cerrno>
|
||||||
#include <cinttypes>
|
|
||||||
#include "esphome/components/network/util.h"
|
#include "esphome/components/network/util.h"
|
||||||
#include "esphome/core/entity_base.h"
|
#include "esphome/core/entity_base.h"
|
||||||
#include "esphome/core/hal.h"
|
#include "esphome/core/hal.h"
|
||||||
@@ -16,9 +15,6 @@
|
|||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
#include "esphome/components/bluetooth_proxy/bluetooth_proxy.h"
|
#include "esphome/components/bluetooth_proxy/bluetooth_proxy.h"
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
#include "esphome/components/voice_assistant/voice_assistant.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace api {
|
namespace api {
|
||||||
@@ -51,14 +47,6 @@ void APIConnection::start() {
|
|||||||
helper_->set_log_info(client_info_);
|
helper_->set_log_info(client_info_);
|
||||||
}
|
}
|
||||||
|
|
||||||
APIConnection::~APIConnection() {
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
if (bluetooth_proxy::global_bluetooth_proxy->get_api_connection() == this) {
|
|
||||||
bluetooth_proxy::global_bluetooth_proxy->unsubscribe_api_connection(this);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
void APIConnection::loop() {
|
void APIConnection::loop() {
|
||||||
if (this->remove_)
|
if (this->remove_)
|
||||||
return;
|
return;
|
||||||
@@ -192,8 +180,7 @@ bool APIConnection::send_binary_sensor_info(binary_sensor::BinarySensor *binary_
|
|||||||
ListEntitiesBinarySensorResponse msg;
|
ListEntitiesBinarySensorResponse msg;
|
||||||
msg.object_id = binary_sensor->get_object_id();
|
msg.object_id = binary_sensor->get_object_id();
|
||||||
msg.key = binary_sensor->get_object_id_hash();
|
msg.key = binary_sensor->get_object_id_hash();
|
||||||
if (binary_sensor->has_own_name())
|
msg.name = binary_sensor->get_name();
|
||||||
msg.name = binary_sensor->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("binary_sensor", binary_sensor);
|
msg.unique_id = get_default_unique_id("binary_sensor", binary_sensor);
|
||||||
msg.device_class = binary_sensor->get_device_class();
|
msg.device_class = binary_sensor->get_device_class();
|
||||||
msg.is_status_binary_sensor = binary_sensor->is_status_binary_sensor();
|
msg.is_status_binary_sensor = binary_sensor->is_status_binary_sensor();
|
||||||
@@ -225,13 +212,11 @@ bool APIConnection::send_cover_info(cover::Cover *cover) {
|
|||||||
ListEntitiesCoverResponse msg;
|
ListEntitiesCoverResponse msg;
|
||||||
msg.key = cover->get_object_id_hash();
|
msg.key = cover->get_object_id_hash();
|
||||||
msg.object_id = cover->get_object_id();
|
msg.object_id = cover->get_object_id();
|
||||||
if (cover->has_own_name())
|
msg.name = cover->get_name();
|
||||||
msg.name = cover->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("cover", cover);
|
msg.unique_id = get_default_unique_id("cover", cover);
|
||||||
msg.assumed_state = traits.get_is_assumed_state();
|
msg.assumed_state = traits.get_is_assumed_state();
|
||||||
msg.supports_position = traits.get_supports_position();
|
msg.supports_position = traits.get_supports_position();
|
||||||
msg.supports_tilt = traits.get_supports_tilt();
|
msg.supports_tilt = traits.get_supports_tilt();
|
||||||
msg.supports_stop = traits.get_supports_stop();
|
|
||||||
msg.device_class = cover->get_device_class();
|
msg.device_class = cover->get_device_class();
|
||||||
msg.disabled_by_default = cover->is_disabled_by_default();
|
msg.disabled_by_default = cover->is_disabled_by_default();
|
||||||
msg.icon = cover->get_icon();
|
msg.icon = cover->get_icon();
|
||||||
@@ -290,8 +275,7 @@ bool APIConnection::send_fan_info(fan::Fan *fan) {
|
|||||||
ListEntitiesFanResponse msg;
|
ListEntitiesFanResponse msg;
|
||||||
msg.key = fan->get_object_id_hash();
|
msg.key = fan->get_object_id_hash();
|
||||||
msg.object_id = fan->get_object_id();
|
msg.object_id = fan->get_object_id();
|
||||||
if (fan->has_own_name())
|
msg.name = fan->get_name();
|
||||||
msg.name = fan->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("fan", fan);
|
msg.unique_id = get_default_unique_id("fan", fan);
|
||||||
msg.supports_oscillation = traits.supports_oscillation();
|
msg.supports_oscillation = traits.supports_oscillation();
|
||||||
msg.supports_speed = traits.supports_speed();
|
msg.supports_speed = traits.supports_speed();
|
||||||
@@ -353,8 +337,7 @@ bool APIConnection::send_light_info(light::LightState *light) {
|
|||||||
ListEntitiesLightResponse msg;
|
ListEntitiesLightResponse msg;
|
||||||
msg.key = light->get_object_id_hash();
|
msg.key = light->get_object_id_hash();
|
||||||
msg.object_id = light->get_object_id();
|
msg.object_id = light->get_object_id();
|
||||||
if (light->has_own_name())
|
msg.name = light->get_name();
|
||||||
msg.name = light->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("light", light);
|
msg.unique_id = get_default_unique_id("light", light);
|
||||||
|
|
||||||
msg.disabled_by_default = light->is_disabled_by_default();
|
msg.disabled_by_default = light->is_disabled_by_default();
|
||||||
@@ -435,8 +418,7 @@ bool APIConnection::send_sensor_info(sensor::Sensor *sensor) {
|
|||||||
ListEntitiesSensorResponse msg;
|
ListEntitiesSensorResponse msg;
|
||||||
msg.key = sensor->get_object_id_hash();
|
msg.key = sensor->get_object_id_hash();
|
||||||
msg.object_id = sensor->get_object_id();
|
msg.object_id = sensor->get_object_id();
|
||||||
if (sensor->has_own_name())
|
msg.name = sensor->get_name();
|
||||||
msg.name = sensor->get_name();
|
|
||||||
msg.unique_id = sensor->unique_id();
|
msg.unique_id = sensor->unique_id();
|
||||||
if (msg.unique_id.empty())
|
if (msg.unique_id.empty())
|
||||||
msg.unique_id = get_default_unique_id("sensor", sensor);
|
msg.unique_id = get_default_unique_id("sensor", sensor);
|
||||||
@@ -466,8 +448,7 @@ bool APIConnection::send_switch_info(switch_::Switch *a_switch) {
|
|||||||
ListEntitiesSwitchResponse msg;
|
ListEntitiesSwitchResponse msg;
|
||||||
msg.key = a_switch->get_object_id_hash();
|
msg.key = a_switch->get_object_id_hash();
|
||||||
msg.object_id = a_switch->get_object_id();
|
msg.object_id = a_switch->get_object_id();
|
||||||
if (a_switch->has_own_name())
|
msg.name = a_switch->get_name();
|
||||||
msg.name = a_switch->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("switch", a_switch);
|
msg.unique_id = get_default_unique_id("switch", a_switch);
|
||||||
msg.icon = a_switch->get_icon();
|
msg.icon = a_switch->get_icon();
|
||||||
msg.assumed_state = a_switch->assumed_state();
|
msg.assumed_state = a_switch->assumed_state();
|
||||||
@@ -539,6 +520,7 @@ bool APIConnection::send_climate_state(climate::Climate *climate) {
|
|||||||
resp.custom_fan_mode = climate->custom_fan_mode.value();
|
resp.custom_fan_mode = climate->custom_fan_mode.value();
|
||||||
if (traits.get_supports_presets() && climate->preset.has_value()) {
|
if (traits.get_supports_presets() && climate->preset.has_value()) {
|
||||||
resp.preset = static_cast<enums::ClimatePreset>(climate->preset.value());
|
resp.preset = static_cast<enums::ClimatePreset>(climate->preset.value());
|
||||||
|
resp.legacy_away = resp.preset == enums::CLIMATE_PRESET_AWAY;
|
||||||
}
|
}
|
||||||
if (!traits.get_supported_custom_presets().empty() && climate->custom_preset.has_value())
|
if (!traits.get_supported_custom_presets().empty() && climate->custom_preset.has_value())
|
||||||
resp.custom_preset = climate->custom_preset.value();
|
resp.custom_preset = climate->custom_preset.value();
|
||||||
@@ -551,8 +533,7 @@ bool APIConnection::send_climate_info(climate::Climate *climate) {
|
|||||||
ListEntitiesClimateResponse msg;
|
ListEntitiesClimateResponse msg;
|
||||||
msg.key = climate->get_object_id_hash();
|
msg.key = climate->get_object_id_hash();
|
||||||
msg.object_id = climate->get_object_id();
|
msg.object_id = climate->get_object_id();
|
||||||
if (climate->has_own_name())
|
msg.name = climate->get_name();
|
||||||
msg.name = climate->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("climate", climate);
|
msg.unique_id = get_default_unique_id("climate", climate);
|
||||||
|
|
||||||
msg.disabled_by_default = climate->is_disabled_by_default();
|
msg.disabled_by_default = climate->is_disabled_by_default();
|
||||||
@@ -599,6 +580,8 @@ void APIConnection::climate_command(const ClimateCommandRequest &msg) {
|
|||||||
call.set_target_temperature_low(msg.target_temperature_low);
|
call.set_target_temperature_low(msg.target_temperature_low);
|
||||||
if (msg.has_target_temperature_high)
|
if (msg.has_target_temperature_high)
|
||||||
call.set_target_temperature_high(msg.target_temperature_high);
|
call.set_target_temperature_high(msg.target_temperature_high);
|
||||||
|
if (msg.has_legacy_away)
|
||||||
|
call.set_preset(msg.legacy_away ? climate::CLIMATE_PRESET_AWAY : climate::CLIMATE_PRESET_HOME);
|
||||||
if (msg.has_fan_mode)
|
if (msg.has_fan_mode)
|
||||||
call.set_fan_mode(static_cast<climate::ClimateFanMode>(msg.fan_mode));
|
call.set_fan_mode(static_cast<climate::ClimateFanMode>(msg.fan_mode));
|
||||||
if (msg.has_custom_fan_mode)
|
if (msg.has_custom_fan_mode)
|
||||||
@@ -628,8 +611,7 @@ bool APIConnection::send_number_info(number::Number *number) {
|
|||||||
ListEntitiesNumberResponse msg;
|
ListEntitiesNumberResponse msg;
|
||||||
msg.key = number->get_object_id_hash();
|
msg.key = number->get_object_id_hash();
|
||||||
msg.object_id = number->get_object_id();
|
msg.object_id = number->get_object_id();
|
||||||
if (number->has_own_name())
|
msg.name = number->get_name();
|
||||||
msg.name = number->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("number", number);
|
msg.unique_id = get_default_unique_id("number", number);
|
||||||
msg.icon = number->get_icon();
|
msg.icon = number->get_icon();
|
||||||
msg.disabled_by_default = number->is_disabled_by_default();
|
msg.disabled_by_default = number->is_disabled_by_default();
|
||||||
@@ -670,8 +652,7 @@ bool APIConnection::send_select_info(select::Select *select) {
|
|||||||
ListEntitiesSelectResponse msg;
|
ListEntitiesSelectResponse msg;
|
||||||
msg.key = select->get_object_id_hash();
|
msg.key = select->get_object_id_hash();
|
||||||
msg.object_id = select->get_object_id();
|
msg.object_id = select->get_object_id();
|
||||||
if (select->has_own_name())
|
msg.name = select->get_name();
|
||||||
msg.name = select->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("select", select);
|
msg.unique_id = get_default_unique_id("select", select);
|
||||||
msg.icon = select->get_icon();
|
msg.icon = select->get_icon();
|
||||||
msg.disabled_by_default = select->is_disabled_by_default();
|
msg.disabled_by_default = select->is_disabled_by_default();
|
||||||
@@ -698,8 +679,7 @@ bool APIConnection::send_button_info(button::Button *button) {
|
|||||||
ListEntitiesButtonResponse msg;
|
ListEntitiesButtonResponse msg;
|
||||||
msg.key = button->get_object_id_hash();
|
msg.key = button->get_object_id_hash();
|
||||||
msg.object_id = button->get_object_id();
|
msg.object_id = button->get_object_id();
|
||||||
if (button->has_own_name())
|
msg.name = button->get_name();
|
||||||
msg.name = button->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("button", button);
|
msg.unique_id = get_default_unique_id("button", button);
|
||||||
msg.icon = button->get_icon();
|
msg.icon = button->get_icon();
|
||||||
msg.disabled_by_default = button->is_disabled_by_default();
|
msg.disabled_by_default = button->is_disabled_by_default();
|
||||||
@@ -730,8 +710,7 @@ bool APIConnection::send_lock_info(lock::Lock *a_lock) {
|
|||||||
ListEntitiesLockResponse msg;
|
ListEntitiesLockResponse msg;
|
||||||
msg.key = a_lock->get_object_id_hash();
|
msg.key = a_lock->get_object_id_hash();
|
||||||
msg.object_id = a_lock->get_object_id();
|
msg.object_id = a_lock->get_object_id();
|
||||||
if (a_lock->has_own_name())
|
msg.name = a_lock->get_name();
|
||||||
msg.name = a_lock->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("lock", a_lock);
|
msg.unique_id = get_default_unique_id("lock", a_lock);
|
||||||
msg.icon = a_lock->get_icon();
|
msg.icon = a_lock->get_icon();
|
||||||
msg.assumed_state = a_lock->traits.get_assumed_state();
|
msg.assumed_state = a_lock->traits.get_assumed_state();
|
||||||
@@ -776,8 +755,7 @@ bool APIConnection::send_media_player_info(media_player::MediaPlayer *media_play
|
|||||||
ListEntitiesMediaPlayerResponse msg;
|
ListEntitiesMediaPlayerResponse msg;
|
||||||
msg.key = media_player->get_object_id_hash();
|
msg.key = media_player->get_object_id_hash();
|
||||||
msg.object_id = media_player->get_object_id();
|
msg.object_id = media_player->get_object_id();
|
||||||
if (media_player->has_own_name())
|
msg.name = media_player->get_name();
|
||||||
msg.name = media_player->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("media_player", media_player);
|
msg.unique_id = get_default_unique_id("media_player", media_player);
|
||||||
msg.icon = media_player->get_icon();
|
msg.icon = media_player->get_icon();
|
||||||
msg.disabled_by_default = media_player->is_disabled_by_default();
|
msg.disabled_by_default = media_player->is_disabled_by_default();
|
||||||
@@ -821,8 +799,7 @@ bool APIConnection::send_camera_info(esp32_camera::ESP32Camera *camera) {
|
|||||||
ListEntitiesCameraResponse msg;
|
ListEntitiesCameraResponse msg;
|
||||||
msg.key = camera->get_object_id_hash();
|
msg.key = camera->get_object_id_hash();
|
||||||
msg.object_id = camera->get_object_id();
|
msg.object_id = camera->get_object_id();
|
||||||
if (camera->has_own_name())
|
msg.name = camera->get_name();
|
||||||
msg.name = camera->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("camera", camera);
|
msg.unique_id = get_default_unique_id("camera", camera);
|
||||||
msg.disabled_by_default = camera->is_disabled_by_default();
|
msg.disabled_by_default = camera->is_disabled_by_default();
|
||||||
msg.icon = camera->get_icon();
|
msg.icon = camera->get_icon();
|
||||||
@@ -853,13 +830,9 @@ void APIConnection::on_get_time_response(const GetTimeResponse &value) {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
void APIConnection::subscribe_bluetooth_le_advertisements(const SubscribeBluetoothLEAdvertisementsRequest &msg) {
|
|
||||||
bluetooth_proxy::global_bluetooth_proxy->subscribe_api_connection(this, msg.flags);
|
|
||||||
}
|
|
||||||
void APIConnection::unsubscribe_bluetooth_le_advertisements(const UnsubscribeBluetoothLEAdvertisementsRequest &msg) {
|
|
||||||
bluetooth_proxy::global_bluetooth_proxy->unsubscribe_api_connection(this);
|
|
||||||
}
|
|
||||||
bool APIConnection::send_bluetooth_le_advertisement(const BluetoothLEAdvertisementResponse &msg) {
|
bool APIConnection::send_bluetooth_le_advertisement(const BluetoothLEAdvertisementResponse &msg) {
|
||||||
|
if (!this->bluetooth_le_advertisement_subscription_)
|
||||||
|
return false;
|
||||||
if (this->client_api_version_major_ < 1 || this->client_api_version_minor_ < 7) {
|
if (this->client_api_version_major_ < 1 || this->client_api_version_minor_ < 7) {
|
||||||
BluetoothLEAdvertisementResponse resp = msg;
|
BluetoothLEAdvertisementResponse resp = msg;
|
||||||
for (auto &service : resp.service_data) {
|
for (auto &service : resp.service_data) {
|
||||||
@@ -906,89 +879,6 @@ BluetoothConnectionsFreeResponse APIConnection::subscribe_bluetooth_connections_
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
bool APIConnection::request_voice_assistant(bool start, const std::string &conversation_id) {
|
|
||||||
if (!this->voice_assistant_subscription_)
|
|
||||||
return false;
|
|
||||||
VoiceAssistantRequest msg;
|
|
||||||
msg.start = start;
|
|
||||||
msg.conversation_id = conversation_id;
|
|
||||||
return this->send_voice_assistant_request(msg);
|
|
||||||
}
|
|
||||||
void APIConnection::on_voice_assistant_response(const VoiceAssistantResponse &msg) {
|
|
||||||
if (voice_assistant::global_voice_assistant != nullptr) {
|
|
||||||
struct sockaddr_storage storage;
|
|
||||||
socklen_t len = sizeof(storage);
|
|
||||||
this->helper_->getpeername((struct sockaddr *) &storage, &len);
|
|
||||||
voice_assistant::global_voice_assistant->start(&storage, msg.port);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
void APIConnection::on_voice_assistant_event_response(const VoiceAssistantEventResponse &msg) {
|
|
||||||
if (voice_assistant::global_voice_assistant != nullptr) {
|
|
||||||
voice_assistant::global_voice_assistant->on_event(msg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
bool APIConnection::send_alarm_control_panel_state(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) {
|
|
||||||
if (!this->state_subscription_)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
AlarmControlPanelStateResponse resp{};
|
|
||||||
resp.key = a_alarm_control_panel->get_object_id_hash();
|
|
||||||
resp.state = static_cast<enums::AlarmControlPanelState>(a_alarm_control_panel->get_state());
|
|
||||||
return this->send_alarm_control_panel_state_response(resp);
|
|
||||||
}
|
|
||||||
bool APIConnection::send_alarm_control_panel_info(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) {
|
|
||||||
ListEntitiesAlarmControlPanelResponse msg;
|
|
||||||
msg.key = a_alarm_control_panel->get_object_id_hash();
|
|
||||||
msg.object_id = a_alarm_control_panel->get_object_id();
|
|
||||||
msg.name = a_alarm_control_panel->get_name();
|
|
||||||
msg.unique_id = get_default_unique_id("alarm_control_panel", a_alarm_control_panel);
|
|
||||||
msg.icon = a_alarm_control_panel->get_icon();
|
|
||||||
msg.disabled_by_default = a_alarm_control_panel->is_disabled_by_default();
|
|
||||||
msg.entity_category = static_cast<enums::EntityCategory>(a_alarm_control_panel->get_entity_category());
|
|
||||||
msg.supported_features = a_alarm_control_panel->get_supported_features();
|
|
||||||
msg.requires_code = a_alarm_control_panel->get_requires_code();
|
|
||||||
msg.requires_code_to_arm = a_alarm_control_panel->get_requires_code_to_arm();
|
|
||||||
return this->send_list_entities_alarm_control_panel_response(msg);
|
|
||||||
}
|
|
||||||
void APIConnection::alarm_control_panel_command(const AlarmControlPanelCommandRequest &msg) {
|
|
||||||
alarm_control_panel::AlarmControlPanel *a_alarm_control_panel = App.get_alarm_control_panel_by_key(msg.key);
|
|
||||||
if (a_alarm_control_panel == nullptr)
|
|
||||||
return;
|
|
||||||
|
|
||||||
auto call = a_alarm_control_panel->make_call();
|
|
||||||
switch (msg.command) {
|
|
||||||
case enums::ALARM_CONTROL_PANEL_DISARM:
|
|
||||||
call.disarm();
|
|
||||||
break;
|
|
||||||
case enums::ALARM_CONTROL_PANEL_ARM_AWAY:
|
|
||||||
call.arm_away();
|
|
||||||
break;
|
|
||||||
case enums::ALARM_CONTROL_PANEL_ARM_HOME:
|
|
||||||
call.arm_home();
|
|
||||||
break;
|
|
||||||
case enums::ALARM_CONTROL_PANEL_ARM_NIGHT:
|
|
||||||
call.arm_night();
|
|
||||||
break;
|
|
||||||
case enums::ALARM_CONTROL_PANEL_ARM_VACATION:
|
|
||||||
call.arm_vacation();
|
|
||||||
break;
|
|
||||||
case enums::ALARM_CONTROL_PANEL_ARM_CUSTOM_BYPASS:
|
|
||||||
call.arm_custom_bypass();
|
|
||||||
break;
|
|
||||||
case enums::ALARM_CONTROL_PANEL_TRIGGER:
|
|
||||||
call.pending();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
call.set_code(msg.code);
|
|
||||||
call.perform();
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
bool APIConnection::send_log_message(int level, const char *tag, const char *line) {
|
bool APIConnection::send_log_message(int level, const char *tag, const char *line) {
|
||||||
if (this->log_subscription_ < level)
|
if (this->log_subscription_ < level)
|
||||||
return false;
|
return false;
|
||||||
@@ -1008,12 +898,12 @@ HelloResponse APIConnection::hello(const HelloRequest &msg) {
|
|||||||
this->helper_->set_log_info(client_info_);
|
this->helper_->set_log_info(client_info_);
|
||||||
this->client_api_version_major_ = msg.api_version_major;
|
this->client_api_version_major_ = msg.api_version_major;
|
||||||
this->client_api_version_minor_ = msg.api_version_minor;
|
this->client_api_version_minor_ = msg.api_version_minor;
|
||||||
ESP_LOGV(TAG, "Hello from client: '%s' | API Version %" PRIu32 ".%" PRIu32, this->client_info_.c_str(),
|
ESP_LOGV(TAG, "Hello from client: '%s' | API Version %d.%d", this->client_info_.c_str(),
|
||||||
this->client_api_version_major_, this->client_api_version_minor_);
|
this->client_api_version_major_, this->client_api_version_minor_);
|
||||||
|
|
||||||
HelloResponse resp;
|
HelloResponse resp;
|
||||||
resp.api_version_major = 1;
|
resp.api_version_major = 1;
|
||||||
resp.api_version_minor = 9;
|
resp.api_version_minor = 7;
|
||||||
resp.server_info = App.get_name() + " (esphome v" ESPHOME_VERSION ")";
|
resp.server_info = App.get_name() + " (esphome v" ESPHOME_VERSION ")";
|
||||||
resp.name = App.get_name();
|
resp.name = App.get_name();
|
||||||
|
|
||||||
@@ -1050,8 +940,6 @@ DeviceInfoResponse APIConnection::device_info(const DeviceInfoRequest &msg) {
|
|||||||
resp.manufacturer = "Espressif";
|
resp.manufacturer = "Espressif";
|
||||||
#elif defined(USE_RP2040)
|
#elif defined(USE_RP2040)
|
||||||
resp.manufacturer = "Raspberry Pi";
|
resp.manufacturer = "Raspberry Pi";
|
||||||
#elif defined(USE_HOST)
|
|
||||||
resp.manufacturer = "Host";
|
|
||||||
#endif
|
#endif
|
||||||
resp.model = ESPHOME_BOARD;
|
resp.model = ESPHOME_BOARD;
|
||||||
#ifdef USE_DEEP_SLEEP
|
#ifdef USE_DEEP_SLEEP
|
||||||
@@ -1065,11 +953,7 @@ DeviceInfoResponse APIConnection::device_info(const DeviceInfoRequest &msg) {
|
|||||||
resp.webserver_port = USE_WEBSERVER_PORT;
|
resp.webserver_port = USE_WEBSERVER_PORT;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
resp.legacy_bluetooth_proxy_version = bluetooth_proxy::global_bluetooth_proxy->get_legacy_version();
|
resp.bluetooth_proxy_version = bluetooth_proxy::global_bluetooth_proxy->has_active() ? 4 : 1;
|
||||||
resp.bluetooth_proxy_feature_flags = bluetooth_proxy::global_bluetooth_proxy->get_feature_flags();
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
resp.voice_assistant_version = voice_assistant::global_voice_assistant->get_version();
|
|
||||||
#endif
|
#endif
|
||||||
return resp;
|
return resp;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,7 +6,6 @@
|
|||||||
#include "api_server.h"
|
#include "api_server.h"
|
||||||
#include "esphome/core/application.h"
|
#include "esphome/core/application.h"
|
||||||
#include "esphome/core/component.h"
|
#include "esphome/core/component.h"
|
||||||
#include "esphome/core/defines.h"
|
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
@@ -16,7 +15,7 @@ namespace api {
|
|||||||
class APIConnection : public APIServerConnection {
|
class APIConnection : public APIServerConnection {
|
||||||
public:
|
public:
|
||||||
APIConnection(std::unique_ptr<socket::Socket> socket, APIServer *parent);
|
APIConnection(std::unique_ptr<socket::Socket> socket, APIServer *parent);
|
||||||
virtual ~APIConnection();
|
virtual ~APIConnection() = default;
|
||||||
|
|
||||||
void start();
|
void start();
|
||||||
void loop();
|
void loop();
|
||||||
@@ -98,8 +97,6 @@ class APIConnection : public APIServerConnection {
|
|||||||
this->send_homeassistant_service_response(call);
|
this->send_homeassistant_service_response(call);
|
||||||
}
|
}
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
void subscribe_bluetooth_le_advertisements(const SubscribeBluetoothLEAdvertisementsRequest &msg) override;
|
|
||||||
void unsubscribe_bluetooth_le_advertisements(const UnsubscribeBluetoothLEAdvertisementsRequest &msg) override;
|
|
||||||
bool send_bluetooth_le_advertisement(const BluetoothLEAdvertisementResponse &msg);
|
bool send_bluetooth_le_advertisement(const BluetoothLEAdvertisementResponse &msg);
|
||||||
|
|
||||||
void bluetooth_device_request(const BluetoothDeviceRequest &msg) override;
|
void bluetooth_device_request(const BluetoothDeviceRequest &msg) override;
|
||||||
@@ -120,21 +117,6 @@ class APIConnection : public APIServerConnection {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
void subscribe_voice_assistant(const SubscribeVoiceAssistantRequest &msg) override {
|
|
||||||
this->voice_assistant_subscription_ = msg.subscribe;
|
|
||||||
}
|
|
||||||
bool request_voice_assistant(bool start, const std::string &conversation_id);
|
|
||||||
void on_voice_assistant_response(const VoiceAssistantResponse &msg) override;
|
|
||||||
void on_voice_assistant_event_response(const VoiceAssistantEventResponse &msg) override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
bool send_alarm_control_panel_state(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel);
|
|
||||||
bool send_alarm_control_panel_info(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel);
|
|
||||||
void alarm_control_panel_command(const AlarmControlPanelCommandRequest &msg) override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
void on_disconnect_response(const DisconnectResponse &value) override;
|
void on_disconnect_response(const DisconnectResponse &value) override;
|
||||||
void on_ping_response(const PingResponse &value) override {
|
void on_ping_response(const PingResponse &value) override {
|
||||||
// we initiated ping
|
// we initiated ping
|
||||||
@@ -168,7 +150,9 @@ class APIConnection : public APIServerConnection {
|
|||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
void execute_service(const ExecuteServiceRequest &msg) override;
|
void execute_service(const ExecuteServiceRequest &msg) override;
|
||||||
|
void subscribe_bluetooth_le_advertisements(const SubscribeBluetoothLEAdvertisementsRequest &msg) override {
|
||||||
|
this->bluetooth_le_advertisement_subscription_ = true;
|
||||||
|
}
|
||||||
bool is_authenticated() override { return this->connection_state_ == ConnectionState::AUTHENTICATED; }
|
bool is_authenticated() override { return this->connection_state_ == ConnectionState::AUTHENTICATED; }
|
||||||
bool is_connection_setup() override {
|
bool is_connection_setup() override {
|
||||||
return this->connection_state_ == ConnectionState ::CONNECTED || this->is_authenticated();
|
return this->connection_state_ == ConnectionState ::CONNECTED || this->is_authenticated();
|
||||||
@@ -213,9 +197,7 @@ class APIConnection : public APIServerConnection {
|
|||||||
uint32_t last_traffic_;
|
uint32_t last_traffic_;
|
||||||
bool sent_ping_{false};
|
bool sent_ping_{false};
|
||||||
bool service_call_subscription_{false};
|
bool service_call_subscription_{false};
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
bool bluetooth_le_advertisement_subscription_{false};
|
||||||
bool voice_assistant_subscription_{false};
|
|
||||||
#endif
|
|
||||||
bool next_close_ = false;
|
bool next_close_ = false;
|
||||||
APIServer *parent_;
|
APIServer *parent_;
|
||||||
InitialStateIterator initial_state_iterator_;
|
InitialStateIterator initial_state_iterator_;
|
||||||
|
|||||||
@@ -10,8 +10,8 @@
|
|||||||
#include "noise/protocol.h"
|
#include "noise/protocol.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "api_noise_context.h"
|
|
||||||
#include "esphome/components/socket/socket.h"
|
#include "esphome/components/socket/socket.h"
|
||||||
|
#include "api_noise_context.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace api {
|
namespace api {
|
||||||
@@ -67,7 +67,6 @@ class APIFrameHelper {
|
|||||||
virtual bool can_write_without_blocking() = 0;
|
virtual bool can_write_without_blocking() = 0;
|
||||||
virtual APIError write_packet(uint16_t type, const uint8_t *data, size_t len) = 0;
|
virtual APIError write_packet(uint16_t type, const uint8_t *data, size_t len) = 0;
|
||||||
virtual std::string getpeername() = 0;
|
virtual std::string getpeername() = 0;
|
||||||
virtual int getpeername(struct sockaddr *addr, socklen_t *addrlen) = 0;
|
|
||||||
virtual APIError close() = 0;
|
virtual APIError close() = 0;
|
||||||
virtual APIError shutdown(int how) = 0;
|
virtual APIError shutdown(int how) = 0;
|
||||||
// Give this helper a name for logging
|
// Give this helper a name for logging
|
||||||
@@ -85,10 +84,7 @@ class APINoiseFrameHelper : public APIFrameHelper {
|
|||||||
APIError read_packet(ReadPacketBuffer *buffer) override;
|
APIError read_packet(ReadPacketBuffer *buffer) override;
|
||||||
bool can_write_without_blocking() override;
|
bool can_write_without_blocking() override;
|
||||||
APIError write_packet(uint16_t type, const uint8_t *payload, size_t len) override;
|
APIError write_packet(uint16_t type, const uint8_t *payload, size_t len) override;
|
||||||
std::string getpeername() override { return this->socket_->getpeername(); }
|
std::string getpeername() override { return socket_->getpeername(); }
|
||||||
int getpeername(struct sockaddr *addr, socklen_t *addrlen) override {
|
|
||||||
return this->socket_->getpeername(addr, addrlen);
|
|
||||||
}
|
|
||||||
APIError close() override;
|
APIError close() override;
|
||||||
APIError shutdown(int how) override;
|
APIError shutdown(int how) override;
|
||||||
// Give this helper a name for logging
|
// Give this helper a name for logging
|
||||||
@@ -148,10 +144,7 @@ class APIPlaintextFrameHelper : public APIFrameHelper {
|
|||||||
APIError read_packet(ReadPacketBuffer *buffer) override;
|
APIError read_packet(ReadPacketBuffer *buffer) override;
|
||||||
bool can_write_without_blocking() override;
|
bool can_write_without_blocking() override;
|
||||||
APIError write_packet(uint16_t type, const uint8_t *payload, size_t len) override;
|
APIError write_packet(uint16_t type, const uint8_t *payload, size_t len) override;
|
||||||
std::string getpeername() override { return this->socket_->getpeername(); }
|
std::string getpeername() override { return socket_->getpeername(); }
|
||||||
int getpeername(struct sockaddr *addr, socklen_t *addrlen) override {
|
|
||||||
return this->socket_->getpeername(addr, addrlen);
|
|
||||||
}
|
|
||||||
APIError close() override;
|
APIError close() override;
|
||||||
APIError shutdown(int how) override;
|
APIError shutdown(int how) override;
|
||||||
// Give this helper a name for logging
|
// Give this helper a name for logging
|
||||||
|
|||||||
@@ -400,85 +400,6 @@ const char *proto_enum_to_string<enums::BluetoothDeviceRequestType>(enums::Bluet
|
|||||||
return "BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITH_CACHE";
|
return "BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITH_CACHE";
|
||||||
case enums::BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITHOUT_CACHE:
|
case enums::BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITHOUT_CACHE:
|
||||||
return "BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITHOUT_CACHE";
|
return "BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITHOUT_CACHE";
|
||||||
case enums::BLUETOOTH_DEVICE_REQUEST_TYPE_CLEAR_CACHE:
|
|
||||||
return "BLUETOOTH_DEVICE_REQUEST_TYPE_CLEAR_CACHE";
|
|
||||||
default:
|
|
||||||
return "UNKNOWN";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
template<> const char *proto_enum_to_string<enums::VoiceAssistantEvent>(enums::VoiceAssistantEvent value) {
|
|
||||||
switch (value) {
|
|
||||||
case enums::VOICE_ASSISTANT_ERROR:
|
|
||||||
return "VOICE_ASSISTANT_ERROR";
|
|
||||||
case enums::VOICE_ASSISTANT_RUN_START:
|
|
||||||
return "VOICE_ASSISTANT_RUN_START";
|
|
||||||
case enums::VOICE_ASSISTANT_RUN_END:
|
|
||||||
return "VOICE_ASSISTANT_RUN_END";
|
|
||||||
case enums::VOICE_ASSISTANT_STT_START:
|
|
||||||
return "VOICE_ASSISTANT_STT_START";
|
|
||||||
case enums::VOICE_ASSISTANT_STT_END:
|
|
||||||
return "VOICE_ASSISTANT_STT_END";
|
|
||||||
case enums::VOICE_ASSISTANT_INTENT_START:
|
|
||||||
return "VOICE_ASSISTANT_INTENT_START";
|
|
||||||
case enums::VOICE_ASSISTANT_INTENT_END:
|
|
||||||
return "VOICE_ASSISTANT_INTENT_END";
|
|
||||||
case enums::VOICE_ASSISTANT_TTS_START:
|
|
||||||
return "VOICE_ASSISTANT_TTS_START";
|
|
||||||
case enums::VOICE_ASSISTANT_TTS_END:
|
|
||||||
return "VOICE_ASSISTANT_TTS_END";
|
|
||||||
default:
|
|
||||||
return "UNKNOWN";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
template<> const char *proto_enum_to_string<enums::AlarmControlPanelState>(enums::AlarmControlPanelState value) {
|
|
||||||
switch (value) {
|
|
||||||
case enums::ALARM_STATE_DISARMED:
|
|
||||||
return "ALARM_STATE_DISARMED";
|
|
||||||
case enums::ALARM_STATE_ARMED_HOME:
|
|
||||||
return "ALARM_STATE_ARMED_HOME";
|
|
||||||
case enums::ALARM_STATE_ARMED_AWAY:
|
|
||||||
return "ALARM_STATE_ARMED_AWAY";
|
|
||||||
case enums::ALARM_STATE_ARMED_NIGHT:
|
|
||||||
return "ALARM_STATE_ARMED_NIGHT";
|
|
||||||
case enums::ALARM_STATE_ARMED_VACATION:
|
|
||||||
return "ALARM_STATE_ARMED_VACATION";
|
|
||||||
case enums::ALARM_STATE_ARMED_CUSTOM_BYPASS:
|
|
||||||
return "ALARM_STATE_ARMED_CUSTOM_BYPASS";
|
|
||||||
case enums::ALARM_STATE_PENDING:
|
|
||||||
return "ALARM_STATE_PENDING";
|
|
||||||
case enums::ALARM_STATE_ARMING:
|
|
||||||
return "ALARM_STATE_ARMING";
|
|
||||||
case enums::ALARM_STATE_DISARMING:
|
|
||||||
return "ALARM_STATE_DISARMING";
|
|
||||||
case enums::ALARM_STATE_TRIGGERED:
|
|
||||||
return "ALARM_STATE_TRIGGERED";
|
|
||||||
default:
|
|
||||||
return "UNKNOWN";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
template<>
|
|
||||||
const char *proto_enum_to_string<enums::AlarmControlPanelStateCommand>(enums::AlarmControlPanelStateCommand value) {
|
|
||||||
switch (value) {
|
|
||||||
case enums::ALARM_CONTROL_PANEL_DISARM:
|
|
||||||
return "ALARM_CONTROL_PANEL_DISARM";
|
|
||||||
case enums::ALARM_CONTROL_PANEL_ARM_AWAY:
|
|
||||||
return "ALARM_CONTROL_PANEL_ARM_AWAY";
|
|
||||||
case enums::ALARM_CONTROL_PANEL_ARM_HOME:
|
|
||||||
return "ALARM_CONTROL_PANEL_ARM_HOME";
|
|
||||||
case enums::ALARM_CONTROL_PANEL_ARM_NIGHT:
|
|
||||||
return "ALARM_CONTROL_PANEL_ARM_NIGHT";
|
|
||||||
case enums::ALARM_CONTROL_PANEL_ARM_VACATION:
|
|
||||||
return "ALARM_CONTROL_PANEL_ARM_VACATION";
|
|
||||||
case enums::ALARM_CONTROL_PANEL_ARM_CUSTOM_BYPASS:
|
|
||||||
return "ALARM_CONTROL_PANEL_ARM_CUSTOM_BYPASS";
|
|
||||||
case enums::ALARM_CONTROL_PANEL_TRIGGER:
|
|
||||||
return "ALARM_CONTROL_PANEL_TRIGGER";
|
|
||||||
default:
|
default:
|
||||||
return "UNKNOWN";
|
return "UNKNOWN";
|
||||||
}
|
}
|
||||||
@@ -668,15 +589,7 @@ bool DeviceInfoResponse::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
case 11: {
|
case 11: {
|
||||||
this->legacy_bluetooth_proxy_version = value.as_uint32();
|
this->bluetooth_proxy_version = value.as_uint32();
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 15: {
|
|
||||||
this->bluetooth_proxy_feature_flags = value.as_uint32();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 14: {
|
|
||||||
this->voice_assistant_version = value.as_uint32();
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
@@ -736,11 +649,9 @@ void DeviceInfoResponse::encode(ProtoWriteBuffer buffer) const {
|
|||||||
buffer.encode_string(8, this->project_name);
|
buffer.encode_string(8, this->project_name);
|
||||||
buffer.encode_string(9, this->project_version);
|
buffer.encode_string(9, this->project_version);
|
||||||
buffer.encode_uint32(10, this->webserver_port);
|
buffer.encode_uint32(10, this->webserver_port);
|
||||||
buffer.encode_uint32(11, this->legacy_bluetooth_proxy_version);
|
buffer.encode_uint32(11, this->bluetooth_proxy_version);
|
||||||
buffer.encode_uint32(15, this->bluetooth_proxy_feature_flags);
|
|
||||||
buffer.encode_string(12, this->manufacturer);
|
buffer.encode_string(12, this->manufacturer);
|
||||||
buffer.encode_string(13, this->friendly_name);
|
buffer.encode_string(13, this->friendly_name);
|
||||||
buffer.encode_uint32(14, this->voice_assistant_version);
|
|
||||||
}
|
}
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
void DeviceInfoResponse::dump_to(std::string &out) const {
|
void DeviceInfoResponse::dump_to(std::string &out) const {
|
||||||
@@ -787,13 +698,8 @@ void DeviceInfoResponse::dump_to(std::string &out) const {
|
|||||||
out.append(buffer);
|
out.append(buffer);
|
||||||
out.append("\n");
|
out.append("\n");
|
||||||
|
|
||||||
out.append(" legacy_bluetooth_proxy_version: ");
|
out.append(" bluetooth_proxy_version: ");
|
||||||
sprintf(buffer, "%u", this->legacy_bluetooth_proxy_version);
|
sprintf(buffer, "%u", this->bluetooth_proxy_version);
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" bluetooth_proxy_feature_flags: ");
|
|
||||||
sprintf(buffer, "%u", this->bluetooth_proxy_feature_flags);
|
|
||||||
out.append(buffer);
|
out.append(buffer);
|
||||||
out.append("\n");
|
out.append("\n");
|
||||||
|
|
||||||
@@ -804,11 +710,6 @@ void DeviceInfoResponse::dump_to(std::string &out) const {
|
|||||||
out.append(" friendly_name: ");
|
out.append(" friendly_name: ");
|
||||||
out.append("'").append(this->friendly_name).append("'");
|
out.append("'").append(this->friendly_name).append("'");
|
||||||
out.append("\n");
|
out.append("\n");
|
||||||
|
|
||||||
out.append(" voice_assistant_version: ");
|
|
||||||
sprintf(buffer, "%u", this->voice_assistant_version);
|
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
out.append("}");
|
out.append("}");
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@@ -1002,10 +903,6 @@ bool ListEntitiesCoverResponse::decode_varint(uint32_t field_id, ProtoVarInt val
|
|||||||
this->entity_category = value.as_enum<enums::EntityCategory>();
|
this->entity_category = value.as_enum<enums::EntityCategory>();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
case 12: {
|
|
||||||
this->supports_stop = value.as_bool();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
default:
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -1058,7 +955,6 @@ void ListEntitiesCoverResponse::encode(ProtoWriteBuffer buffer) const {
|
|||||||
buffer.encode_bool(9, this->disabled_by_default);
|
buffer.encode_bool(9, this->disabled_by_default);
|
||||||
buffer.encode_string(10, this->icon);
|
buffer.encode_string(10, this->icon);
|
||||||
buffer.encode_enum<enums::EntityCategory>(11, this->entity_category);
|
buffer.encode_enum<enums::EntityCategory>(11, this->entity_category);
|
||||||
buffer.encode_bool(12, this->supports_stop);
|
|
||||||
}
|
}
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
void ListEntitiesCoverResponse::dump_to(std::string &out) const {
|
void ListEntitiesCoverResponse::dump_to(std::string &out) const {
|
||||||
@@ -1108,10 +1004,6 @@ void ListEntitiesCoverResponse::dump_to(std::string &out) const {
|
|||||||
out.append(" entity_category: ");
|
out.append(" entity_category: ");
|
||||||
out.append(proto_enum_to_string<enums::EntityCategory>(this->entity_category));
|
out.append(proto_enum_to_string<enums::EntityCategory>(this->entity_category));
|
||||||
out.append("\n");
|
out.append("\n");
|
||||||
|
|
||||||
out.append(" supports_stop: ");
|
|
||||||
out.append(YESNO(this->supports_stop));
|
|
||||||
out.append("\n");
|
|
||||||
out.append("}");
|
out.append("}");
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@@ -3719,7 +3611,7 @@ bool ClimateStateResponse::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
case 7: {
|
case 7: {
|
||||||
this->unused_legacy_away = value.as_bool();
|
this->legacy_away = value.as_bool();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
case 8: {
|
case 8: {
|
||||||
@@ -3789,7 +3681,7 @@ void ClimateStateResponse::encode(ProtoWriteBuffer buffer) const {
|
|||||||
buffer.encode_float(4, this->target_temperature);
|
buffer.encode_float(4, this->target_temperature);
|
||||||
buffer.encode_float(5, this->target_temperature_low);
|
buffer.encode_float(5, this->target_temperature_low);
|
||||||
buffer.encode_float(6, this->target_temperature_high);
|
buffer.encode_float(6, this->target_temperature_high);
|
||||||
buffer.encode_bool(7, this->unused_legacy_away);
|
buffer.encode_bool(7, this->legacy_away);
|
||||||
buffer.encode_enum<enums::ClimateAction>(8, this->action);
|
buffer.encode_enum<enums::ClimateAction>(8, this->action);
|
||||||
buffer.encode_enum<enums::ClimateFanMode>(9, this->fan_mode);
|
buffer.encode_enum<enums::ClimateFanMode>(9, this->fan_mode);
|
||||||
buffer.encode_enum<enums::ClimateSwingMode>(10, this->swing_mode);
|
buffer.encode_enum<enums::ClimateSwingMode>(10, this->swing_mode);
|
||||||
@@ -3830,8 +3722,8 @@ void ClimateStateResponse::dump_to(std::string &out) const {
|
|||||||
out.append(buffer);
|
out.append(buffer);
|
||||||
out.append("\n");
|
out.append("\n");
|
||||||
|
|
||||||
out.append(" unused_legacy_away: ");
|
out.append(" legacy_away: ");
|
||||||
out.append(YESNO(this->unused_legacy_away));
|
out.append(YESNO(this->legacy_away));
|
||||||
out.append("\n");
|
out.append("\n");
|
||||||
|
|
||||||
out.append(" action: ");
|
out.append(" action: ");
|
||||||
@@ -3883,11 +3775,11 @@ bool ClimateCommandRequest::decode_varint(uint32_t field_id, ProtoVarInt value)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
case 10: {
|
case 10: {
|
||||||
this->unused_has_legacy_away = value.as_bool();
|
this->has_legacy_away = value.as_bool();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
case 11: {
|
case 11: {
|
||||||
this->unused_legacy_away = value.as_bool();
|
this->legacy_away = value.as_bool();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
case 12: {
|
case 12: {
|
||||||
@@ -3972,8 +3864,8 @@ void ClimateCommandRequest::encode(ProtoWriteBuffer buffer) const {
|
|||||||
buffer.encode_float(7, this->target_temperature_low);
|
buffer.encode_float(7, this->target_temperature_low);
|
||||||
buffer.encode_bool(8, this->has_target_temperature_high);
|
buffer.encode_bool(8, this->has_target_temperature_high);
|
||||||
buffer.encode_float(9, this->target_temperature_high);
|
buffer.encode_float(9, this->target_temperature_high);
|
||||||
buffer.encode_bool(10, this->unused_has_legacy_away);
|
buffer.encode_bool(10, this->has_legacy_away);
|
||||||
buffer.encode_bool(11, this->unused_legacy_away);
|
buffer.encode_bool(11, this->legacy_away);
|
||||||
buffer.encode_bool(12, this->has_fan_mode);
|
buffer.encode_bool(12, this->has_fan_mode);
|
||||||
buffer.encode_enum<enums::ClimateFanMode>(13, this->fan_mode);
|
buffer.encode_enum<enums::ClimateFanMode>(13, this->fan_mode);
|
||||||
buffer.encode_bool(14, this->has_swing_mode);
|
buffer.encode_bool(14, this->has_swing_mode);
|
||||||
@@ -4029,12 +3921,12 @@ void ClimateCommandRequest::dump_to(std::string &out) const {
|
|||||||
out.append(buffer);
|
out.append(buffer);
|
||||||
out.append("\n");
|
out.append("\n");
|
||||||
|
|
||||||
out.append(" unused_has_legacy_away: ");
|
out.append(" has_legacy_away: ");
|
||||||
out.append(YESNO(this->unused_has_legacy_away));
|
out.append(YESNO(this->has_legacy_away));
|
||||||
out.append("\n");
|
out.append("\n");
|
||||||
|
|
||||||
out.append(" unused_legacy_away: ");
|
out.append(" legacy_away: ");
|
||||||
out.append(YESNO(this->unused_legacy_away));
|
out.append(YESNO(this->legacy_away));
|
||||||
out.append("\n");
|
out.append("\n");
|
||||||
|
|
||||||
out.append(" has_fan_mode: ");
|
out.append(" has_fan_mode: ");
|
||||||
@@ -5102,28 +4994,10 @@ void MediaPlayerCommandRequest::dump_to(std::string &out) const {
|
|||||||
out.append("}");
|
out.append("}");
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
bool SubscribeBluetoothLEAdvertisementsRequest::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
void SubscribeBluetoothLEAdvertisementsRequest::encode(ProtoWriteBuffer buffer) const {}
|
||||||
switch (field_id) {
|
|
||||||
case 1: {
|
|
||||||
this->flags = value.as_uint32();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void SubscribeBluetoothLEAdvertisementsRequest::encode(ProtoWriteBuffer buffer) const {
|
|
||||||
buffer.encode_uint32(1, this->flags);
|
|
||||||
}
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
void SubscribeBluetoothLEAdvertisementsRequest::dump_to(std::string &out) const {
|
void SubscribeBluetoothLEAdvertisementsRequest::dump_to(std::string &out) const {
|
||||||
__attribute__((unused)) char buffer[64];
|
out.append("SubscribeBluetoothLEAdvertisementsRequest {}");
|
||||||
out.append("SubscribeBluetoothLEAdvertisementsRequest {\n");
|
|
||||||
out.append(" flags: ");
|
|
||||||
sprintf(buffer, "%u", this->flags);
|
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
out.append("}");
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
bool BluetoothServiceData::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
bool BluetoothServiceData::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
||||||
@@ -5276,92 +5150,6 @@ void BluetoothLEAdvertisementResponse::dump_to(std::string &out) const {
|
|||||||
out.append("}");
|
out.append("}");
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
bool BluetoothLERawAdvertisement::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 1: {
|
|
||||||
this->address = value.as_uint64();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 2: {
|
|
||||||
this->rssi = value.as_sint32();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 3: {
|
|
||||||
this->address_type = value.as_uint32();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
bool BluetoothLERawAdvertisement::decode_length(uint32_t field_id, ProtoLengthDelimited value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 4: {
|
|
||||||
this->data = value.as_string();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void BluetoothLERawAdvertisement::encode(ProtoWriteBuffer buffer) const {
|
|
||||||
buffer.encode_uint64(1, this->address);
|
|
||||||
buffer.encode_sint32(2, this->rssi);
|
|
||||||
buffer.encode_uint32(3, this->address_type);
|
|
||||||
buffer.encode_string(4, this->data);
|
|
||||||
}
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void BluetoothLERawAdvertisement::dump_to(std::string &out) const {
|
|
||||||
__attribute__((unused)) char buffer[64];
|
|
||||||
out.append("BluetoothLERawAdvertisement {\n");
|
|
||||||
out.append(" address: ");
|
|
||||||
sprintf(buffer, "%llu", this->address);
|
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" rssi: ");
|
|
||||||
sprintf(buffer, "%d", this->rssi);
|
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" address_type: ");
|
|
||||||
sprintf(buffer, "%u", this->address_type);
|
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" data: ");
|
|
||||||
out.append("'").append(this->data).append("'");
|
|
||||||
out.append("\n");
|
|
||||||
out.append("}");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
bool BluetoothLERawAdvertisementsResponse::decode_length(uint32_t field_id, ProtoLengthDelimited value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 1: {
|
|
||||||
this->advertisements.push_back(value.as_message<BluetoothLERawAdvertisement>());
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void BluetoothLERawAdvertisementsResponse::encode(ProtoWriteBuffer buffer) const {
|
|
||||||
for (auto &it : this->advertisements) {
|
|
||||||
buffer.encode_message<BluetoothLERawAdvertisement>(1, it, true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void BluetoothLERawAdvertisementsResponse::dump_to(std::string &out) const {
|
|
||||||
__attribute__((unused)) char buffer[64];
|
|
||||||
out.append("BluetoothLERawAdvertisementsResponse {\n");
|
|
||||||
for (const auto &it : this->advertisements) {
|
|
||||||
out.append(" advertisements: ");
|
|
||||||
it.dump_to(out);
|
|
||||||
out.append("\n");
|
|
||||||
}
|
|
||||||
out.append("}");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
bool BluetoothDeviceRequest::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
bool BluetoothDeviceRequest::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
||||||
switch (field_id) {
|
switch (field_id) {
|
||||||
case 1: {
|
case 1: {
|
||||||
@@ -6272,432 +6060,6 @@ void BluetoothDeviceUnpairingResponse::dump_to(std::string &out) const {
|
|||||||
out.append("}");
|
out.append("}");
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
void UnsubscribeBluetoothLEAdvertisementsRequest::encode(ProtoWriteBuffer buffer) const {}
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void UnsubscribeBluetoothLEAdvertisementsRequest::dump_to(std::string &out) const {
|
|
||||||
out.append("UnsubscribeBluetoothLEAdvertisementsRequest {}");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
bool BluetoothDeviceClearCacheResponse::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 1: {
|
|
||||||
this->address = value.as_uint64();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 2: {
|
|
||||||
this->success = value.as_bool();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 3: {
|
|
||||||
this->error = value.as_int32();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void BluetoothDeviceClearCacheResponse::encode(ProtoWriteBuffer buffer) const {
|
|
||||||
buffer.encode_uint64(1, this->address);
|
|
||||||
buffer.encode_bool(2, this->success);
|
|
||||||
buffer.encode_int32(3, this->error);
|
|
||||||
}
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void BluetoothDeviceClearCacheResponse::dump_to(std::string &out) const {
|
|
||||||
__attribute__((unused)) char buffer[64];
|
|
||||||
out.append("BluetoothDeviceClearCacheResponse {\n");
|
|
||||||
out.append(" address: ");
|
|
||||||
sprintf(buffer, "%llu", this->address);
|
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" success: ");
|
|
||||||
out.append(YESNO(this->success));
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" error: ");
|
|
||||||
sprintf(buffer, "%d", this->error);
|
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
out.append("}");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
bool SubscribeVoiceAssistantRequest::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 1: {
|
|
||||||
this->subscribe = value.as_bool();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void SubscribeVoiceAssistantRequest::encode(ProtoWriteBuffer buffer) const { buffer.encode_bool(1, this->subscribe); }
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void SubscribeVoiceAssistantRequest::dump_to(std::string &out) const {
|
|
||||||
__attribute__((unused)) char buffer[64];
|
|
||||||
out.append("SubscribeVoiceAssistantRequest {\n");
|
|
||||||
out.append(" subscribe: ");
|
|
||||||
out.append(YESNO(this->subscribe));
|
|
||||||
out.append("\n");
|
|
||||||
out.append("}");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
bool VoiceAssistantRequest::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 1: {
|
|
||||||
this->start = value.as_bool();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
bool VoiceAssistantRequest::decode_length(uint32_t field_id, ProtoLengthDelimited value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 2: {
|
|
||||||
this->conversation_id = value.as_string();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void VoiceAssistantRequest::encode(ProtoWriteBuffer buffer) const {
|
|
||||||
buffer.encode_bool(1, this->start);
|
|
||||||
buffer.encode_string(2, this->conversation_id);
|
|
||||||
}
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void VoiceAssistantRequest::dump_to(std::string &out) const {
|
|
||||||
__attribute__((unused)) char buffer[64];
|
|
||||||
out.append("VoiceAssistantRequest {\n");
|
|
||||||
out.append(" start: ");
|
|
||||||
out.append(YESNO(this->start));
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" conversation_id: ");
|
|
||||||
out.append("'").append(this->conversation_id).append("'");
|
|
||||||
out.append("\n");
|
|
||||||
out.append("}");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
bool VoiceAssistantResponse::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 1: {
|
|
||||||
this->port = value.as_uint32();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 2: {
|
|
||||||
this->error = value.as_bool();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void VoiceAssistantResponse::encode(ProtoWriteBuffer buffer) const {
|
|
||||||
buffer.encode_uint32(1, this->port);
|
|
||||||
buffer.encode_bool(2, this->error);
|
|
||||||
}
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void VoiceAssistantResponse::dump_to(std::string &out) const {
|
|
||||||
__attribute__((unused)) char buffer[64];
|
|
||||||
out.append("VoiceAssistantResponse {\n");
|
|
||||||
out.append(" port: ");
|
|
||||||
sprintf(buffer, "%u", this->port);
|
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" error: ");
|
|
||||||
out.append(YESNO(this->error));
|
|
||||||
out.append("\n");
|
|
||||||
out.append("}");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
bool VoiceAssistantEventData::decode_length(uint32_t field_id, ProtoLengthDelimited value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 1: {
|
|
||||||
this->name = value.as_string();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 2: {
|
|
||||||
this->value = value.as_string();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void VoiceAssistantEventData::encode(ProtoWriteBuffer buffer) const {
|
|
||||||
buffer.encode_string(1, this->name);
|
|
||||||
buffer.encode_string(2, this->value);
|
|
||||||
}
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void VoiceAssistantEventData::dump_to(std::string &out) const {
|
|
||||||
__attribute__((unused)) char buffer[64];
|
|
||||||
out.append("VoiceAssistantEventData {\n");
|
|
||||||
out.append(" name: ");
|
|
||||||
out.append("'").append(this->name).append("'");
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" value: ");
|
|
||||||
out.append("'").append(this->value).append("'");
|
|
||||||
out.append("\n");
|
|
||||||
out.append("}");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
bool VoiceAssistantEventResponse::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 1: {
|
|
||||||
this->event_type = value.as_enum<enums::VoiceAssistantEvent>();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
bool VoiceAssistantEventResponse::decode_length(uint32_t field_id, ProtoLengthDelimited value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 2: {
|
|
||||||
this->data.push_back(value.as_message<VoiceAssistantEventData>());
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void VoiceAssistantEventResponse::encode(ProtoWriteBuffer buffer) const {
|
|
||||||
buffer.encode_enum<enums::VoiceAssistantEvent>(1, this->event_type);
|
|
||||||
for (auto &it : this->data) {
|
|
||||||
buffer.encode_message<VoiceAssistantEventData>(2, it, true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void VoiceAssistantEventResponse::dump_to(std::string &out) const {
|
|
||||||
__attribute__((unused)) char buffer[64];
|
|
||||||
out.append("VoiceAssistantEventResponse {\n");
|
|
||||||
out.append(" event_type: ");
|
|
||||||
out.append(proto_enum_to_string<enums::VoiceAssistantEvent>(this->event_type));
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
for (const auto &it : this->data) {
|
|
||||||
out.append(" data: ");
|
|
||||||
it.dump_to(out);
|
|
||||||
out.append("\n");
|
|
||||||
}
|
|
||||||
out.append("}");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
bool ListEntitiesAlarmControlPanelResponse::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 6: {
|
|
||||||
this->disabled_by_default = value.as_bool();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 7: {
|
|
||||||
this->entity_category = value.as_enum<enums::EntityCategory>();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 8: {
|
|
||||||
this->supported_features = value.as_uint32();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 9: {
|
|
||||||
this->requires_code = value.as_bool();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 10: {
|
|
||||||
this->requires_code_to_arm = value.as_bool();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
bool ListEntitiesAlarmControlPanelResponse::decode_length(uint32_t field_id, ProtoLengthDelimited value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 1: {
|
|
||||||
this->object_id = value.as_string();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 3: {
|
|
||||||
this->name = value.as_string();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 4: {
|
|
||||||
this->unique_id = value.as_string();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
case 5: {
|
|
||||||
this->icon = value.as_string();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
bool ListEntitiesAlarmControlPanelResponse::decode_32bit(uint32_t field_id, Proto32Bit value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 2: {
|
|
||||||
this->key = value.as_fixed32();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void ListEntitiesAlarmControlPanelResponse::encode(ProtoWriteBuffer buffer) const {
|
|
||||||
buffer.encode_string(1, this->object_id);
|
|
||||||
buffer.encode_fixed32(2, this->key);
|
|
||||||
buffer.encode_string(3, this->name);
|
|
||||||
buffer.encode_string(4, this->unique_id);
|
|
||||||
buffer.encode_string(5, this->icon);
|
|
||||||
buffer.encode_bool(6, this->disabled_by_default);
|
|
||||||
buffer.encode_enum<enums::EntityCategory>(7, this->entity_category);
|
|
||||||
buffer.encode_uint32(8, this->supported_features);
|
|
||||||
buffer.encode_bool(9, this->requires_code);
|
|
||||||
buffer.encode_bool(10, this->requires_code_to_arm);
|
|
||||||
}
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void ListEntitiesAlarmControlPanelResponse::dump_to(std::string &out) const {
|
|
||||||
__attribute__((unused)) char buffer[64];
|
|
||||||
out.append("ListEntitiesAlarmControlPanelResponse {\n");
|
|
||||||
out.append(" object_id: ");
|
|
||||||
out.append("'").append(this->object_id).append("'");
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" key: ");
|
|
||||||
sprintf(buffer, "%u", this->key);
|
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" name: ");
|
|
||||||
out.append("'").append(this->name).append("'");
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" unique_id: ");
|
|
||||||
out.append("'").append(this->unique_id).append("'");
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" icon: ");
|
|
||||||
out.append("'").append(this->icon).append("'");
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" disabled_by_default: ");
|
|
||||||
out.append(YESNO(this->disabled_by_default));
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" entity_category: ");
|
|
||||||
out.append(proto_enum_to_string<enums::EntityCategory>(this->entity_category));
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" supported_features: ");
|
|
||||||
sprintf(buffer, "%u", this->supported_features);
|
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" requires_code: ");
|
|
||||||
out.append(YESNO(this->requires_code));
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" requires_code_to_arm: ");
|
|
||||||
out.append(YESNO(this->requires_code_to_arm));
|
|
||||||
out.append("\n");
|
|
||||||
out.append("}");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
bool AlarmControlPanelStateResponse::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 2: {
|
|
||||||
this->state = value.as_enum<enums::AlarmControlPanelState>();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
bool AlarmControlPanelStateResponse::decode_32bit(uint32_t field_id, Proto32Bit value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 1: {
|
|
||||||
this->key = value.as_fixed32();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void AlarmControlPanelStateResponse::encode(ProtoWriteBuffer buffer) const {
|
|
||||||
buffer.encode_fixed32(1, this->key);
|
|
||||||
buffer.encode_enum<enums::AlarmControlPanelState>(2, this->state);
|
|
||||||
}
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void AlarmControlPanelStateResponse::dump_to(std::string &out) const {
|
|
||||||
__attribute__((unused)) char buffer[64];
|
|
||||||
out.append("AlarmControlPanelStateResponse {\n");
|
|
||||||
out.append(" key: ");
|
|
||||||
sprintf(buffer, "%u", this->key);
|
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" state: ");
|
|
||||||
out.append(proto_enum_to_string<enums::AlarmControlPanelState>(this->state));
|
|
||||||
out.append("\n");
|
|
||||||
out.append("}");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
bool AlarmControlPanelCommandRequest::decode_varint(uint32_t field_id, ProtoVarInt value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 2: {
|
|
||||||
this->command = value.as_enum<enums::AlarmControlPanelStateCommand>();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
bool AlarmControlPanelCommandRequest::decode_length(uint32_t field_id, ProtoLengthDelimited value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 3: {
|
|
||||||
this->code = value.as_string();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
bool AlarmControlPanelCommandRequest::decode_32bit(uint32_t field_id, Proto32Bit value) {
|
|
||||||
switch (field_id) {
|
|
||||||
case 1: {
|
|
||||||
this->key = value.as_fixed32();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void AlarmControlPanelCommandRequest::encode(ProtoWriteBuffer buffer) const {
|
|
||||||
buffer.encode_fixed32(1, this->key);
|
|
||||||
buffer.encode_enum<enums::AlarmControlPanelStateCommand>(2, this->command);
|
|
||||||
buffer.encode_string(3, this->code);
|
|
||||||
}
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void AlarmControlPanelCommandRequest::dump_to(std::string &out) const {
|
|
||||||
__attribute__((unused)) char buffer[64];
|
|
||||||
out.append("AlarmControlPanelCommandRequest {\n");
|
|
||||||
out.append(" key: ");
|
|
||||||
sprintf(buffer, "%u", this->key);
|
|
||||||
out.append(buffer);
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" command: ");
|
|
||||||
out.append(proto_enum_to_string<enums::AlarmControlPanelStateCommand>(this->command));
|
|
||||||
out.append("\n");
|
|
||||||
|
|
||||||
out.append(" code: ");
|
|
||||||
out.append("'").append(this->code).append("'");
|
|
||||||
out.append("\n");
|
|
||||||
out.append("}");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -163,39 +163,6 @@ enum BluetoothDeviceRequestType : uint32_t {
|
|||||||
BLUETOOTH_DEVICE_REQUEST_TYPE_UNPAIR = 3,
|
BLUETOOTH_DEVICE_REQUEST_TYPE_UNPAIR = 3,
|
||||||
BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITH_CACHE = 4,
|
BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITH_CACHE = 4,
|
||||||
BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITHOUT_CACHE = 5,
|
BLUETOOTH_DEVICE_REQUEST_TYPE_CONNECT_V3_WITHOUT_CACHE = 5,
|
||||||
BLUETOOTH_DEVICE_REQUEST_TYPE_CLEAR_CACHE = 6,
|
|
||||||
};
|
|
||||||
enum VoiceAssistantEvent : uint32_t {
|
|
||||||
VOICE_ASSISTANT_ERROR = 0,
|
|
||||||
VOICE_ASSISTANT_RUN_START = 1,
|
|
||||||
VOICE_ASSISTANT_RUN_END = 2,
|
|
||||||
VOICE_ASSISTANT_STT_START = 3,
|
|
||||||
VOICE_ASSISTANT_STT_END = 4,
|
|
||||||
VOICE_ASSISTANT_INTENT_START = 5,
|
|
||||||
VOICE_ASSISTANT_INTENT_END = 6,
|
|
||||||
VOICE_ASSISTANT_TTS_START = 7,
|
|
||||||
VOICE_ASSISTANT_TTS_END = 8,
|
|
||||||
};
|
|
||||||
enum AlarmControlPanelState : uint32_t {
|
|
||||||
ALARM_STATE_DISARMED = 0,
|
|
||||||
ALARM_STATE_ARMED_HOME = 1,
|
|
||||||
ALARM_STATE_ARMED_AWAY = 2,
|
|
||||||
ALARM_STATE_ARMED_NIGHT = 3,
|
|
||||||
ALARM_STATE_ARMED_VACATION = 4,
|
|
||||||
ALARM_STATE_ARMED_CUSTOM_BYPASS = 5,
|
|
||||||
ALARM_STATE_PENDING = 6,
|
|
||||||
ALARM_STATE_ARMING = 7,
|
|
||||||
ALARM_STATE_DISARMING = 8,
|
|
||||||
ALARM_STATE_TRIGGERED = 9,
|
|
||||||
};
|
|
||||||
enum AlarmControlPanelStateCommand : uint32_t {
|
|
||||||
ALARM_CONTROL_PANEL_DISARM = 0,
|
|
||||||
ALARM_CONTROL_PANEL_ARM_AWAY = 1,
|
|
||||||
ALARM_CONTROL_PANEL_ARM_HOME = 2,
|
|
||||||
ALARM_CONTROL_PANEL_ARM_NIGHT = 3,
|
|
||||||
ALARM_CONTROL_PANEL_ARM_VACATION = 4,
|
|
||||||
ALARM_CONTROL_PANEL_ARM_CUSTOM_BYPASS = 5,
|
|
||||||
ALARM_CONTROL_PANEL_TRIGGER = 6,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace enums
|
} // namespace enums
|
||||||
@@ -308,11 +275,9 @@ class DeviceInfoResponse : public ProtoMessage {
|
|||||||
std::string project_name{};
|
std::string project_name{};
|
||||||
std::string project_version{};
|
std::string project_version{};
|
||||||
uint32_t webserver_port{0};
|
uint32_t webserver_port{0};
|
||||||
uint32_t legacy_bluetooth_proxy_version{0};
|
uint32_t bluetooth_proxy_version{0};
|
||||||
uint32_t bluetooth_proxy_feature_flags{0};
|
|
||||||
std::string manufacturer{};
|
std::string manufacturer{};
|
||||||
std::string friendly_name{};
|
std::string friendly_name{};
|
||||||
uint32_t voice_assistant_version{0};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
void encode(ProtoWriteBuffer buffer) const override;
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
void dump_to(std::string &out) const override;
|
void dump_to(std::string &out) const override;
|
||||||
@@ -397,7 +362,6 @@ class ListEntitiesCoverResponse : public ProtoMessage {
|
|||||||
bool disabled_by_default{false};
|
bool disabled_by_default{false};
|
||||||
std::string icon{};
|
std::string icon{};
|
||||||
enums::EntityCategory entity_category{};
|
enums::EntityCategory entity_category{};
|
||||||
bool supports_stop{false};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
void encode(ProtoWriteBuffer buffer) const override;
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
void dump_to(std::string &out) const override;
|
void dump_to(std::string &out) const override;
|
||||||
@@ -981,7 +945,7 @@ class ClimateStateResponse : public ProtoMessage {
|
|||||||
float target_temperature{0.0f};
|
float target_temperature{0.0f};
|
||||||
float target_temperature_low{0.0f};
|
float target_temperature_low{0.0f};
|
||||||
float target_temperature_high{0.0f};
|
float target_temperature_high{0.0f};
|
||||||
bool unused_legacy_away{false};
|
bool legacy_away{false};
|
||||||
enums::ClimateAction action{};
|
enums::ClimateAction action{};
|
||||||
enums::ClimateFanMode fan_mode{};
|
enums::ClimateFanMode fan_mode{};
|
||||||
enums::ClimateSwingMode swing_mode{};
|
enums::ClimateSwingMode swing_mode{};
|
||||||
@@ -1009,8 +973,8 @@ class ClimateCommandRequest : public ProtoMessage {
|
|||||||
float target_temperature_low{0.0f};
|
float target_temperature_low{0.0f};
|
||||||
bool has_target_temperature_high{false};
|
bool has_target_temperature_high{false};
|
||||||
float target_temperature_high{0.0f};
|
float target_temperature_high{0.0f};
|
||||||
bool unused_has_legacy_away{false};
|
bool has_legacy_away{false};
|
||||||
bool unused_legacy_away{false};
|
bool legacy_away{false};
|
||||||
bool has_fan_mode{false};
|
bool has_fan_mode{false};
|
||||||
enums::ClimateFanMode fan_mode{};
|
enums::ClimateFanMode fan_mode{};
|
||||||
bool has_swing_mode{false};
|
bool has_swing_mode{false};
|
||||||
@@ -1269,14 +1233,12 @@ class MediaPlayerCommandRequest : public ProtoMessage {
|
|||||||
};
|
};
|
||||||
class SubscribeBluetoothLEAdvertisementsRequest : public ProtoMessage {
|
class SubscribeBluetoothLEAdvertisementsRequest : public ProtoMessage {
|
||||||
public:
|
public:
|
||||||
uint32_t flags{0};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
void encode(ProtoWriteBuffer buffer) const override;
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
void dump_to(std::string &out) const override;
|
void dump_to(std::string &out) const override;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
|
||||||
};
|
};
|
||||||
class BluetoothServiceData : public ProtoMessage {
|
class BluetoothServiceData : public ProtoMessage {
|
||||||
public:
|
public:
|
||||||
@@ -1310,32 +1272,6 @@ class BluetoothLEAdvertisementResponse : public ProtoMessage {
|
|||||||
bool decode_length(uint32_t field_id, ProtoLengthDelimited value) override;
|
bool decode_length(uint32_t field_id, ProtoLengthDelimited value) override;
|
||||||
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
||||||
};
|
};
|
||||||
class BluetoothLERawAdvertisement : public ProtoMessage {
|
|
||||||
public:
|
|
||||||
uint64_t address{0};
|
|
||||||
int32_t rssi{0};
|
|
||||||
uint32_t address_type{0};
|
|
||||||
std::string data{};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void dump_to(std::string &out) const override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool decode_length(uint32_t field_id, ProtoLengthDelimited value) override;
|
|
||||||
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
|
||||||
};
|
|
||||||
class BluetoothLERawAdvertisementsResponse : public ProtoMessage {
|
|
||||||
public:
|
|
||||||
std::vector<BluetoothLERawAdvertisement> advertisements{};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void dump_to(std::string &out) const override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool decode_length(uint32_t field_id, ProtoLengthDelimited value) override;
|
|
||||||
};
|
|
||||||
class BluetoothDeviceRequest : public ProtoMessage {
|
class BluetoothDeviceRequest : public ProtoMessage {
|
||||||
public:
|
public:
|
||||||
uint64_t address{0};
|
uint64_t address{0};
|
||||||
@@ -1618,139 +1554,6 @@ class BluetoothDeviceUnpairingResponse : public ProtoMessage {
|
|||||||
protected:
|
protected:
|
||||||
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
||||||
};
|
};
|
||||||
class UnsubscribeBluetoothLEAdvertisementsRequest : public ProtoMessage {
|
|
||||||
public:
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void dump_to(std::string &out) const override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
protected:
|
|
||||||
};
|
|
||||||
class BluetoothDeviceClearCacheResponse : public ProtoMessage {
|
|
||||||
public:
|
|
||||||
uint64_t address{0};
|
|
||||||
bool success{false};
|
|
||||||
int32_t error{0};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void dump_to(std::string &out) const override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
|
||||||
};
|
|
||||||
class SubscribeVoiceAssistantRequest : public ProtoMessage {
|
|
||||||
public:
|
|
||||||
bool subscribe{false};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void dump_to(std::string &out) const override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
|
||||||
};
|
|
||||||
class VoiceAssistantRequest : public ProtoMessage {
|
|
||||||
public:
|
|
||||||
bool start{false};
|
|
||||||
std::string conversation_id{};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void dump_to(std::string &out) const override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool decode_length(uint32_t field_id, ProtoLengthDelimited value) override;
|
|
||||||
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
|
||||||
};
|
|
||||||
class VoiceAssistantResponse : public ProtoMessage {
|
|
||||||
public:
|
|
||||||
uint32_t port{0};
|
|
||||||
bool error{false};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void dump_to(std::string &out) const override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
|
||||||
};
|
|
||||||
class VoiceAssistantEventData : public ProtoMessage {
|
|
||||||
public:
|
|
||||||
std::string name{};
|
|
||||||
std::string value{};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void dump_to(std::string &out) const override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool decode_length(uint32_t field_id, ProtoLengthDelimited value) override;
|
|
||||||
};
|
|
||||||
class VoiceAssistantEventResponse : public ProtoMessage {
|
|
||||||
public:
|
|
||||||
enums::VoiceAssistantEvent event_type{};
|
|
||||||
std::vector<VoiceAssistantEventData> data{};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void dump_to(std::string &out) const override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool decode_length(uint32_t field_id, ProtoLengthDelimited value) override;
|
|
||||||
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
|
||||||
};
|
|
||||||
class ListEntitiesAlarmControlPanelResponse : public ProtoMessage {
|
|
||||||
public:
|
|
||||||
std::string object_id{};
|
|
||||||
uint32_t key{0};
|
|
||||||
std::string name{};
|
|
||||||
std::string unique_id{};
|
|
||||||
std::string icon{};
|
|
||||||
bool disabled_by_default{false};
|
|
||||||
enums::EntityCategory entity_category{};
|
|
||||||
uint32_t supported_features{0};
|
|
||||||
bool requires_code{false};
|
|
||||||
bool requires_code_to_arm{false};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void dump_to(std::string &out) const override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool decode_32bit(uint32_t field_id, Proto32Bit value) override;
|
|
||||||
bool decode_length(uint32_t field_id, ProtoLengthDelimited value) override;
|
|
||||||
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
|
||||||
};
|
|
||||||
class AlarmControlPanelStateResponse : public ProtoMessage {
|
|
||||||
public:
|
|
||||||
uint32_t key{0};
|
|
||||||
enums::AlarmControlPanelState state{};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void dump_to(std::string &out) const override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool decode_32bit(uint32_t field_id, Proto32Bit value) override;
|
|
||||||
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
|
||||||
};
|
|
||||||
class AlarmControlPanelCommandRequest : public ProtoMessage {
|
|
||||||
public:
|
|
||||||
uint32_t key{0};
|
|
||||||
enums::AlarmControlPanelStateCommand command{};
|
|
||||||
std::string code{};
|
|
||||||
void encode(ProtoWriteBuffer buffer) const override;
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
void dump_to(std::string &out) const override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
protected:
|
|
||||||
bool decode_32bit(uint32_t field_id, Proto32Bit value) override;
|
|
||||||
bool decode_length(uint32_t field_id, ProtoLengthDelimited value) override;
|
|
||||||
bool decode_varint(uint32_t field_id, ProtoVarInt value) override;
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -329,8 +329,6 @@ bool APIServerConnectionBase::send_media_player_state_response(const MediaPlayer
|
|||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool APIServerConnectionBase::send_bluetooth_le_advertisement_response(const BluetoothLEAdvertisementResponse &msg) {
|
bool APIServerConnectionBase::send_bluetooth_le_advertisement_response(const BluetoothLEAdvertisementResponse &msg) {
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
ESP_LOGVV(TAG, "send_bluetooth_le_advertisement_response: %s", msg.dump().c_str());
|
ESP_LOGVV(TAG, "send_bluetooth_le_advertisement_response: %s", msg.dump().c_str());
|
||||||
@@ -339,15 +337,6 @@ bool APIServerConnectionBase::send_bluetooth_le_advertisement_response(const Blu
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
bool APIServerConnectionBase::send_bluetooth_le_raw_advertisements_response(
|
|
||||||
const BluetoothLERawAdvertisementsResponse &msg) {
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
ESP_LOGVV(TAG, "send_bluetooth_le_raw_advertisements_response: %s", msg.dump().c_str());
|
|
||||||
#endif
|
|
||||||
return this->send_message_<BluetoothLERawAdvertisementsResponse>(msg, 93);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
bool APIServerConnectionBase::send_bluetooth_device_connection_response(const BluetoothDeviceConnectionResponse &msg) {
|
bool APIServerConnectionBase::send_bluetooth_device_connection_response(const BluetoothDeviceConnectionResponse &msg) {
|
||||||
@@ -452,49 +441,6 @@ bool APIServerConnectionBase::send_bluetooth_device_unpairing_response(const Blu
|
|||||||
return this->send_message_<BluetoothDeviceUnpairingResponse>(msg, 86);
|
return this->send_message_<BluetoothDeviceUnpairingResponse>(msg, 86);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool APIServerConnectionBase::send_bluetooth_device_clear_cache_response(const BluetoothDeviceClearCacheResponse &msg) {
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
ESP_LOGVV(TAG, "send_bluetooth_device_clear_cache_response: %s", msg.dump().c_str());
|
|
||||||
#endif
|
|
||||||
return this->send_message_<BluetoothDeviceClearCacheResponse>(msg, 88);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
bool APIServerConnectionBase::send_voice_assistant_request(const VoiceAssistantRequest &msg) {
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
ESP_LOGVV(TAG, "send_voice_assistant_request: %s", msg.dump().c_str());
|
|
||||||
#endif
|
|
||||||
return this->send_message_<VoiceAssistantRequest>(msg, 90);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
bool APIServerConnectionBase::send_list_entities_alarm_control_panel_response(
|
|
||||||
const ListEntitiesAlarmControlPanelResponse &msg) {
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
ESP_LOGVV(TAG, "send_list_entities_alarm_control_panel_response: %s", msg.dump().c_str());
|
|
||||||
#endif
|
|
||||||
return this->send_message_<ListEntitiesAlarmControlPanelResponse>(msg, 94);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
bool APIServerConnectionBase::send_alarm_control_panel_state_response(const AlarmControlPanelStateResponse &msg) {
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
ESP_LOGVV(TAG, "send_alarm_control_panel_state_response: %s", msg.dump().c_str());
|
|
||||||
#endif
|
|
||||||
return this->send_message_<AlarmControlPanelStateResponse>(msg, 95);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
#endif
|
|
||||||
bool APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type, uint8_t *msg_data) {
|
bool APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type, uint8_t *msg_data) {
|
||||||
switch (msg_type) {
|
switch (msg_type) {
|
||||||
case 1: {
|
case 1: {
|
||||||
@@ -763,14 +709,12 @@ bool APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type,
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case 66: {
|
case 66: {
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
SubscribeBluetoothLEAdvertisementsRequest msg;
|
SubscribeBluetoothLEAdvertisementsRequest msg;
|
||||||
msg.decode(msg_data, msg_size);
|
msg.decode(msg_data, msg_size);
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
#ifdef HAS_PROTO_MESSAGE_DUMP
|
||||||
ESP_LOGVV(TAG, "on_subscribe_bluetooth_le_advertisements_request: %s", msg.dump().c_str());
|
ESP_LOGVV(TAG, "on_subscribe_bluetooth_le_advertisements_request: %s", msg.dump().c_str());
|
||||||
#endif
|
#endif
|
||||||
this->on_subscribe_bluetooth_le_advertisements_request(msg);
|
this->on_subscribe_bluetooth_le_advertisements_request(msg);
|
||||||
#endif
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case 68: {
|
case 68: {
|
||||||
@@ -858,61 +802,6 @@ bool APIServerConnectionBase::read_message(uint32_t msg_size, uint32_t msg_type,
|
|||||||
ESP_LOGVV(TAG, "on_subscribe_bluetooth_connections_free_request: %s", msg.dump().c_str());
|
ESP_LOGVV(TAG, "on_subscribe_bluetooth_connections_free_request: %s", msg.dump().c_str());
|
||||||
#endif
|
#endif
|
||||||
this->on_subscribe_bluetooth_connections_free_request(msg);
|
this->on_subscribe_bluetooth_connections_free_request(msg);
|
||||||
#endif
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case 87: {
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
UnsubscribeBluetoothLEAdvertisementsRequest msg;
|
|
||||||
msg.decode(msg_data, msg_size);
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
ESP_LOGVV(TAG, "on_unsubscribe_bluetooth_le_advertisements_request: %s", msg.dump().c_str());
|
|
||||||
#endif
|
|
||||||
this->on_unsubscribe_bluetooth_le_advertisements_request(msg);
|
|
||||||
#endif
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case 89: {
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
SubscribeVoiceAssistantRequest msg;
|
|
||||||
msg.decode(msg_data, msg_size);
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
ESP_LOGVV(TAG, "on_subscribe_voice_assistant_request: %s", msg.dump().c_str());
|
|
||||||
#endif
|
|
||||||
this->on_subscribe_voice_assistant_request(msg);
|
|
||||||
#endif
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case 91: {
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
VoiceAssistantResponse msg;
|
|
||||||
msg.decode(msg_data, msg_size);
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
ESP_LOGVV(TAG, "on_voice_assistant_response: %s", msg.dump().c_str());
|
|
||||||
#endif
|
|
||||||
this->on_voice_assistant_response(msg);
|
|
||||||
#endif
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case 92: {
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
VoiceAssistantEventResponse msg;
|
|
||||||
msg.decode(msg_data, msg_size);
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
ESP_LOGVV(TAG, "on_voice_assistant_event_response: %s", msg.dump().c_str());
|
|
||||||
#endif
|
|
||||||
this->on_voice_assistant_event_response(msg);
|
|
||||||
#endif
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case 96: {
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
AlarmControlPanelCommandRequest msg;
|
|
||||||
msg.decode(msg_data, msg_size);
|
|
||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
|
|
||||||
ESP_LOGVV(TAG, "on_alarm_control_panel_command_request: %s", msg.dump().c_str());
|
|
||||||
#endif
|
|
||||||
this->on_alarm_control_panel_command_request(msg);
|
|
||||||
#endif
|
#endif
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -1176,7 +1065,6 @@ void APIServerConnection::on_media_player_command_request(const MediaPlayerComma
|
|||||||
this->media_player_command(msg);
|
this->media_player_command(msg);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
void APIServerConnection::on_subscribe_bluetooth_le_advertisements_request(
|
void APIServerConnection::on_subscribe_bluetooth_le_advertisements_request(
|
||||||
const SubscribeBluetoothLEAdvertisementsRequest &msg) {
|
const SubscribeBluetoothLEAdvertisementsRequest &msg) {
|
||||||
if (!this->is_connection_setup()) {
|
if (!this->is_connection_setup()) {
|
||||||
@@ -1189,7 +1077,6 @@ void APIServerConnection::on_subscribe_bluetooth_le_advertisements_request(
|
|||||||
}
|
}
|
||||||
this->subscribe_bluetooth_le_advertisements(msg);
|
this->subscribe_bluetooth_le_advertisements(msg);
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
void APIServerConnection::on_bluetooth_device_request(const BluetoothDeviceRequest &msg) {
|
void APIServerConnection::on_bluetooth_device_request(const BluetoothDeviceRequest &msg) {
|
||||||
if (!this->is_connection_setup()) {
|
if (!this->is_connection_setup()) {
|
||||||
@@ -1298,46 +1185,6 @@ void APIServerConnection::on_subscribe_bluetooth_connections_free_request(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
void APIServerConnection::on_unsubscribe_bluetooth_le_advertisements_request(
|
|
||||||
const UnsubscribeBluetoothLEAdvertisementsRequest &msg) {
|
|
||||||
if (!this->is_connection_setup()) {
|
|
||||||
this->on_no_setup_connection();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (!this->is_authenticated()) {
|
|
||||||
this->on_unauthenticated_access();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this->unsubscribe_bluetooth_le_advertisements(msg);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
void APIServerConnection::on_subscribe_voice_assistant_request(const SubscribeVoiceAssistantRequest &msg) {
|
|
||||||
if (!this->is_connection_setup()) {
|
|
||||||
this->on_no_setup_connection();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (!this->is_authenticated()) {
|
|
||||||
this->on_unauthenticated_access();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this->subscribe_voice_assistant(msg);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
void APIServerConnection::on_alarm_control_panel_command_request(const AlarmControlPanelCommandRequest &msg) {
|
|
||||||
if (!this->is_connection_setup()) {
|
|
||||||
this->on_no_setup_connection();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (!this->is_authenticated()) {
|
|
||||||
this->on_unauthenticated_access();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this->alarm_control_panel_command(msg);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -154,16 +154,11 @@ class APIServerConnectionBase : public ProtoService {
|
|||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
virtual void on_media_player_command_request(const MediaPlayerCommandRequest &value){};
|
virtual void on_media_player_command_request(const MediaPlayerCommandRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
virtual void on_subscribe_bluetooth_le_advertisements_request(
|
virtual void on_subscribe_bluetooth_le_advertisements_request(
|
||||||
const SubscribeBluetoothLEAdvertisementsRequest &value){};
|
const SubscribeBluetoothLEAdvertisementsRequest &value){};
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
bool send_bluetooth_le_advertisement_response(const BluetoothLEAdvertisementResponse &msg);
|
bool send_bluetooth_le_advertisement_response(const BluetoothLEAdvertisementResponse &msg);
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_le_raw_advertisements_response(const BluetoothLERawAdvertisementsResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual void on_bluetooth_device_request(const BluetoothDeviceRequest &value){};
|
virtual void on_bluetooth_device_request(const BluetoothDeviceRequest &value){};
|
||||||
#endif
|
#endif
|
||||||
@@ -220,34 +215,6 @@ class APIServerConnectionBase : public ProtoService {
|
|||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
bool send_bluetooth_device_unpairing_response(const BluetoothDeviceUnpairingResponse &msg);
|
bool send_bluetooth_device_unpairing_response(const BluetoothDeviceUnpairingResponse &msg);
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
virtual void on_unsubscribe_bluetooth_le_advertisements_request(
|
|
||||||
const UnsubscribeBluetoothLEAdvertisementsRequest &value){};
|
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
bool send_bluetooth_device_clear_cache_response(const BluetoothDeviceClearCacheResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
virtual void on_subscribe_voice_assistant_request(const SubscribeVoiceAssistantRequest &value){};
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
bool send_voice_assistant_request(const VoiceAssistantRequest &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
virtual void on_voice_assistant_response(const VoiceAssistantResponse &value){};
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
virtual void on_voice_assistant_event_response(const VoiceAssistantEventResponse &value){};
|
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
bool send_list_entities_alarm_control_panel_response(const ListEntitiesAlarmControlPanelResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
bool send_alarm_control_panel_state_response(const AlarmControlPanelStateResponse &msg);
|
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
virtual void on_alarm_control_panel_command_request(const AlarmControlPanelCommandRequest &value){};
|
|
||||||
#endif
|
#endif
|
||||||
protected:
|
protected:
|
||||||
bool read_message(uint32_t msg_size, uint32_t msg_type, uint8_t *msg_data) override;
|
bool read_message(uint32_t msg_size, uint32_t msg_type, uint8_t *msg_data) override;
|
||||||
@@ -300,9 +267,7 @@ class APIServerConnection : public APIServerConnectionBase {
|
|||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
virtual void media_player_command(const MediaPlayerCommandRequest &msg) = 0;
|
virtual void media_player_command(const MediaPlayerCommandRequest &msg) = 0;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
virtual void subscribe_bluetooth_le_advertisements(const SubscribeBluetoothLEAdvertisementsRequest &msg) = 0;
|
virtual void subscribe_bluetooth_le_advertisements(const SubscribeBluetoothLEAdvertisementsRequest &msg) = 0;
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual void bluetooth_device_request(const BluetoothDeviceRequest &msg) = 0;
|
virtual void bluetooth_device_request(const BluetoothDeviceRequest &msg) = 0;
|
||||||
#endif
|
#endif
|
||||||
@@ -327,15 +292,6 @@ class APIServerConnection : public APIServerConnectionBase {
|
|||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
virtual BluetoothConnectionsFreeResponse subscribe_bluetooth_connections_free(
|
virtual BluetoothConnectionsFreeResponse subscribe_bluetooth_connections_free(
|
||||||
const SubscribeBluetoothConnectionsFreeRequest &msg) = 0;
|
const SubscribeBluetoothConnectionsFreeRequest &msg) = 0;
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
virtual void unsubscribe_bluetooth_le_advertisements(const UnsubscribeBluetoothLEAdvertisementsRequest &msg) = 0;
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
virtual void subscribe_voice_assistant(const SubscribeVoiceAssistantRequest &msg) = 0;
|
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
virtual void alarm_control_panel_command(const AlarmControlPanelCommandRequest &msg) = 0;
|
|
||||||
#endif
|
#endif
|
||||||
protected:
|
protected:
|
||||||
void on_hello_request(const HelloRequest &msg) override;
|
void on_hello_request(const HelloRequest &msg) override;
|
||||||
@@ -383,9 +339,7 @@ class APIServerConnection : public APIServerConnectionBase {
|
|||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
void on_media_player_command_request(const MediaPlayerCommandRequest &msg) override;
|
void on_media_player_command_request(const MediaPlayerCommandRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
void on_subscribe_bluetooth_le_advertisements_request(const SubscribeBluetoothLEAdvertisementsRequest &msg) override;
|
void on_subscribe_bluetooth_le_advertisements_request(const SubscribeBluetoothLEAdvertisementsRequest &msg) override;
|
||||||
#endif
|
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
void on_bluetooth_device_request(const BluetoothDeviceRequest &msg) override;
|
void on_bluetooth_device_request(const BluetoothDeviceRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
@@ -410,16 +364,6 @@ class APIServerConnection : public APIServerConnectionBase {
|
|||||||
#ifdef USE_BLUETOOTH_PROXY
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
void on_subscribe_bluetooth_connections_free_request(const SubscribeBluetoothConnectionsFreeRequest &msg) override;
|
void on_subscribe_bluetooth_connections_free_request(const SubscribeBluetoothConnectionsFreeRequest &msg) override;
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_BLUETOOTH_PROXY
|
|
||||||
void on_unsubscribe_bluetooth_le_advertisements_request(
|
|
||||||
const UnsubscribeBluetoothLEAdvertisementsRequest &msg) override;
|
|
||||||
#endif
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
void on_subscribe_voice_assistant_request(const SubscribeVoiceAssistantRequest &msg) override;
|
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
void on_alarm_control_panel_command_request(const AlarmControlPanelCommandRequest &msg) override;
|
|
||||||
#endif
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
|
|||||||
@@ -45,7 +45,7 @@ void APIServer::setup() {
|
|||||||
|
|
||||||
struct sockaddr_storage server;
|
struct sockaddr_storage server;
|
||||||
|
|
||||||
socklen_t sl = socket::set_sockaddr_any((struct sockaddr *) &server, sizeof(server), this->port_);
|
socklen_t sl = socket::set_sockaddr_any((struct sockaddr *) &server, sizeof(server), htons(this->port_));
|
||||||
if (sl == 0) {
|
if (sl == 0) {
|
||||||
ESP_LOGW(TAG, "Socket unable to set sockaddr: errno %d", errno);
|
ESP_LOGW(TAG, "Socket unable to set sockaddr: errno %d", errno);
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
@@ -291,7 +291,101 @@ void APIServer::send_homeassistant_service_call(const HomeassistantServiceRespon
|
|||||||
client->send_homeassistant_service_call(call);
|
client->send_homeassistant_service_call(call);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void APIServer::send_bluetooth_le_advertisement(const BluetoothLEAdvertisementResponse &call) {
|
||||||
|
for (auto &client : this->clients_) {
|
||||||
|
client->send_bluetooth_le_advertisement(call);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void APIServer::send_bluetooth_device_connection(uint64_t address, bool connected, uint16_t mtu, esp_err_t error) {
|
||||||
|
BluetoothDeviceConnectionResponse call;
|
||||||
|
call.address = address;
|
||||||
|
call.connected = connected;
|
||||||
|
call.mtu = mtu;
|
||||||
|
call.error = error;
|
||||||
|
|
||||||
|
for (auto &client : this->clients_) {
|
||||||
|
client->send_bluetooth_device_connection_response(call);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void APIServer::send_bluetooth_device_pairing(uint64_t address, bool paired, esp_err_t error) {
|
||||||
|
BluetoothDevicePairingResponse call;
|
||||||
|
call.address = address;
|
||||||
|
call.paired = paired;
|
||||||
|
call.error = error;
|
||||||
|
|
||||||
|
for (auto &client : this->clients_) {
|
||||||
|
client->send_bluetooth_device_pairing_response(call);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void APIServer::send_bluetooth_device_unpairing(uint64_t address, bool success, esp_err_t error) {
|
||||||
|
BluetoothDeviceUnpairingResponse call;
|
||||||
|
call.address = address;
|
||||||
|
call.success = success;
|
||||||
|
call.error = error;
|
||||||
|
|
||||||
|
for (auto &client : this->clients_) {
|
||||||
|
client->send_bluetooth_device_unpairing_response(call);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void APIServer::send_bluetooth_connections_free(uint8_t free, uint8_t limit) {
|
||||||
|
BluetoothConnectionsFreeResponse call;
|
||||||
|
call.free = free;
|
||||||
|
call.limit = limit;
|
||||||
|
|
||||||
|
for (auto &client : this->clients_) {
|
||||||
|
client->send_bluetooth_connections_free_response(call);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void APIServer::send_bluetooth_gatt_read_response(const BluetoothGATTReadResponse &call) {
|
||||||
|
for (auto &client : this->clients_) {
|
||||||
|
client->send_bluetooth_gatt_read_response(call);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void APIServer::send_bluetooth_gatt_write_response(const BluetoothGATTWriteResponse &call) {
|
||||||
|
for (auto &client : this->clients_) {
|
||||||
|
client->send_bluetooth_gatt_write_response(call);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void APIServer::send_bluetooth_gatt_notify_data_response(const BluetoothGATTNotifyDataResponse &call) {
|
||||||
|
for (auto &client : this->clients_) {
|
||||||
|
client->send_bluetooth_gatt_notify_data_response(call);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void APIServer::send_bluetooth_gatt_notify_response(const BluetoothGATTNotifyResponse &call) {
|
||||||
|
for (auto &client : this->clients_) {
|
||||||
|
client->send_bluetooth_gatt_notify_response(call);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void APIServer::send_bluetooth_gatt_services(const BluetoothGATTGetServicesResponse &call) {
|
||||||
|
for (auto &client : this->clients_) {
|
||||||
|
client->send_bluetooth_gatt_get_services_response(call);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void APIServer::send_bluetooth_gatt_services_done(uint64_t address) {
|
||||||
|
BluetoothGATTGetServicesDoneResponse call;
|
||||||
|
call.address = address;
|
||||||
|
|
||||||
|
for (auto &client : this->clients_) {
|
||||||
|
client->send_bluetooth_gatt_get_services_done_response(call);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void APIServer::send_bluetooth_gatt_error(uint64_t address, uint16_t handle, esp_err_t error) {
|
||||||
|
BluetoothGATTErrorResponse call;
|
||||||
|
call.address = address;
|
||||||
|
call.handle = handle;
|
||||||
|
call.error = error;
|
||||||
|
|
||||||
|
for (auto &client : this->clients_) {
|
||||||
|
client->send_bluetooth_gatt_error_response(call);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
APIServer::APIServer() { global_api_server = this; }
|
APIServer::APIServer() { global_api_server = this; }
|
||||||
void APIServer::subscribe_home_assistant_state(std::string entity_id, optional<std::string> attribute,
|
void APIServer::subscribe_home_assistant_state(std::string entity_id, optional<std::string> attribute,
|
||||||
std::function<void(std::string)> f) {
|
std::function<void(std::string)> f) {
|
||||||
@@ -322,30 +416,5 @@ void APIServer::on_shutdown() {
|
|||||||
delay(10);
|
delay(10);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
bool APIServer::start_voice_assistant(const std::string &conversation_id) {
|
|
||||||
for (auto &c : this->clients_) {
|
|
||||||
if (c->request_voice_assistant(true, conversation_id))
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
void APIServer::stop_voice_assistant() {
|
|
||||||
for (auto &c : this->clients_) {
|
|
||||||
if (c->request_voice_assistant(false, ""))
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
void APIServer::on_alarm_control_panel_update(alarm_control_panel::AlarmControlPanel *obj) {
|
|
||||||
if (obj->is_internal())
|
|
||||||
return;
|
|
||||||
for (auto &c : this->clients_)
|
|
||||||
c->send_alarm_control_panel_state(obj);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -75,20 +75,25 @@ class APIServer : public Component, public Controller {
|
|||||||
void on_media_player_update(media_player::MediaPlayer *obj) override;
|
void on_media_player_update(media_player::MediaPlayer *obj) override;
|
||||||
#endif
|
#endif
|
||||||
void send_homeassistant_service_call(const HomeassistantServiceResponse &call);
|
void send_homeassistant_service_call(const HomeassistantServiceResponse &call);
|
||||||
|
#ifdef USE_BLUETOOTH_PROXY
|
||||||
|
void send_bluetooth_le_advertisement(const BluetoothLEAdvertisementResponse &call);
|
||||||
|
void send_bluetooth_device_connection(uint64_t address, bool connected, uint16_t mtu = 0, esp_err_t error = ESP_OK);
|
||||||
|
void send_bluetooth_device_pairing(uint64_t address, bool paired, esp_err_t error = ESP_OK);
|
||||||
|
void send_bluetooth_device_unpairing(uint64_t address, bool success, esp_err_t error = ESP_OK);
|
||||||
|
void send_bluetooth_connections_free(uint8_t free, uint8_t limit);
|
||||||
|
void send_bluetooth_gatt_read_response(const BluetoothGATTReadResponse &call);
|
||||||
|
void send_bluetooth_gatt_write_response(const BluetoothGATTWriteResponse &call);
|
||||||
|
void send_bluetooth_gatt_notify_data_response(const BluetoothGATTNotifyDataResponse &call);
|
||||||
|
void send_bluetooth_gatt_notify_response(const BluetoothGATTNotifyResponse &call);
|
||||||
|
void send_bluetooth_gatt_services(const BluetoothGATTGetServicesResponse &call);
|
||||||
|
void send_bluetooth_gatt_services_done(uint64_t address);
|
||||||
|
void send_bluetooth_gatt_error(uint64_t address, uint16_t handle, esp_err_t error);
|
||||||
|
#endif
|
||||||
void register_user_service(UserServiceDescriptor *descriptor) { this->user_services_.push_back(descriptor); }
|
void register_user_service(UserServiceDescriptor *descriptor) { this->user_services_.push_back(descriptor); }
|
||||||
#ifdef USE_HOMEASSISTANT_TIME
|
#ifdef USE_HOMEASSISTANT_TIME
|
||||||
void request_time();
|
void request_time();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef USE_VOICE_ASSISTANT
|
|
||||||
bool start_voice_assistant(const std::string &conversation_id);
|
|
||||||
void stop_voice_assistant();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
void on_alarm_control_panel_update(alarm_control_panel::AlarmControlPanel *obj) override;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
bool is_connected() const;
|
bool is_connected() const;
|
||||||
|
|
||||||
struct HomeAssistantStateSubscription {
|
struct HomeAssistantStateSubscription {
|
||||||
|
|||||||
@@ -69,11 +69,6 @@ bool ListEntitiesIterator::on_media_player(media_player::MediaPlayer *media_play
|
|||||||
return this->client_->send_media_player_info(media_player);
|
return this->client_->send_media_player_info(media_player);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
bool ListEntitiesIterator::on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) {
|
|
||||||
return this->client_->send_alarm_control_panel_info(a_alarm_control_panel);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -54,9 +54,6 @@ class ListEntitiesIterator : public ComponentIterator {
|
|||||||
#endif
|
#endif
|
||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
bool on_media_player(media_player::MediaPlayer *media_player) override;
|
bool on_media_player(media_player::MediaPlayer *media_player) override;
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
bool on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) override;
|
|
||||||
#endif
|
#endif
|
||||||
bool on_end() override;
|
bool on_end() override;
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
#include "proto.h"
|
#include "proto.h"
|
||||||
#include <cinttypes>
|
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
@@ -14,7 +13,7 @@ void ProtoMessage::decode(const uint8_t *buffer, size_t length) {
|
|||||||
uint32_t consumed;
|
uint32_t consumed;
|
||||||
auto res = ProtoVarInt::parse(&buffer[i], length - i, &consumed);
|
auto res = ProtoVarInt::parse(&buffer[i], length - i, &consumed);
|
||||||
if (!res.has_value()) {
|
if (!res.has_value()) {
|
||||||
ESP_LOGV(TAG, "Invalid field start at %" PRIu32, i);
|
ESP_LOGV(TAG, "Invalid field start at %u", i);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -26,12 +25,12 @@ void ProtoMessage::decode(const uint8_t *buffer, size_t length) {
|
|||||||
case 0: { // VarInt
|
case 0: { // VarInt
|
||||||
res = ProtoVarInt::parse(&buffer[i], length - i, &consumed);
|
res = ProtoVarInt::parse(&buffer[i], length - i, &consumed);
|
||||||
if (!res.has_value()) {
|
if (!res.has_value()) {
|
||||||
ESP_LOGV(TAG, "Invalid VarInt at %" PRIu32, i);
|
ESP_LOGV(TAG, "Invalid VarInt at %u", i);
|
||||||
error = true;
|
error = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (!this->decode_varint(field_id, *res)) {
|
if (!this->decode_varint(field_id, *res)) {
|
||||||
ESP_LOGV(TAG, "Cannot decode VarInt field %" PRIu32 " with value %" PRIu32 "!", field_id, res->as_uint32());
|
ESP_LOGV(TAG, "Cannot decode VarInt field %u with value %u!", field_id, res->as_uint32());
|
||||||
}
|
}
|
||||||
i += consumed;
|
i += consumed;
|
||||||
break;
|
break;
|
||||||
@@ -39,38 +38,38 @@ void ProtoMessage::decode(const uint8_t *buffer, size_t length) {
|
|||||||
case 2: { // Length-delimited
|
case 2: { // Length-delimited
|
||||||
res = ProtoVarInt::parse(&buffer[i], length - i, &consumed);
|
res = ProtoVarInt::parse(&buffer[i], length - i, &consumed);
|
||||||
if (!res.has_value()) {
|
if (!res.has_value()) {
|
||||||
ESP_LOGV(TAG, "Invalid Length Delimited at %" PRIu32, i);
|
ESP_LOGV(TAG, "Invalid Length Delimited at %u", i);
|
||||||
error = true;
|
error = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
uint32_t field_length = res->as_uint32();
|
uint32_t field_length = res->as_uint32();
|
||||||
i += consumed;
|
i += consumed;
|
||||||
if (field_length > length - i) {
|
if (field_length > length - i) {
|
||||||
ESP_LOGV(TAG, "Out-of-bounds Length Delimited at %" PRIu32, i);
|
ESP_LOGV(TAG, "Out-of-bounds Length Delimited at %u", i);
|
||||||
error = true;
|
error = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (!this->decode_length(field_id, ProtoLengthDelimited(&buffer[i], field_length))) {
|
if (!this->decode_length(field_id, ProtoLengthDelimited(&buffer[i], field_length))) {
|
||||||
ESP_LOGV(TAG, "Cannot decode Length Delimited field %" PRIu32 "!", field_id);
|
ESP_LOGV(TAG, "Cannot decode Length Delimited field %u!", field_id);
|
||||||
}
|
}
|
||||||
i += field_length;
|
i += field_length;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case 5: { // 32-bit
|
case 5: { // 32-bit
|
||||||
if (length - i < 4) {
|
if (length - i < 4) {
|
||||||
ESP_LOGV(TAG, "Out-of-bounds Fixed32-bit at %" PRIu32, i);
|
ESP_LOGV(TAG, "Out-of-bounds Fixed32-bit at %u", i);
|
||||||
error = true;
|
error = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
uint32_t val = encode_uint32(buffer[i + 3], buffer[i + 2], buffer[i + 1], buffer[i]);
|
uint32_t val = encode_uint32(buffer[i + 3], buffer[i + 2], buffer[i + 1], buffer[i]);
|
||||||
if (!this->decode_32bit(field_id, Proto32Bit(val))) {
|
if (!this->decode_32bit(field_id, Proto32Bit(val))) {
|
||||||
ESP_LOGV(TAG, "Cannot decode 32-bit field %" PRIu32 " with value %" PRIu32 "!", field_id, val);
|
ESP_LOGV(TAG, "Cannot decode 32-bit field %u with value %u!", field_id, val);
|
||||||
}
|
}
|
||||||
i += 4;
|
i += 4;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
ESP_LOGV(TAG, "Invalid field type at %" PRIu32, i);
|
ESP_LOGV(TAG, "Invalid field type at %u", i);
|
||||||
error = true;
|
error = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -55,11 +55,6 @@ bool InitialStateIterator::on_media_player(media_player::MediaPlayer *media_play
|
|||||||
return this->client_->send_media_player_state(media_player);
|
return this->client_->send_media_player_state(media_player);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
bool InitialStateIterator::on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) {
|
|
||||||
return this->client_->send_alarm_control_panel_state(a_alarm_control_panel);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
InitialStateIterator::InitialStateIterator(APIConnection *client) : client_(client) {}
|
InitialStateIterator::InitialStateIterator(APIConnection *client) : client_(client) {}
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
|
|||||||
@@ -51,9 +51,6 @@ class InitialStateIterator : public ComponentIterator {
|
|||||||
#endif
|
#endif
|
||||||
#ifdef USE_MEDIA_PLAYER
|
#ifdef USE_MEDIA_PLAYER
|
||||||
bool on_media_player(media_player::MediaPlayer *media_player) override;
|
bool on_media_player(media_player::MediaPlayer *media_player) override;
|
||||||
#endif
|
|
||||||
#ifdef USE_ALARM_CONTROL_PANEL
|
|
||||||
bool on_alarm_control_panel(alarm_control_panel::AlarmControlPanel *a_alarm_control_panel) override;
|
|
||||||
#endif
|
#endif
|
||||||
protected:
|
protected:
|
||||||
APIConnection *client_;
|
APIConnection *client_;
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ from esphome.const import (
|
|||||||
CONF_CAPACITANCE,
|
CONF_CAPACITANCE,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
AUTO_LOAD = ["sensor", "binary_sensor"]
|
||||||
MULTI_CONF = True
|
MULTI_CONF = True
|
||||||
|
|
||||||
CONF_AS3935_ID = "as3935_id"
|
CONF_AS3935_ID = "as3935_id"
|
||||||
|
|||||||
@@ -26,13 +26,9 @@ void AS3935Component::setup() {
|
|||||||
void AS3935Component::dump_config() {
|
void AS3935Component::dump_config() {
|
||||||
ESP_LOGCONFIG(TAG, "AS3935:");
|
ESP_LOGCONFIG(TAG, "AS3935:");
|
||||||
LOG_PIN(" Interrupt Pin: ", this->irq_pin_);
|
LOG_PIN(" Interrupt Pin: ", this->irq_pin_);
|
||||||
#ifdef USE_BINARY_SENSOR
|
|
||||||
LOG_BINARY_SENSOR(" ", "Thunder alert", this->thunder_alert_binary_sensor_);
|
LOG_BINARY_SENSOR(" ", "Thunder alert", this->thunder_alert_binary_sensor_);
|
||||||
#endif
|
|
||||||
#ifdef USE_SENSOR
|
|
||||||
LOG_SENSOR(" ", "Distance", this->distance_sensor_);
|
LOG_SENSOR(" ", "Distance", this->distance_sensor_);
|
||||||
LOG_SENSOR(" ", "Lightning energy", this->energy_sensor_);
|
LOG_SENSOR(" ", "Lightning energy", this->energy_sensor_);
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
float AS3935Component::get_setup_priority() const { return setup_priority::DATA; }
|
float AS3935Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
@@ -48,22 +44,16 @@ void AS3935Component::loop() {
|
|||||||
ESP_LOGI(TAG, "Disturber was detected - try increasing the spike rejection value!");
|
ESP_LOGI(TAG, "Disturber was detected - try increasing the spike rejection value!");
|
||||||
} else if (int_value == LIGHTNING_INT) {
|
} else if (int_value == LIGHTNING_INT) {
|
||||||
ESP_LOGI(TAG, "Lightning has been detected!");
|
ESP_LOGI(TAG, "Lightning has been detected!");
|
||||||
#ifdef USE_BINARY_SENSOR
|
if (this->thunder_alert_binary_sensor_ != nullptr)
|
||||||
if (this->thunder_alert_binary_sensor_ != nullptr) {
|
|
||||||
this->thunder_alert_binary_sensor_->publish_state(true);
|
this->thunder_alert_binary_sensor_->publish_state(true);
|
||||||
this->set_timeout(10, [this]() { this->thunder_alert_binary_sensor_->publish_state(false); });
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef USE_SENSOR
|
|
||||||
uint8_t distance = this->get_distance_to_storm_();
|
uint8_t distance = this->get_distance_to_storm_();
|
||||||
if (this->distance_sensor_ != nullptr)
|
if (this->distance_sensor_ != nullptr)
|
||||||
this->distance_sensor_->publish_state(distance);
|
this->distance_sensor_->publish_state(distance);
|
||||||
|
|
||||||
uint32_t energy = this->get_lightning_energy_();
|
uint32_t energy = this->get_lightning_energy_();
|
||||||
if (this->energy_sensor_ != nullptr)
|
if (this->energy_sensor_ != nullptr)
|
||||||
this->energy_sensor_->publish_state(energy);
|
this->energy_sensor_->publish_state(energy);
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
this->thunder_alert_binary_sensor_->publish_state(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
void AS3935Component::write_indoor(bool indoor) {
|
void AS3935Component::write_indoor(bool indoor) {
|
||||||
|
|||||||
@@ -1,14 +1,9 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "esphome/core/component.h"
|
#include "esphome/core/component.h"
|
||||||
#include "esphome/core/defines.h"
|
|
||||||
#include "esphome/core/hal.h"
|
#include "esphome/core/hal.h"
|
||||||
#ifdef USE_SENSOR
|
|
||||||
#include "esphome/components/sensor/sensor.h"
|
#include "esphome/components/sensor/sensor.h"
|
||||||
#endif
|
|
||||||
#ifdef USE_BINARY_SENSOR
|
|
||||||
#include "esphome/components/binary_sensor/binary_sensor.h"
|
#include "esphome/components/binary_sensor/binary_sensor.h"
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace as3935 {
|
namespace as3935 {
|
||||||
@@ -57,15 +52,6 @@ enum AS3935Values {
|
|||||||
};
|
};
|
||||||
|
|
||||||
class AS3935Component : public Component {
|
class AS3935Component : public Component {
|
||||||
#ifdef USE_SENSOR
|
|
||||||
SUB_SENSOR(distance)
|
|
||||||
SUB_SENSOR(energy)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef USE_BINARY_SENSOR
|
|
||||||
SUB_BINARY_SENSOR(thunder_alert)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
void setup() override;
|
void setup() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
@@ -73,7 +59,11 @@ class AS3935Component : public Component {
|
|||||||
void loop() override;
|
void loop() override;
|
||||||
|
|
||||||
void set_irq_pin(GPIOPin *irq_pin) { irq_pin_ = irq_pin; }
|
void set_irq_pin(GPIOPin *irq_pin) { irq_pin_ = irq_pin; }
|
||||||
|
void set_distance_sensor(sensor::Sensor *distance_sensor) { distance_sensor_ = distance_sensor; }
|
||||||
|
void set_energy_sensor(sensor::Sensor *energy_sensor) { energy_sensor_ = energy_sensor; }
|
||||||
|
void set_thunder_alert_binary_sensor(binary_sensor::BinarySensor *thunder_alert_binary_sensor) {
|
||||||
|
thunder_alert_binary_sensor_ = thunder_alert_binary_sensor;
|
||||||
|
}
|
||||||
void set_indoor(bool indoor) { indoor_ = indoor; }
|
void set_indoor(bool indoor) { indoor_ = indoor; }
|
||||||
void write_indoor(bool indoor);
|
void write_indoor(bool indoor);
|
||||||
void set_noise_level(uint8_t noise_level) { noise_level_ = noise_level; }
|
void set_noise_level(uint8_t noise_level) { noise_level_ = noise_level; }
|
||||||
@@ -102,6 +92,9 @@ class AS3935Component : public Component {
|
|||||||
|
|
||||||
virtual void write_register(uint8_t reg, uint8_t mask, uint8_t bits, uint8_t start_position) = 0;
|
virtual void write_register(uint8_t reg, uint8_t mask, uint8_t bits, uint8_t start_position) = 0;
|
||||||
|
|
||||||
|
sensor::Sensor *distance_sensor_{nullptr};
|
||||||
|
sensor::Sensor *energy_sensor_{nullptr};
|
||||||
|
binary_sensor::BinarySensor *thunder_alert_binary_sensor_{nullptr};
|
||||||
GPIOPin *irq_pin_;
|
GPIOPin *irq_pin_;
|
||||||
|
|
||||||
bool indoor_;
|
bool indoor_;
|
||||||
|
|||||||
@@ -18,5 +18,5 @@ async def to_code(config):
|
|||||||
# https://github.com/esphome/AsyncTCP/blob/master/library.json
|
# https://github.com/esphome/AsyncTCP/blob/master/library.json
|
||||||
cg.add_library("esphome/AsyncTCP-esphome", "1.2.2")
|
cg.add_library("esphome/AsyncTCP-esphome", "1.2.2")
|
||||||
elif CORE.is_esp8266:
|
elif CORE.is_esp8266:
|
||||||
# https://github.com/esphome/ESPAsyncTCP
|
# https://github.com/OttoWinter/ESPAsyncTCP
|
||||||
cg.add_library("esphome/ESPAsyncTCP-esphome", "1.2.3")
|
cg.add_library("ottowinter/ESPAsyncTCP-esphome", "1.2.3")
|
||||||
|
|||||||
@@ -442,7 +442,7 @@ uint8_t BedJetHub::write_notify_config_descriptor_(bool enable) {
|
|||||||
void BedJetHub::send_local_time() {
|
void BedJetHub::send_local_time() {
|
||||||
if (this->time_id_.has_value()) {
|
if (this->time_id_.has_value()) {
|
||||||
auto *time_id = *this->time_id_;
|
auto *time_id = *this->time_id_;
|
||||||
ESPTime now = time_id->now();
|
time::ESPTime now = time_id->now();
|
||||||
if (now.is_valid()) {
|
if (now.is_valid()) {
|
||||||
this->set_clock(now.hour, now.minute);
|
this->set_clock(now.hour, now.minute);
|
||||||
ESP_LOGD(TAG, "Using time component to set BedJet clock: %d:%02d", now.hour, now.minute);
|
ESP_LOGD(TAG, "Using time component to set BedJet clock: %d:%02d", now.hour, now.minute);
|
||||||
|
|||||||
@@ -13,7 +13,6 @@
|
|||||||
|
|
||||||
#ifdef USE_TIME
|
#ifdef USE_TIME
|
||||||
#include "esphome/components/time/real_time_clock.h"
|
#include "esphome/components/time/real_time_clock.h"
|
||||||
#include "esphome/core/time.h"
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include <esp_gattc_api.h>
|
#include <esp_gattc_api.h>
|
||||||
@@ -117,7 +116,7 @@ class BedJetHub : public esphome::ble_client::BLEClientNode, public PollingCompo
|
|||||||
void update() override;
|
void update() override;
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
void setup() override { this->codec_ = make_unique<BedjetCodec>(); }
|
void setup() override { this->codec_ = make_unique<BedjetCodec>(); }
|
||||||
float get_setup_priority() const override { return setup_priority::BLUETOOTH; }
|
float get_setup_priority() const override { return setup_priority::AFTER_WIFI; }
|
||||||
|
|
||||||
/** @return The BedJet's configured name, or the MAC address if not discovered yet. */
|
/** @return The BedJet's configured name, or the MAC address if not discovered yet. */
|
||||||
std::string get_name() {
|
std::string get_name() {
|
||||||
|
|||||||
@@ -43,7 +43,12 @@ void BinarySensor::send_state_internal(bool state, bool is_initial) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
BinarySensor::BinarySensor() : state(false) {}
|
BinarySensor::BinarySensor() : state(false) {}
|
||||||
|
void BinarySensor::set_device_class(const std::string &device_class) { this->device_class_ = device_class; }
|
||||||
|
std::string BinarySensor::get_device_class() {
|
||||||
|
if (this->device_class_.has_value())
|
||||||
|
return *this->device_class_;
|
||||||
|
return "";
|
||||||
|
}
|
||||||
void BinarySensor::add_filter(Filter *filter) {
|
void BinarySensor::add_filter(Filter *filter) {
|
||||||
filter->parent_ = this;
|
filter->parent_ = this;
|
||||||
if (this->filter_list_ == nullptr) {
|
if (this->filter_list_ == nullptr) {
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ namespace binary_sensor {
|
|||||||
* The sub classes should notify the front-end of new states via the publish_state() method which
|
* The sub classes should notify the front-end of new states via the publish_state() method which
|
||||||
* handles inverted inputs for you.
|
* handles inverted inputs for you.
|
||||||
*/
|
*/
|
||||||
class BinarySensor : public EntityBase, public EntityBase_DeviceClass {
|
class BinarySensor : public EntityBase {
|
||||||
public:
|
public:
|
||||||
explicit BinarySensor();
|
explicit BinarySensor();
|
||||||
|
|
||||||
@@ -60,6 +60,12 @@ class BinarySensor : public EntityBase, public EntityBase_DeviceClass {
|
|||||||
/// The current reported state of the binary sensor.
|
/// The current reported state of the binary sensor.
|
||||||
bool state;
|
bool state;
|
||||||
|
|
||||||
|
/// Manually set the Home Assistant device class (see binary_sensor::device_class)
|
||||||
|
void set_device_class(const std::string &device_class);
|
||||||
|
|
||||||
|
/// Get the device class for this binary sensor, using the manual override if specified.
|
||||||
|
std::string get_device_class();
|
||||||
|
|
||||||
void add_filter(Filter *filter);
|
void add_filter(Filter *filter);
|
||||||
void add_filters(const std::vector<Filter *> &filters);
|
void add_filters(const std::vector<Filter *> &filters);
|
||||||
|
|
||||||
@@ -76,6 +82,7 @@ class BinarySensor : public EntityBase, public EntityBase_DeviceClass {
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
CallbackManager<void(bool)> state_callback_{};
|
CallbackManager<void(bool)> state_callback_{};
|
||||||
|
optional<std::string> device_class_{}; ///< Stores the override of the device class
|
||||||
Filter *filter_list_{nullptr};
|
Filter *filter_list_{nullptr};
|
||||||
bool has_state_{false};
|
bool has_state_{false};
|
||||||
bool publish_initial_state_{false};
|
bool publish_initial_state_{false};
|
||||||
|
|||||||
@@ -16,9 +16,6 @@ void BinarySensorMap::loop() {
|
|||||||
case BINARY_SENSOR_MAP_TYPE_SUM:
|
case BINARY_SENSOR_MAP_TYPE_SUM:
|
||||||
this->process_sum_();
|
this->process_sum_();
|
||||||
break;
|
break;
|
||||||
case BINARY_SENSOR_MAP_TYPE_BAYESIAN:
|
|
||||||
this->process_bayesian_();
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -26,117 +23,69 @@ void BinarySensorMap::process_group_() {
|
|||||||
float total_current_value = 0.0;
|
float total_current_value = 0.0;
|
||||||
uint8_t num_active_sensors = 0;
|
uint8_t num_active_sensors = 0;
|
||||||
uint64_t mask = 0x00;
|
uint64_t mask = 0x00;
|
||||||
|
// check all binary_sensors for its state. when active add its value to total_current_value.
|
||||||
// - check all binary_sensors for its state
|
// create a bitmask for the binary_sensor status on all channels
|
||||||
// - if active, add its value to total_current_value.
|
|
||||||
// - creates a bitmask for the binary_sensor states on all channels
|
|
||||||
for (size_t i = 0; i < this->channels_.size(); i++) {
|
for (size_t i = 0; i < this->channels_.size(); i++) {
|
||||||
auto bs = this->channels_[i];
|
auto bs = this->channels_[i];
|
||||||
if (bs.binary_sensor->state) {
|
if (bs.binary_sensor->state) {
|
||||||
num_active_sensors++;
|
num_active_sensors++;
|
||||||
total_current_value += bs.parameters.sensor_value;
|
total_current_value += bs.sensor_value;
|
||||||
mask |= 1ULL << i;
|
mask |= 1 << i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// check if the sensor map was touched
|
||||||
// potentially update state only if a binary_sensor is active
|
|
||||||
if (mask != 0ULL) {
|
if (mask != 0ULL) {
|
||||||
// publish the average if the bitmask has changed
|
// did the bit_mask change or is it a new sensor touch
|
||||||
if (this->last_mask_ != mask) {
|
if (this->last_mask_ != mask) {
|
||||||
float publish_value = total_current_value / num_active_sensors;
|
float publish_value = total_current_value / num_active_sensors;
|
||||||
|
ESP_LOGD(TAG, "'%s' - Publishing %.2f", this->name_.c_str(), publish_value);
|
||||||
this->publish_state(publish_value);
|
this->publish_state(publish_value);
|
||||||
}
|
}
|
||||||
} else if (this->last_mask_ != 0ULL) {
|
} else if (this->last_mask_ != 0ULL) {
|
||||||
// no buttons are pressed and the states have changed since last run, so publish NAN
|
// is this a new sensor release
|
||||||
ESP_LOGV(TAG, "'%s' - No binary sensor active, publishing NAN", this->name_.c_str());
|
ESP_LOGD(TAG, "'%s' - No binary sensor active, publishing NAN", this->name_.c_str());
|
||||||
this->publish_state(NAN);
|
this->publish_state(NAN);
|
||||||
}
|
}
|
||||||
|
|
||||||
this->last_mask_ = mask;
|
this->last_mask_ = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
void BinarySensorMap::process_sum_() {
|
void BinarySensorMap::process_sum_() {
|
||||||
float total_current_value = 0.0;
|
float total_current_value = 0.0;
|
||||||
uint64_t mask = 0x00;
|
uint64_t mask = 0x00;
|
||||||
|
// check all binary_sensors for its state. when active add its value to total_current_value.
|
||||||
// - check all binary_sensor states
|
// create a bitmask for the binary_sensor status on all channels
|
||||||
// - if active, add its value to total_current_value
|
|
||||||
// - creates a bitmask for the binary_sensor states on all channels
|
|
||||||
for (size_t i = 0; i < this->channels_.size(); i++) {
|
for (size_t i = 0; i < this->channels_.size(); i++) {
|
||||||
auto bs = this->channels_[i];
|
auto bs = this->channels_[i];
|
||||||
if (bs.binary_sensor->state) {
|
if (bs.binary_sensor->state) {
|
||||||
total_current_value += bs.parameters.sensor_value;
|
total_current_value += bs.sensor_value;
|
||||||
mask |= 1ULL << i;
|
mask |= 1 << i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// check if the sensor map was touched
|
||||||
// update state only if any binary_sensor states have changed or if no state has ever been sent on boot
|
if (mask != 0ULL) {
|
||||||
if ((this->last_mask_ != mask) || (!this->has_state())) {
|
// did the bit_mask change or is it a new sensor touch
|
||||||
this->publish_state(total_current_value);
|
if (this->last_mask_ != mask) {
|
||||||
|
float publish_value = total_current_value;
|
||||||
|
ESP_LOGD(TAG, "'%s' - Publishing %.2f", this->name_.c_str(), publish_value);
|
||||||
|
this->publish_state(publish_value);
|
||||||
|
}
|
||||||
|
} else if (this->last_mask_ != 0ULL) {
|
||||||
|
// is this a new sensor release
|
||||||
|
ESP_LOGD(TAG, "'%s' - No binary sensor active, publishing 0", this->name_.c_str());
|
||||||
|
this->publish_state(0.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
this->last_mask_ = mask;
|
this->last_mask_ = mask;
|
||||||
}
|
}
|
||||||
|
|
||||||
void BinarySensorMap::process_bayesian_() {
|
|
||||||
float posterior_probability = this->bayesian_prior_;
|
|
||||||
uint64_t mask = 0x00;
|
|
||||||
|
|
||||||
// - compute the posterior probability by taking the product of the predicate probablities for each observation
|
|
||||||
// - create a bitmask for the binary_sensor states on all channels/observations
|
|
||||||
for (size_t i = 0; i < this->channels_.size(); i++) {
|
|
||||||
auto bs = this->channels_[i];
|
|
||||||
|
|
||||||
posterior_probability *=
|
|
||||||
this->bayesian_predicate_(bs.binary_sensor->state, posterior_probability,
|
|
||||||
bs.parameters.probabilities.given_true, bs.parameters.probabilities.given_false);
|
|
||||||
|
|
||||||
mask |= ((uint64_t) (bs.binary_sensor->state)) << i;
|
|
||||||
}
|
|
||||||
|
|
||||||
// update state only if any binary_sensor states have changed or if no state has ever been sent on boot
|
|
||||||
if ((this->last_mask_ != mask) || (!this->has_state())) {
|
|
||||||
this->publish_state(posterior_probability);
|
|
||||||
}
|
|
||||||
|
|
||||||
this->last_mask_ = mask;
|
|
||||||
}
|
|
||||||
|
|
||||||
float BinarySensorMap::bayesian_predicate_(bool sensor_state, float prior, float prob_given_true,
|
|
||||||
float prob_given_false) {
|
|
||||||
float prob_state_source_true = prob_given_true;
|
|
||||||
float prob_state_source_false = prob_given_false;
|
|
||||||
|
|
||||||
// if sensor is off, then we use the probabilities for the observation's complement
|
|
||||||
if (!sensor_state) {
|
|
||||||
prob_state_source_true = 1 - prob_given_true;
|
|
||||||
prob_state_source_false = 1 - prob_given_false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return prob_state_source_true / (prior * prob_state_source_true + (1.0 - prior) * prob_state_source_false);
|
|
||||||
}
|
|
||||||
|
|
||||||
void BinarySensorMap::add_channel(binary_sensor::BinarySensor *sensor, float value) {
|
void BinarySensorMap::add_channel(binary_sensor::BinarySensor *sensor, float value) {
|
||||||
BinarySensorMapChannel sensor_channel{
|
BinarySensorMapChannel sensor_channel{
|
||||||
.binary_sensor = sensor,
|
.binary_sensor = sensor,
|
||||||
.parameters{
|
.sensor_value = value,
|
||||||
.sensor_value = value,
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
this->channels_.push_back(sensor_channel);
|
this->channels_.push_back(sensor_channel);
|
||||||
}
|
}
|
||||||
|
|
||||||
void BinarySensorMap::add_channel(binary_sensor::BinarySensor *sensor, float prob_given_true, float prob_given_false) {
|
void BinarySensorMap::set_sensor_type(BinarySensorMapType sensor_type) { this->sensor_type_ = sensor_type; }
|
||||||
BinarySensorMapChannel sensor_channel{
|
|
||||||
.binary_sensor = sensor,
|
|
||||||
.parameters{
|
|
||||||
.probabilities{
|
|
||||||
.given_true = prob_given_true,
|
|
||||||
.given_false = prob_given_false,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
};
|
|
||||||
this->channels_.push_back(sensor_channel);
|
|
||||||
}
|
|
||||||
} // namespace binary_sensor_map
|
} // namespace binary_sensor_map
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -12,88 +12,51 @@ namespace binary_sensor_map {
|
|||||||
enum BinarySensorMapType {
|
enum BinarySensorMapType {
|
||||||
BINARY_SENSOR_MAP_TYPE_GROUP,
|
BINARY_SENSOR_MAP_TYPE_GROUP,
|
||||||
BINARY_SENSOR_MAP_TYPE_SUM,
|
BINARY_SENSOR_MAP_TYPE_SUM,
|
||||||
BINARY_SENSOR_MAP_TYPE_BAYESIAN,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct BinarySensorMapChannel {
|
struct BinarySensorMapChannel {
|
||||||
binary_sensor::BinarySensor *binary_sensor;
|
binary_sensor::BinarySensor *binary_sensor;
|
||||||
union {
|
float sensor_value;
|
||||||
float sensor_value;
|
|
||||||
struct {
|
|
||||||
float given_true;
|
|
||||||
float given_false;
|
|
||||||
} probabilities;
|
|
||||||
} parameters;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/** Class to map one or more binary_sensors to one Sensor.
|
/** Class to group binary_sensors to one Sensor.
|
||||||
*
|
*
|
||||||
* Each binary sensor has configured parameters that each mapping type uses to compute the single numerical result
|
* Each binary sensor represents a float value in the group.
|
||||||
*/
|
*/
|
||||||
class BinarySensorMap : public sensor::Sensor, public Component {
|
class BinarySensorMap : public sensor::Sensor, public Component {
|
||||||
public:
|
public:
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The loop calls the configured type processing method
|
* The loop checks all binary_sensor states
|
||||||
|
* When the binary_sensor reports a true value for its state, then the float value it represents is added to the
|
||||||
|
* total_current_value
|
||||||
*
|
*
|
||||||
* The processing method loops through all sensors and calculates the numerical result
|
* Only when the total_current_value changed and at least one sensor reports an active state we publish the sensors
|
||||||
* The result is only published if a binary sensor state has changed or, for some types, on initial boot
|
* average value. When the value changed and no sensors ar active we publish NAN.
|
||||||
*/
|
* */
|
||||||
void loop() override;
|
void loop() override;
|
||||||
|
float get_setup_priority() const override { return setup_priority::DATA; }
|
||||||
/**
|
/** Add binary_sensors to the group.
|
||||||
* Add binary_sensors to the group when only one parameter is needed for the configured mapping type.
|
* Each binary_sensor represents a float value when its state is true
|
||||||
*
|
*
|
||||||
* @param *sensor The binary sensor.
|
* @param *sensor The binary sensor.
|
||||||
* @param value The value this binary_sensor represents
|
* @param value The value this binary_sensor represents
|
||||||
*/
|
*/
|
||||||
void add_channel(binary_sensor::BinarySensor *sensor, float value);
|
void add_channel(binary_sensor::BinarySensor *sensor, float value);
|
||||||
|
void set_sensor_type(BinarySensorMapType sensor_type);
|
||||||
/**
|
|
||||||
* Add binary_sensors to the group when two parameters are needed for the Bayesian mapping type.
|
|
||||||
*
|
|
||||||
* @param *sensor The binary sensor.
|
|
||||||
* @param prob_given_true Probability this observation is on when the Bayesian event is true
|
|
||||||
* @param prob_given_false Probability this observation is on when the Bayesian event is false
|
|
||||||
*/
|
|
||||||
void add_channel(binary_sensor::BinarySensor *sensor, float prob_given_true, float prob_given_false);
|
|
||||||
|
|
||||||
void set_sensor_type(BinarySensorMapType sensor_type) { this->sensor_type_ = sensor_type; }
|
|
||||||
|
|
||||||
void set_bayesian_prior(float prior) { this->bayesian_prior_ = prior; };
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
std::vector<BinarySensorMapChannel> channels_{};
|
std::vector<BinarySensorMapChannel> channels_{};
|
||||||
BinarySensorMapType sensor_type_{BINARY_SENSOR_MAP_TYPE_GROUP};
|
BinarySensorMapType sensor_type_{BINARY_SENSOR_MAP_TYPE_GROUP};
|
||||||
|
// this gives max 64 channels per binary_sensor_map
|
||||||
// this allows a max of 64 channels/observations in order to keep track of binary_sensor states
|
|
||||||
uint64_t last_mask_{0x00};
|
uint64_t last_mask_{0x00};
|
||||||
|
|
||||||
// Bayesian event prior probability before taking into account any observations
|
|
||||||
float bayesian_prior_{};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Methods to process the binary_sensor_maps types
|
* methods to process the types of binary_sensor_maps
|
||||||
*
|
* GROUP: process_group_() just map to a value
|
||||||
* GROUP: process_group_() averages all the values
|
|
||||||
* ADD: process_add_() adds all the values
|
* ADD: process_add_() adds all the values
|
||||||
* BAYESIAN: process_bayesian_() computes the predicate probability
|
|
||||||
* */
|
* */
|
||||||
void process_group_();
|
void process_group_();
|
||||||
void process_sum_();
|
void process_sum_();
|
||||||
void process_bayesian_();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Computes the Bayesian predicate for a specific observation
|
|
||||||
* If the sensor state is false, then we use the parameters' probabilities for the observatiosn complement
|
|
||||||
*
|
|
||||||
* @param sensor_state State of observation
|
|
||||||
* @param prior Prior probability before accounting for this observation
|
|
||||||
* @param prob_given_true Probability this observation is on when the Bayesian event is true
|
|
||||||
* @param prob_given_false Probability this observation is on when the Bayesian event is false
|
|
||||||
* */
|
|
||||||
float bayesian_predicate_(bool sensor_state, float prior, float prob_given_true, float prob_given_false);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace binary_sensor_map
|
} // namespace binary_sensor_map
|
||||||
|
|||||||
@@ -20,29 +20,16 @@ BinarySensorMap = binary_sensor_map_ns.class_(
|
|||||||
)
|
)
|
||||||
SensorMapType = binary_sensor_map_ns.enum("SensorMapType")
|
SensorMapType = binary_sensor_map_ns.enum("SensorMapType")
|
||||||
|
|
||||||
CONF_BAYESIAN = "bayesian"
|
|
||||||
CONF_PRIOR = "prior"
|
|
||||||
CONF_PROB_GIVEN_TRUE = "prob_given_true"
|
|
||||||
CONF_PROB_GIVEN_FALSE = "prob_given_false"
|
|
||||||
CONF_OBSERVATIONS = "observations"
|
|
||||||
|
|
||||||
SENSOR_MAP_TYPES = {
|
SENSOR_MAP_TYPES = {
|
||||||
CONF_GROUP: SensorMapType.BINARY_SENSOR_MAP_TYPE_GROUP,
|
CONF_GROUP: SensorMapType.BINARY_SENSOR_MAP_TYPE_GROUP,
|
||||||
CONF_SUM: SensorMapType.BINARY_SENSOR_MAP_TYPE_SUM,
|
CONF_SUM: SensorMapType.BINARY_SENSOR_MAP_TYPE_SUM,
|
||||||
CONF_BAYESIAN: SensorMapType.BINARY_SENSOR_MAP_TYPE_BAYESIAN,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
entry_one_parameter = {
|
entry = {
|
||||||
cv.Required(CONF_BINARY_SENSOR): cv.use_id(binary_sensor.BinarySensor),
|
cv.Required(CONF_BINARY_SENSOR): cv.use_id(binary_sensor.BinarySensor),
|
||||||
cv.Required(CONF_VALUE): cv.float_,
|
cv.Required(CONF_VALUE): cv.float_,
|
||||||
}
|
}
|
||||||
|
|
||||||
entry_bayesian_parameters = {
|
|
||||||
cv.Required(CONF_BINARY_SENSOR): cv.use_id(binary_sensor.BinarySensor),
|
|
||||||
cv.Required(CONF_PROB_GIVEN_TRUE): cv.float_range(min=0, max=1),
|
|
||||||
cv.Required(CONF_PROB_GIVEN_FALSE): cv.float_range(min=0, max=1),
|
|
||||||
}
|
|
||||||
|
|
||||||
CONFIG_SCHEMA = cv.typed_schema(
|
CONFIG_SCHEMA = cv.typed_schema(
|
||||||
{
|
{
|
||||||
CONF_GROUP: sensor.sensor_schema(
|
CONF_GROUP: sensor.sensor_schema(
|
||||||
@@ -52,7 +39,7 @@ CONFIG_SCHEMA = cv.typed_schema(
|
|||||||
).extend(
|
).extend(
|
||||||
{
|
{
|
||||||
cv.Required(CONF_CHANNELS): cv.All(
|
cv.Required(CONF_CHANNELS): cv.All(
|
||||||
cv.ensure_list(entry_one_parameter), cv.Length(min=1, max=64)
|
cv.ensure_list(entry), cv.Length(min=1)
|
||||||
),
|
),
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
@@ -63,18 +50,7 @@ CONFIG_SCHEMA = cv.typed_schema(
|
|||||||
).extend(
|
).extend(
|
||||||
{
|
{
|
||||||
cv.Required(CONF_CHANNELS): cv.All(
|
cv.Required(CONF_CHANNELS): cv.All(
|
||||||
cv.ensure_list(entry_one_parameter), cv.Length(min=1, max=64)
|
cv.ensure_list(entry), cv.Length(min=1)
|
||||||
),
|
|
||||||
}
|
|
||||||
),
|
|
||||||
CONF_BAYESIAN: sensor.sensor_schema(
|
|
||||||
BinarySensorMap,
|
|
||||||
accuracy_decimals=2,
|
|
||||||
).extend(
|
|
||||||
{
|
|
||||||
cv.Required(CONF_PRIOR): cv.float_range(min=0, max=1),
|
|
||||||
cv.Required(CONF_OBSERVATIONS): cv.All(
|
|
||||||
cv.ensure_list(entry_bayesian_parameters), cv.Length(min=1, max=64)
|
|
||||||
),
|
),
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
@@ -90,17 +66,6 @@ async def to_code(config):
|
|||||||
constant = SENSOR_MAP_TYPES[config[CONF_TYPE]]
|
constant = SENSOR_MAP_TYPES[config[CONF_TYPE]]
|
||||||
cg.add(var.set_sensor_type(constant))
|
cg.add(var.set_sensor_type(constant))
|
||||||
|
|
||||||
if config[CONF_TYPE] == CONF_BAYESIAN:
|
for ch in config[CONF_CHANNELS]:
|
||||||
cg.add(var.set_bayesian_prior(config[CONF_PRIOR]))
|
input_var = await cg.get_variable(ch[CONF_BINARY_SENSOR])
|
||||||
|
cg.add(var.add_channel(input_var, ch[CONF_VALUE]))
|
||||||
for obs in config[CONF_OBSERVATIONS]:
|
|
||||||
input_var = await cg.get_variable(obs[CONF_BINARY_SENSOR])
|
|
||||||
cg.add(
|
|
||||||
var.add_channel(
|
|
||||||
input_var, obs[CONF_PROB_GIVEN_TRUE], obs[CONF_PROB_GIVEN_FALSE]
|
|
||||||
)
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
for ch in config[CONF_CHANNELS]:
|
|
||||||
input_var = await cg.get_variable(ch[CONF_BINARY_SENSOR])
|
|
||||||
cg.add(var.add_channel(input_var, ch[CONF_VALUE]))
|
|
||||||
|
|||||||
@@ -29,35 +29,8 @@ BLEClientConnectTrigger = ble_client_ns.class_(
|
|||||||
BLEClientDisconnectTrigger = ble_client_ns.class_(
|
BLEClientDisconnectTrigger = ble_client_ns.class_(
|
||||||
"BLEClientDisconnectTrigger", automation.Trigger.template(BLEClientNodeConstRef)
|
"BLEClientDisconnectTrigger", automation.Trigger.template(BLEClientNodeConstRef)
|
||||||
)
|
)
|
||||||
BLEClientPasskeyRequestTrigger = ble_client_ns.class_(
|
|
||||||
"BLEClientPasskeyRequestTrigger", automation.Trigger.template(BLEClientNodeConstRef)
|
|
||||||
)
|
|
||||||
BLEClientPasskeyNotificationTrigger = ble_client_ns.class_(
|
|
||||||
"BLEClientPasskeyNotificationTrigger",
|
|
||||||
automation.Trigger.template(BLEClientNodeConstRef, cg.uint32),
|
|
||||||
)
|
|
||||||
BLEClientNumericComparisonRequestTrigger = ble_client_ns.class_(
|
|
||||||
"BLEClientNumericComparisonRequestTrigger",
|
|
||||||
automation.Trigger.template(BLEClientNodeConstRef, cg.uint32),
|
|
||||||
)
|
|
||||||
|
|
||||||
# Actions
|
# Actions
|
||||||
BLEWriteAction = ble_client_ns.class_("BLEClientWriteAction", automation.Action)
|
BLEWriteAction = ble_client_ns.class_("BLEClientWriteAction", automation.Action)
|
||||||
BLEPasskeyReplyAction = ble_client_ns.class_(
|
|
||||||
"BLEClientPasskeyReplyAction", automation.Action
|
|
||||||
)
|
|
||||||
BLENumericComparisonReplyAction = ble_client_ns.class_(
|
|
||||||
"BLEClientNumericComparisonReplyAction", automation.Action
|
|
||||||
)
|
|
||||||
BLERemoveBondAction = ble_client_ns.class_(
|
|
||||||
"BLEClientRemoveBondAction", automation.Action
|
|
||||||
)
|
|
||||||
|
|
||||||
CONF_PASSKEY = "passkey"
|
|
||||||
CONF_ACCEPT = "accept"
|
|
||||||
CONF_ON_PASSKEY_REQUEST = "on_passkey_request"
|
|
||||||
CONF_ON_PASSKEY_NOTIFICATION = "on_passkey_notification"
|
|
||||||
CONF_ON_NUMERIC_COMPARISON_REQUEST = "on_numeric_comparison_request"
|
|
||||||
|
|
||||||
# Espressif platformio framework is built with MAX_BLE_CONN to 3, so
|
# Espressif platformio framework is built with MAX_BLE_CONN to 3, so
|
||||||
# enforce this in yaml checks.
|
# enforce this in yaml checks.
|
||||||
@@ -83,29 +56,6 @@ CONFIG_SCHEMA = (
|
|||||||
),
|
),
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
cv.Optional(CONF_ON_PASSKEY_REQUEST): automation.validate_automation(
|
|
||||||
{
|
|
||||||
cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(
|
|
||||||
BLEClientPasskeyRequestTrigger
|
|
||||||
),
|
|
||||||
}
|
|
||||||
),
|
|
||||||
cv.Optional(CONF_ON_PASSKEY_NOTIFICATION): automation.validate_automation(
|
|
||||||
{
|
|
||||||
cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(
|
|
||||||
BLEClientPasskeyNotificationTrigger
|
|
||||||
),
|
|
||||||
}
|
|
||||||
),
|
|
||||||
cv.Optional(
|
|
||||||
CONF_ON_NUMERIC_COMPARISON_REQUEST
|
|
||||||
): automation.validate_automation(
|
|
||||||
{
|
|
||||||
cv.GenerateID(CONF_TRIGGER_ID): cv.declare_id(
|
|
||||||
BLEClientNumericComparisonRequestTrigger
|
|
||||||
),
|
|
||||||
}
|
|
||||||
),
|
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
.extend(cv.COMPONENT_SCHEMA)
|
.extend(cv.COMPONENT_SCHEMA)
|
||||||
@@ -135,34 +85,13 @@ BLE_WRITE_ACTION_SCHEMA = cv.Schema(
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
BLE_NUMERIC_COMPARISON_REPLY_ACTION_SCHEMA = cv.Schema(
|
|
||||||
{
|
|
||||||
cv.GenerateID(CONF_ID): cv.use_id(BLEClient),
|
|
||||||
cv.Required(CONF_ACCEPT): cv.templatable(cv.boolean),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
BLE_PASSKEY_REPLY_ACTION_SCHEMA = cv.Schema(
|
|
||||||
{
|
|
||||||
cv.GenerateID(CONF_ID): cv.use_id(BLEClient),
|
|
||||||
cv.Required(CONF_PASSKEY): cv.templatable(cv.int_range(min=0, max=999999)),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
BLE_REMOVE_BOND_ACTION_SCHEMA = cv.Schema(
|
|
||||||
{
|
|
||||||
cv.GenerateID(CONF_ID): cv.use_id(BLEClient),
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
@automation.register_action(
|
@automation.register_action(
|
||||||
"ble_client.ble_write", BLEWriteAction, BLE_WRITE_ACTION_SCHEMA
|
"ble_client.ble_write", BLEWriteAction, BLE_WRITE_ACTION_SCHEMA
|
||||||
)
|
)
|
||||||
async def ble_write_to_code(config, action_id, template_arg, args):
|
async def ble_write_to_code(config, action_id, template_arg, args):
|
||||||
parent = await cg.get_variable(config[CONF_ID])
|
paren = await cg.get_variable(config[CONF_ID])
|
||||||
var = cg.new_Pvariable(action_id, template_arg, parent)
|
var = cg.new_Pvariable(action_id, template_arg, paren)
|
||||||
|
|
||||||
value = config[CONF_VALUE]
|
value = config[CONF_VALUE]
|
||||||
if cg.is_template(value):
|
if cg.is_template(value):
|
||||||
@@ -208,54 +137,6 @@ async def ble_write_to_code(config, action_id, template_arg, args):
|
|||||||
return var
|
return var
|
||||||
|
|
||||||
|
|
||||||
@automation.register_action(
|
|
||||||
"ble_client.numeric_comparison_reply",
|
|
||||||
BLENumericComparisonReplyAction,
|
|
||||||
BLE_NUMERIC_COMPARISON_REPLY_ACTION_SCHEMA,
|
|
||||||
)
|
|
||||||
async def numeric_comparison_reply_to_code(config, action_id, template_arg, args):
|
|
||||||
parent = await cg.get_variable(config[CONF_ID])
|
|
||||||
var = cg.new_Pvariable(action_id, template_arg, parent)
|
|
||||||
|
|
||||||
accept = config[CONF_ACCEPT]
|
|
||||||
if cg.is_template(accept):
|
|
||||||
templ = await cg.templatable(accept, args, cg.bool_)
|
|
||||||
cg.add(var.set_value_template(templ))
|
|
||||||
else:
|
|
||||||
cg.add(var.set_value_simple(accept))
|
|
||||||
|
|
||||||
return var
|
|
||||||
|
|
||||||
|
|
||||||
@automation.register_action(
|
|
||||||
"ble_client.passkey_reply", BLEPasskeyReplyAction, BLE_PASSKEY_REPLY_ACTION_SCHEMA
|
|
||||||
)
|
|
||||||
async def passkey_reply_to_code(config, action_id, template_arg, args):
|
|
||||||
parent = await cg.get_variable(config[CONF_ID])
|
|
||||||
var = cg.new_Pvariable(action_id, template_arg, parent)
|
|
||||||
|
|
||||||
passkey = config[CONF_PASSKEY]
|
|
||||||
if cg.is_template(passkey):
|
|
||||||
templ = await cg.templatable(passkey, args, cg.uint32)
|
|
||||||
cg.add(var.set_value_template(templ))
|
|
||||||
else:
|
|
||||||
cg.add(var.set_value_simple(passkey))
|
|
||||||
|
|
||||||
return var
|
|
||||||
|
|
||||||
|
|
||||||
@automation.register_action(
|
|
||||||
"ble_client.remove_bond",
|
|
||||||
BLERemoveBondAction,
|
|
||||||
BLE_REMOVE_BOND_ACTION_SCHEMA,
|
|
||||||
)
|
|
||||||
async def remove_bond_to_code(config, action_id, template_arg, args):
|
|
||||||
parent = await cg.get_variable(config[CONF_ID])
|
|
||||||
var = cg.new_Pvariable(action_id, template_arg, parent)
|
|
||||||
|
|
||||||
return var
|
|
||||||
|
|
||||||
|
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
var = cg.new_Pvariable(config[CONF_ID])
|
var = cg.new_Pvariable(config[CONF_ID])
|
||||||
await cg.register_component(var, config)
|
await cg.register_component(var, config)
|
||||||
@@ -267,12 +148,3 @@ async def to_code(config):
|
|||||||
for conf in config.get(CONF_ON_DISCONNECT, []):
|
for conf in config.get(CONF_ON_DISCONNECT, []):
|
||||||
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
||||||
await automation.build_automation(trigger, [], conf)
|
await automation.build_automation(trigger, [], conf)
|
||||||
for conf in config.get(CONF_ON_PASSKEY_REQUEST, []):
|
|
||||||
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
|
||||||
await automation.build_automation(trigger, [], conf)
|
|
||||||
for conf in config.get(CONF_ON_PASSKEY_NOTIFICATION, []):
|
|
||||||
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
|
||||||
await automation.build_automation(trigger, [(cg.uint32, "passkey")], conf)
|
|
||||||
for conf in config.get(CONF_ON_NUMERIC_COMPARISON_REQUEST, []):
|
|
||||||
trigger = cg.new_Pvariable(conf[CONF_TRIGGER_ID], var)
|
|
||||||
await automation.build_automation(trigger, [(cg.uint32, "passkey")], conf)
|
|
||||||
|
|||||||
@@ -37,44 +37,6 @@ class BLEClientDisconnectTrigger : public Trigger<>, public BLEClientNode {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
class BLEClientPasskeyRequestTrigger : public Trigger<>, public BLEClientNode {
|
|
||||||
public:
|
|
||||||
explicit BLEClientPasskeyRequestTrigger(BLEClient *parent) { parent->register_ble_node(this); }
|
|
||||||
void loop() override {}
|
|
||||||
void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) override {
|
|
||||||
if (event == ESP_GAP_BLE_PASSKEY_REQ_EVT &&
|
|
||||||
memcmp(param->ble_security.auth_cmpl.bd_addr, this->parent_->get_remote_bda(), 6) == 0) {
|
|
||||||
this->trigger();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
class BLEClientPasskeyNotificationTrigger : public Trigger<uint32_t>, public BLEClientNode {
|
|
||||||
public:
|
|
||||||
explicit BLEClientPasskeyNotificationTrigger(BLEClient *parent) { parent->register_ble_node(this); }
|
|
||||||
void loop() override {}
|
|
||||||
void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) override {
|
|
||||||
if (event == ESP_GAP_BLE_PASSKEY_NOTIF_EVT &&
|
|
||||||
memcmp(param->ble_security.auth_cmpl.bd_addr, this->parent_->get_remote_bda(), 6) == 0) {
|
|
||||||
uint32_t passkey = param->ble_security.key_notif.passkey;
|
|
||||||
this->trigger(passkey);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
class BLEClientNumericComparisonRequestTrigger : public Trigger<uint32_t>, public BLEClientNode {
|
|
||||||
public:
|
|
||||||
explicit BLEClientNumericComparisonRequestTrigger(BLEClient *parent) { parent->register_ble_node(this); }
|
|
||||||
void loop() override {}
|
|
||||||
void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) override {
|
|
||||||
if (event == ESP_GAP_BLE_NC_REQ_EVT &&
|
|
||||||
memcmp(param->ble_security.auth_cmpl.bd_addr, this->parent_->get_remote_bda(), 6) == 0) {
|
|
||||||
uint32_t passkey = param->ble_security.key_notif.passkey;
|
|
||||||
this->trigger(passkey);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
class BLEWriterClientNode : public BLEClientNode {
|
class BLEWriterClientNode : public BLEClientNode {
|
||||||
public:
|
public:
|
||||||
BLEWriterClientNode(BLEClient *ble_client) {
|
BLEWriterClientNode(BLEClient *ble_client) {
|
||||||
@@ -132,86 +94,6 @@ template<typename... Ts> class BLEClientWriteAction : public Action<Ts...>, publ
|
|||||||
std::function<std::vector<uint8_t>(Ts...)> value_template_{};
|
std::function<std::vector<uint8_t>(Ts...)> value_template_{};
|
||||||
};
|
};
|
||||||
|
|
||||||
template<typename... Ts> class BLEClientPasskeyReplyAction : public Action<Ts...> {
|
|
||||||
public:
|
|
||||||
BLEClientPasskeyReplyAction(BLEClient *ble_client) { parent_ = ble_client; }
|
|
||||||
|
|
||||||
void play(Ts... x) override {
|
|
||||||
uint32_t passkey;
|
|
||||||
if (has_simple_value_) {
|
|
||||||
passkey = this->value_simple_;
|
|
||||||
} else {
|
|
||||||
passkey = this->value_template_(x...);
|
|
||||||
}
|
|
||||||
if (passkey > 999999)
|
|
||||||
return;
|
|
||||||
esp_bd_addr_t remote_bda;
|
|
||||||
memcpy(remote_bda, parent_->get_remote_bda(), sizeof(esp_bd_addr_t));
|
|
||||||
esp_ble_passkey_reply(remote_bda, true, passkey);
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_value_template(std::function<uint32_t(Ts...)> func) {
|
|
||||||
this->value_template_ = std::move(func);
|
|
||||||
has_simple_value_ = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_value_simple(const uint32_t &value) {
|
|
||||||
this->value_simple_ = value;
|
|
||||||
has_simple_value_ = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
BLEClient *parent_{nullptr};
|
|
||||||
bool has_simple_value_ = true;
|
|
||||||
uint32_t value_simple_{0};
|
|
||||||
std::function<uint32_t(Ts...)> value_template_{};
|
|
||||||
};
|
|
||||||
|
|
||||||
template<typename... Ts> class BLEClientNumericComparisonReplyAction : public Action<Ts...> {
|
|
||||||
public:
|
|
||||||
BLEClientNumericComparisonReplyAction(BLEClient *ble_client) { parent_ = ble_client; }
|
|
||||||
|
|
||||||
void play(Ts... x) override {
|
|
||||||
esp_bd_addr_t remote_bda;
|
|
||||||
memcpy(remote_bda, parent_->get_remote_bda(), sizeof(esp_bd_addr_t));
|
|
||||||
if (has_simple_value_) {
|
|
||||||
esp_ble_confirm_reply(remote_bda, this->value_simple_);
|
|
||||||
} else {
|
|
||||||
esp_ble_confirm_reply(remote_bda, this->value_template_(x...));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_value_template(std::function<bool(Ts...)> func) {
|
|
||||||
this->value_template_ = std::move(func);
|
|
||||||
has_simple_value_ = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_value_simple(const bool &value) {
|
|
||||||
this->value_simple_ = value;
|
|
||||||
has_simple_value_ = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
BLEClient *parent_{nullptr};
|
|
||||||
bool has_simple_value_ = true;
|
|
||||||
bool value_simple_{false};
|
|
||||||
std::function<bool(Ts...)> value_template_{};
|
|
||||||
};
|
|
||||||
|
|
||||||
template<typename... Ts> class BLEClientRemoveBondAction : public Action<Ts...> {
|
|
||||||
public:
|
|
||||||
BLEClientRemoveBondAction(BLEClient *ble_client) { parent_ = ble_client; }
|
|
||||||
|
|
||||||
void play(Ts... x) override {
|
|
||||||
esp_bd_addr_t remote_bda;
|
|
||||||
memcpy(remote_bda, parent_->get_remote_bda(), sizeof(esp_bd_addr_t));
|
|
||||||
esp_ble_remove_bond_device(remote_bda);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
BLEClient *parent_{nullptr};
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace ble_client
|
} // namespace ble_client
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ class BLEClient;
|
|||||||
class BLEClientNode {
|
class BLEClientNode {
|
||||||
public:
|
public:
|
||||||
virtual void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
virtual void gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
|
||||||
esp_ble_gattc_cb_param_t *param){};
|
esp_ble_gattc_cb_param_t *param) = 0;
|
||||||
virtual void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) {}
|
virtual void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) {}
|
||||||
virtual void loop() {}
|
virtual void loop() {}
|
||||||
void set_address(uint64_t address) { address_ = address; }
|
void set_address(uint64_t address) { address_ = address; }
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ from esphome.const import (
|
|||||||
CONF_IBEACON_MAJOR,
|
CONF_IBEACON_MAJOR,
|
||||||
CONF_IBEACON_MINOR,
|
CONF_IBEACON_MINOR,
|
||||||
CONF_IBEACON_UUID,
|
CONF_IBEACON_UUID,
|
||||||
CONF_MIN_RSSI,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
DEPENDENCIES = ["esp32_ble_tracker"]
|
DEPENDENCIES = ["esp32_ble_tracker"]
|
||||||
@@ -38,9 +37,6 @@ CONFIG_SCHEMA = cv.All(
|
|||||||
cv.Optional(CONF_IBEACON_MAJOR): cv.uint16_t,
|
cv.Optional(CONF_IBEACON_MAJOR): cv.uint16_t,
|
||||||
cv.Optional(CONF_IBEACON_MINOR): cv.uint16_t,
|
cv.Optional(CONF_IBEACON_MINOR): cv.uint16_t,
|
||||||
cv.Optional(CONF_IBEACON_UUID): cv.uuid,
|
cv.Optional(CONF_IBEACON_UUID): cv.uuid,
|
||||||
cv.Optional(CONF_MIN_RSSI): cv.All(
|
|
||||||
cv.decibel, cv.int_range(min=-90, max=-30)
|
|
||||||
),
|
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
.extend(esp32_ble_tracker.ESP_BLE_DEVICE_SCHEMA)
|
.extend(esp32_ble_tracker.ESP_BLE_DEVICE_SCHEMA)
|
||||||
@@ -55,9 +51,6 @@ async def to_code(config):
|
|||||||
await cg.register_component(var, config)
|
await cg.register_component(var, config)
|
||||||
await esp32_ble_tracker.register_ble_device(var, config)
|
await esp32_ble_tracker.register_ble_device(var, config)
|
||||||
|
|
||||||
if CONF_MIN_RSSI in config:
|
|
||||||
cg.add(var.set_minimum_rssi(config[CONF_MIN_RSSI]))
|
|
||||||
|
|
||||||
if CONF_MAC_ADDRESS in config:
|
if CONF_MAC_ADDRESS in config:
|
||||||
cg.add(var.set_address(config[CONF_MAC_ADDRESS].as_hex))
|
cg.add(var.set_address(config[CONF_MAC_ADDRESS].as_hex))
|
||||||
|
|
||||||
|
|||||||
@@ -41,19 +41,12 @@ class BLEPresenceDevice : public binary_sensor::BinarySensorInitiallyOff,
|
|||||||
this->check_ibeacon_minor_ = true;
|
this->check_ibeacon_minor_ = true;
|
||||||
this->ibeacon_minor_ = minor;
|
this->ibeacon_minor_ = minor;
|
||||||
}
|
}
|
||||||
void set_minimum_rssi(int rssi) {
|
|
||||||
this->check_minimum_rssi_ = true;
|
|
||||||
this->minimum_rssi_ = rssi;
|
|
||||||
}
|
|
||||||
void on_scan_end() override {
|
void on_scan_end() override {
|
||||||
if (!this->found_)
|
if (!this->found_)
|
||||||
this->publish_state(false);
|
this->publish_state(false);
|
||||||
this->found_ = false;
|
this->found_ = false;
|
||||||
}
|
}
|
||||||
bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override {
|
bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override {
|
||||||
if (this->check_minimum_rssi_ && this->minimum_rssi_ <= device.get_rssi()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
switch (this->match_by_) {
|
switch (this->match_by_) {
|
||||||
case MATCH_BY_MAC_ADDRESS:
|
case MATCH_BY_MAC_ADDRESS:
|
||||||
if (device.address_uint64() == this->address_) {
|
if (device.address_uint64() == this->address_) {
|
||||||
@@ -103,21 +96,17 @@ class BLEPresenceDevice : public binary_sensor::BinarySensorInitiallyOff,
|
|||||||
enum MatchType { MATCH_BY_MAC_ADDRESS, MATCH_BY_SERVICE_UUID, MATCH_BY_IBEACON_UUID };
|
enum MatchType { MATCH_BY_MAC_ADDRESS, MATCH_BY_SERVICE_UUID, MATCH_BY_IBEACON_UUID };
|
||||||
MatchType match_by_;
|
MatchType match_by_;
|
||||||
|
|
||||||
|
bool found_{false};
|
||||||
|
|
||||||
uint64_t address_;
|
uint64_t address_;
|
||||||
|
|
||||||
esp32_ble_tracker::ESPBTUUID uuid_;
|
esp32_ble_tracker::ESPBTUUID uuid_;
|
||||||
|
|
||||||
esp32_ble_tracker::ESPBTUUID ibeacon_uuid_;
|
esp32_ble_tracker::ESPBTUUID ibeacon_uuid_;
|
||||||
uint16_t ibeacon_major_{0};
|
uint16_t ibeacon_major_;
|
||||||
uint16_t ibeacon_minor_{0};
|
bool check_ibeacon_major_;
|
||||||
|
uint16_t ibeacon_minor_;
|
||||||
int minimum_rssi_{0};
|
bool check_ibeacon_minor_;
|
||||||
|
|
||||||
bool check_ibeacon_major_{false};
|
|
||||||
bool check_ibeacon_minor_{false};
|
|
||||||
bool check_minimum_rssi_{false};
|
|
||||||
|
|
||||||
bool found_{false};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace ble_presence
|
} // namespace ble_presence
|
||||||
|
|||||||
@@ -102,9 +102,8 @@ class BLERSSISensor : public sensor::Sensor, public esp32_ble_tracker::ESPBTDevi
|
|||||||
|
|
||||||
esp32_ble_tracker::ESPBTUUID ibeacon_uuid_;
|
esp32_ble_tracker::ESPBTUUID ibeacon_uuid_;
|
||||||
uint16_t ibeacon_major_;
|
uint16_t ibeacon_major_;
|
||||||
uint16_t ibeacon_minor_;
|
|
||||||
|
|
||||||
bool check_ibeacon_major_;
|
bool check_ibeacon_major_;
|
||||||
|
uint16_t ibeacon_minor_;
|
||||||
bool check_ibeacon_minor_;
|
bool check_ibeacon_minor_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
#include "bluetooth_connection.h"
|
#include "bluetooth_connection.h"
|
||||||
|
|
||||||
#include "esphome/components/api/api_pb2.h"
|
#include "esphome/components/api/api_server.h"
|
||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
@@ -20,21 +20,24 @@ bool BluetoothConnection::gattc_event_handler(esp_gattc_cb_event_t event, esp_ga
|
|||||||
|
|
||||||
switch (event) {
|
switch (event) {
|
||||||
case ESP_GATTC_DISCONNECT_EVT: {
|
case ESP_GATTC_DISCONNECT_EVT: {
|
||||||
this->proxy_->send_device_connection(this->address_, false, 0, param->disconnect.reason);
|
api::global_api_server->send_bluetooth_device_connection(this->address_, false, 0, param->disconnect.reason);
|
||||||
this->set_address(0);
|
this->set_address(0);
|
||||||
this->proxy_->send_connections_free();
|
api::global_api_server->send_bluetooth_connections_free(this->proxy_->get_bluetooth_connections_free(),
|
||||||
|
this->proxy_->get_bluetooth_connections_limit());
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ESP_GATTC_OPEN_EVT: {
|
case ESP_GATTC_OPEN_EVT: {
|
||||||
if (param->open.conn_id != this->conn_id_)
|
if (param->open.conn_id != this->conn_id_)
|
||||||
break;
|
break;
|
||||||
if (param->open.status != ESP_GATT_OK && param->open.status != ESP_GATT_ALREADY_OPEN) {
|
if (param->open.status != ESP_GATT_OK && param->open.status != ESP_GATT_ALREADY_OPEN) {
|
||||||
this->proxy_->send_device_connection(this->address_, false, 0, param->open.status);
|
api::global_api_server->send_bluetooth_device_connection(this->address_, false, 0, param->open.status);
|
||||||
this->set_address(0);
|
this->set_address(0);
|
||||||
this->proxy_->send_connections_free();
|
api::global_api_server->send_bluetooth_connections_free(this->proxy_->get_bluetooth_connections_free(),
|
||||||
|
this->proxy_->get_bluetooth_connections_limit());
|
||||||
} else if (this->connection_type_ == espbt::ConnectionType::V3_WITH_CACHE) {
|
} else if (this->connection_type_ == espbt::ConnectionType::V3_WITH_CACHE) {
|
||||||
this->proxy_->send_device_connection(this->address_, true, this->mtu_);
|
api::global_api_server->send_bluetooth_device_connection(this->address_, true, this->mtu_);
|
||||||
this->proxy_->send_connections_free();
|
api::global_api_server->send_bluetooth_connections_free(this->proxy_->get_bluetooth_connections_free(),
|
||||||
|
this->proxy_->get_bluetooth_connections_limit());
|
||||||
}
|
}
|
||||||
this->seen_mtu_or_services_ = false;
|
this->seen_mtu_or_services_ = false;
|
||||||
break;
|
break;
|
||||||
@@ -49,8 +52,9 @@ bool BluetoothConnection::gattc_event_handler(esp_gattc_cb_event_t event, esp_ga
|
|||||||
this->seen_mtu_or_services_ = true;
|
this->seen_mtu_or_services_ = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
this->proxy_->send_device_connection(this->address_, true, this->mtu_);
|
api::global_api_server->send_bluetooth_device_connection(this->address_, true, this->mtu_);
|
||||||
this->proxy_->send_connections_free();
|
api::global_api_server->send_bluetooth_connections_free(this->proxy_->get_bluetooth_connections_free(),
|
||||||
|
this->proxy_->get_bluetooth_connections_limit());
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ESP_GATTC_SEARCH_CMPL_EVT: {
|
case ESP_GATTC_SEARCH_CMPL_EVT: {
|
||||||
@@ -63,8 +67,9 @@ bool BluetoothConnection::gattc_event_handler(esp_gattc_cb_event_t event, esp_ga
|
|||||||
this->seen_mtu_or_services_ = true;
|
this->seen_mtu_or_services_ = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
this->proxy_->send_device_connection(this->address_, true, this->mtu_);
|
api::global_api_server->send_bluetooth_device_connection(this->address_, true, this->mtu_);
|
||||||
this->proxy_->send_connections_free();
|
api::global_api_server->send_bluetooth_connections_free(this->proxy_->get_bluetooth_connections_free(),
|
||||||
|
this->proxy_->get_bluetooth_connections_limit());
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ESP_GATTC_READ_DESCR_EVT:
|
case ESP_GATTC_READ_DESCR_EVT:
|
||||||
@@ -74,7 +79,7 @@ bool BluetoothConnection::gattc_event_handler(esp_gattc_cb_event_t event, esp_ga
|
|||||||
if (param->read.status != ESP_GATT_OK) {
|
if (param->read.status != ESP_GATT_OK) {
|
||||||
ESP_LOGW(TAG, "[%d] [%s] Error reading char/descriptor at handle 0x%2X, status=%d", this->connection_index_,
|
ESP_LOGW(TAG, "[%d] [%s] Error reading char/descriptor at handle 0x%2X, status=%d", this->connection_index_,
|
||||||
this->address_str_.c_str(), param->read.handle, param->read.status);
|
this->address_str_.c_str(), param->read.handle, param->read.status);
|
||||||
this->proxy_->send_gatt_error(this->address_, param->read.handle, param->read.status);
|
api::global_api_server->send_bluetooth_gatt_error(this->address_, param->read.handle, param->read.status);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
api::BluetoothGATTReadResponse resp;
|
api::BluetoothGATTReadResponse resp;
|
||||||
@@ -84,7 +89,7 @@ bool BluetoothConnection::gattc_event_handler(esp_gattc_cb_event_t event, esp_ga
|
|||||||
for (uint16_t i = 0; i < param->read.value_len; i++) {
|
for (uint16_t i = 0; i < param->read.value_len; i++) {
|
||||||
resp.data.push_back(param->read.value[i]);
|
resp.data.push_back(param->read.value[i]);
|
||||||
}
|
}
|
||||||
this->proxy_->get_api_connection()->send_bluetooth_gatt_read_response(resp);
|
api::global_api_server->send_bluetooth_gatt_read_response(resp);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ESP_GATTC_WRITE_CHAR_EVT:
|
case ESP_GATTC_WRITE_CHAR_EVT:
|
||||||
@@ -94,13 +99,13 @@ bool BluetoothConnection::gattc_event_handler(esp_gattc_cb_event_t event, esp_ga
|
|||||||
if (param->write.status != ESP_GATT_OK) {
|
if (param->write.status != ESP_GATT_OK) {
|
||||||
ESP_LOGW(TAG, "[%d] [%s] Error writing char/descriptor at handle 0x%2X, status=%d", this->connection_index_,
|
ESP_LOGW(TAG, "[%d] [%s] Error writing char/descriptor at handle 0x%2X, status=%d", this->connection_index_,
|
||||||
this->address_str_.c_str(), param->write.handle, param->write.status);
|
this->address_str_.c_str(), param->write.handle, param->write.status);
|
||||||
this->proxy_->send_gatt_error(this->address_, param->write.handle, param->write.status);
|
api::global_api_server->send_bluetooth_gatt_error(this->address_, param->write.handle, param->write.status);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
api::BluetoothGATTWriteResponse resp;
|
api::BluetoothGATTWriteResponse resp;
|
||||||
resp.address = this->address_;
|
resp.address = this->address_;
|
||||||
resp.handle = param->write.handle;
|
resp.handle = param->write.handle;
|
||||||
this->proxy_->get_api_connection()->send_bluetooth_gatt_write_response(resp);
|
api::global_api_server->send_bluetooth_gatt_write_response(resp);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ESP_GATTC_UNREG_FOR_NOTIFY_EVT: {
|
case ESP_GATTC_UNREG_FOR_NOTIFY_EVT: {
|
||||||
@@ -108,26 +113,28 @@ bool BluetoothConnection::gattc_event_handler(esp_gattc_cb_event_t event, esp_ga
|
|||||||
ESP_LOGW(TAG, "[%d] [%s] Error unregistering notifications for handle 0x%2X, status=%d",
|
ESP_LOGW(TAG, "[%d] [%s] Error unregistering notifications for handle 0x%2X, status=%d",
|
||||||
this->connection_index_, this->address_str_.c_str(), param->unreg_for_notify.handle,
|
this->connection_index_, this->address_str_.c_str(), param->unreg_for_notify.handle,
|
||||||
param->unreg_for_notify.status);
|
param->unreg_for_notify.status);
|
||||||
this->proxy_->send_gatt_error(this->address_, param->unreg_for_notify.handle, param->unreg_for_notify.status);
|
api::global_api_server->send_bluetooth_gatt_error(this->address_, param->unreg_for_notify.handle,
|
||||||
|
param->unreg_for_notify.status);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
api::BluetoothGATTNotifyResponse resp;
|
api::BluetoothGATTNotifyResponse resp;
|
||||||
resp.address = this->address_;
|
resp.address = this->address_;
|
||||||
resp.handle = param->unreg_for_notify.handle;
|
resp.handle = param->unreg_for_notify.handle;
|
||||||
this->proxy_->get_api_connection()->send_bluetooth_gatt_notify_response(resp);
|
api::global_api_server->send_bluetooth_gatt_notify_response(resp);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ESP_GATTC_REG_FOR_NOTIFY_EVT: {
|
case ESP_GATTC_REG_FOR_NOTIFY_EVT: {
|
||||||
if (param->reg_for_notify.status != ESP_GATT_OK) {
|
if (param->reg_for_notify.status != ESP_GATT_OK) {
|
||||||
ESP_LOGW(TAG, "[%d] [%s] Error registering notifications for handle 0x%2X, status=%d", this->connection_index_,
|
ESP_LOGW(TAG, "[%d] [%s] Error registering notifications for handle 0x%2X, status=%d", this->connection_index_,
|
||||||
this->address_str_.c_str(), param->reg_for_notify.handle, param->reg_for_notify.status);
|
this->address_str_.c_str(), param->reg_for_notify.handle, param->reg_for_notify.status);
|
||||||
this->proxy_->send_gatt_error(this->address_, param->reg_for_notify.handle, param->reg_for_notify.status);
|
api::global_api_server->send_bluetooth_gatt_error(this->address_, param->reg_for_notify.handle,
|
||||||
|
param->reg_for_notify.status);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
api::BluetoothGATTNotifyResponse resp;
|
api::BluetoothGATTNotifyResponse resp;
|
||||||
resp.address = this->address_;
|
resp.address = this->address_;
|
||||||
resp.handle = param->reg_for_notify.handle;
|
resp.handle = param->reg_for_notify.handle;
|
||||||
this->proxy_->get_api_connection()->send_bluetooth_gatt_notify_response(resp);
|
api::global_api_server->send_bluetooth_gatt_notify_response(resp);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ESP_GATTC_NOTIFY_EVT: {
|
case ESP_GATTC_NOTIFY_EVT: {
|
||||||
@@ -142,7 +149,7 @@ bool BluetoothConnection::gattc_event_handler(esp_gattc_cb_event_t event, esp_ga
|
|||||||
for (uint16_t i = 0; i < param->notify.value_len; i++) {
|
for (uint16_t i = 0; i < param->notify.value_len; i++) {
|
||||||
resp.data.push_back(param->notify.value[i]);
|
resp.data.push_back(param->notify.value[i]);
|
||||||
}
|
}
|
||||||
this->proxy_->get_api_connection()->send_bluetooth_gatt_notify_data_response(resp);
|
api::global_api_server->send_bluetooth_gatt_notify_data_response(resp);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
@@ -159,9 +166,10 @@ void BluetoothConnection::gap_event_handler(esp_gap_ble_cb_event_t event, esp_bl
|
|||||||
if (memcmp(param->ble_security.auth_cmpl.bd_addr, this->remote_bda_, 6) != 0)
|
if (memcmp(param->ble_security.auth_cmpl.bd_addr, this->remote_bda_, 6) != 0)
|
||||||
break;
|
break;
|
||||||
if (param->ble_security.auth_cmpl.success) {
|
if (param->ble_security.auth_cmpl.success) {
|
||||||
this->proxy_->send_device_pairing(this->address_, true);
|
api::global_api_server->send_bluetooth_device_pairing(this->address_, true);
|
||||||
} else {
|
} else {
|
||||||
this->proxy_->send_device_pairing(this->address_, false, param->ble_security.auth_cmpl.fail_reason);
|
api::global_api_server->send_bluetooth_device_pairing(this->address_, false,
|
||||||
|
param->ble_security.auth_cmpl.fail_reason);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
|||||||
@@ -1,10 +1,11 @@
|
|||||||
#include "bluetooth_proxy.h"
|
#include "bluetooth_proxy.h"
|
||||||
|
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
#include "esphome/core/macros.h"
|
|
||||||
|
|
||||||
#ifdef USE_ESP32
|
#ifdef USE_ESP32
|
||||||
|
|
||||||
|
#include "esphome/components/api/api_server.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace bluetooth_proxy {
|
namespace bluetooth_proxy {
|
||||||
|
|
||||||
@@ -26,39 +27,15 @@ std::vector<uint64_t> get_128bit_uuid_vec(esp_bt_uuid_t uuid_source) {
|
|||||||
BluetoothProxy::BluetoothProxy() { global_bluetooth_proxy = this; }
|
BluetoothProxy::BluetoothProxy() { global_bluetooth_proxy = this; }
|
||||||
|
|
||||||
bool BluetoothProxy::parse_device(const esp32_ble_tracker::ESPBTDevice &device) {
|
bool BluetoothProxy::parse_device(const esp32_ble_tracker::ESPBTDevice &device) {
|
||||||
if (!api::global_api_server->is_connected() || this->api_connection_ == nullptr || this->raw_advertisements_)
|
if (!api::global_api_server->is_connected())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
ESP_LOGV(TAG, "Proxying packet from %s - %s. RSSI: %d dB", device.get_name().c_str(), device.address_str().c_str(),
|
ESP_LOGV(TAG, "Proxying packet from %s - %s. RSSI: %d dB", device.get_name().c_str(), device.address_str().c_str(),
|
||||||
device.get_rssi());
|
device.get_rssi());
|
||||||
this->send_api_packet_(device);
|
this->send_api_packet_(device);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool BluetoothProxy::parse_devices(esp_ble_gap_cb_param_t::ble_scan_result_evt_param *advertisements, size_t count) {
|
|
||||||
if (!api::global_api_server->is_connected() || this->api_connection_ == nullptr || !this->raw_advertisements_)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
api::BluetoothLERawAdvertisementsResponse resp;
|
|
||||||
for (size_t i = 0; i < count; i++) {
|
|
||||||
auto &result = advertisements[i];
|
|
||||||
api::BluetoothLERawAdvertisement adv;
|
|
||||||
adv.address = esp32_ble::ble_addr_to_uint64(result.bda);
|
|
||||||
adv.rssi = result.rssi;
|
|
||||||
adv.address_type = result.ble_addr_type;
|
|
||||||
|
|
||||||
uint8_t length = result.adv_data_len + result.scan_rsp_len;
|
|
||||||
adv.data.reserve(length);
|
|
||||||
for (uint16_t i = 0; i < length; i++) {
|
|
||||||
adv.data.push_back(result.ble_adv[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
resp.advertisements.push_back(std::move(adv));
|
|
||||||
}
|
|
||||||
ESP_LOGV(TAG, "Proxying %d packets", count);
|
|
||||||
this->api_connection_->send_bluetooth_le_raw_advertisements_response(resp);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
void BluetoothProxy::send_api_packet_(const esp32_ble_tracker::ESPBTDevice &device) {
|
void BluetoothProxy::send_api_packet_(const esp32_ble_tracker::ESPBTDevice &device) {
|
||||||
api::BluetoothLEAdvertisementResponse resp;
|
api::BluetoothLEAdvertisementResponse resp;
|
||||||
resp.address = device.address_uint64();
|
resp.address = device.address_uint64();
|
||||||
@@ -81,7 +58,7 @@ void BluetoothProxy::send_api_packet_(const esp32_ble_tracker::ESPBTDevice &devi
|
|||||||
manufacturer_data.data.assign(data.data.begin(), data.data.end());
|
manufacturer_data.data.assign(data.data.begin(), data.data.end());
|
||||||
resp.manufacturer_data.push_back(std::move(manufacturer_data));
|
resp.manufacturer_data.push_back(std::move(manufacturer_data));
|
||||||
}
|
}
|
||||||
this->api_connection_->send_bluetooth_le_advertisement(resp);
|
api::global_api_server->send_bluetooth_le_advertisement(resp);
|
||||||
}
|
}
|
||||||
|
|
||||||
void BluetoothProxy::dump_config() {
|
void BluetoothProxy::dump_config() {
|
||||||
@@ -104,7 +81,7 @@ int BluetoothProxy::get_bluetooth_connections_free() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void BluetoothProxy::loop() {
|
void BluetoothProxy::loop() {
|
||||||
if (!api::global_api_server->is_connected() || this->api_connection_ == nullptr) {
|
if (!api::global_api_server->is_connected()) {
|
||||||
for (auto *connection : this->connections_) {
|
for (auto *connection : this->connections_) {
|
||||||
if (connection->get_address() != 0) {
|
if (connection->get_address() != 0) {
|
||||||
connection->disconnect();
|
connection->disconnect();
|
||||||
@@ -115,7 +92,7 @@ void BluetoothProxy::loop() {
|
|||||||
for (auto *connection : this->connections_) {
|
for (auto *connection : this->connections_) {
|
||||||
if (connection->send_service_ == connection->service_count_) {
|
if (connection->send_service_ == connection->service_count_) {
|
||||||
connection->send_service_ = DONE_SENDING_SERVICES;
|
connection->send_service_ = DONE_SENDING_SERVICES;
|
||||||
this->send_gatt_services_done(connection->get_address());
|
api::global_api_server->send_bluetooth_gatt_services_done(connection->get_address());
|
||||||
if (connection->connection_type_ == espbt::ConnectionType::V3_WITH_CACHE ||
|
if (connection->connection_type_ == espbt::ConnectionType::V3_WITH_CACHE ||
|
||||||
connection->connection_type_ == espbt::ConnectionType::V3_WITHOUT_CACHE) {
|
connection->connection_type_ == espbt::ConnectionType::V3_WITHOUT_CACHE) {
|
||||||
connection->release_services();
|
connection->release_services();
|
||||||
@@ -193,7 +170,7 @@ void BluetoothProxy::loop() {
|
|||||||
service_resp.characteristics.push_back(std::move(characteristic_resp));
|
service_resp.characteristics.push_back(std::move(characteristic_resp));
|
||||||
}
|
}
|
||||||
resp.services.push_back(std::move(service_resp));
|
resp.services.push_back(std::move(service_resp));
|
||||||
this->api_connection_->send_bluetooth_gatt_get_services_response(resp);
|
api::global_api_server->send_bluetooth_gatt_services(resp);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -231,15 +208,16 @@ void BluetoothProxy::bluetooth_device_request(const api::BluetoothDeviceRequest
|
|||||||
auto *connection = this->get_connection_(msg.address, true);
|
auto *connection = this->get_connection_(msg.address, true);
|
||||||
if (connection == nullptr) {
|
if (connection == nullptr) {
|
||||||
ESP_LOGW(TAG, "No free connections available");
|
ESP_LOGW(TAG, "No free connections available");
|
||||||
this->send_device_connection(msg.address, false);
|
api::global_api_server->send_bluetooth_device_connection(msg.address, false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (connection->state() == espbt::ClientState::CONNECTED ||
|
if (connection->state() == espbt::ClientState::CONNECTED ||
|
||||||
connection->state() == espbt::ClientState::ESTABLISHED) {
|
connection->state() == espbt::ClientState::ESTABLISHED) {
|
||||||
ESP_LOGW(TAG, "[%d] [%s] Connection already established", connection->get_connection_index(),
|
ESP_LOGW(TAG, "[%d] [%s] Connection already established", connection->get_connection_index(),
|
||||||
connection->address_str().c_str());
|
connection->address_str().c_str());
|
||||||
this->send_device_connection(msg.address, true);
|
api::global_api_server->send_bluetooth_device_connection(msg.address, true);
|
||||||
this->send_connections_free();
|
api::global_api_server->send_bluetooth_connections_free(this->get_bluetooth_connections_free(),
|
||||||
|
this->get_bluetooth_connections_limit());
|
||||||
return;
|
return;
|
||||||
} else if (connection->state() == espbt::ClientState::SEARCHING) {
|
} else if (connection->state() == espbt::ClientState::SEARCHING) {
|
||||||
ESP_LOGW(TAG, "[%d] [%s] Connection request ignored, already searching for device",
|
ESP_LOGW(TAG, "[%d] [%s] Connection request ignored, already searching for device",
|
||||||
@@ -285,22 +263,25 @@ void BluetoothProxy::bluetooth_device_request(const api::BluetoothDeviceRequest
|
|||||||
} else {
|
} else {
|
||||||
connection->set_state(espbt::ClientState::SEARCHING);
|
connection->set_state(espbt::ClientState::SEARCHING);
|
||||||
}
|
}
|
||||||
this->send_connections_free();
|
api::global_api_server->send_bluetooth_connections_free(this->get_bluetooth_connections_free(),
|
||||||
|
this->get_bluetooth_connections_limit());
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case api::enums::BLUETOOTH_DEVICE_REQUEST_TYPE_DISCONNECT: {
|
case api::enums::BLUETOOTH_DEVICE_REQUEST_TYPE_DISCONNECT: {
|
||||||
auto *connection = this->get_connection_(msg.address, false);
|
auto *connection = this->get_connection_(msg.address, false);
|
||||||
if (connection == nullptr) {
|
if (connection == nullptr) {
|
||||||
this->send_device_connection(msg.address, false);
|
api::global_api_server->send_bluetooth_device_connection(msg.address, false);
|
||||||
this->send_connections_free();
|
api::global_api_server->send_bluetooth_connections_free(this->get_bluetooth_connections_free(),
|
||||||
|
this->get_bluetooth_connections_limit());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (connection->state() != espbt::ClientState::IDLE) {
|
if (connection->state() != espbt::ClientState::IDLE) {
|
||||||
connection->disconnect();
|
connection->disconnect();
|
||||||
} else {
|
} else {
|
||||||
connection->set_address(0);
|
connection->set_address(0);
|
||||||
this->send_device_connection(msg.address, false);
|
api::global_api_server->send_bluetooth_device_connection(msg.address, false);
|
||||||
this->send_connections_free();
|
api::global_api_server->send_bluetooth_connections_free(this->get_bluetooth_connections_free(),
|
||||||
|
this->get_bluetooth_connections_limit());
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -310,10 +291,10 @@ void BluetoothProxy::bluetooth_device_request(const api::BluetoothDeviceRequest
|
|||||||
if (!connection->is_paired()) {
|
if (!connection->is_paired()) {
|
||||||
auto err = connection->pair();
|
auto err = connection->pair();
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
this->send_device_pairing(msg.address, false, err);
|
api::global_api_server->send_bluetooth_device_pairing(msg.address, false, err);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
this->send_device_pairing(msg.address, true);
|
api::global_api_server->send_bluetooth_device_pairing(msg.address, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -322,20 +303,7 @@ void BluetoothProxy::bluetooth_device_request(const api::BluetoothDeviceRequest
|
|||||||
esp_bd_addr_t address;
|
esp_bd_addr_t address;
|
||||||
uint64_to_bd_addr(msg.address, address);
|
uint64_to_bd_addr(msg.address, address);
|
||||||
esp_err_t ret = esp_ble_remove_bond_device(address);
|
esp_err_t ret = esp_ble_remove_bond_device(address);
|
||||||
this->send_device_pairing(msg.address, ret == ESP_OK, ret);
|
api::global_api_server->send_bluetooth_device_unpairing(msg.address, ret == ESP_OK, ret);
|
||||||
break;
|
|
||||||
}
|
|
||||||
case api::enums::BLUETOOTH_DEVICE_REQUEST_TYPE_CLEAR_CACHE: {
|
|
||||||
esp_bd_addr_t address;
|
|
||||||
uint64_to_bd_addr(msg.address, address);
|
|
||||||
esp_err_t ret = esp_ble_gattc_cache_clean(address);
|
|
||||||
api::BluetoothDeviceClearCacheResponse call;
|
|
||||||
call.address = msg.address;
|
|
||||||
call.success = ret == ESP_OK;
|
|
||||||
call.error = ret;
|
|
||||||
|
|
||||||
this->api_connection_->send_bluetooth_device_clear_cache_response(call);
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -345,13 +313,13 @@ void BluetoothProxy::bluetooth_gatt_read(const api::BluetoothGATTReadRequest &ms
|
|||||||
auto *connection = this->get_connection_(msg.address, false);
|
auto *connection = this->get_connection_(msg.address, false);
|
||||||
if (connection == nullptr) {
|
if (connection == nullptr) {
|
||||||
ESP_LOGW(TAG, "Cannot read GATT characteristic, not connected");
|
ESP_LOGW(TAG, "Cannot read GATT characteristic, not connected");
|
||||||
this->send_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED);
|
api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto err = connection->read_characteristic(msg.handle);
|
auto err = connection->read_characteristic(msg.handle);
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
this->send_gatt_error(msg.address, msg.handle, err);
|
api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -359,13 +327,13 @@ void BluetoothProxy::bluetooth_gatt_write(const api::BluetoothGATTWriteRequest &
|
|||||||
auto *connection = this->get_connection_(msg.address, false);
|
auto *connection = this->get_connection_(msg.address, false);
|
||||||
if (connection == nullptr) {
|
if (connection == nullptr) {
|
||||||
ESP_LOGW(TAG, "Cannot write GATT characteristic, not connected");
|
ESP_LOGW(TAG, "Cannot write GATT characteristic, not connected");
|
||||||
this->send_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED);
|
api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto err = connection->write_characteristic(msg.handle, msg.data, msg.response);
|
auto err = connection->write_characteristic(msg.handle, msg.data, msg.response);
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
this->send_gatt_error(msg.address, msg.handle, err);
|
api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -373,13 +341,13 @@ void BluetoothProxy::bluetooth_gatt_read_descriptor(const api::BluetoothGATTRead
|
|||||||
auto *connection = this->get_connection_(msg.address, false);
|
auto *connection = this->get_connection_(msg.address, false);
|
||||||
if (connection == nullptr) {
|
if (connection == nullptr) {
|
||||||
ESP_LOGW(TAG, "Cannot read GATT descriptor, not connected");
|
ESP_LOGW(TAG, "Cannot read GATT descriptor, not connected");
|
||||||
this->send_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED);
|
api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto err = connection->read_descriptor(msg.handle);
|
auto err = connection->read_descriptor(msg.handle);
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
this->send_gatt_error(msg.address, msg.handle, err);
|
api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -387,13 +355,13 @@ void BluetoothProxy::bluetooth_gatt_write_descriptor(const api::BluetoothGATTWri
|
|||||||
auto *connection = this->get_connection_(msg.address, false);
|
auto *connection = this->get_connection_(msg.address, false);
|
||||||
if (connection == nullptr) {
|
if (connection == nullptr) {
|
||||||
ESP_LOGW(TAG, "Cannot write GATT descriptor, not connected");
|
ESP_LOGW(TAG, "Cannot write GATT descriptor, not connected");
|
||||||
this->send_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED);
|
api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto err = connection->write_descriptor(msg.handle, msg.data, true);
|
auto err = connection->write_descriptor(msg.handle, msg.data, true);
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
this->send_gatt_error(msg.address, msg.handle, err);
|
api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -401,12 +369,12 @@ void BluetoothProxy::bluetooth_gatt_send_services(const api::BluetoothGATTGetSer
|
|||||||
auto *connection = this->get_connection_(msg.address, false);
|
auto *connection = this->get_connection_(msg.address, false);
|
||||||
if (connection == nullptr || !connection->connected()) {
|
if (connection == nullptr || !connection->connected()) {
|
||||||
ESP_LOGW(TAG, "Cannot get GATT services, not connected");
|
ESP_LOGW(TAG, "Cannot get GATT services, not connected");
|
||||||
this->send_gatt_error(msg.address, 0, ESP_GATT_NOT_CONNECTED);
|
api::global_api_server->send_bluetooth_gatt_error(msg.address, 0, ESP_GATT_NOT_CONNECTED);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!connection->service_count_) {
|
if (!connection->service_count_) {
|
||||||
ESP_LOGW(TAG, "[%d] [%s] No GATT services found", connection->connection_index_, connection->address_str().c_str());
|
ESP_LOGW(TAG, "[%d] [%s] No GATT services found", connection->connection_index_, connection->address_str().c_str());
|
||||||
this->send_gatt_services_done(msg.address);
|
api::global_api_server->send_bluetooth_gatt_services_done(msg.address);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (connection->send_service_ ==
|
if (connection->send_service_ ==
|
||||||
@@ -418,89 +386,16 @@ void BluetoothProxy::bluetooth_gatt_notify(const api::BluetoothGATTNotifyRequest
|
|||||||
auto *connection = this->get_connection_(msg.address, false);
|
auto *connection = this->get_connection_(msg.address, false);
|
||||||
if (connection == nullptr) {
|
if (connection == nullptr) {
|
||||||
ESP_LOGW(TAG, "Cannot notify GATT characteristic, not connected");
|
ESP_LOGW(TAG, "Cannot notify GATT characteristic, not connected");
|
||||||
this->send_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED);
|
api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, ESP_GATT_NOT_CONNECTED);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto err = connection->notify_characteristic(msg.handle, msg.enable);
|
auto err = connection->notify_characteristic(msg.handle, msg.enable);
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
this->send_gatt_error(msg.address, msg.handle, err);
|
api::global_api_server->send_bluetooth_gatt_error(msg.address, msg.handle, err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void BluetoothProxy::subscribe_api_connection(api::APIConnection *api_connection, uint32_t flags) {
|
|
||||||
if (this->api_connection_ != nullptr) {
|
|
||||||
ESP_LOGE(TAG, "Only one API subscription is allowed at a time");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this->api_connection_ = api_connection;
|
|
||||||
this->raw_advertisements_ = flags & BluetoothProxySubscriptionFlag::SUBSCRIPTION_RAW_ADVERTISEMENTS;
|
|
||||||
}
|
|
||||||
|
|
||||||
void BluetoothProxy::unsubscribe_api_connection(api::APIConnection *api_connection) {
|
|
||||||
if (this->api_connection_ != api_connection) {
|
|
||||||
ESP_LOGV(TAG, "API connection is not subscribed");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this->api_connection_ = nullptr;
|
|
||||||
this->raw_advertisements_ = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void BluetoothProxy::send_device_connection(uint64_t address, bool connected, uint16_t mtu, esp_err_t error) {
|
|
||||||
if (this->api_connection_ == nullptr)
|
|
||||||
return;
|
|
||||||
api::BluetoothDeviceConnectionResponse call;
|
|
||||||
call.address = address;
|
|
||||||
call.connected = connected;
|
|
||||||
call.mtu = mtu;
|
|
||||||
call.error = error;
|
|
||||||
this->api_connection_->send_bluetooth_device_connection_response(call);
|
|
||||||
}
|
|
||||||
void BluetoothProxy::send_connections_free() {
|
|
||||||
if (this->api_connection_ == nullptr)
|
|
||||||
return;
|
|
||||||
api::BluetoothConnectionsFreeResponse call;
|
|
||||||
call.free = this->get_bluetooth_connections_free();
|
|
||||||
call.limit = this->get_bluetooth_connections_limit();
|
|
||||||
this->api_connection_->send_bluetooth_connections_free_response(call);
|
|
||||||
}
|
|
||||||
|
|
||||||
void BluetoothProxy::send_gatt_services_done(uint64_t address) {
|
|
||||||
if (this->api_connection_ == nullptr)
|
|
||||||
return;
|
|
||||||
api::BluetoothGATTGetServicesDoneResponse call;
|
|
||||||
call.address = address;
|
|
||||||
this->api_connection_->send_bluetooth_gatt_get_services_done_response(call);
|
|
||||||
}
|
|
||||||
|
|
||||||
void BluetoothProxy::send_gatt_error(uint64_t address, uint16_t handle, esp_err_t error) {
|
|
||||||
if (this->api_connection_ == nullptr)
|
|
||||||
return;
|
|
||||||
api::BluetoothGATTErrorResponse call;
|
|
||||||
call.address = address;
|
|
||||||
call.handle = handle;
|
|
||||||
call.error = error;
|
|
||||||
this->api_connection_->send_bluetooth_gatt_error_response(call);
|
|
||||||
}
|
|
||||||
|
|
||||||
void BluetoothProxy::send_device_pairing(uint64_t address, bool paired, esp_err_t error) {
|
|
||||||
api::BluetoothDevicePairingResponse call;
|
|
||||||
call.address = address;
|
|
||||||
call.paired = paired;
|
|
||||||
call.error = error;
|
|
||||||
|
|
||||||
this->api_connection_->send_bluetooth_device_pairing_response(call);
|
|
||||||
}
|
|
||||||
|
|
||||||
void BluetoothProxy::send_device_unpairing(uint64_t address, bool success, esp_err_t error) {
|
|
||||||
api::BluetoothDeviceUnpairingResponse call;
|
|
||||||
call.address = address;
|
|
||||||
call.success = success;
|
|
||||||
call.error = error;
|
|
||||||
|
|
||||||
this->api_connection_->send_bluetooth_device_unpairing_response(call);
|
|
||||||
}
|
|
||||||
|
|
||||||
BluetoothProxy *global_bluetooth_proxy = nullptr; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
|
BluetoothProxy *global_bluetooth_proxy = nullptr; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
|
||||||
|
|
||||||
} // namespace bluetooth_proxy
|
} // namespace bluetooth_proxy
|
||||||
|
|||||||
@@ -5,7 +5,6 @@
|
|||||||
#include <map>
|
#include <map>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include "esphome/components/api/api_connection.h"
|
|
||||||
#include "esphome/components/api/api_pb2.h"
|
#include "esphome/components/api/api_pb2.h"
|
||||||
#include "esphome/components/esp32_ble_client/ble_client_base.h"
|
#include "esphome/components/esp32_ble_client/ble_client_base.h"
|
||||||
#include "esphome/components/esp32_ble_tracker/esp32_ble_tracker.h"
|
#include "esphome/components/esp32_ble_tracker/esp32_ble_tracker.h"
|
||||||
@@ -22,33 +21,10 @@ static const esp_err_t ESP_GATT_NOT_CONNECTED = -1;
|
|||||||
|
|
||||||
using namespace esp32_ble_client;
|
using namespace esp32_ble_client;
|
||||||
|
|
||||||
// Legacy versions:
|
|
||||||
// Version 1: Initial version without active connections
|
|
||||||
// Version 2: Support for active connections
|
|
||||||
// Version 3: New connection API
|
|
||||||
// Version 4: Pairing support
|
|
||||||
// Version 5: Cache clear support
|
|
||||||
static const uint32_t LEGACY_ACTIVE_CONNECTIONS_VERSION = 5;
|
|
||||||
static const uint32_t LEGACY_PASSIVE_ONLY_VERSION = 1;
|
|
||||||
|
|
||||||
enum BluetoothProxyFeature : uint32_t {
|
|
||||||
FEATURE_PASSIVE_SCAN = 1 << 0,
|
|
||||||
FEATURE_ACTIVE_CONNECTIONS = 1 << 1,
|
|
||||||
FEATURE_REMOTE_CACHING = 1 << 2,
|
|
||||||
FEATURE_PAIRING = 1 << 3,
|
|
||||||
FEATURE_CACHE_CLEARING = 1 << 4,
|
|
||||||
FEATURE_RAW_ADVERTISEMENTS = 1 << 5,
|
|
||||||
};
|
|
||||||
|
|
||||||
enum BluetoothProxySubscriptionFlag : uint32_t {
|
|
||||||
SUBSCRIPTION_RAW_ADVERTISEMENTS = 1 << 0,
|
|
||||||
};
|
|
||||||
|
|
||||||
class BluetoothProxy : public esp32_ble_tracker::ESPBTDeviceListener, public Component {
|
class BluetoothProxy : public esp32_ble_tracker::ESPBTDeviceListener, public Component {
|
||||||
public:
|
public:
|
||||||
BluetoothProxy();
|
BluetoothProxy();
|
||||||
bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override;
|
bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override;
|
||||||
bool parse_devices(esp_ble_gap_cb_param_t::ble_scan_result_evt_param *advertisements, size_t count) override;
|
|
||||||
void dump_config() override;
|
void dump_config() override;
|
||||||
void loop() override;
|
void loop() override;
|
||||||
|
|
||||||
@@ -68,18 +44,6 @@ class BluetoothProxy : public esp32_ble_tracker::ESPBTDeviceListener, public Com
|
|||||||
int get_bluetooth_connections_free();
|
int get_bluetooth_connections_free();
|
||||||
int get_bluetooth_connections_limit() { return this->connections_.size(); }
|
int get_bluetooth_connections_limit() { return this->connections_.size(); }
|
||||||
|
|
||||||
void subscribe_api_connection(api::APIConnection *api_connection, uint32_t flags);
|
|
||||||
void unsubscribe_api_connection(api::APIConnection *api_connection);
|
|
||||||
api::APIConnection *get_api_connection() { return this->api_connection_; }
|
|
||||||
|
|
||||||
void send_device_connection(uint64_t address, bool connected, uint16_t mtu = 0, esp_err_t error = ESP_OK);
|
|
||||||
void send_connections_free();
|
|
||||||
void send_gatt_services_done(uint64_t address);
|
|
||||||
void send_gatt_error(uint64_t address, uint16_t handle, esp_err_t error);
|
|
||||||
void send_device_pairing(uint64_t address, bool paired, esp_err_t error = ESP_OK);
|
|
||||||
void send_device_unpairing(uint64_t address, bool success, esp_err_t error = ESP_OK);
|
|
||||||
void send_device_clear_cache(uint64_t address, bool success, esp_err_t error = ESP_OK);
|
|
||||||
|
|
||||||
static void uint64_to_bd_addr(uint64_t address, esp_bd_addr_t bd_addr) {
|
static void uint64_to_bd_addr(uint64_t address, esp_bd_addr_t bd_addr) {
|
||||||
bd_addr[0] = (address >> 40) & 0xff;
|
bd_addr[0] = (address >> 40) & 0xff;
|
||||||
bd_addr[1] = (address >> 32) & 0xff;
|
bd_addr[1] = (address >> 32) & 0xff;
|
||||||
@@ -92,27 +56,6 @@ class BluetoothProxy : public esp32_ble_tracker::ESPBTDeviceListener, public Com
|
|||||||
void set_active(bool active) { this->active_ = active; }
|
void set_active(bool active) { this->active_ = active; }
|
||||||
bool has_active() { return this->active_; }
|
bool has_active() { return this->active_; }
|
||||||
|
|
||||||
uint32_t get_legacy_version() const {
|
|
||||||
if (this->active_) {
|
|
||||||
return LEGACY_ACTIVE_CONNECTIONS_VERSION;
|
|
||||||
}
|
|
||||||
return LEGACY_PASSIVE_ONLY_VERSION;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t get_feature_flags() const {
|
|
||||||
uint32_t flags = 0;
|
|
||||||
flags |= BluetoothProxyFeature::FEATURE_PASSIVE_SCAN;
|
|
||||||
flags |= BluetoothProxyFeature::FEATURE_RAW_ADVERTISEMENTS;
|
|
||||||
if (this->active_) {
|
|
||||||
flags |= BluetoothProxyFeature::FEATURE_ACTIVE_CONNECTIONS;
|
|
||||||
flags |= BluetoothProxyFeature::FEATURE_REMOTE_CACHING;
|
|
||||||
flags |= BluetoothProxyFeature::FEATURE_PAIRING;
|
|
||||||
flags |= BluetoothProxyFeature::FEATURE_CACHE_CLEARING;
|
|
||||||
}
|
|
||||||
|
|
||||||
return flags;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void send_api_packet_(const esp32_ble_tracker::ESPBTDevice &device);
|
void send_api_packet_(const esp32_ble_tracker::ESPBTDevice &device);
|
||||||
|
|
||||||
@@ -121,8 +64,6 @@ class BluetoothProxy : public esp32_ble_tracker::ESPBTDeviceListener, public Com
|
|||||||
bool active_;
|
bool active_;
|
||||||
|
|
||||||
std::vector<BluetoothConnection *> connections_{};
|
std::vector<BluetoothConnection *> connections_{};
|
||||||
api::APIConnection *api_connection_{nullptr};
|
|
||||||
bool raw_advertisements_{false};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
extern BluetoothProxy *global_bluetooth_proxy; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
|
extern BluetoothProxy *global_bluetooth_proxy; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
from esphome.components import i2c, esp32
|
from esphome.components import i2c
|
||||||
from esphome.const import CONF_ID
|
from esphome.const import CONF_ID
|
||||||
|
|
||||||
CODEOWNERS = ["@trvrnrth"]
|
CODEOWNERS = ["@trvrnrth"]
|
||||||
@@ -32,31 +32,22 @@ BME680BSECComponent = bme680_bsec_ns.class_(
|
|||||||
"BME680BSECComponent", cg.Component, i2c.I2CDevice
|
"BME680BSECComponent", cg.Component, i2c.I2CDevice
|
||||||
)
|
)
|
||||||
|
|
||||||
CONFIG_SCHEMA = cv.All(
|
CONFIG_SCHEMA = cv.Schema(
|
||||||
cv.Schema(
|
{
|
||||||
{
|
cv.GenerateID(): cv.declare_id(BME680BSECComponent),
|
||||||
cv.GenerateID(): cv.declare_id(BME680BSECComponent),
|
cv.Optional(CONF_TEMPERATURE_OFFSET, default=0): cv.temperature,
|
||||||
cv.Optional(CONF_TEMPERATURE_OFFSET, default=0): cv.temperature,
|
cv.Optional(CONF_IAQ_MODE, default="STATIC"): cv.enum(
|
||||||
cv.Optional(CONF_IAQ_MODE, default="STATIC"): cv.enum(
|
IAQ_MODE_OPTIONS, upper=True
|
||||||
IAQ_MODE_OPTIONS, upper=True
|
|
||||||
),
|
|
||||||
cv.Optional(CONF_SAMPLE_RATE, default="LP"): cv.enum(
|
|
||||||
SAMPLE_RATE_OPTIONS, upper=True
|
|
||||||
),
|
|
||||||
cv.Optional(
|
|
||||||
CONF_STATE_SAVE_INTERVAL, default="6hours"
|
|
||||||
): cv.positive_time_period_minutes,
|
|
||||||
}
|
|
||||||
).extend(i2c.i2c_device_schema(0x76)),
|
|
||||||
cv.only_with_arduino,
|
|
||||||
cv.Any(
|
|
||||||
cv.only_on_esp8266,
|
|
||||||
cv.All(
|
|
||||||
cv.only_on_esp32,
|
|
||||||
esp32.only_on_variant(supported=[esp32.const.VARIANT_ESP32]),
|
|
||||||
),
|
),
|
||||||
),
|
cv.Optional(CONF_SAMPLE_RATE, default="LP"): cv.enum(
|
||||||
)
|
SAMPLE_RATE_OPTIONS, upper=True
|
||||||
|
),
|
||||||
|
cv.Optional(
|
||||||
|
CONF_STATE_SAVE_INTERVAL, default="6hours"
|
||||||
|
): cv.positive_time_period_minutes,
|
||||||
|
},
|
||||||
|
cv.only_with_arduino,
|
||||||
|
).extend(i2c.i2c_device_schema(0x76))
|
||||||
|
|
||||||
|
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
|
|||||||
@@ -94,5 +94,3 @@ async def to_code(config):
|
|||||||
sens = await sensor.new_sensor(conf)
|
sens = await sensor.new_sensor(conf)
|
||||||
cg.add(var.set_pressure_sensor(sens))
|
cg.add(var.set_pressure_sensor(sens))
|
||||||
cg.add(var.set_pressure_oversampling(conf[CONF_OVERSAMPLING]))
|
cg.add(var.set_pressure_oversampling(conf[CONF_OVERSAMPLING]))
|
||||||
|
|
||||||
cg.add(var.set_iir_filter(config[CONF_IIR_FILTER]))
|
|
||||||
|
|||||||
@@ -6,6 +6,9 @@ namespace button {
|
|||||||
|
|
||||||
static const char *const TAG = "button";
|
static const char *const TAG = "button";
|
||||||
|
|
||||||
|
Button::Button(const std::string &name) : EntityBase(name) {}
|
||||||
|
Button::Button() : Button("") {}
|
||||||
|
|
||||||
void Button::press() {
|
void Button::press() {
|
||||||
ESP_LOGD(TAG, "'%s' Pressed.", this->get_name().c_str());
|
ESP_LOGD(TAG, "'%s' Pressed.", this->get_name().c_str());
|
||||||
this->press_action();
|
this->press_action();
|
||||||
@@ -13,5 +16,8 @@ void Button::press() {
|
|||||||
}
|
}
|
||||||
void Button::add_on_press_callback(std::function<void()> &&callback) { this->press_callback_.add(std::move(callback)); }
|
void Button::add_on_press_callback(std::function<void()> &&callback) { this->press_callback_.add(std::move(callback)); }
|
||||||
|
|
||||||
|
void Button::set_device_class(const std::string &device_class) { this->device_class_ = device_class; }
|
||||||
|
std::string Button::get_device_class() { return this->device_class_; }
|
||||||
|
|
||||||
} // namespace button
|
} // namespace button
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -26,8 +26,11 @@ namespace button {
|
|||||||
*
|
*
|
||||||
* A button is just a momentary switch that does not have a state, only a trigger.
|
* A button is just a momentary switch that does not have a state, only a trigger.
|
||||||
*/
|
*/
|
||||||
class Button : public EntityBase, public EntityBase_DeviceClass {
|
class Button : public EntityBase {
|
||||||
public:
|
public:
|
||||||
|
explicit Button();
|
||||||
|
explicit Button(const std::string &name);
|
||||||
|
|
||||||
/** Press this button. This is called by the front-end.
|
/** Press this button. This is called by the front-end.
|
||||||
*
|
*
|
||||||
* For implementing buttons, please override press_action.
|
* For implementing buttons, please override press_action.
|
||||||
@@ -40,12 +43,19 @@ class Button : public EntityBase, public EntityBase_DeviceClass {
|
|||||||
*/
|
*/
|
||||||
void add_on_press_callback(std::function<void()> &&callback);
|
void add_on_press_callback(std::function<void()> &&callback);
|
||||||
|
|
||||||
|
/// Set the Home Assistant device class (see button::device_class).
|
||||||
|
void set_device_class(const std::string &device_class);
|
||||||
|
|
||||||
|
/// Get the device class for this button.
|
||||||
|
std::string get_device_class();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/** You should implement this virtual method if you want to create your own button.
|
/** You should implement this virtual method if you want to create your own button.
|
||||||
*/
|
*/
|
||||||
virtual void press_action() = 0;
|
virtual void press_action() = 0;
|
||||||
|
|
||||||
CallbackManager<void()> press_callback_{};
|
CallbackManager<void()> press_callback_{};
|
||||||
|
std::string device_class_{};
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace button
|
} // namespace button
|
||||||
|
|||||||
@@ -6,100 +6,102 @@ namespace esphome {
|
|||||||
namespace captive_portal {
|
namespace captive_portal {
|
||||||
|
|
||||||
const uint8_t INDEX_GZ[] PROGMEM = {
|
const uint8_t INDEX_GZ[] PROGMEM = {
|
||||||
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0xdd, 0x58, 0x5b, 0x8f, 0xdb, 0x36, 0x16, 0x7e, 0xef,
|
0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0a, 0xdd, 0x58, 0x09, 0x6f, 0xdc, 0x36, 0x16, 0xfe, 0x2b,
|
||||||
0xaf, 0xe0, 0x2a, 0x49, 0x2d, 0x37, 0x23, 0xea, 0x66, 0xf9, 0x2a, 0xa9, 0x48, 0xb2, 0x29, 0x5a, 0x20, 0x69, 0x03,
|
0xac, 0x92, 0x74, 0x34, 0x8d, 0xc5, 0xd1, 0x31, 0x97, 0x35, 0xd2, 0x14, 0x89, 0x37, 0x45, 0x0b, 0x24, 0x69, 0x00,
|
||||||
0xcc, 0xb4, 0xfb, 0x10, 0x04, 0x18, 0x5a, 0xa2, 0x2c, 0x66, 0x24, 0x4a, 0x15, 0xe9, 0x5b, 0x0c, 0xef, 0x6f, 0xdf,
|
0xbb, 0x5d, 0x14, 0x69, 0x00, 0x73, 0x24, 0x6a, 0xc4, 0x58, 0xa2, 0x54, 0x91, 0x9a, 0x23, 0x83, 0xd9, 0xdf, 0xde,
|
||||||
0x43, 0x52, 0xf6, 0x38, 0xb3, 0x99, 0x05, 0x52, 0xec, 0x62, 0xd1, 0x4e, 0x26, 0x1c, 0x92, 0x3a, 0xd7, 0x4f, 0x3c,
|
0x47, 0x52, 0x73, 0x38, 0x6b, 0x2f, 0x90, 0x62, 0x8b, 0xa2, 0x4d, 0x6c, 0x9a, 0xc7, 0x3b, 0x3f, 0xf2, 0xf1, 0x3d,
|
||||||
0x17, 0x2a, 0xfe, 0x5b, 0xde, 0x64, 0x72, 0xdf, 0x52, 0x54, 0xca, 0xba, 0x4a, 0x63, 0x35, 0xa2, 0x8a, 0xf0, 0x55,
|
0x2a, 0xfa, 0x2a, 0xad, 0x12, 0xb9, 0xad, 0x29, 0xca, 0x65, 0x59, 0xcc, 0x23, 0xd5, 0xa2, 0x82, 0xf0, 0x65, 0x4c,
|
||||||
0x42, 0x39, 0xac, 0x28, 0xc9, 0xd3, 0xb8, 0xa6, 0x92, 0xa0, 0xac, 0x24, 0x9d, 0xa0, 0x32, 0xf9, 0xf5, 0xe6, 0x07,
|
0x39, 0x8c, 0x28, 0x49, 0xe7, 0x51, 0x49, 0x25, 0x41, 0x49, 0x4e, 0x1a, 0x41, 0x65, 0xfc, 0xd3, 0xcd, 0x77, 0xce,
|
||||||
0x67, 0x8a, 0xdc, 0x34, 0xae, 0x18, 0xbf, 0x43, 0x1d, 0xad, 0x12, 0x96, 0x35, 0x1c, 0x95, 0x1d, 0x2d, 0x92, 0x9c,
|
0x14, 0x0d, 0xe6, 0x51, 0xc1, 0xf8, 0x1d, 0x6a, 0x68, 0x11, 0xb3, 0xa4, 0xe2, 0x28, 0x6f, 0x68, 0x16, 0xa7, 0x44,
|
||||||
0x48, 0x32, 0x67, 0x35, 0x59, 0x51, 0x45, 0xa0, 0xd9, 0x38, 0xa9, 0x69, 0xb2, 0x61, 0x74, 0xdb, 0x36, 0x9d, 0x44,
|
0x92, 0x90, 0x95, 0x64, 0x49, 0x15, 0x81, 0x66, 0xe3, 0xa4, 0xa4, 0xf1, 0x8a, 0xd1, 0x75, 0x5d, 0x35, 0x12, 0x01,
|
||||||
0x40, 0x29, 0x29, 0x97, 0x89, 0xb5, 0x65, 0xb9, 0x2c, 0x93, 0x9c, 0x6e, 0x58, 0x46, 0x1d, 0xbd, 0xb8, 0x62, 0x9c,
|
0xa5, 0xa4, 0x5c, 0xc6, 0xd6, 0x9a, 0xa5, 0x32, 0x8f, 0x53, 0xba, 0x62, 0x09, 0x75, 0xf4, 0xe0, 0x82, 0x71, 0x26,
|
||||||
0x49, 0x46, 0x2a, 0x47, 0x64, 0xa4, 0xa2, 0x89, 0x7f, 0xb5, 0x16, 0xb4, 0xd3, 0x0b, 0xb2, 0x84, 0x35, 0x6f, 0x2c,
|
0x19, 0x29, 0x1c, 0x91, 0x90, 0x82, 0xc6, 0xde, 0x45, 0x2b, 0x68, 0xa3, 0x07, 0x64, 0x01, 0x63, 0x5e, 0x59, 0x20,
|
||||||
0x10, 0x29, 0xb2, 0x8e, 0xb5, 0x12, 0x29, 0x7b, 0x93, 0xba, 0xc9, 0xd7, 0x15, 0x4d, 0x5d, 0x97, 0x08, 0xb0, 0x4b,
|
0x52, 0x24, 0x0d, 0xab, 0x25, 0x52, 0xf6, 0xc6, 0x65, 0x95, 0xb6, 0x05, 0x9d, 0x67, 0x2d, 0x4f, 0x24, 0x03, 0x0b,
|
||||||
0xb8, 0x8c, 0xe7, 0x74, 0x87, 0xa7, 0xb3, 0x68, 0x32, 0x9e, 0xe6, 0x13, 0xfc, 0x51, 0x7c, 0x03, 0x9e, 0xad, 0x6b,
|
0x84, 0xcd, 0xfb, 0xbb, 0x82, 0x4a, 0x44, 0xe3, 0x37, 0x44, 0xe6, 0xb8, 0x24, 0x1b, 0xdb, 0x74, 0x18, 0xb7, 0xfd,
|
||||||
0x50, 0x87, 0xab, 0x26, 0x23, 0x92, 0x35, 0x1c, 0x0b, 0x4a, 0xba, 0xac, 0x4c, 0x92, 0xc4, 0xfa, 0x5e, 0x90, 0x0d,
|
0x6f, 0x6c, 0xfe, 0xdc, 0x73, 0xdd, 0xfe, 0x85, 0x6e, 0xdc, 0xfe, 0x00, 0xfe, 0xce, 0x1a, 0x2a, 0xdb, 0x86, 0x23,
|
||||||
0xb5, 0xbe, 0xfd, 0xd6, 0x3e, 0x13, 0xad, 0xa8, 0x7c, 0x5d, 0x51, 0x35, 0x15, 0x2f, 0xf7, 0x37, 0x64, 0xf5, 0x33,
|
0x62, 0xdf, 0x46, 0x35, 0x50, 0xa2, 0x34, 0xb6, 0x4a, 0xcf, 0xc7, 0xae, 0x3b, 0x45, 0xde, 0x25, 0xf6, 0x47, 0x8e,
|
||||||
0x58, 0x6e, 0x5b, 0x44, 0xb0, 0x9c, 0x5a, 0xc3, 0xf7, 0xde, 0x07, 0x2c, 0xe4, 0xbe, 0xa2, 0x38, 0x67, 0xa2, 0xad,
|
0xe7, 0xe1, 0xc0, 0xf1, 0x46, 0xc9, 0xc4, 0x19, 0x21, 0x6f, 0x08, 0x8d, 0xef, 0xe3, 0x11, 0x72, 0x3f, 0x59, 0x28,
|
||||||
0xc8, 0x3e, 0xb1, 0x96, 0x20, 0xf5, 0xce, 0x1a, 0x2e, 0x8a, 0x35, 0xcf, 0x94, 0x70, 0x24, 0x6c, 0x3a, 0x3c, 0x54,
|
0x63, 0x45, 0x11, 0x5b, 0xbc, 0xe2, 0xd4, 0x42, 0x42, 0x36, 0xd5, 0x1d, 0x8d, 0xad, 0xa4, 0x6d, 0x1a, 0xf0, 0xee,
|
||||||
0x14, 0xcc, 0x4b, 0xde, 0x12, 0x59, 0xe2, 0x9a, 0xec, 0x6c, 0x33, 0x61, 0xdc, 0x0e, 0xbe, 0xb3, 0xe9, 0x73, 0xdf,
|
0xaa, 0x2a, 0xaa, 0x06, 0xac, 0xfd, 0x95, 0xa3, 0x7b, 0xff, 0xbe, 0x58, 0x87, 0x6c, 0x08, 0x17, 0x59, 0xd5, 0x94,
|
||||||
0xf3, 0x86, 0x57, 0x7a, 0xf0, 0x86, 0x2e, 0xfc, 0x5d, 0x74, 0x54, 0xae, 0x3b, 0x8e, 0x88, 0x7d, 0x1b, 0xb7, 0x40,
|
0xb1, 0xa5, 0x41, 0xb1, 0x9f, 0xee, 0xe8, 0x1e, 0xa9, 0xa6, 0x7f, 0xb6, 0xe8, 0x54, 0x0d, 0x5b, 0x32, 0x1e, 0x5b,
|
||||||
0x89, 0xf2, 0xc4, 0xaa, 0xfd, 0x00, 0x7b, 0xde, 0x14, 0xf9, 0x33, 0x1c, 0x44, 0x8e, 0xef, 0xe3, 0xd0, 0xf1, 0xa3,
|
0x9e, 0x8f, 0xbc, 0x29, 0xe8, 0xbd, 0xed, 0xef, 0x8f, 0xa0, 0x10, 0x05, 0x4a, 0xe7, 0x66, 0x65, 0xbf, 0xbf, 0x8d,
|
||||||
0x6c, 0xe2, 0x44, 0xc8, 0x1f, 0xc1, 0x10, 0x04, 0x38, 0x42, 0xde, 0x27, 0x0b, 0x15, 0xac, 0xaa, 0x12, 0x8b, 0x37,
|
0xc4, 0x6a, 0x89, 0x36, 0x65, 0xc1, 0x45, 0x6c, 0xe5, 0x52, 0xd6, 0xe1, 0x60, 0xb0, 0x5e, 0xaf, 0xf1, 0x3a, 0xc0,
|
||||||
0x9c, 0x5a, 0x48, 0xc8, 0xae, 0xb9, 0xa3, 0x89, 0x95, 0xad, 0xbb, 0x0e, 0xec, 0x7f, 0xd5, 0x54, 0x4d, 0x07, 0x70,
|
0x55, 0xb3, 0x1c, 0xf8, 0xae, 0xeb, 0x0e, 0x80, 0xc2, 0x42, 0x66, 0x7f, 0x2c, 0x7f, 0x68, 0xa1, 0x9c, 0xb2, 0x65,
|
||||||
0x7d, 0x83, 0x3e, 0xfb, 0xf9, 0x6a, 0x15, 0xb2, 0x23, 0x5c, 0x14, 0x4d, 0x57, 0x27, 0x96, 0x7e, 0x29, 0xf6, 0xd3,
|
0x2e, 0x75, 0x7f, 0xfe, 0x74, 0xc7, 0xf7, 0x91, 0xa2, 0x98, 0xdf, 0x7e, 0x38, 0xd3, 0xd2, 0x9c, 0x69, 0xe1, 0xdf,
|
||||||
0x83, 0x3c, 0x22, 0x35, 0x0c, 0x2f, 0x1e, 0x3a, 0x4d, 0xc7, 0x56, 0x8c, 0x27, 0x96, 0x1f, 0x20, 0x7f, 0x0a, 0x6a,
|
0x12, 0xdb, 0x3a, 0xb8, 0xda, 0x7b, 0xa3, 0x8c, 0x9a, 0x10, 0x1f, 0xf9, 0xc8, 0xd5, 0xff, 0x7d, 0x47, 0xf5, 0xbb,
|
||||||
0x6f, 0x87, 0xc7, 0x33, 0x26, 0x44, 0x61, 0xd2, 0x7b, 0xd9, 0xd8, 0xef, 0x6f, 0x63, 0xb1, 0x59, 0xa1, 0x5d, 0x5d,
|
0x91, 0xf3, 0xd9, 0x08, 0x9d, 0x8d, 0xe0, 0xaf, 0x02, 0xd0, 0x2f, 0xc7, 0xce, 0xe5, 0x91, 0xdf, 0x53, 0xeb, 0x2b,
|
||||||
0x71, 0x91, 0x58, 0xa5, 0x94, 0xed, 0xdc, 0x75, 0xb7, 0xdb, 0x2d, 0xde, 0x86, 0xb8, 0xe9, 0x56, 0x6e, 0xe0, 0x79,
|
0xcf, 0x3d, 0x4d, 0x28, 0xa6, 0xef, 0xc7, 0xe7, 0x63, 0xc7, 0xff, 0x59, 0x11, 0x68, 0xf4, 0x8f, 0x5c, 0x8e, 0x9f,
|
||||||
0x9e, 0x0b, 0x14, 0x16, 0x32, 0xe7, 0xc3, 0x0a, 0x46, 0x16, 0x2a, 0x29, 0x5b, 0x95, 0x52, 0xcf, 0xd3, 0xa7, 0x07,
|
0x7b, 0x3f, 0x8f, 0xc9, 0x08, 0x8d, 0xba, 0x99, 0x91, 0xa3, 0xfa, 0xc7, 0x91, 0xd6, 0x85, 0x46, 0x2b, 0x20, 0x2b,
|
||||||
0x7a, 0x8c, 0x15, 0x45, 0x7a, 0xfb, 0xe1, 0x42, 0x4b, 0x77, 0xa1, 0x85, 0x7e, 0x7f, 0x81, 0xe6, 0xe0, 0xad, 0x32,
|
0x9d, 0xb1, 0x33, 0x22, 0x01, 0x0a, 0x3a, 0xab, 0xa0, 0x07, 0xd3, 0x63, 0xe0, 0x3e, 0x9b, 0x73, 0x82, 0x4f, 0xbd,
|
||||||
0x6a, 0x42, 0x02, 0x14, 0x20, 0x4f, 0xff, 0x0b, 0x1c, 0x35, 0xef, 0x57, 0xce, 0x83, 0x15, 0xba, 0x58, 0xc1, 0x5f,
|
0xc1, 0xdc, 0xea, 0x87, 0x96, 0x75, 0x82, 0xa1, 0x3a, 0x87, 0x01, 0x7f, 0xac, 0xe0, 0xdc, 0x59, 0x56, 0x7f, 0x6f,
|
||||||
0xc0, 0x2f, 0xa8, 0xc7, 0xce, 0xec, 0xcc, 0xee, 0xab, 0xc7, 0x1b, 0xdf, 0xbb, 0xdf, 0x50, 0x3c, 0x3f, 0x8e, 0x2f,
|
0x7d, 0x2b, 0xc8, 0x8a, 0x5a, 0x71, 0x1c, 0x43, 0xa8, 0xb5, 0x25, 0x9c, 0x10, 0x5c, 0x54, 0x09, 0x51, 0x2c, 0x58,
|
||||||
0xd7, 0x4e, 0xf0, 0x9b, 0x22, 0x50, 0xd8, 0x9f, 0x99, 0x9c, 0xa0, 0xf4, 0x7f, 0x1b, 0x93, 0x08, 0x45, 0xfd, 0x4e,
|
0x50, 0xd2, 0x24, 0xf9, 0xd7, 0x5f, 0xdb, 0xc7, 0xa5, 0x25, 0x95, 0xaf, 0x0a, 0xaa, 0xba, 0xe2, 0xe5, 0xf6, 0x86,
|
||||||
0xe4, 0xa8, 0xf9, 0x79, 0xa5, 0x34, 0xa1, 0x68, 0x03, 0x54, 0xb5, 0x33, 0x76, 0x22, 0x12, 0xa2, 0xb0, 0x37, 0x09,
|
0x2c, 0xdf, 0x42, 0x00, 0xd9, 0x16, 0x11, 0x2c, 0xa5, 0x56, 0xff, 0xbd, 0xfb, 0x01, 0x0b, 0xb9, 0x2d, 0x28, 0x4e,
|
||||||
0x66, 0xb0, 0x3d, 0x06, 0xe6, 0x8b, 0x3d, 0x27, 0xfc, 0x34, 0x50, 0x30, 0xcf, 0x2d, 0xeb, 0x1e, 0x83, 0xe6, 0x12,
|
0x99, 0xa8, 0x0b, 0xb2, 0x8d, 0xad, 0x05, 0xc8, 0xba, 0xb3, 0xfa, 0x17, 0x19, 0x95, 0x49, 0x6e, 0x5b, 0x03, 0x08,
|
||||||
0x03, 0xfc, 0xb1, 0x81, 0x33, 0x67, 0x59, 0x80, 0x11, 0x95, 0x59, 0x69, 0x5b, 0x2e, 0x44, 0x5e, 0xc1, 0x56, 0x10,
|
0xb1, 0x8c, 0x2d, 0xf1, 0x47, 0x51, 0x71, 0xab, 0x8f, 0x65, 0x4e, 0xb9, 0x6d, 0x1f, 0x2c, 0x54, 0xf6, 0x71, 0xbd,
|
||||||
0x15, 0x0d, 0xb7, 0x86, 0x58, 0x96, 0x94, 0xdb, 0x27, 0x56, 0xc5, 0x48, 0xf5, 0x13, 0xfb, 0xe1, 0x13, 0x39, 0x3c,
|
0x64, 0x3f, 0xb4, 0x74, 0xb4, 0x41, 0x32, 0xa9, 0x42, 0x0e, 0xab, 0xe0, 0xbd, 0x38, 0xce, 0x2e, 0xaa, 0x74, 0xfb,
|
||||||
0x9c, 0xe3, 0x43, 0x32, 0x09, 0x71, 0x28, 0xb1, 0x8a, 0xe8, 0xab, 0xf3, 0xee, 0xb2, 0xc9, 0xf7, 0x8f, 0x84, 0x4e,
|
0x88, 0x79, 0xb9, 0x67, 0x6c, 0x63, 0x9c, 0xd3, 0xe6, 0x86, 0x6e, 0xe0, 0xb8, 0xfc, 0x9b, 0x7d, 0xc7, 0xd0, 0x5b,
|
||||||
0xe9, 0x9b, 0xb8, 0x61, 0x9c, 0xd3, 0xee, 0x86, 0xee, 0xe0, 0x1d, 0xfe, 0x83, 0xfd, 0xc0, 0xd0, 0xcf, 0x54, 0x6e,
|
0x2a, 0xd7, 0x55, 0x73, 0x27, 0x42, 0x64, 0x3d, 0x37, 0xe2, 0x66, 0x26, 0x42, 0x39, 0x26, 0xb5, 0xc0, 0xa2, 0x80,
|
||||||
0x9b, 0xee, 0x4e, 0xcc, 0x91, 0xf5, 0xdc, 0x88, 0x5b, 0xa8, 0xa8, 0x61, 0x20, 0x9b, 0xb4, 0x02, 0x8b, 0x0a, 0x72,
|
0xf0, 0xb7, 0xbd, 0x3e, 0xc4, 0x6a, 0x7d, 0xdf, 0x14, 0x83, 0xe2, 0x6d, 0x94, 0xb2, 0x15, 0x4a, 0x0a, 0x22, 0xe0,
|
||||||
0x82, 0xed, 0x0f, 0x21, 0x7e, 0xda, 0x7b, 0x4b, 0xf8, 0xc9, 0xb9, 0xdb, 0x38, 0x67, 0x1b, 0x94, 0x55, 0x10, 0xf5,
|
0xb8, 0x72, 0x23, 0xcb, 0x42, 0x87, 0xb8, 0xaa, 0x78, 0x02, 0xfc, 0x77, 0xb1, 0xf5, 0x00, 0x76, 0x2f, 0xb7, 0x3f,
|
||||||
0x70, 0xfc, 0x8d, 0x28, 0x0b, 0xf5, 0x47, 0xbd, 0xe1, 0x19, 0x70, 0xdf, 0x25, 0xd6, 0x17, 0xa2, 0xfa, 0xe5, 0xfe,
|
0xa4, 0x76, 0x4f, 0x00, 0x6a, 0xbd, 0x3e, 0x5e, 0x91, 0xa2, 0xa5, 0x28, 0x46, 0x32, 0x67, 0xe2, 0x64, 0xe2, 0xec,
|
||||||
0xa7, 0xdc, 0x1e, 0x08, 0x88, 0xe7, 0xc1, 0x10, 0x6f, 0x48, 0xb5, 0xa6, 0x28, 0x41, 0xb2, 0x64, 0xe2, 0xde, 0xc0,
|
0x51, 0xb6, 0x5a, 0xdc, 0x01, 0x57, 0x06, 0xcb, 0xc2, 0xee, 0x5b, 0xc7, 0x38, 0x8e, 0x88, 0xb9, 0xe5, 0xac, 0x27,
|
||||||
0xc5, 0xa3, 0x6c, 0xad, 0xb8, 0x03, 0xae, 0x02, 0x1e, 0x0b, 0x7b, 0x68, 0x9d, 0x22, 0x2b, 0x26, 0x26, 0xef, 0x59,
|
0xd6, 0x67, 0x36, 0x39, 0x05, 0xcd, 0xa4, 0x75, 0x16, 0xf0, 0x4f, 0x77, 0x70, 0x1b, 0xe1, 0x06, 0xf4, 0xf7, 0xf7,
|
||||||
0x4f, 0xac, 0x07, 0x16, 0x39, 0x15, 0x2d, 0xa4, 0x75, 0x1f, 0x81, 0x4f, 0x0f, 0xc2, 0xe6, 0xb8, 0x03, 0xed, 0xc3,
|
0xa7, 0xd9, 0x48, 0xd4, 0x84, 0x7f, 0xce, 0xaa, 0x6c, 0xd4, 0x81, 0x85, 0x55, 0x4f, 0x45, 0x17, 0x10, 0x9d, 0x74,
|
||||||
0xe3, 0x79, 0x33, 0x16, 0x2d, 0xe1, 0x0f, 0x19, 0x95, 0x81, 0xea, 0xa0, 0x43, 0xb2, 0x82, 0x99, 0x3a, 0xed, 0x40,
|
0x0e, 0xc8, 0xb1, 0xff, 0x74, 0x07, 0x71, 0xa6, 0x8e, 0xce, 0xdd, 0x49, 0x68, 0x34, 0x00, 0x84, 0xe6, 0xb7, 0xfb,
|
||||||
0x74, 0x56, 0xe8, 0x92, 0xd3, 0xf4, 0xe9, 0xa1, 0x03, 0x89, 0x2a, 0x07, 0x9d, 0x25, 0xc6, 0x2e, 0x40, 0x93, 0xde,
|
0x7e, 0xff, 0xe4, 0xce, 0x6f, 0x2d, 0x6d, 0xb6, 0xd7, 0xb4, 0xa0, 0x89, 0xac, 0x1a, 0xdb, 0x7a, 0x02, 0x9a, 0xe0,
|
||||||
0x1e, 0x87, 0xf7, 0x7e, 0xfc, 0xbe, 0xa6, 0xdd, 0xfe, 0x9a, 0x56, 0x34, 0x93, 0x4d, 0x67, 0x5b, 0x4f, 0x40, 0x0b,
|
0x24, 0x68, 0xbf, 0xbf, 0xbf, 0x79, 0xf3, 0x3a, 0xae, 0x6c, 0xda, 0xbf, 0x78, 0x8c, 0x5a, 0xdd, 0xea, 0xef, 0xe1,
|
||||||
0xbc, 0x7e, 0xed, 0xf0, 0x8f, 0x37, 0x6f, 0xdf, 0x24, 0x8d, 0xcd, 0x86, 0x57, 0x8f, 0x51, 0xab, 0x0c, 0xff, 0x1e,
|
0x56, 0xff, 0x4f, 0xdc, 0x53, 0xf7, 0x7a, 0xef, 0x03, 0xb0, 0x1a, 0xaf, 0x4f, 0x97, 0xbb, 0xba, 0x00, 0x9e, 0xc3,
|
||||||
0x32, 0xfc, 0x3f, 0x93, 0x81, 0xca, 0xf1, 0x83, 0x0f, 0xc0, 0xaa, 0xfd, 0xbd, 0xbd, 0x4f, 0xf4, 0x2a, 0x18, 0x9f,
|
0x25, 0x72, 0x61, 0x3d, 0x17, 0xb6, 0x33, 0x1e, 0xf5, 0x41, 0x3d, 0xfc, 0x80, 0xe9, 0xfa, 0x7a, 0x86, 0x6b, 0x5a,
|
||||||
0x43, 0x40, 0x5f, 0x29, 0x0f, 0x9d, 0x71, 0x34, 0x3c, 0x82, 0x7e, 0xb0, 0x00, 0xec, 0xd6, 0xb9, 0x1a, 0x72, 0xb6,
|
0x1d, 0xd1, 0xf9, 0x37, 0xbb, 0x45, 0xb5, 0x71, 0x04, 0xfb, 0xc4, 0xf8, 0x32, 0x64, 0x3c, 0xa7, 0x0d, 0x93, 0x7b,
|
||||||
0x4a, 0x9b, 0xe9, 0x77, 0x87, 0x65, 0xb3, 0x73, 0x04, 0xfb, 0xc4, 0xf8, 0x6a, 0xce, 0x78, 0x49, 0x3b, 0x26, 0x8f,
|
0x30, 0x17, 0x6e, 0xfa, 0xba, 0x95, 0xbb, 0x9a, 0xa4, 0xa9, 0x5a, 0x19, 0xd5, 0x9b, 0x59, 0x06, 0x79, 0x41, 0x51,
|
||||||
0x60, 0x2e, 0xa4, 0xfd, 0x76, 0x2d, 0x0f, 0x2d, 0xc9, 0x73, 0xf5, 0x24, 0x6a, 0x77, 0x8b, 0x02, 0x8a, 0x84, 0xa2,
|
0xd2, 0xd0, 0xa3, 0xe5, 0xde, 0xac, 0xeb, 0x2b, 0x28, 0xbc, 0x1c, 0x3d, 0xdb, 0xab, 0x83, 0xb7, 0x93, 0xb0, 0x65,
|
||||||
0xa4, 0x73, 0x9f, 0xd6, 0x47, 0xf3, 0x5c, 0xe7, 0x83, 0xf9, 0x2c, 0x7a, 0x76, 0x54, 0x07, 0xee, 0x20, 0xe1, 0x65,
|
0x0e, 0x29, 0xd8, 0x92, 0x87, 0x09, 0xd8, 0x4d, 0x1b, 0xc3, 0x94, 0x91, 0x92, 0x15, 0xdb, 0x50, 0xc0, 0x65, 0xe8,
|
||||||
0x39, 0xa4, 0x62, 0x2b, 0x3e, 0xcf, 0xc0, 0x70, 0xda, 0x19, 0xa6, 0x82, 0xd4, 0xac, 0xda, 0xcf, 0x05, 0x64, 0x26,
|
0x40, 0xc2, 0x60, 0xd9, 0x7e, 0xd1, 0x4a, 0x59, 0x71, 0xd0, 0xdd, 0xa4, 0xb4, 0x09, 0xdd, 0x99, 0xe9, 0x38, 0x0d,
|
||||||
0x07, 0xaa, 0x07, 0x2b, 0x8e, 0xcb, 0xb5, 0x94, 0x0d, 0x07, 0xdd, 0x5d, 0x4e, 0xbb, 0xb9, 0xb7, 0x30, 0x13, 0xa7,
|
0x49, 0x59, 0x2b, 0x42, 0x1c, 0x34, 0xb4, 0x9c, 0x2d, 0x48, 0x72, 0xb7, 0x6c, 0xaa, 0x96, 0xa7, 0x4e, 0xa2, 0x6e,
|
||||||
0x23, 0x39, 0x5b, 0x8b, 0x39, 0x0e, 0x3b, 0x5a, 0x2f, 0x96, 0x24, 0xbb, 0x5b, 0x75, 0xcd, 0x9a, 0xe7, 0x4e, 0xa6,
|
0xeb, 0xf0, 0x89, 0x97, 0x91, 0x80, 0x26, 0xb3, 0x6e, 0x94, 0x65, 0xd9, 0x0c, 0x90, 0xa0, 0x8e, 0xb9, 0xfc, 0x42,
|
||||||
0x32, 0xe7, 0xfc, 0x89, 0x5f, 0x90, 0x90, 0x66, 0x8b, 0x7e, 0x55, 0x14, 0xc5, 0x02, 0xa0, 0xa0, 0x8e, 0xc9, 0x44,
|
0x1f, 0x0f, 0x15, 0xdb, 0x99, 0x99, 0xd8, 0x57, 0x13, 0xc6, 0x46, 0x48, 0x25, 0xcf, 0x66, 0x07, 0x77, 0xdc, 0x19,
|
||||||
0xf3, 0x00, 0x8f, 0x14, 0xdb, 0x85, 0x99, 0x38, 0x50, 0x1b, 0xc6, 0x46, 0x48, 0xeb, 0xcf, 0x16, 0x27, 0x77, 0xbc,
|
0xa4, 0x01, 0x01, 0x42, 0x6a, 0x88, 0x7f, 0x30, 0x73, 0x5f, 0x12, 0xc6, 0xcf, 0xad, 0x57, 0x67, 0x65, 0xd6, 0x85,
|
||||||
0x05, 0xa4, 0x64, 0x01, 0x42, 0x5a, 0x88, 0x47, 0x30, 0xf3, 0x58, 0x13, 0xc6, 0x2f, 0xad, 0x57, 0xc7, 0x64, 0xd1,
|
0x2f, 0xc0, 0xa2, 0xd5, 0xe8, 0x20, 0x9e, 0x41, 0xa2, 0x32, 0xb9, 0x30, 0xf4, 0xc7, 0x6e, 0xbd, 0xd9, 0xe3, 0xee,
|
||||||
0x97, 0x14, 0x80, 0x45, 0xab, 0xd1, 0x85, 0x65, 0x01, 0x45, 0xc3, 0x14, 0xc6, 0x79, 0x30, 0xf6, 0xda, 0xdd, 0x11,
|
0x8c, 0xec, 0x0e, 0xd4, 0x59, 0x41, 0x37, 0xb3, 0x8f, 0xad, 0x90, 0x2c, 0xdb, 0x3a, 0x5d, 0x2e, 0x0d, 0xe1, 0xbc,
|
||||||
0xf7, 0x07, 0xe4, 0x70, 0xa2, 0x2e, 0x2a, 0xba, 0x5b, 0x7c, 0x5c, 0x0b, 0xc9, 0x8a, 0xbd, 0xd3, 0x17, 0xd6, 0x39,
|
0x40, 0x0e, 0x5d, 0x00, 0x29, 0xa5, 0x7c, 0xa6, 0x75, 0x38, 0x4c, 0xd2, 0x52, 0x74, 0x38, 0x1d, 0xc5, 0xe8, 0x53,
|
||||||
0x1c, 0x16, 0x28, 0xa8, 0x4b, 0x20, 0xa5, 0x94, 0x2f, 0xb4, 0x0e, 0x87, 0x49, 0x5a, 0x8b, 0x1e, 0xa7, 0xb3, 0x18,
|
0x7a, 0x5f, 0xd6, 0xff, 0xa2, 0x56, 0xc7, 0x71, 0x57, 0x92, 0x06, 0x72, 0x8b, 0xb3, 0xa8, 0x00, 0xd3, 0x32, 0x74,
|
||||||
0x7d, 0x40, 0x3f, 0x97, 0xf5, 0x9f, 0xa8, 0xd5, 0x59, 0x3c, 0xd4, 0xa4, 0x83, 0x44, 0xef, 0x2c, 0x1b, 0xc0, 0xb4,
|
0x26, 0xb0, 0x57, 0xdd, 0x94, 0x12, 0x06, 0x9e, 0x83, 0x99, 0xfa, 0x6e, 0x3a, 0xe0, 0xed, 0xd5, 0x1b, 0x24, 0xaa,
|
||||||
0x9e, 0x3b, 0x13, 0x78, 0x57, 0xfd, 0x96, 0x12, 0x06, 0x9e, 0x83, 0x99, 0xba, 0x5e, 0x9e, 0xf0, 0xf6, 0xdb, 0x1d,
|
0x82, 0xa5, 0x1d, 0x9d, 0x26, 0x41, 0xee, 0x11, 0x1e, 0x0f, 0xb6, 0x1b, 0xa9, 0xb9, 0x03, 0xd4, 0xc3, 0x6c, 0x4a,
|
||||||
0x12, 0x4d, 0xc5, 0xf2, 0x9e, 0x4e, 0x93, 0x20, 0xef, 0x0c, 0x8f, 0x0f, 0xaf, 0x1b, 0xa9, 0xbd, 0x13, 0xd4, 0xa3,
|
0x3c, 0xf7, 0x81, 0x1d, 0x49, 0xb3, 0xcc, 0x5f, 0x64, 0x47, 0xa4, 0x54, 0xaa, 0xdd, 0xb3, 0xee, 0x54, 0xf8, 0x43,
|
||||||
0x62, 0x4a, 0x7c, 0xef, 0x0b, 0x6f, 0x24, 0x2f, 0x8a, 0x60, 0x59, 0x9c, 0x91, 0x52, 0x65, 0xef, 0xc8, 0xfa, 0x53,
|
0x10, 0x70, 0xd8, 0x1b, 0xe8, 0xef, 0x99, 0x8e, 0x8b, 0xdd, 0x99, 0x14, 0x7d, 0x52, 0xc3, 0xb6, 0x29, 0xec, 0x87,
|
||||||
0x11, 0x8c, 0x40, 0xc0, 0xe9, 0xdd, 0xc0, 0xfc, 0xc8, 0x74, 0x58, 0x1c, 0x2e, 0xa4, 0xe8, 0xa3, 0x3a, 0x5f, 0x77,
|
0x4e, 0xee, 0xb3, 0xe0, 0xea, 0x94, 0x09, 0x7b, 0x8f, 0x67, 0xc2, 0x1e, 0x52, 0xb5, 0xcb, 0xcb, 0x6a, 0x13, 0xf7,
|
||||||
0x95, 0x6d, 0x7d, 0xe1, 0xe8, 0x3e, 0x0b, 0x5f, 0xdd, 0x97, 0xa5, 0xc1, 0xe3, 0x65, 0x69, 0x80, 0x54, 0x23, 0xf3,
|
0x74, 0x4e, 0x1a, 0xc2, 0x4f, 0xef, 0x59, 0xf0, 0x0a, 0xf8, 0xff, 0x2f, 0x29, 0xee, 0x0f, 0xa7, 0xb7, 0x2f, 0x48,
|
||||||
0xb2, 0xd9, 0x25, 0x03, 0x5d, 0x20, 0x46, 0xf0, 0x3b, 0x78, 0x16, 0xbe, 0x06, 0xfe, 0xff, 0x4a, 0xbd, 0xf9, 0xc3,
|
0x6d, 0x5f, 0x98, 0xd5, 0x8c, 0x77, 0xca, 0x79, 0xe8, 0x41, 0xfa, 0x62, 0x58, 0xb0, 0xa5, 0xf7, 0x67, 0x40, 0xfb,
|
||||||
0xc5, 0xe6, 0x2b, 0x2a, 0xcd, 0x57, 0x56, 0x19, 0xe3, 0x9d, 0x72, 0x1e, 0x66, 0x50, 0x4e, 0x18, 0x16, 0x6c, 0xe5,
|
0xdf, 0x38, 0x06, 0x2f, 0xbc, 0x29, 0xbe, 0x44, 0xba, 0x31, 0x10, 0xe1, 0x60, 0x8a, 0x26, 0x57, 0x43, 0x3c, 0xf4,
|
||||||
0xff, 0x2f, 0xa0, 0xfd, 0x77, 0x1c, 0xc3, 0x17, 0xfe, 0x14, 0xcf, 0x90, 0x1e, 0x0c, 0x44, 0x38, 0x9c, 0xa2, 0xc9,
|
0x90, 0xaa, 0x9a, 0xc6, 0x68, 0x82, 0xa7, 0x40, 0x30, 0xc6, 0xc1, 0x04, 0x26, 0x90, 0xef, 0xe1, 0xd1, 0x6b, 0x3f,
|
||||||
0xab, 0x11, 0x1e, 0xf9, 0x48, 0xb5, 0x30, 0x63, 0x34, 0x81, 0x7e, 0x0f, 0xf9, 0x63, 0x1c, 0x4e, 0x60, 0x03, 0x05,
|
0xc0, 0xe3, 0x11, 0x50, 0xf9, 0x2e, 0x0e, 0x7c, 0x64, 0x68, 0xc7, 0xd8, 0x07, 0x71, 0x8a, 0x24, 0x28, 0x01, 0xe8,
|
||||||
0x3e, 0x8e, 0xde, 0x04, 0x21, 0x1e, 0x47, 0x40, 0x15, 0x78, 0x38, 0x0c, 0x90, 0xa1, 0x1d, 0xe3, 0x00, 0xc4, 0x29,
|
0x24, 0xc0, 0xee, 0x04, 0xc4, 0x8d, 0xb1, 0x7b, 0x89, 0xa7, 0x63, 0x34, 0xc5, 0x13, 0x80, 0x0e, 0x0f, 0x47, 0x85,
|
||||||
0x92, 0xb0, 0x06, 0xa0, 0xb3, 0x10, 0x7b, 0x13, 0x10, 0x37, 0xc6, 0xde, 0x0c, 0x4f, 0xc7, 0x68, 0x8a, 0x27, 0x00,
|
0x33, 0xc2, 0x1e, 0x4c, 0x07, 0x63, 0x32, 0xc5, 0xc3, 0x00, 0xe9, 0xc6, 0xc0, 0x31, 0x01, 0x11, 0x0e, 0x76, 0xbd,
|
||||||
0x1d, 0x1e, 0x45, 0x95, 0x13, 0x61, 0x1f, 0xb6, 0xc3, 0x31, 0x99, 0xe2, 0x51, 0x88, 0xf4, 0x60, 0xe0, 0x98, 0x80,
|
0xd7, 0x01, 0xf6, 0x27, 0xa0, 0x77, 0x38, 0x7c, 0x01, 0x62, 0x2f, 0x87, 0xc8, 0xb4, 0x06, 0x5e, 0x50, 0x30, 0x7a,
|
||||||
0x08, 0x07, 0x7b, 0xfe, 0x9b, 0x10, 0x07, 0x13, 0xd0, 0x3b, 0x1a, 0xbd, 0x00, 0xb1, 0xb3, 0x11, 0x32, 0xa3, 0x81,
|
0x0c, 0x34, 0xff, 0x9f, 0x0b, 0x1a, 0x40, 0xe2, 0xa1, 0x00, 0x5f, 0x42, 0xec, 0x7a, 0x8a, 0xdf, 0xb4, 0x06, 0x37,
|
||||||
0x17, 0x14, 0x44, 0x8f, 0x81, 0x16, 0xfc, 0x75, 0x41, 0x03, 0x48, 0x7c, 0x14, 0xe2, 0x19, 0xc4, 0xae, 0xaf, 0xf8,
|
0xcf, 0x43, 0xee, 0x1f, 0xc6, 0x2c, 0xf8, 0xe7, 0x62, 0xe6, 0x29, 0x04, 0xa0, 0x0b, 0xba, 0x41, 0x0e, 0xd2, 0x8d,
|
||||||
0xcd, 0x68, 0x70, 0xf3, 0x7d, 0xe4, 0xfd, 0x61, 0xcc, 0xc2, 0xbf, 0x2e, 0x66, 0xbe, 0x42, 0x00, 0xa6, 0xa0, 0x1b,
|
0xd1, 0x0d, 0xcc, 0xd3, 0xab, 0x4b, 0x34, 0x05, 0xae, 0xf1, 0x14, 0x5d, 0xa2, 0x91, 0x42, 0x17, 0xd8, 0x87, 0x86,
|
||||||
0xe4, 0x20, 0x3d, 0x18, 0xdd, 0xc0, 0x3c, 0x7d, 0x35, 0x43, 0x53, 0xe0, 0x1a, 0x4f, 0xd1, 0x0c, 0x45, 0x0a, 0x5d,
|
0xc9, 0x01, 0xa6, 0x2f, 0x84, 0x71, 0xf8, 0x37, 0x86, 0xf1, 0x31, 0x9f, 0xfe, 0xc6, 0x2e, 0xfd, 0x15, 0x57, 0x10,
|
||||||
0x60, 0x1f, 0x19, 0x26, 0x07, 0x98, 0xbe, 0x12, 0xc6, 0xd1, 0x9f, 0x18, 0xc6, 0xc7, 0x7c, 0xfa, 0x13, 0xbb, 0xf4,
|
0x94, 0x63, 0xba, 0x0c, 0x8b, 0x06, 0xe6, 0x15, 0xaf, 0xaa, 0x28, 0x78, 0x94, 0x43, 0x35, 0x02, 0xef, 0x7a, 0x0f,
|
||||||
0xff, 0x48, 0x41, 0xd0, 0x8e, 0xe9, 0x36, 0x2c, 0x76, 0xcd, 0x95, 0x5e, 0x75, 0x51, 0x70, 0x43, 0x87, 0x6e, 0x04,
|
0xb1, 0x34, 0xce, 0xbd, 0xf9, 0xbd, 0x2a, 0x1d, 0x28, 0xbd, 0x79, 0xa4, 0xd3, 0xf9, 0xfc, 0x26, 0xa7, 0xe8, 0xd5,
|
||||||
0x2e, 0xf9, 0x3e, 0x62, 0x79, 0x52, 0xfa, 0xe9, 0x67, 0xdd, 0x39, 0x50, 0xfa, 0x69, 0xac, 0xcb, 0x79, 0x7a, 0x53,
|
0xf5, 0x3b, 0x78, 0x08, 0x16, 0x05, 0xe2, 0xd5, 0x1a, 0xde, 0x9b, 0x5b, 0x24, 0x2b, 0xf5, 0x82, 0xe7, 0x50, 0x2a,
|
||||||
0x52, 0xf4, 0xfa, 0xfa, 0x1d, 0xdc, 0xca, 0xaa, 0x0a, 0xf1, 0x66, 0x0b, 0x97, 0xbf, 0x3d, 0x92, 0x8d, 0xba, 0xce,
|
0xaa, 0x2e, 0x3c, 0x20, 0x50, 0x57, 0x2c, 0x60, 0x8c, 0xa3, 0x45, 0x33, 0x7f, 0x57, 0x50, 0x22, 0x28, 0x5a, 0xb2,
|
||||||
0x73, 0xe8, 0x15, 0xd5, 0x14, 0xee, 0x0d, 0xa8, 0x6f, 0x16, 0x30, 0xc6, 0xf1, 0xb2, 0x4b, 0xdf, 0x55, 0x94, 0x08,
|
0x15, 0x45, 0x4c, 0x42, 0x1d, 0x50, 0x52, 0x24, 0x99, 0x6a, 0x8e, 0x8c, 0x9a, 0xee, 0x6d, 0x25, 0x69, 0x88, 0xae,
|
||||||
0x8a, 0x56, 0x6c, 0x43, 0x11, 0x93, 0xd0, 0x07, 0xd4, 0x14, 0x49, 0xa6, 0x86, 0x33, 0xa3, 0xa6, 0x83, 0x9e, 0x56,
|
0xaa, 0x7a, 0xab, 0x85, 0x24, 0x39, 0xe1, 0x4b, 0x9a, 0x1e, 0x84, 0x29, 0xea, 0x6d, 0xd5, 0x36, 0xe8, 0x97, 0x17,
|
||||||
0x2b, 0x31, 0xdd, 0x30, 0x58, 0x02, 0x62, 0xd2, 0xbe, 0xed, 0x8d, 0xcb, 0xd0, 0x58, 0x75, 0x4d, 0xa5, 0x84, 0x8e,
|
0x6f, 0x5e, 0xab, 0x87, 0x36, 0x45, 0x4e, 0xa7, 0x6c, 0x23, 0xd1, 0x8f, 0x37, 0x2f, 0x50, 0x5b, 0xc3, 0xa6, 0x53,
|
||||||
0x41, 0x59, 0x15, 0xa6, 0xb1, 0xba, 0x76, 0x22, 0xa2, 0x2f, 0x06, 0x89, 0xbb, 0x65, 0x05, 0x53, 0x97, 0xf9, 0x34,
|
0x63, 0x5b, 0xb5, 0xa2, 0xcd, 0x1a, 0x2a, 0x4b, 0xaa, 0x48, 0x40, 0xb9, 0xa0, 0x52, 0x42, 0xa1, 0x21, 0x30, 0x94,
|
||||||
0xd6, 0xad, 0xa2, 0x92, 0xa0, 0xba, 0x15, 0xf3, 0xe5, 0x41, 0xcf, 0x2a, 0xca, 0x57, 0x70, 0x9b, 0x84, 0x77, 0x01,
|
0xce, 0xda, 0x13, 0x53, 0x75, 0x83, 0xbb, 0x20, 0x7e, 0xde, 0x95, 0xd7, 0x51, 0x1e, 0x18, 0xd7, 0xaf, 0x3b, 0x6a,
|
||||||
0xcd, 0x43, 0x46, 0xcb, 0xa6, 0x82, 0xe6, 0x24, 0xb9, 0xbe, 0xfe, 0xe9, 0xef, 0xea, 0x33, 0x85, 0x32, 0xe1, 0xcc,
|
0x70, 0x3d, 0x98, 0x47, 0xea, 0x39, 0x8d, 0x88, 0x7e, 0x84, 0xc4, 0x83, 0x35, 0xcb, 0x98, 0x7a, 0xb8, 0xcd, 0x23,
|
||||||
0x09, 0x7d, 0xbe, 0x61, 0x54, 0x93, 0x9e, 0x6f, 0x3c, 0x32, 0x1f, 0x1c, 0x5a, 0xe8, 0xd3, 0xc1, 0xbf, 0xfc, 0x33,
|
0x5d, 0x8f, 0x2a, 0x09, 0xaa, 0x24, 0x32, 0x5f, 0x34, 0x74, 0xaf, 0xa0, 0x7c, 0x09, 0xaf, 0x64, 0xd8, 0x70, 0xa8,
|
||||||
0x29, 0xef, 0x4e, 0x9b, 0xbd, 0x24, 0xfd, 0x5f, 0x37, 0x9d, 0x86, 0x49, 0xac, 0x97, 0x35, 0x93, 0xe9, 0x35, 0x18,
|
0x50, 0x12, 0x9a, 0x57, 0x05, 0x54, 0x40, 0xf1, 0xf5, 0xf5, 0x0f, 0xff, 0x52, 0x9f, 0x3f, 0xc0, 0xcf, 0x13, 0x27,
|
||||||
0x18, 0xbb, 0xe6, 0x01, 0x38, 0xa7, 0x1c, 0x30, 0xb4, 0x65, 0xcf, 0x03, 0x60, 0xff, 0x72, 0xf3, 0x02, 0xfd, 0xda,
|
0x3c, 0x29, 0x0c, 0xa3, 0xea, 0x74, 0x7c, 0xe3, 0xa1, 0xf9, 0x90, 0x51, 0xc3, 0x7b, 0x00, 0xfc, 0x4e, 0xef, 0x49,
|
||||||
0xc2, 0x09, 0xa6, 0x06, 0x7b, 0xed, 0x65, 0x4d, 0x65, 0xd9, 0xe4, 0xc9, 0xbb, 0x5f, 0xae, 0x6f, 0xce, 0x1e, 0xaf,
|
0x79, 0x77, 0x98, 0xec, 0x24, 0xe9, 0x5f, 0x5d, 0xd9, 0x1a, 0x26, 0xd1, 0x2e, 0x4a, 0x26, 0xe7, 0xd7, 0x60, 0x60,
|
||||||
0x35, 0x11, 0xa2, 0x3c, 0x33, 0x1f, 0x42, 0xd6, 0x95, 0x64, 0x2d, 0xe9, 0xa4, 0x16, 0xeb, 0xa8, 0x10, 0x38, 0x79,
|
0x34, 0x30, 0x0b, 0xe0, 0x9c, 0x72, 0xc0, 0xd0, 0xe6, 0x1d, 0x0f, 0xec, 0xa8, 0x42, 0xec, 0x27, 0x8d, 0x98, 0xd9,
|
||||||
0xa4, 0x9f, 0x17, 0xac, 0xa2, 0xc6, 0xa9, 0x9e, 0xd1, 0x4d, 0xd1, 0x97, 0x6c, 0x3c, 0xe9, 0x7e, 0x60, 0xa5, 0x6b,
|
0x60, 0xed, 0x65, 0x49, 0x65, 0x5e, 0xa5, 0xf1, 0xbb, 0x1f, 0xaf, 0x6f, 0x8e, 0x1e, 0x77, 0xb0, 0x52, 0x9e, 0x98,
|
||||||
0x4e, 0x89, 0x6b, 0x8e, 0x8c, 0xab, 0x3f, 0x13, 0xfd, 0x0b, 0x65, 0x37, 0xa3, 0x8e, 0x36, 0x12, 0x00, 0x00};
|
0x0f, 0x2c, 0x6d, 0x21, 0x59, 0x4d, 0x1a, 0xa9, 0xc5, 0x3a, 0x2a, 0xce, 0x0e, 0x1e, 0xe9, 0x75, 0xbd, 0x33, 0xda,
|
||||||
|
0xa9, 0x8e, 0x71, 0x30, 0x47, 0x0f, 0xd9, 0x78, 0xd0, 0xfd, 0x99, 0x95, 0x03, 0x73, 0x14, 0x07, 0xe6, 0x5c, 0x0e,
|
||||||
|
0xf4, 0xe7, 0xa7, 0xdf, 0x01, 0xf1, 0x69, 0xfc, 0xac, 0x8e, 0x12, 0x00, 0x00};
|
||||||
|
|
||||||
} // namespace captive_portal
|
} // namespace captive_portal
|
||||||
} // namespace esphome
|
} // namespace esphome
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ from esphome.const import (
|
|||||||
ICON_RADIATOR,
|
ICON_RADIATOR,
|
||||||
ICON_RESTART,
|
ICON_RESTART,
|
||||||
DEVICE_CLASS_CARBON_DIOXIDE,
|
DEVICE_CLASS_CARBON_DIOXIDE,
|
||||||
DEVICE_CLASS_VOLATILE_ORGANIC_COMPOUNDS_PARTS,
|
DEVICE_CLASS_VOLATILE_ORGANIC_COMPOUNDS,
|
||||||
STATE_CLASS_MEASUREMENT,
|
STATE_CLASS_MEASUREMENT,
|
||||||
UNIT_PARTS_PER_MILLION,
|
UNIT_PARTS_PER_MILLION,
|
||||||
UNIT_PARTS_PER_BILLION,
|
UNIT_PARTS_PER_BILLION,
|
||||||
@@ -43,7 +43,7 @@ CONFIG_SCHEMA = (
|
|||||||
unit_of_measurement=UNIT_PARTS_PER_BILLION,
|
unit_of_measurement=UNIT_PARTS_PER_BILLION,
|
||||||
icon=ICON_RADIATOR,
|
icon=ICON_RADIATOR,
|
||||||
accuracy_decimals=0,
|
accuracy_decimals=0,
|
||||||
device_class=DEVICE_CLASS_VOLATILE_ORGANIC_COMPOUNDS_PARTS,
|
device_class=DEVICE_CLASS_VOLATILE_ORGANIC_COMPOUNDS,
|
||||||
state_class=STATE_CLASS_MEASUREMENT,
|
state_class=STATE_CLASS_MEASUREMENT,
|
||||||
),
|
),
|
||||||
cv.Optional(CONF_VERSION): text_sensor.text_sensor_schema(
|
cv.Optional(CONF_VERSION): text_sensor.text_sensor_schema(
|
||||||
|
|||||||
@@ -343,7 +343,7 @@ CLIMATE_CONTROL_ACTION_SCHEMA = cv.Schema(
|
|||||||
cv.Optional(CONF_TARGET_TEMPERATURE): cv.templatable(cv.temperature),
|
cv.Optional(CONF_TARGET_TEMPERATURE): cv.templatable(cv.temperature),
|
||||||
cv.Optional(CONF_TARGET_TEMPERATURE_LOW): cv.templatable(cv.temperature),
|
cv.Optional(CONF_TARGET_TEMPERATURE_LOW): cv.templatable(cv.temperature),
|
||||||
cv.Optional(CONF_TARGET_TEMPERATURE_HIGH): cv.templatable(cv.temperature),
|
cv.Optional(CONF_TARGET_TEMPERATURE_HIGH): cv.templatable(cv.temperature),
|
||||||
cv.Optional(CONF_AWAY): cv.invalid("Use preset instead"),
|
cv.Optional(CONF_AWAY): cv.templatable(cv.boolean),
|
||||||
cv.Exclusive(CONF_FAN_MODE, "fan_mode"): cv.templatable(
|
cv.Exclusive(CONF_FAN_MODE, "fan_mode"): cv.templatable(
|
||||||
validate_climate_fan_mode
|
validate_climate_fan_mode
|
||||||
),
|
),
|
||||||
@@ -379,6 +379,9 @@ async def climate_control_to_code(config, action_id, template_arg, args):
|
|||||||
config[CONF_TARGET_TEMPERATURE_HIGH], args, float
|
config[CONF_TARGET_TEMPERATURE_HIGH], args, float
|
||||||
)
|
)
|
||||||
cg.add(var.set_target_temperature_high(template_))
|
cg.add(var.set_target_temperature_high(template_))
|
||||||
|
if CONF_AWAY in config:
|
||||||
|
template_ = await cg.templatable(config[CONF_AWAY], args, bool)
|
||||||
|
cg.add(var.set_away(template_))
|
||||||
if CONF_FAN_MODE in config:
|
if CONF_FAN_MODE in config:
|
||||||
template_ = await cg.templatable(config[CONF_FAN_MODE], args, ClimateFanMode)
|
template_ = await cg.templatable(config[CONF_FAN_MODE], args, ClimateFanMode)
|
||||||
cg.add(var.set_fan_mode(template_))
|
cg.add(var.set_fan_mode(template_))
|
||||||
|
|||||||
@@ -264,11 +264,25 @@ const optional<ClimateMode> &ClimateCall::get_mode() const { return this->mode_;
|
|||||||
const optional<float> &ClimateCall::get_target_temperature() const { return this->target_temperature_; }
|
const optional<float> &ClimateCall::get_target_temperature() const { return this->target_temperature_; }
|
||||||
const optional<float> &ClimateCall::get_target_temperature_low() const { return this->target_temperature_low_; }
|
const optional<float> &ClimateCall::get_target_temperature_low() const { return this->target_temperature_low_; }
|
||||||
const optional<float> &ClimateCall::get_target_temperature_high() const { return this->target_temperature_high_; }
|
const optional<float> &ClimateCall::get_target_temperature_high() const { return this->target_temperature_high_; }
|
||||||
|
optional<bool> ClimateCall::get_away() const {
|
||||||
|
if (!this->preset_.has_value())
|
||||||
|
return {};
|
||||||
|
return *this->preset_ == ClimatePreset::CLIMATE_PRESET_AWAY;
|
||||||
|
}
|
||||||
const optional<ClimateFanMode> &ClimateCall::get_fan_mode() const { return this->fan_mode_; }
|
const optional<ClimateFanMode> &ClimateCall::get_fan_mode() const { return this->fan_mode_; }
|
||||||
const optional<std::string> &ClimateCall::get_custom_fan_mode() const { return this->custom_fan_mode_; }
|
const optional<std::string> &ClimateCall::get_custom_fan_mode() const { return this->custom_fan_mode_; }
|
||||||
const optional<ClimatePreset> &ClimateCall::get_preset() const { return this->preset_; }
|
const optional<ClimatePreset> &ClimateCall::get_preset() const { return this->preset_; }
|
||||||
const optional<std::string> &ClimateCall::get_custom_preset() const { return this->custom_preset_; }
|
const optional<std::string> &ClimateCall::get_custom_preset() const { return this->custom_preset_; }
|
||||||
const optional<ClimateSwingMode> &ClimateCall::get_swing_mode() const { return this->swing_mode_; }
|
const optional<ClimateSwingMode> &ClimateCall::get_swing_mode() const { return this->swing_mode_; }
|
||||||
|
ClimateCall &ClimateCall::set_away(bool away) {
|
||||||
|
this->preset_ = away ? CLIMATE_PRESET_AWAY : CLIMATE_PRESET_HOME;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
ClimateCall &ClimateCall::set_away(optional<bool> away) {
|
||||||
|
if (away.has_value())
|
||||||
|
this->preset_ = *away ? CLIMATE_PRESET_AWAY : CLIMATE_PRESET_HOME;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
ClimateCall &ClimateCall::set_target_temperature_high(optional<float> target_temperature_high) {
|
ClimateCall &ClimateCall::set_target_temperature_high(optional<float> target_temperature_high) {
|
||||||
this->target_temperature_high_ = target_temperature_high;
|
this->target_temperature_high_ = target_temperature_high;
|
||||||
return *this;
|
return *this;
|
||||||
@@ -439,7 +453,12 @@ void Climate::set_visual_temperature_step_override(float target, float current)
|
|||||||
this->visual_target_temperature_step_override_ = target;
|
this->visual_target_temperature_step_override_ = target;
|
||||||
this->visual_current_temperature_step_override_ = current;
|
this->visual_current_temperature_step_override_ = current;
|
||||||
}
|
}
|
||||||
|
#pragma GCC diagnostic push
|
||||||
|
#pragma GCC diagnostic ignored "-Wdeprecated-declarations"
|
||||||
|
Climate::Climate(const std::string &name) : EntityBase(name) {}
|
||||||
|
#pragma GCC diagnostic pop
|
||||||
|
|
||||||
|
Climate::Climate() : Climate("") {}
|
||||||
ClimateCall Climate::make_call() { return ClimateCall(this); }
|
ClimateCall Climate::make_call() { return ClimateCall(this); }
|
||||||
|
|
||||||
ClimateCall ClimateDeviceRestoreState::to_call(Climate *climate) {
|
ClimateCall ClimateDeviceRestoreState::to_call(Climate *climate) {
|
||||||
|
|||||||
@@ -64,6 +64,10 @@ class ClimateCall {
|
|||||||
* For climate devices with two point target temperature control
|
* For climate devices with two point target temperature control
|
||||||
*/
|
*/
|
||||||
ClimateCall &set_target_temperature_high(optional<float> target_temperature_high);
|
ClimateCall &set_target_temperature_high(optional<float> target_temperature_high);
|
||||||
|
ESPDEPRECATED("set_away() is deprecated, please use .set_preset(CLIMATE_PRESET_AWAY) instead", "v1.20")
|
||||||
|
ClimateCall &set_away(bool away);
|
||||||
|
ESPDEPRECATED("set_away() is deprecated, please use .set_preset(CLIMATE_PRESET_AWAY) instead", "v1.20")
|
||||||
|
ClimateCall &set_away(optional<bool> away);
|
||||||
/// Set the fan mode of the climate device.
|
/// Set the fan mode of the climate device.
|
||||||
ClimateCall &set_fan_mode(ClimateFanMode fan_mode);
|
ClimateCall &set_fan_mode(ClimateFanMode fan_mode);
|
||||||
/// Set the fan mode of the climate device.
|
/// Set the fan mode of the climate device.
|
||||||
@@ -93,6 +97,8 @@ class ClimateCall {
|
|||||||
const optional<float> &get_target_temperature() const;
|
const optional<float> &get_target_temperature() const;
|
||||||
const optional<float> &get_target_temperature_low() const;
|
const optional<float> &get_target_temperature_low() const;
|
||||||
const optional<float> &get_target_temperature_high() const;
|
const optional<float> &get_target_temperature_high() const;
|
||||||
|
ESPDEPRECATED("get_away() is deprecated, please use .get_preset() instead", "v1.20")
|
||||||
|
optional<bool> get_away() const;
|
||||||
const optional<ClimateFanMode> &get_fan_mode() const;
|
const optional<ClimateFanMode> &get_fan_mode() const;
|
||||||
const optional<ClimateSwingMode> &get_swing_mode() const;
|
const optional<ClimateSwingMode> &get_swing_mode() const;
|
||||||
const optional<std::string> &get_custom_fan_mode() const;
|
const optional<std::string> &get_custom_fan_mode() const;
|
||||||
@@ -160,6 +166,11 @@ struct ClimateDeviceRestoreState {
|
|||||||
*/
|
*/
|
||||||
class Climate : public EntityBase {
|
class Climate : public EntityBase {
|
||||||
public:
|
public:
|
||||||
|
/// Construct a climate device with empty name (will be set later).
|
||||||
|
Climate();
|
||||||
|
/// Construct a climate device with a name.
|
||||||
|
Climate(const std::string &name);
|
||||||
|
|
||||||
/// The active mode of the climate device.
|
/// The active mode of the climate device.
|
||||||
ClimateMode mode{CLIMATE_MODE_OFF};
|
ClimateMode mode{CLIMATE_MODE_OFF};
|
||||||
/// The active state of the climate device.
|
/// The active state of the climate device.
|
||||||
@@ -178,6 +189,14 @@ class Climate : public EntityBase {
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/** Whether the climate device is in away mode.
|
||||||
|
*
|
||||||
|
* Away allows climate devices to have two different target temperature configs:
|
||||||
|
* one for normal mode and one for away mode.
|
||||||
|
*/
|
||||||
|
ESPDEPRECATED("away is deprecated, use preset instead", "v1.20")
|
||||||
|
bool away{false};
|
||||||
|
|
||||||
/// The active fan mode of the climate device.
|
/// The active fan mode of the climate device.
|
||||||
optional<ClimateFanMode> fan_mode;
|
optional<ClimateFanMode> fan_mode;
|
||||||
|
|
||||||
|
|||||||
@@ -117,6 +117,15 @@ class ClimateTraits {
|
|||||||
bool supports_custom_preset(const std::string &custom_preset) const {
|
bool supports_custom_preset(const std::string &custom_preset) const {
|
||||||
return supported_custom_presets_.count(custom_preset);
|
return supported_custom_presets_.count(custom_preset);
|
||||||
}
|
}
|
||||||
|
ESPDEPRECATED("This method is deprecated, use set_supported_presets() instead", "v1.20")
|
||||||
|
void set_supports_away(bool supports) {
|
||||||
|
if (supports) {
|
||||||
|
supported_presets_.insert(CLIMATE_PRESET_AWAY);
|
||||||
|
supported_presets_.insert(CLIMATE_PRESET_HOME);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ESPDEPRECATED("This method is deprecated, use supports_preset() instead", "v1.20")
|
||||||
|
bool get_supports_away() const { return supports_preset(CLIMATE_PRESET_AWAY); }
|
||||||
|
|
||||||
void set_supported_swing_modes(std::set<ClimateSwingMode> modes) { supported_swing_modes_ = std::move(modes); }
|
void set_supported_swing_modes(std::set<ClimateSwingMode> modes) { supported_swing_modes_ = std::move(modes); }
|
||||||
void add_supported_swing_mode(ClimateSwingMode mode) { supported_swing_modes_.insert(mode); }
|
void add_supported_swing_mode(ClimateSwingMode mode) { supported_swing_modes_.insert(mode); }
|
||||||
|
|||||||
@@ -28,7 +28,6 @@ cover::CoverTraits CopyCover::get_traits() {
|
|||||||
// copy traits manually so it doesn't break when new options are added
|
// copy traits manually so it doesn't break when new options are added
|
||||||
// but the control() method hasn't implemented them yet.
|
// but the control() method hasn't implemented them yet.
|
||||||
traits.set_is_assumed_state(base.get_is_assumed_state());
|
traits.set_is_assumed_state(base.get_is_assumed_state());
|
||||||
traits.set_supports_stop(base.get_supports_stop());
|
|
||||||
traits.set_supports_position(base.get_supports_position());
|
traits.set_supports_position(base.get_supports_position());
|
||||||
traits.set_supports_tilt(base.get_supports_tilt());
|
traits.set_supports_tilt(base.get_supports_tilt());
|
||||||
traits.set_supports_toggle(base.get_supports_toggle());
|
traits.set_supports_toggle(base.get_supports_toggle());
|
||||||
|
|||||||
@@ -14,15 +14,12 @@ from .. import copy_ns
|
|||||||
CopySelect = copy_ns.class_("CopySelect", select.Select, cg.Component)
|
CopySelect = copy_ns.class_("CopySelect", select.Select, cg.Component)
|
||||||
|
|
||||||
|
|
||||||
CONFIG_SCHEMA = (
|
CONFIG_SCHEMA = select.SELECT_SCHEMA.extend(
|
||||||
select.select_schema(CopySelect)
|
{
|
||||||
.extend(
|
cv.GenerateID(): cv.declare_id(CopySelect),
|
||||||
{
|
cv.Required(CONF_SOURCE_ID): cv.use_id(select.Select),
|
||||||
cv.Required(CONF_SOURCE_ID): cv.use_id(select.Select),
|
}
|
||||||
}
|
).extend(cv.COMPONENT_SCHEMA)
|
||||||
)
|
|
||||||
.extend(cv.COMPONENT_SCHEMA)
|
|
||||||
)
|
|
||||||
|
|
||||||
FINAL_VALIDATE_SCHEMA = cv.All(
|
FINAL_VALIDATE_SCHEMA = cv.All(
|
||||||
inherit_property_from(CONF_ICON, CONF_SOURCE_ID),
|
inherit_property_from(CONF_ICON, CONF_SOURCE_ID),
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ const char *cover_operation_to_str(CoverOperation op) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Cover::Cover() : position{COVER_OPEN} {}
|
Cover::Cover(const std::string &name) : EntityBase(name), position{COVER_OPEN} {}
|
||||||
|
|
||||||
CoverCall::CoverCall(Cover *parent) : parent_(parent) {}
|
CoverCall::CoverCall(Cover *parent) : parent_(parent) {}
|
||||||
CoverCall &CoverCall::set_command(const char *command) {
|
CoverCall &CoverCall::set_command(const char *command) {
|
||||||
@@ -145,7 +145,7 @@ CoverCall &CoverCall::set_stop(bool stop) {
|
|||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
bool CoverCall::get_stop() const { return this->stop_; }
|
bool CoverCall::get_stop() const { return this->stop_; }
|
||||||
|
void Cover::set_device_class(const std::string &device_class) { this->device_class_override_ = device_class; }
|
||||||
CoverCall Cover::make_call() { return {this}; }
|
CoverCall Cover::make_call() { return {this}; }
|
||||||
void Cover::open() {
|
void Cover::open() {
|
||||||
auto call = this->make_call();
|
auto call = this->make_call();
|
||||||
@@ -204,7 +204,12 @@ optional<CoverRestoreState> Cover::restore_state_() {
|
|||||||
return {};
|
return {};
|
||||||
return recovered;
|
return recovered;
|
||||||
}
|
}
|
||||||
|
Cover::Cover() : Cover("") {}
|
||||||
|
std::string Cover::get_device_class() {
|
||||||
|
if (this->device_class_override_.has_value())
|
||||||
|
return *this->device_class_override_;
|
||||||
|
return "";
|
||||||
|
}
|
||||||
bool Cover::is_fully_open() const { return this->position == COVER_OPEN; }
|
bool Cover::is_fully_open() const { return this->position == COVER_OPEN; }
|
||||||
bool Cover::is_fully_closed() const { return this->position == COVER_CLOSED; }
|
bool Cover::is_fully_closed() const { return this->position == COVER_CLOSED; }
|
||||||
|
|
||||||
|
|||||||
@@ -108,9 +108,10 @@ const char *cover_operation_to_str(CoverOperation op);
|
|||||||
* to control all values of the cover. Also implement get_traits() to return what operations
|
* to control all values of the cover. Also implement get_traits() to return what operations
|
||||||
* the cover supports.
|
* the cover supports.
|
||||||
*/
|
*/
|
||||||
class Cover : public EntityBase, public EntityBase_DeviceClass {
|
class Cover : public EntityBase {
|
||||||
public:
|
public:
|
||||||
explicit Cover();
|
explicit Cover();
|
||||||
|
explicit Cover(const std::string &name);
|
||||||
|
|
||||||
/// The current operation of the cover (idle, opening, closing).
|
/// The current operation of the cover (idle, opening, closing).
|
||||||
CoverOperation current_operation{COVER_OPERATION_IDLE};
|
CoverOperation current_operation{COVER_OPERATION_IDLE};
|
||||||
@@ -140,9 +141,8 @@ class Cover : public EntityBase, public EntityBase_DeviceClass {
|
|||||||
/** Stop the cover.
|
/** Stop the cover.
|
||||||
*
|
*
|
||||||
* This is a legacy method and may be removed later, please use `.make_call()` instead.
|
* This is a legacy method and may be removed later, please use `.make_call()` instead.
|
||||||
* As per solution from issue #2885 the call should include perform()
|
|
||||||
*/
|
*/
|
||||||
ESPDEPRECATED("stop() is deprecated, use make_call().set_command_stop().perform() instead.", "2021.9")
|
ESPDEPRECATED("stop() is deprecated, use make_call().set_command_stop() instead.", "2021.9")
|
||||||
void stop();
|
void stop();
|
||||||
|
|
||||||
void add_on_state_callback(std::function<void()> &&f);
|
void add_on_state_callback(std::function<void()> &&f);
|
||||||
@@ -157,6 +157,8 @@ class Cover : public EntityBase, public EntityBase_DeviceClass {
|
|||||||
void publish_state(bool save = true);
|
void publish_state(bool save = true);
|
||||||
|
|
||||||
virtual CoverTraits get_traits() = 0;
|
virtual CoverTraits get_traits() = 0;
|
||||||
|
void set_device_class(const std::string &device_class);
|
||||||
|
std::string get_device_class();
|
||||||
|
|
||||||
/// Helper method to check if the cover is fully open. Equivalent to comparing .position against 1.0
|
/// Helper method to check if the cover is fully open. Equivalent to comparing .position against 1.0
|
||||||
bool is_fully_open() const;
|
bool is_fully_open() const;
|
||||||
@@ -171,6 +173,7 @@ class Cover : public EntityBase, public EntityBase_DeviceClass {
|
|||||||
optional<CoverRestoreState> restore_state_();
|
optional<CoverRestoreState> restore_state_();
|
||||||
|
|
||||||
CallbackManager<void()> state_callback_{};
|
CallbackManager<void()> state_callback_{};
|
||||||
|
optional<std::string> device_class_override_{};
|
||||||
|
|
||||||
ESPPreferenceObject rtc_;
|
ESPPreferenceObject rtc_;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -15,15 +15,12 @@ class CoverTraits {
|
|||||||
void set_supports_tilt(bool supports_tilt) { this->supports_tilt_ = supports_tilt; }
|
void set_supports_tilt(bool supports_tilt) { this->supports_tilt_ = supports_tilt; }
|
||||||
bool get_supports_toggle() const { return this->supports_toggle_; }
|
bool get_supports_toggle() const { return this->supports_toggle_; }
|
||||||
void set_supports_toggle(bool supports_toggle) { this->supports_toggle_ = supports_toggle; }
|
void set_supports_toggle(bool supports_toggle) { this->supports_toggle_ = supports_toggle; }
|
||||||
bool get_supports_stop() const { return this->supports_stop_; }
|
|
||||||
void set_supports_stop(bool supports_stop) { this->supports_stop_ = supports_stop; }
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
bool is_assumed_state_{false};
|
bool is_assumed_state_{false};
|
||||||
bool supports_position_{false};
|
bool supports_position_{false};
|
||||||
bool supports_tilt_{false};
|
bool supports_tilt_{false};
|
||||||
bool supports_toggle_{false};
|
bool supports_toggle_{false};
|
||||||
bool supports_stop_{false};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace cover
|
} // namespace cover
|
||||||
|
|||||||
@@ -12,7 +12,6 @@ using namespace esphome::cover;
|
|||||||
|
|
||||||
CoverTraits CurrentBasedCover::get_traits() {
|
CoverTraits CurrentBasedCover::get_traits() {
|
||||||
auto traits = CoverTraits();
|
auto traits = CoverTraits();
|
||||||
traits.set_supports_stop(true);
|
|
||||||
traits.set_supports_position(true);
|
traits.set_supports_position(true);
|
||||||
traits.set_supports_toggle(true);
|
traits.set_supports_toggle(true);
|
||||||
traits.set_is_assumed_state(false);
|
traits.set_is_assumed_state(false);
|
||||||
|
|||||||
@@ -7,10 +7,9 @@ import requests
|
|||||||
|
|
||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
import esphome.final_validate as fv
|
|
||||||
from esphome import git
|
from esphome import git
|
||||||
from esphome.components.packages import validate_source_shorthand
|
from esphome.components.packages import validate_source_shorthand
|
||||||
from esphome.const import CONF_REF, CONF_WIFI, CONF_ESPHOME, CONF_PROJECT
|
from esphome.const import CONF_REF, CONF_WIFI
|
||||||
from esphome.wizard import wizard_file
|
from esphome.wizard import wizard_file
|
||||||
from esphome.yaml_util import dump
|
from esphome.yaml_util import dump
|
||||||
|
|
||||||
@@ -53,17 +52,6 @@ CONFIG_SCHEMA = cv.All(
|
|||||||
validate_full_url,
|
validate_full_url,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
def _final_validate(config):
|
|
||||||
full_config = fv.full_config.get()[CONF_ESPHOME]
|
|
||||||
if CONF_PROJECT not in full_config:
|
|
||||||
raise cv.Invalid(
|
|
||||||
"Dashboard import requires the `esphome` -> `project` information to be provided."
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
FINAL_VALIDATE_SCHEMA = _final_validate
|
|
||||||
|
|
||||||
WIFI_CONFIG = """
|
WIFI_CONFIG = """
|
||||||
|
|
||||||
wifi:
|
wifi:
|
||||||
|
|||||||
@@ -1,11 +1,15 @@
|
|||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
|
import esphome.final_validate as fv
|
||||||
|
from esphome.components import logger
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
CONF_BLOCK,
|
CONF_BLOCK,
|
||||||
CONF_DEVICE,
|
CONF_DEVICE,
|
||||||
CONF_FRAGMENTATION,
|
CONF_FRAGMENTATION,
|
||||||
CONF_FREE,
|
CONF_FREE,
|
||||||
CONF_ID,
|
CONF_ID,
|
||||||
|
CONF_LEVEL,
|
||||||
|
CONF_LOGGER,
|
||||||
CONF_LOOP_TIME,
|
CONF_LOOP_TIME,
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -17,29 +21,38 @@ debug_ns = cg.esphome_ns.namespace("debug")
|
|||||||
DebugComponent = debug_ns.class_("DebugComponent", cg.PollingComponent)
|
DebugComponent = debug_ns.class_("DebugComponent", cg.PollingComponent)
|
||||||
|
|
||||||
|
|
||||||
CONFIG_SCHEMA = cv.All(
|
CONFIG_SCHEMA = cv.Schema(
|
||||||
cv.Schema(
|
{
|
||||||
{
|
cv.GenerateID(): cv.declare_id(DebugComponent),
|
||||||
cv.GenerateID(): cv.declare_id(DebugComponent),
|
cv.Optional(CONF_DEVICE): cv.invalid(
|
||||||
cv.Optional(CONF_DEVICE): cv.invalid(
|
"The 'device' option has been moved to the 'debug' text_sensor component"
|
||||||
"The 'device' option has been moved to the 'debug' text_sensor component"
|
),
|
||||||
),
|
cv.Optional(CONF_FREE): cv.invalid(
|
||||||
cv.Optional(CONF_FREE): cv.invalid(
|
"The 'free' option has been moved to the 'debug' sensor component"
|
||||||
"The 'free' option has been moved to the 'debug' sensor component"
|
),
|
||||||
),
|
cv.Optional(CONF_BLOCK): cv.invalid(
|
||||||
cv.Optional(CONF_BLOCK): cv.invalid(
|
"The 'block' option has been moved to the 'debug' sensor component"
|
||||||
"The 'block' option has been moved to the 'debug' sensor component"
|
),
|
||||||
),
|
cv.Optional(CONF_FRAGMENTATION): cv.invalid(
|
||||||
cv.Optional(CONF_FRAGMENTATION): cv.invalid(
|
"The 'fragmentation' option has been moved to the 'debug' sensor component"
|
||||||
"The 'fragmentation' option has been moved to the 'debug' sensor component"
|
),
|
||||||
),
|
cv.Optional(CONF_LOOP_TIME): cv.invalid(
|
||||||
cv.Optional(CONF_LOOP_TIME): cv.invalid(
|
"The 'loop_time' option has been moved to the 'debug' sensor component"
|
||||||
"The 'loop_time' option has been moved to the 'debug' sensor component"
|
),
|
||||||
),
|
}
|
||||||
}
|
).extend(cv.polling_component_schema("60s"))
|
||||||
).extend(cv.polling_component_schema("60s")),
|
|
||||||
cv.only_on(["esp32", "esp8266"]),
|
|
||||||
)
|
def _final_validate(_):
|
||||||
|
logger_conf = fv.full_config.get()[CONF_LOGGER]
|
||||||
|
severity = logger.LOG_LEVEL_SEVERITY.index(logger_conf[CONF_LEVEL])
|
||||||
|
if severity < logger.LOG_LEVEL_SEVERITY.index("DEBUG"):
|
||||||
|
raise cv.Invalid(
|
||||||
|
"The debug component requires the logger to be at least at DEBUG level"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
FINAL_VALIDATE_SCHEMA = _final_validate
|
||||||
|
|
||||||
|
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
|
|||||||
@@ -37,10 +37,6 @@ static uint32_t get_free_heap() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void DebugComponent::dump_config() {
|
void DebugComponent::dump_config() {
|
||||||
#ifndef ESPHOME_LOG_HAS_DEBUG
|
|
||||||
return; // Can't log below if debug logging is disabled
|
|
||||||
#endif
|
|
||||||
|
|
||||||
std::string device_info;
|
std::string device_info;
|
||||||
std::string reset_reason;
|
std::string reset_reason;
|
||||||
device_info.reserve(256);
|
device_info.reserve(256);
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "esphome/core/automation.h"
|
|
||||||
#include "esphome/core/component.h"
|
#include "esphome/core/component.h"
|
||||||
#include "esphome/core/hal.h"
|
|
||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
|
#include "esphome/core/automation.h"
|
||||||
|
#include "esphome/core/hal.h"
|
||||||
|
|
||||||
#ifdef USE_ESP32
|
#ifdef USE_ESP32
|
||||||
#include <esp_sleep.h>
|
#include <esp_sleep.h>
|
||||||
@@ -11,7 +11,6 @@
|
|||||||
|
|
||||||
#ifdef USE_TIME
|
#ifdef USE_TIME
|
||||||
#include "esphome/components/time/real_time_clock.h"
|
#include "esphome/components/time/real_time_clock.h"
|
||||||
#include "esphome/core/time.h"
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
@@ -171,7 +170,7 @@ template<typename... Ts> class EnterDeepSleepAction : public Action<Ts...> {
|
|||||||
if (after_time)
|
if (after_time)
|
||||||
timestamp += 60 * 60 * 24;
|
timestamp += 60 * 60 * 24;
|
||||||
|
|
||||||
int32_t offset = ESPTime::timezone_offset();
|
int32_t offset = time::ESPTime::timezone_offset();
|
||||||
timestamp -= offset; // Change timestamp to utc
|
timestamp -= offset; // Change timestamp to utc
|
||||||
const uint32_t ms_left = (timestamp - timestamp_now) * 1000;
|
const uint32_t ms_left = (timestamp - timestamp_now) * 1000;
|
||||||
this->deep_sleep_->set_sleep_duration(ms_left);
|
this->deep_sleep_->set_sleep_duration(ms_left);
|
||||||
|
|||||||
@@ -72,7 +72,6 @@ class DemoCover : public cover::Cover, public Component {
|
|||||||
traits.set_supports_tilt(true);
|
traits.set_supports_tilt(true);
|
||||||
break;
|
break;
|
||||||
case DemoCoverType::TYPE_4:
|
case DemoCoverType::TYPE_4:
|
||||||
traits.set_supports_stop(true);
|
|
||||||
traits.set_is_assumed_state(true);
|
traits.set_is_assumed_state(true);
|
||||||
traits.set_supports_tilt(true);
|
traits.set_supports_tilt(true);
|
||||||
break;
|
break;
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user