mirror of
				https://github.com/esphome/esphome.git
				synced 2025-11-03 16:41:50 +00:00 
			
		
		
		
	Compare commits
	
		
			1 Commits
		
	
	
		
			2023.4.0b2
			...
			jesserockz
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					69fe18f1ef | 
@@ -4,60 +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",
 | 
				
			||||||
          "!include_dir_named scalar",
 | 
					      "!include_dir_merge_list scalar",
 | 
				
			||||||
          "!include_dir_list scalar",
 | 
					      "!include_dir_merge_named scalar"
 | 
				
			||||||
          "!include_dir_merge_list scalar",
 | 
					    ],
 | 
				
			||||||
          "!include_dir_merge_named scalar"
 | 
					    "files.exclude": {
 | 
				
			||||||
        ],
 | 
					      "**/.git": true,
 | 
				
			||||||
        "files.exclude": {
 | 
					      "**/.DS_Store": true,
 | 
				
			||||||
          "**/.git": true,
 | 
					      "**/*.pyc": {
 | 
				
			||||||
          "**/.DS_Store": true,
 | 
					        "when": "$(basename).py"
 | 
				
			||||||
          "**/*.pyc": {
 | 
					      },
 | 
				
			||||||
            "when": "$(basename).py"
 | 
					      "**/__pycache__": true
 | 
				
			||||||
          },
 | 
					    },
 | 
				
			||||||
          "**/__pycache__": true
 | 
					    "files.associations": {
 | 
				
			||||||
        },
 | 
					      "**/.vscode/*.json": "jsonc"
 | 
				
			||||||
        "files.associations": {
 | 
					    },
 | 
				
			||||||
          "**/.vscode/*.json": "jsonc"
 | 
					    "C_Cpp.clang_format_path": "/usr/bin/clang-format-11",
 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        "C_Cpp.clang_format_path": "/usr/bin/clang-format-13"
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										5
									
								
								.github/workflows/ci-docker.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/ci-docker.yml
									
									
									
									
										vendored
									
									
								
							@@ -23,11 +23,6 @@ permissions:
 | 
				
			|||||||
  contents: read
 | 
					  contents: read
 | 
				
			||||||
  packages: read
 | 
					  packages: read
 | 
				
			||||||
 | 
					
 | 
				
			||||||
concurrency:
 | 
					 | 
				
			||||||
  # yamllint disable-line rule:line-length
 | 
					 | 
				
			||||||
  group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}
 | 
					 | 
				
			||||||
  cancel-in-progress: true
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
jobs:
 | 
					jobs:
 | 
				
			||||||
  check-docker:
 | 
					  check-docker:
 | 
				
			||||||
    name: Build docker containers
 | 
					    name: Build docker containers
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										26
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										26
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							@@ -7,7 +7,6 @@ on:
 | 
				
			|||||||
    branches: [dev, beta, release]
 | 
					    branches: [dev, beta, release]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  pull_request:
 | 
					  pull_request:
 | 
				
			||||||
  merge_group:
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
permissions:
 | 
					permissions:
 | 
				
			||||||
  contents: read
 | 
					  contents: read
 | 
				
			||||||
@@ -41,10 +40,6 @@ jobs:
 | 
				
			|||||||
            file: tests/test3.yaml
 | 
					            file: tests/test3.yaml
 | 
				
			||||||
            name: Test tests/test3.yaml
 | 
					            name: Test tests/test3.yaml
 | 
				
			||||||
            pio_cache_key: test3
 | 
					            pio_cache_key: test3
 | 
				
			||||||
          - id: test
 | 
					 | 
				
			||||||
            file: tests/test3.1.yaml
 | 
					 | 
				
			||||||
            name: Test tests/test3.1.yaml
 | 
					 | 
				
			||||||
            pio_cache_key: test3.1
 | 
					 | 
				
			||||||
          - id: test
 | 
					          - id: test
 | 
				
			||||||
            file: tests/test4.yaml
 | 
					            file: tests/test4.yaml
 | 
				
			||||||
            name: Test tests/test4.yaml
 | 
					            name: Test tests/test4.yaml
 | 
				
			||||||
@@ -57,10 +52,6 @@ jobs:
 | 
				
			|||||||
            file: tests/test6.yaml
 | 
					            file: tests/test6.yaml
 | 
				
			||||||
            name: Test tests/test6.yaml
 | 
					            name: Test tests/test6.yaml
 | 
				
			||||||
            pio_cache_key: test6
 | 
					            pio_cache_key: test6
 | 
				
			||||||
          - id: test
 | 
					 | 
				
			||||||
            file: tests/test7.yaml
 | 
					 | 
				
			||||||
            name: Test tests/test7.yaml
 | 
					 | 
				
			||||||
            pio_cache_key: test7
 | 
					 | 
				
			||||||
          - id: pytest
 | 
					          - id: pytest
 | 
				
			||||||
            name: Run pytest
 | 
					            name: Run pytest
 | 
				
			||||||
          - id: clang-format
 | 
					          - id: clang-format
 | 
				
			||||||
@@ -133,7 +124,7 @@ jobs:
 | 
				
			|||||||
      - name: Install clang tools
 | 
					      - name: Install clang tools
 | 
				
			||||||
        run: |
 | 
					        run: |
 | 
				
			||||||
          sudo apt-get install \
 | 
					          sudo apt-get install \
 | 
				
			||||||
              clang-format-13 \
 | 
					              clang-format-11 \
 | 
				
			||||||
              clang-tidy-11
 | 
					              clang-tidy-11
 | 
				
			||||||
        if: matrix.id == 'clang-tidy' || matrix.id == 'clang-format'
 | 
					        if: matrix.id == 'clang-tidy' || matrix.id == 'clang-format'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -186,22 +177,9 @@ jobs:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
      - name: Run yamllint
 | 
					      - name: Run yamllint
 | 
				
			||||||
        if: matrix.id == 'yamllint'
 | 
					        if: matrix.id == 'yamllint'
 | 
				
			||||||
        uses: frenck/action-yamllint@v1.4.0
 | 
					        uses: frenck/action-yamllint@v1.3.1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      - 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() && (matrix.id == 'clang-tidy' || matrix.id == 'clang-format' || matrix.id == 'lint-python')
 | 
					        if: always() && (matrix.id == 'clang-tidy' || matrix.id == 'clang-format' || matrix.id == 'lint-python')
 | 
				
			||||||
 | 
					 | 
				
			||||||
  ci-status:
 | 
					 | 
				
			||||||
    name: CI Status
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					 | 
				
			||||||
    needs: [ci]
 | 
					 | 
				
			||||||
    if: always()
 | 
					 | 
				
			||||||
    steps:
 | 
					 | 
				
			||||||
      - name: Successful deploy
 | 
					 | 
				
			||||||
        if: ${{ !(contains(needs.*.result, 'failure')) }}
 | 
					 | 
				
			||||||
        run: exit 0
 | 
					 | 
				
			||||||
      - name: Failing deploy
 | 
					 | 
				
			||||||
        if: ${{ contains(needs.*.result, 'failure') }}
 | 
					 | 
				
			||||||
        run: exit 1
 | 
					 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										117
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										117
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							@@ -30,10 +30,6 @@ jobs:
 | 
				
			|||||||
            TAG=$(cat esphome/const.py | sed -n -E "s/^__version__\s+=\s+\"(.+)\"$/\1/p")
 | 
					            TAG=$(cat esphome/const.py | sed -n -E "s/^__version__\s+=\s+\"(.+)\"$/\1/p")
 | 
				
			||||||
            today="$(date --utc '+%Y%m%d')"
 | 
					            today="$(date --utc '+%Y%m%d')"
 | 
				
			||||||
            TAG="${TAG}${today}"
 | 
					            TAG="${TAG}${today}"
 | 
				
			||||||
            BRANCH=${GITHUB_REF#refs/heads/}
 | 
					 | 
				
			||||||
            if [[ "$BRANCH" != "dev" ]]; then
 | 
					 | 
				
			||||||
              TAG="${TAG}-${BRANCH}"
 | 
					 | 
				
			||||||
            fi
 | 
					 | 
				
			||||||
          fi
 | 
					          fi
 | 
				
			||||||
          echo "tag=${TAG}" >> $GITHUB_OUTPUT
 | 
					          echo "tag=${TAG}" >> $GITHUB_OUTPUT
 | 
				
			||||||
        # yamllint enable rule:line-length
 | 
					        # yamllint enable rule:line-length
 | 
				
			||||||
@@ -61,30 +57,17 @@ jobs:
 | 
				
			|||||||
        run: twine upload dist/*
 | 
					        run: twine upload dist/*
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  deploy-docker:
 | 
					  deploy-docker:
 | 
				
			||||||
    name: Build and publish ESPHome ${{ matrix.image.title}}
 | 
					    name: Build and publish docker containers
 | 
				
			||||||
    if: github.repository == 'esphome/esphome'
 | 
					    if: github.repository == 'esphome/esphome'
 | 
				
			||||||
    permissions:
 | 
					    permissions:
 | 
				
			||||||
      contents: read
 | 
					      contents: read
 | 
				
			||||||
      packages: write
 | 
					      packages: write
 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    continue-on-error: ${{ matrix.image.title == 'lint' }}
 | 
					 | 
				
			||||||
    needs: [init]
 | 
					    needs: [init]
 | 
				
			||||||
    strategy:
 | 
					    strategy:
 | 
				
			||||||
      fail-fast: false
 | 
					 | 
				
			||||||
      matrix:
 | 
					      matrix:
 | 
				
			||||||
        image:
 | 
					        arch: [amd64, armv7, aarch64]
 | 
				
			||||||
          - title: "ha-addon"
 | 
					        build_type: ["ha-addon", "docker", "lint"]
 | 
				
			||||||
            suffix: "hassio"
 | 
					 | 
				
			||||||
            target: "hassio"
 | 
					 | 
				
			||||||
            baseimg: "hassio"
 | 
					 | 
				
			||||||
          - title: "docker"
 | 
					 | 
				
			||||||
            suffix: ""
 | 
					 | 
				
			||||||
            target: "docker"
 | 
					 | 
				
			||||||
            baseimg: "docker"
 | 
					 | 
				
			||||||
          - title: "lint"
 | 
					 | 
				
			||||||
            suffix: "lint"
 | 
					 | 
				
			||||||
            target: "lint"
 | 
					 | 
				
			||||||
            baseimg: "docker"
 | 
					 | 
				
			||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
      - uses: actions/checkout@v3
 | 
					      - uses: actions/checkout@v3
 | 
				
			||||||
      - name: Set up Python
 | 
					      - name: Set up Python
 | 
				
			||||||
@@ -109,47 +92,69 @@ jobs:
 | 
				
			|||||||
          username: ${{ github.actor }}
 | 
					          username: ${{ github.actor }}
 | 
				
			||||||
          password: ${{ secrets.GITHUB_TOKEN }}
 | 
					          password: ${{ secrets.GITHUB_TOKEN }}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      - name: Generate short tags
 | 
					 | 
				
			||||||
        id: tags
 | 
					 | 
				
			||||||
        run: |
 | 
					 | 
				
			||||||
          docker/generate_tags.py \
 | 
					 | 
				
			||||||
            --tag "${{ needs.init.outputs.tag }}" \
 | 
					 | 
				
			||||||
            --suffix "${{ matrix.image.suffix }}"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      - name: Build and push
 | 
					      - name: Build and push
 | 
				
			||||||
        uses: docker/build-push-action@v3
 | 
					        run: |
 | 
				
			||||||
 | 
					          docker/build.py \
 | 
				
			||||||
 | 
					            --tag "${{ needs.init.outputs.tag }}" \
 | 
				
			||||||
 | 
					            --arch "${{ matrix.arch }}" \
 | 
				
			||||||
 | 
					            --build-type "${{ matrix.build_type }}" \
 | 
				
			||||||
 | 
					            build \
 | 
				
			||||||
 | 
					            --push
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  deploy-docker-manifest:
 | 
				
			||||||
 | 
					    if: github.repository == 'esphome/esphome'
 | 
				
			||||||
 | 
					    permissions:
 | 
				
			||||||
 | 
					      contents: read
 | 
				
			||||||
 | 
					      packages: write
 | 
				
			||||||
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
 | 
					    needs: [init, deploy-docker]
 | 
				
			||||||
 | 
					    strategy:
 | 
				
			||||||
 | 
					      matrix:
 | 
				
			||||||
 | 
					        build_type: ["ha-addon", "docker", "lint"]
 | 
				
			||||||
 | 
					    steps:
 | 
				
			||||||
 | 
					      - uses: actions/checkout@v3
 | 
				
			||||||
 | 
					      - name: Set up Python
 | 
				
			||||||
 | 
					        uses: actions/setup-python@v4
 | 
				
			||||||
        with:
 | 
					        with:
 | 
				
			||||||
          context: .
 | 
					          python-version: "3.9"
 | 
				
			||||||
          file: ./docker/Dockerfile
 | 
					      - name: Enable experimental manifest support
 | 
				
			||||||
          platforms: linux/amd64,linux/arm/v7,linux/arm64
 | 
					        run: |
 | 
				
			||||||
          target: ${{ matrix.image.target }}
 | 
					          mkdir -p ~/.docker
 | 
				
			||||||
          push: true
 | 
					          echo "{\"experimental\": \"enabled\"}" > ~/.docker/config.json
 | 
				
			||||||
          # yamllint disable rule:line-length
 | 
					
 | 
				
			||||||
          cache-from: type=registry,ref=ghcr.io/${{ steps.tags.outputs.image }}:cache-${{ steps.tags.outputs.channel }}
 | 
					      - name: Log in to docker hub
 | 
				
			||||||
          cache-to: type=registry,ref=ghcr.io/${{ steps.tags.outputs.image }}:cache-${{ steps.tags.outputs.channel }},mode=max
 | 
					        uses: docker/login-action@v2
 | 
				
			||||||
          # yamllint enable rule:line-length
 | 
					        with:
 | 
				
			||||||
          tags: ${{ steps.tags.outputs.tags }}
 | 
					          username: ${{ secrets.DOCKER_USER }}
 | 
				
			||||||
          build-args: |
 | 
					          password: ${{ secrets.DOCKER_PASSWORD }}
 | 
				
			||||||
            BASEIMGTYPE=${{ matrix.image.baseimg }}
 | 
					      - name: Log in to the GitHub container registry
 | 
				
			||||||
            BUILD_VERSION=${{ needs.init.outputs.tag }}
 | 
					        uses: docker/login-action@v2
 | 
				
			||||||
 | 
					        with:
 | 
				
			||||||
 | 
					          registry: ghcr.io
 | 
				
			||||||
 | 
					          username: ${{ github.actor }}
 | 
				
			||||||
 | 
					          password: ${{ secrets.GITHUB_TOKEN }}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      - name: Run manifest
 | 
				
			||||||
 | 
					        run: |
 | 
				
			||||||
 | 
					          docker/build.py \
 | 
				
			||||||
 | 
					            --tag "${{ needs.init.outputs.tag }}" \
 | 
				
			||||||
 | 
					            --build-type "${{ matrix.build_type }}" \
 | 
				
			||||||
 | 
					            manifest
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  deploy-ha-addon-repo:
 | 
					  deploy-ha-addon-repo:
 | 
				
			||||||
    if: github.repository == 'esphome/esphome' && github.event_name == 'release'
 | 
					    if: github.repository == 'esphome/esphome' && github.event_name == 'release'
 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    needs: [deploy-docker]
 | 
					    needs: [deploy-docker]
 | 
				
			||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
      - name: Trigger Workflow
 | 
					      - env:
 | 
				
			||||||
        uses: actions/github-script@v6
 | 
					          TOKEN: ${{ secrets.DEPLOY_HA_ADDON_REPO_TOKEN }}
 | 
				
			||||||
        with:
 | 
					        # yamllint disable rule:line-length
 | 
				
			||||||
          github-token: ${{ secrets.DEPLOY_HA_ADDON_REPO_TOKEN }}
 | 
					        run: |
 | 
				
			||||||
          script: |
 | 
					          TAG="${GITHUB_REF#refs/tags/}"
 | 
				
			||||||
            github.rest.actions.createWorkflowDispatch({
 | 
					          curl \
 | 
				
			||||||
              owner: "esphome",
 | 
					            -u ":$TOKEN" \
 | 
				
			||||||
              repo: "home-assistant-addon",
 | 
					            -X POST \
 | 
				
			||||||
              workflow_id: "bump-version.yml",
 | 
					            -H "Accept: application/vnd.github.v3+json" \
 | 
				
			||||||
              ref: "main",
 | 
					            https://api.github.com/repos/esphome/home-assistant-addon/actions/workflows/bump-version.yml/dispatches \
 | 
				
			||||||
              inputs: {
 | 
					            -d "{\"ref\":\"main\",\"inputs\":{\"version\":\"$TAG\"}}"
 | 
				
			||||||
                version: "${{ github.event.release.tag_name }}",
 | 
					        # yamllint enable rule:line-length
 | 
				
			||||||
                content: ${{ toJSON(github.event.release.body) }}
 | 
					 | 
				
			||||||
              }
 | 
					 | 
				
			||||||
            })
 | 
					 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										4
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.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@v6
 | 
				
			||||||
        with:
 | 
					        with:
 | 
				
			||||||
          days-before-pr-stale: 90
 | 
					          days-before-pr-stale: 90
 | 
				
			||||||
          days-before-pr-close: 7
 | 
					          days-before-pr-close: 7
 | 
				
			||||||
@@ -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@v6
 | 
				
			||||||
        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@v4
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          commit-message: "Synchronise Device Classes from Home Assistant"
 | 
					 | 
				
			||||||
          committer: esphomebot <esphome@nabucasa.com>
 | 
					 | 
				
			||||||
          author: esphomebot <esphome@nabucasa.com>
 | 
					 | 
				
			||||||
          branch: sync/device-classes/
 | 
					 | 
				
			||||||
          branch-suffix: timestamp
 | 
					 | 
				
			||||||
          delete-branch: true
 | 
					 | 
				
			||||||
          title: "Synchronise Device Classes from Home Assistant"
 | 
					 | 
				
			||||||
          body: ${{ steps.pr-template-body.outputs.body }}
 | 
					 | 
				
			||||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -128,5 +128,3 @@ tests/.esphome/
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
sdkconfig.*
 | 
					sdkconfig.*
 | 
				
			||||||
!sdkconfig.defaults
 | 
					!sdkconfig.defaults
 | 
				
			||||||
 | 
					 | 
				
			||||||
.tests/
 | 
					 | 
				
			||||||
@@ -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: 22.10.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.3.1
 | 
					    rev: v3.3.0
 | 
				
			||||||
    hooks:
 | 
					    hooks:
 | 
				
			||||||
      - id: pyupgrade
 | 
					      - id: pyupgrade
 | 
				
			||||||
        args: [--py39-plus]
 | 
					        args: [--py39-plus]
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										15
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										15
									
								
								.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
									
								
								CODEOWNERS
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								CODEOWNERS
									
									
									
									
									
								
							@@ -11,7 +11,6 @@ esphome/*.py @esphome/core
 | 
				
			|||||||
esphome/core/* @esphome/core
 | 
					esphome/core/* @esphome/core
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# Integrations
 | 
					# Integrations
 | 
				
			||||||
esphome/components/absolute_humidity/* @DAVe3283
 | 
					 | 
				
			||||||
esphome/components/ac_dimmer/* @glmnet
 | 
					esphome/components/ac_dimmer/* @glmnet
 | 
				
			||||||
esphome/components/adc/* @esphome/core
 | 
					esphome/components/adc/* @esphome/core
 | 
				
			||||||
esphome/components/adc128s102/* @DeerMaximum
 | 
					esphome/components/adc128s102/* @DeerMaximum
 | 
				
			||||||
@@ -25,7 +24,6 @@ esphome/components/analog_threshold/* @ianchi
 | 
				
			|||||||
esphome/components/animation/* @syndlex
 | 
					esphome/components/animation/* @syndlex
 | 
				
			||||||
esphome/components/anova/* @buxtronix
 | 
					esphome/components/anova/* @buxtronix
 | 
				
			||||||
esphome/components/api/* @OttoWinter
 | 
					esphome/components/api/* @OttoWinter
 | 
				
			||||||
esphome/components/as7341/* @mrgnr
 | 
					 | 
				
			||||||
esphome/components/async_tcp/* @OttoWinter
 | 
					esphome/components/async_tcp/* @OttoWinter
 | 
				
			||||||
esphome/components/atc_mithermometer/* @ahpohl
 | 
					esphome/components/atc_mithermometer/* @ahpohl
 | 
				
			||||||
esphome/components/b_parasite/* @rbaron
 | 
					esphome/components/b_parasite/* @rbaron
 | 
				
			||||||
@@ -43,8 +41,6 @@ esphome/components/ble_client/* @buxtronix
 | 
				
			|||||||
esphome/components/bluetooth_proxy/* @jesserockz
 | 
					esphome/components/bluetooth_proxy/* @jesserockz
 | 
				
			||||||
esphome/components/bme680_bsec/* @trvrnrth
 | 
					esphome/components/bme680_bsec/* @trvrnrth
 | 
				
			||||||
esphome/components/bmp3xx/* @martgras
 | 
					esphome/components/bmp3xx/* @martgras
 | 
				
			||||||
esphome/components/bp1658cj/* @Cossid
 | 
					 | 
				
			||||||
esphome/components/bp5758d/* @Cossid
 | 
					 | 
				
			||||||
esphome/components/button/* @esphome/core
 | 
					esphome/components/button/* @esphome/core
 | 
				
			||||||
esphome/components/canbus/* @danielschramm @mvturnho
 | 
					esphome/components/canbus/* @danielschramm @mvturnho
 | 
				
			||||||
esphome/components/cap1188/* @MrEditor97
 | 
					esphome/components/cap1188/* @MrEditor97
 | 
				
			||||||
@@ -73,7 +69,6 @@ esphome/components/display_menu_base/* @numo68
 | 
				
			|||||||
esphome/components/dps310/* @kbx81
 | 
					esphome/components/dps310/* @kbx81
 | 
				
			||||||
esphome/components/ds1307/* @badbadc0ffee
 | 
					esphome/components/ds1307/* @badbadc0ffee
 | 
				
			||||||
esphome/components/dsmr/* @glmnet @zuidwijk
 | 
					esphome/components/dsmr/* @glmnet @zuidwijk
 | 
				
			||||||
esphome/components/ee895/* @Stock-M
 | 
					 | 
				
			||||||
esphome/components/ektf2232/* @jesserockz
 | 
					esphome/components/ektf2232/* @jesserockz
 | 
				
			||||||
esphome/components/ens210/* @itn3rd77
 | 
					esphome/components/ens210/* @itn3rd77
 | 
				
			||||||
esphome/components/esp32/* @esphome/core
 | 
					esphome/components/esp32/* @esphome/core
 | 
				
			||||||
@@ -92,13 +87,11 @@ esphome/components/factory_reset/* @anatoly-savchenkov
 | 
				
			|||||||
esphome/components/fastled_base/* @OttoWinter
 | 
					esphome/components/fastled_base/* @OttoWinter
 | 
				
			||||||
esphome/components/feedback/* @ianchi
 | 
					esphome/components/feedback/* @ianchi
 | 
				
			||||||
esphome/components/fingerprint_grow/* @OnFreund @loongyh
 | 
					esphome/components/fingerprint_grow/* @OnFreund @loongyh
 | 
				
			||||||
esphome/components/fs3000/* @kahrendt
 | 
					 | 
				
			||||||
esphome/components/globals/* @esphome/core
 | 
					esphome/components/globals/* @esphome/core
 | 
				
			||||||
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
 | 
				
			||||||
esphome/components/growatt_solar/* @leeuwte
 | 
					esphome/components/growatt_solar/* @leeuwte
 | 
				
			||||||
esphome/components/haier/* @Yarikx
 | 
					 | 
				
			||||||
esphome/components/havells_solar/* @sourabhjaiswal
 | 
					esphome/components/havells_solar/* @sourabhjaiswal
 | 
				
			||||||
esphome/components/hbridge/fan/* @WeekendWarrior
 | 
					esphome/components/hbridge/fan/* @WeekendWarrior
 | 
				
			||||||
esphome/components/hbridge/light/* @DotNetDann
 | 
					esphome/components/hbridge/light/* @DotNetDann
 | 
				
			||||||
@@ -107,35 +100,24 @@ esphome/components/hitachi_ac424/* @sourabhjaiswal
 | 
				
			|||||||
esphome/components/homeassistant/* @OttoWinter
 | 
					esphome/components/homeassistant/* @OttoWinter
 | 
				
			||||||
esphome/components/honeywellabp/* @RubyBailey
 | 
					esphome/components/honeywellabp/* @RubyBailey
 | 
				
			||||||
esphome/components/hrxl_maxsonar_wr/* @netmikey
 | 
					esphome/components/hrxl_maxsonar_wr/* @netmikey
 | 
				
			||||||
esphome/components/hte501/* @Stock-M
 | 
					 | 
				
			||||||
esphome/components/hydreon_rgxx/* @functionpointer
 | 
					esphome/components/hydreon_rgxx/* @functionpointer
 | 
				
			||||||
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/ili9xxx/* @nielsnl68
 | 
					 | 
				
			||||||
esphome/components/improv_base/* @esphome/core
 | 
					 | 
				
			||||||
esphome/components/improv_serial/* @esphome/core
 | 
					esphome/components/improv_serial/* @esphome/core
 | 
				
			||||||
esphome/components/ina260/* @MrEditor97
 | 
					esphome/components/ina260/* @MrEditor97
 | 
				
			||||||
esphome/components/inkbird_ibsth1_mini/* @fkirill
 | 
					esphome/components/inkbird_ibsth1_mini/* @fkirill
 | 
				
			||||||
esphome/components/inkplate6/* @jesserockz
 | 
					esphome/components/inkplate6/* @jesserockz
 | 
				
			||||||
esphome/components/integration/* @OttoWinter
 | 
					esphome/components/integration/* @OttoWinter
 | 
				
			||||||
esphome/components/internal_temperature/* @Mat931
 | 
					 | 
				
			||||||
esphome/components/interval/* @esphome/core
 | 
					esphome/components/interval/* @esphome/core
 | 
				
			||||||
esphome/components/json/* @OttoWinter
 | 
					esphome/components/json/* @OttoWinter
 | 
				
			||||||
esphome/components/kalman_combinator/* @Cat-Ion
 | 
					esphome/components/kalman_combinator/* @Cat-Ion
 | 
				
			||||||
esphome/components/key_collector/* @ssieb
 | 
					 | 
				
			||||||
esphome/components/key_provider/* @ssieb
 | 
					 | 
				
			||||||
esphome/components/kuntze/* @ssieb
 | 
					 | 
				
			||||||
esphome/components/lcd_menu/* @numo68
 | 
					esphome/components/lcd_menu/* @numo68
 | 
				
			||||||
esphome/components/ld2410/* @sebcaps
 | 
					 | 
				
			||||||
esphome/components/ledc/* @OttoWinter
 | 
					esphome/components/ledc/* @OttoWinter
 | 
				
			||||||
esphome/components/light/* @esphome/core
 | 
					esphome/components/light/* @esphome/core
 | 
				
			||||||
esphome/components/lilygo_t5_47/touchscreen/* @jesserockz
 | 
					esphome/components/lilygo_t5_47/touchscreen/* @jesserockz
 | 
				
			||||||
esphome/components/lock/* @esphome/core
 | 
					esphome/components/lock/* @esphome/core
 | 
				
			||||||
esphome/components/logger/* @esphome/core
 | 
					esphome/components/logger/* @esphome/core
 | 
				
			||||||
esphome/components/ltr390/* @sjtrny
 | 
					esphome/components/ltr390/* @sjtrny
 | 
				
			||||||
esphome/components/matrix_keypad/* @ssieb
 | 
					 | 
				
			||||||
esphome/components/max31865/* @DAVe3283
 | 
					esphome/components/max31865/* @DAVe3283
 | 
				
			||||||
esphome/components/max44009/* @berfenger
 | 
					esphome/components/max44009/* @berfenger
 | 
				
			||||||
esphome/components/max7219digit/* @rspaargaren
 | 
					esphome/components/max7219digit/* @rspaargaren
 | 
				
			||||||
@@ -156,13 +138,10 @@ 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/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/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
 | 
				
			||||||
@@ -171,9 +150,8 @@ esphome/components/modbus_controller/select/* @martgras @stegm
 | 
				
			|||||||
esphome/components/modbus_controller/sensor/* @martgras
 | 
					esphome/components/modbus_controller/sensor/* @martgras
 | 
				
			||||||
esphome/components/modbus_controller/switch/* @martgras
 | 
					esphome/components/modbus_controller/switch/* @martgras
 | 
				
			||||||
esphome/components/modbus_controller/text_sensor/* @martgras
 | 
					esphome/components/modbus_controller/text_sensor/* @martgras
 | 
				
			||||||
esphome/components/mopeka_ble/* @Fabian-Schmidt @spbrogan
 | 
					esphome/components/mopeka_ble/* @spbrogan
 | 
				
			||||||
esphome/components/mopeka_pro_check/* @spbrogan
 | 
					esphome/components/mopeka_pro_check/* @spbrogan
 | 
				
			||||||
esphome/components/mopeka_std_check/* @Fabian-Schmidt
 | 
					 | 
				
			||||||
esphome/components/mpl3115a2/* @kbickar
 | 
					esphome/components/mpl3115a2/* @kbickar
 | 
				
			||||||
esphome/components/mpu6886/* @fabaff
 | 
					esphome/components/mpu6886/* @fabaff
 | 
				
			||||||
esphome/components/network/* @esphome/core
 | 
					esphome/components/network/* @esphome/core
 | 
				
			||||||
@@ -186,8 +164,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/pca9554/* @hwstar
 | 
					 | 
				
			||||||
esphome/components/pcf85063/* @brogon
 | 
					 | 
				
			||||||
esphome/components/pid/* @OttoWinter
 | 
					esphome/components/pid/* @OttoWinter
 | 
				
			||||||
esphome/components/pipsolar/* @andreashergert1984
 | 
					esphome/components/pipsolar/* @andreashergert1984
 | 
				
			||||||
esphome/components/pm1006/* @habbie
 | 
					esphome/components/pm1006/* @habbie
 | 
				
			||||||
@@ -220,7 +196,6 @@ esphome/components/sdm_meter/* @jesserockz @polyfaces
 | 
				
			|||||||
esphome/components/sdp3x/* @Azimath
 | 
					esphome/components/sdp3x/* @Azimath
 | 
				
			||||||
esphome/components/selec_meter/* @sourabhjaiswal
 | 
					esphome/components/selec_meter/* @sourabhjaiswal
 | 
				
			||||||
esphome/components/select/* @esphome/core
 | 
					esphome/components/select/* @esphome/core
 | 
				
			||||||
esphome/components/sen21231/* @shreyaskarnik
 | 
					 | 
				
			||||||
esphome/components/sen5x/* @martgras
 | 
					esphome/components/sen5x/* @martgras
 | 
				
			||||||
esphome/components/sensirion_common/* @martgras
 | 
					esphome/components/sensirion_common/* @martgras
 | 
				
			||||||
esphome/components/sensor/* @esphome/core
 | 
					esphome/components/sensor/* @esphome/core
 | 
				
			||||||
@@ -229,12 +204,8 @@ esphome/components/sgp4x/* @SenexCrenshaw @martgras
 | 
				
			|||||||
esphome/components/shelly_dimmer/* @edge90 @rnauber
 | 
					esphome/components/shelly_dimmer/* @edge90 @rnauber
 | 
				
			||||||
esphome/components/sht4x/* @sjtrny
 | 
					esphome/components/sht4x/* @sjtrny
 | 
				
			||||||
esphome/components/shutdown/* @esphome/core @jsuanet
 | 
					esphome/components/shutdown/* @esphome/core @jsuanet
 | 
				
			||||||
esphome/components/sigma_delta_output/* @Cat-Ion
 | 
					 | 
				
			||||||
esphome/components/sim800l/* @glmnet
 | 
					esphome/components/sim800l/* @glmnet
 | 
				
			||||||
esphome/components/sm10bit_base/* @Cossid
 | 
					 | 
				
			||||||
esphome/components/sm2135/* @BoukeHaarsma23
 | 
					esphome/components/sm2135/* @BoukeHaarsma23
 | 
				
			||||||
esphome/components/sm2235/* @Cossid
 | 
					 | 
				
			||||||
esphome/components/sm2335/* @Cossid
 | 
					 | 
				
			||||||
esphome/components/sml/* @alengwenus
 | 
					esphome/components/sml/* @alengwenus
 | 
				
			||||||
esphome/components/smt100/* @piechade
 | 
					esphome/components/smt100/* @piechade
 | 
				
			||||||
esphome/components/sn74hc165/* @jesserockz
 | 
					esphome/components/sn74hc165/* @jesserockz
 | 
				
			||||||
@@ -263,7 +234,6 @@ esphome/components/switch/* @esphome/core
 | 
				
			|||||||
esphome/components/t6615/* @tylermenezes
 | 
					esphome/components/t6615/* @tylermenezes
 | 
				
			||||||
esphome/components/tca9548a/* @andreashergert1984
 | 
					esphome/components/tca9548a/* @andreashergert1984
 | 
				
			||||||
esphome/components/tcl112/* @glmnet
 | 
					esphome/components/tcl112/* @glmnet
 | 
				
			||||||
esphome/components/tee501/* @Stock-M
 | 
					 | 
				
			||||||
esphome/components/teleinfo/* @0hax
 | 
					esphome/components/teleinfo/* @0hax
 | 
				
			||||||
esphome/components/thermostat/* @kbx81
 | 
					esphome/components/thermostat/* @kbx81
 | 
				
			||||||
esphome/components/time/* @OttoWinter
 | 
					esphome/components/time/* @OttoWinter
 | 
				
			||||||
@@ -288,16 +258,12 @@ esphome/components/uart/* @esphome/core
 | 
				
			|||||||
esphome/components/ufire_ec/* @pvizeli
 | 
					esphome/components/ufire_ec/* @pvizeli
 | 
				
			||||||
esphome/components/ufire_ise/* @pvizeli
 | 
					esphome/components/ufire_ise/* @pvizeli
 | 
				
			||||||
esphome/components/ultrasonic/* @OttoWinter
 | 
					esphome/components/ultrasonic/* @OttoWinter
 | 
				
			||||||
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
 | 
				
			||||||
esphome/components/whynter/* @aeonsablaze
 | 
					esphome/components/whynter/* @aeonsablaze
 | 
				
			||||||
esphome/components/wiegand/* @ssieb
 | 
					 | 
				
			||||||
esphome/components/wl_134/* @hobbypunk90
 | 
					esphome/components/wl_134/* @hobbypunk90
 | 
				
			||||||
esphome/components/x9c/* @EtienneMD
 | 
					 | 
				
			||||||
esphome/components/xiaomi_lywsd03mmc/* @ahpohl
 | 
					esphome/components/xiaomi_lywsd03mmc/* @ahpohl
 | 
				
			||||||
esphome/components/xiaomi_mhoc303/* @drug123
 | 
					esphome/components/xiaomi_mhoc303/* @drug123
 | 
				
			||||||
esphome/components/xiaomi_mhoc401/* @vevsvevs
 | 
					esphome/components/xiaomi_mhoc401/* @vevsvevs
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,6 +1,8 @@
 | 
				
			|||||||
include LICENSE
 | 
					include LICENSE
 | 
				
			||||||
include README.md
 | 
					include README.md
 | 
				
			||||||
include requirements.txt
 | 
					include requirements.txt
 | 
				
			||||||
recursive-include esphome *.cpp *.h *.tcc *.c
 | 
					include esphome/dashboard/templates/*.html
 | 
				
			||||||
 | 
					recursive-include esphome/dashboard/static *.ico *.js *.css *.woff* LICENSE
 | 
				
			||||||
 | 
					recursive-include esphome *.cpp *.h *.tcc
 | 
				
			||||||
recursive-include esphome *.py.script
 | 
					recursive-include esphome *.py.script
 | 
				
			||||||
recursive-include esphome LICENSE.txt
 | 
					recursive-include esphome LICENSE.txt
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -6,9 +6,9 @@
 | 
				
			|||||||
ARG BASEIMGTYPE=docker
 | 
					ARG BASEIMGTYPE=docker
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# https://github.com/hassio-addons/addon-debian-base/releases
 | 
					# https://github.com/hassio-addons/addon-debian-base/releases
 | 
				
			||||||
FROM ghcr.io/hassio-addons/debian-base:6.2.3 AS base-hassio
 | 
					FROM ghcr.io/hassio-addons/debian-base:6.1.3 AS base-hassio
 | 
				
			||||||
# https://hub.docker.com/_/debian?tab=tags&page=1&name=bullseye
 | 
					# https://hub.docker.com/_/debian?tab=tags&page=1&name=bullseye
 | 
				
			||||||
FROM debian:bullseye-20230208-slim AS base-docker
 | 
					FROM debian:bullseye-20221024-slim AS base-docker
 | 
				
			||||||
 | 
					
 | 
				
			||||||
FROM base-${BASEIMGTYPE} AS base
 | 
					FROM base-${BASEIMGTYPE} AS base
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -26,7 +26,7 @@ RUN \
 | 
				
			|||||||
        python3-cryptography=3.3.2-1 \
 | 
					        python3-cryptography=3.3.2-1 \
 | 
				
			||||||
        iputils-ping=3:20210202-1 \
 | 
					        iputils-ping=3:20210202-1 \
 | 
				
			||||||
        git=1:2.30.2-1 \
 | 
					        git=1:2.30.2-1 \
 | 
				
			||||||
        curl=7.74.0-1.3+deb11u7 \
 | 
					        curl=7.74.0-1.3+deb11u3 \
 | 
				
			||||||
        openssh-client=1:8.4p1-5+deb11u1 \
 | 
					        openssh-client=1:8.4p1-5+deb11u1 \
 | 
				
			||||||
    && rm -rf \
 | 
					    && rm -rf \
 | 
				
			||||||
        /tmp/* \
 | 
					        /tmp/* \
 | 
				
			||||||
@@ -51,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.6 \
 | 
					        platformio==6.1.5 \
 | 
				
			||||||
    # 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 \
 | 
				
			||||||
@@ -135,11 +135,11 @@ RUN \
 | 
				
			|||||||
    apt-get update \
 | 
					    apt-get update \
 | 
				
			||||||
    # Use pinned versions so that we get updates with build caching
 | 
					    # Use pinned versions so that we get updates with build caching
 | 
				
			||||||
    && apt-get install -y --no-install-recommends \
 | 
					    && apt-get install -y --no-install-recommends \
 | 
				
			||||||
        clang-format-13=1:13.0.1-6~deb11u1 \
 | 
					        clang-format-11=1:11.0.1-2 \
 | 
				
			||||||
        clang-tidy-11=1:11.0.1-2 \
 | 
					        clang-tidy-11=1:11.0.1-2 \
 | 
				
			||||||
        patch=2.7.6-7 \
 | 
					        patch=2.7.6-7 \
 | 
				
			||||||
        software-properties-common=0.96.20.2-2.1 \
 | 
					        software-properties-common=0.96.20.2-2.1 \
 | 
				
			||||||
        nano=5.4-2+deb11u2 \
 | 
					        nano=5.4-2+deb11u1 \
 | 
				
			||||||
        build-essential=12.9 \
 | 
					        build-essential=12.9 \
 | 
				
			||||||
        python3-dev=3.9.2-3 \
 | 
					        python3-dev=3.9.2-3 \
 | 
				
			||||||
    && rm -rf \
 | 
					    && rm -rf \
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -8,49 +8,32 @@ import re
 | 
				
			|||||||
import sys
 | 
					import sys
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
CHANNEL_DEV = "dev"
 | 
					CHANNEL_DEV = 'dev'
 | 
				
			||||||
CHANNEL_BETA = "beta"
 | 
					CHANNEL_BETA = 'beta'
 | 
				
			||||||
CHANNEL_RELEASE = "release"
 | 
					CHANNEL_RELEASE = 'release'
 | 
				
			||||||
CHANNELS = [CHANNEL_DEV, CHANNEL_BETA, CHANNEL_RELEASE]
 | 
					CHANNELS = [CHANNEL_DEV, CHANNEL_BETA, CHANNEL_RELEASE]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
ARCH_AMD64 = "amd64"
 | 
					ARCH_AMD64 = 'amd64'
 | 
				
			||||||
ARCH_ARMV7 = "armv7"
 | 
					ARCH_ARMV7 = 'armv7'
 | 
				
			||||||
ARCH_AARCH64 = "aarch64"
 | 
					ARCH_AARCH64 = 'aarch64'
 | 
				
			||||||
ARCHS = [ARCH_AMD64, ARCH_ARMV7, ARCH_AARCH64]
 | 
					ARCHS = [ARCH_AMD64, ARCH_ARMV7, ARCH_AARCH64]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
TYPE_DOCKER = "docker"
 | 
					TYPE_DOCKER = 'docker'
 | 
				
			||||||
TYPE_HA_ADDON = "ha-addon"
 | 
					TYPE_HA_ADDON = 'ha-addon'
 | 
				
			||||||
TYPE_LINT = "lint"
 | 
					TYPE_LINT = 'lint'
 | 
				
			||||||
TYPES = [TYPE_DOCKER, TYPE_HA_ADDON, TYPE_LINT]
 | 
					TYPES = [TYPE_DOCKER, TYPE_HA_ADDON, TYPE_LINT]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
parser = argparse.ArgumentParser()
 | 
					parser = argparse.ArgumentParser()
 | 
				
			||||||
parser.add_argument(
 | 
					parser.add_argument("--tag", type=str, required=True, help="The main docker tag to push to. If a version number also adds latest and/or beta tag")
 | 
				
			||||||
    "--tag",
 | 
					parser.add_argument("--arch", choices=ARCHS, required=False, help="The architecture to build for")
 | 
				
			||||||
    type=str,
 | 
					parser.add_argument("--build-type", choices=TYPES, required=True, help="The type of build to run")
 | 
				
			||||||
    required=True,
 | 
					parser.add_argument("--dry-run", action="store_true", help="Don't run any commands, just print them")
 | 
				
			||||||
    help="The main docker tag to push to. If a version number also adds latest and/or beta tag",
 | 
					subparsers = parser.add_subparsers(help="Action to perform", dest="command", required=True)
 | 
				
			||||||
)
 | 
					 | 
				
			||||||
parser.add_argument(
 | 
					 | 
				
			||||||
    "--arch", choices=ARCHS, required=False, help="The architecture to build for"
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
parser.add_argument(
 | 
					 | 
				
			||||||
    "--build-type", choices=TYPES, required=True, help="The type of build to run"
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
parser.add_argument(
 | 
					 | 
				
			||||||
    "--dry-run", action="store_true", help="Don't run any commands, just print them"
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
subparsers = parser.add_subparsers(
 | 
					 | 
				
			||||||
    help="Action to perform", dest="command", required=True
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
build_parser = subparsers.add_parser("build", help="Build the image")
 | 
					build_parser = subparsers.add_parser("build", help="Build the image")
 | 
				
			||||||
build_parser.add_argument("--push", help="Also push the images", action="store_true")
 | 
					build_parser.add_argument("--push", help="Also push the images", action="store_true")
 | 
				
			||||||
build_parser.add_argument(
 | 
					build_parser.add_argument("--load", help="Load the docker image locally", action="store_true")
 | 
				
			||||||
    "--load", help="Load the docker image locally", action="store_true"
 | 
					manifest_parser = subparsers.add_parser("manifest", help="Create a manifest from already pushed images")
 | 
				
			||||||
)
 | 
					 | 
				
			||||||
manifest_parser = subparsers.add_parser(
 | 
					 | 
				
			||||||
    "manifest", help="Create a manifest from already pushed images"
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@dataclass(frozen=True)
 | 
					@dataclass(frozen=True)
 | 
				
			||||||
@@ -66,7 +49,7 @@ class DockerParams:
 | 
				
			|||||||
        prefix = {
 | 
					        prefix = {
 | 
				
			||||||
            TYPE_DOCKER: "esphome/esphome",
 | 
					            TYPE_DOCKER: "esphome/esphome",
 | 
				
			||||||
            TYPE_HA_ADDON: "esphome/esphome-hassio",
 | 
					            TYPE_HA_ADDON: "esphome/esphome-hassio",
 | 
				
			||||||
            TYPE_LINT: "esphome/esphome-lint",
 | 
					            TYPE_LINT: "esphome/esphome-lint"
 | 
				
			||||||
        }[build_type]
 | 
					        }[build_type]
 | 
				
			||||||
        build_to = f"{prefix}-{arch}"
 | 
					        build_to = f"{prefix}-{arch}"
 | 
				
			||||||
        baseimgtype = {
 | 
					        baseimgtype = {
 | 
				
			||||||
@@ -145,21 +128,13 @@ def main():
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        # 3. build
 | 
					        # 3. build
 | 
				
			||||||
        cmd = [
 | 
					        cmd = [
 | 
				
			||||||
            "docker",
 | 
					            "docker", "buildx", "build",
 | 
				
			||||||
            "buildx",
 | 
					            "--build-arg", f"BASEIMGTYPE={params.baseimgtype}",
 | 
				
			||||||
            "build",
 | 
					            "--build-arg", f"BUILD_VERSION={args.tag}",
 | 
				
			||||||
            "--build-arg",
 | 
					            "--cache-from", f"type=registry,ref={cache_img}",
 | 
				
			||||||
            f"BASEIMGTYPE={params.baseimgtype}",
 | 
					            "--file", "docker/Dockerfile",
 | 
				
			||||||
            "--build-arg",
 | 
					            "--platform", params.platform,
 | 
				
			||||||
            f"BUILD_VERSION={args.tag}",
 | 
					            "--target", params.target,
 | 
				
			||||||
            "--cache-from",
 | 
					 | 
				
			||||||
            f"type=registry,ref={cache_img}",
 | 
					 | 
				
			||||||
            "--file",
 | 
					 | 
				
			||||||
            "docker/Dockerfile",
 | 
					 | 
				
			||||||
            "--platform",
 | 
					 | 
				
			||||||
            params.platform,
 | 
					 | 
				
			||||||
            "--target",
 | 
					 | 
				
			||||||
            params.target,
 | 
					 | 
				
			||||||
        ]
 | 
					        ]
 | 
				
			||||||
        for img in imgs:
 | 
					        for img in imgs:
 | 
				
			||||||
            cmd += ["--tag", img]
 | 
					            cmd += ["--tag", img]
 | 
				
			||||||
@@ -185,7 +160,9 @@ def main():
 | 
				
			|||||||
            run_command(*cmd)
 | 
					            run_command(*cmd)
 | 
				
			||||||
        # 2. Push manifests
 | 
					        # 2. Push manifests
 | 
				
			||||||
        for target in targets:
 | 
					        for target in targets:
 | 
				
			||||||
            run_command("docker", "manifest", "push", target)
 | 
					            run_command(
 | 
				
			||||||
 | 
					                "docker", "manifest", "push", target
 | 
				
			||||||
 | 
					            )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
if __name__ == "__main__":
 | 
					if __name__ == "__main__":
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,68 +0,0 @@
 | 
				
			|||||||
#!/usr/bin/env python3
 | 
					 | 
				
			||||||
import re
 | 
					 | 
				
			||||||
import os
 | 
					 | 
				
			||||||
import argparse
 | 
					 | 
				
			||||||
import json
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
CHANNEL_DEV = "dev"
 | 
					 | 
				
			||||||
CHANNEL_BETA = "beta"
 | 
					 | 
				
			||||||
CHANNEL_RELEASE = "release"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
parser = argparse.ArgumentParser()
 | 
					 | 
				
			||||||
parser.add_argument(
 | 
					 | 
				
			||||||
    "--tag",
 | 
					 | 
				
			||||||
    type=str,
 | 
					 | 
				
			||||||
    required=True,
 | 
					 | 
				
			||||||
    help="The main docker tag to push to. If a version number also adds latest and/or beta tag",
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
parser.add_argument(
 | 
					 | 
				
			||||||
    "--suffix",
 | 
					 | 
				
			||||||
    type=str,
 | 
					 | 
				
			||||||
    required=True,
 | 
					 | 
				
			||||||
    help="The suffix of the tag.",
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
def main():
 | 
					 | 
				
			||||||
    args = parser.parse_args()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    # detect channel from tag
 | 
					 | 
				
			||||||
    match = re.match(r"^(\d+\.\d+)(?:\.\d+)?(b\d+)?$", args.tag)
 | 
					 | 
				
			||||||
    major_minor_version = None
 | 
					 | 
				
			||||||
    if match is None:
 | 
					 | 
				
			||||||
        channel = CHANNEL_DEV
 | 
					 | 
				
			||||||
    elif match.group(2) is None:
 | 
					 | 
				
			||||||
        major_minor_version = match.group(1)
 | 
					 | 
				
			||||||
        channel = CHANNEL_RELEASE
 | 
					 | 
				
			||||||
    else:
 | 
					 | 
				
			||||||
        channel = CHANNEL_BETA
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    tags_to_push = [args.tag]
 | 
					 | 
				
			||||||
    if channel == CHANNEL_DEV:
 | 
					 | 
				
			||||||
        tags_to_push.append("dev")
 | 
					 | 
				
			||||||
    elif channel == CHANNEL_BETA:
 | 
					 | 
				
			||||||
        tags_to_push.append("beta")
 | 
					 | 
				
			||||||
    elif channel == CHANNEL_RELEASE:
 | 
					 | 
				
			||||||
        # Additionally push to beta
 | 
					 | 
				
			||||||
        tags_to_push.append("beta")
 | 
					 | 
				
			||||||
        tags_to_push.append("latest")
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if major_minor_version:
 | 
					 | 
				
			||||||
            tags_to_push.append("stable")
 | 
					 | 
				
			||||||
            tags_to_push.append(major_minor_version)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    suffix = f"-{args.suffix}" if args.suffix else ""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    with open(os.environ["GITHUB_OUTPUT"], "w") as f:
 | 
					 | 
				
			||||||
        print(f"channel={channel}", file=f)
 | 
					 | 
				
			||||||
        print(f"image=esphome/esphome{suffix}", file=f)
 | 
					 | 
				
			||||||
        full_tags = []
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        for tag in tags_to_push:
 | 
					 | 
				
			||||||
            full_tags += [f"ghcr.io/esphome/esphome{suffix}:{tag}"]
 | 
					 | 
				
			||||||
            full_tags += [f"esphome/esphome{suffix}:{tag}"]
 | 
					 | 
				
			||||||
        print(f"tags={','.join(full_tags)}", file=f)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
if __name__ == "__main__":
 | 
					 | 
				
			||||||
    main()
 | 
					 | 
				
			||||||
							
								
								
									
										41
									
								
								docker/ha-addon-rootfs/etc/cont-init.d/10-requirements.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										41
									
								
								docker/ha-addon-rootfs/etc/cont-init.d/10-requirements.sh
									
									
									
									
									
										Executable file
									
								
							@@ -0,0 +1,41 @@
 | 
				
			|||||||
 | 
					#!/usr/bin/with-contenv bashio
 | 
				
			||||||
 | 
					# ==============================================================================
 | 
				
			||||||
 | 
					# Community Hass.io Add-ons: ESPHome
 | 
				
			||||||
 | 
					# This files check if all user configuration requirements are met
 | 
				
			||||||
 | 
					# ==============================================================================
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					# Check SSL requirements, if enabled
 | 
				
			||||||
 | 
					if bashio::config.true 'ssl'; then
 | 
				
			||||||
 | 
					    if ! bashio::config.has_value 'certfile'; then
 | 
				
			||||||
 | 
					        bashio::log.fatal 'SSL is enabled, but no certfile was specified.'
 | 
				
			||||||
 | 
					        bashio::exit.nok
 | 
				
			||||||
 | 
					    fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if ! bashio::config.has_value 'keyfile'; then
 | 
				
			||||||
 | 
					        bashio::log.fatal 'SSL is enabled, but no keyfile was specified'
 | 
				
			||||||
 | 
					        bashio::exit.nok
 | 
				
			||||||
 | 
					    fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    certfile="/ssl/$(bashio::config 'certfile')"
 | 
				
			||||||
 | 
					    keyfile="/ssl/$(bashio::config 'keyfile')"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if ! bashio::fs.file_exists "${certfile}"; then
 | 
				
			||||||
 | 
					        if ! bashio::fs.file_exists "${keyfile}"; then
 | 
				
			||||||
 | 
					            # Both files are missing, let's print a friendlier error message
 | 
				
			||||||
 | 
					            bashio::log.fatal 'You enabled encrypted connections using the "ssl": true option.'
 | 
				
			||||||
 | 
					            bashio::log.fatal "However, the SSL files '${certfile}' and '${keyfile}'"
 | 
				
			||||||
 | 
					            bashio::log.fatal "were not found. If you're using Hass.io on your local network and don't want"
 | 
				
			||||||
 | 
					            bashio::log.fatal 'to encrypt connections to the ESPHome dashboard, you can manually disable'
 | 
				
			||||||
 | 
					            bashio::log.fatal 'SSL by setting "ssl" to false."'
 | 
				
			||||||
 | 
					            bashio::exit.nok
 | 
				
			||||||
 | 
					        fi
 | 
				
			||||||
 | 
					        bashio::log.fatal "The configured certfile '${certfile}' was not found."
 | 
				
			||||||
 | 
					        bashio::exit.nok
 | 
				
			||||||
 | 
					    fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if ! bashio::fs.file_exists "/ssl/$(bashio::config 'keyfile')"; then
 | 
				
			||||||
 | 
					        bashio::log.fatal "The configured keyfile '${keyfile}' was not found."
 | 
				
			||||||
 | 
					        bashio::exit.nok
 | 
				
			||||||
 | 
					    fi
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
							
								
								
									
										34
									
								
								docker/ha-addon-rootfs/etc/cont-init.d/20-nginx.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										34
									
								
								docker/ha-addon-rootfs/etc/cont-init.d/20-nginx.sh
									
									
									
									
									
										Executable file
									
								
							@@ -0,0 +1,34 @@
 | 
				
			|||||||
 | 
					#!/usr/bin/with-contenv bashio
 | 
				
			||||||
 | 
					# ==============================================================================
 | 
				
			||||||
 | 
					# Community Hass.io Add-ons: ESPHome
 | 
				
			||||||
 | 
					# Configures NGINX for use with ESPHome
 | 
				
			||||||
 | 
					# ==============================================================================
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					declare certfile
 | 
				
			||||||
 | 
					declare keyfile
 | 
				
			||||||
 | 
					declare direct_port
 | 
				
			||||||
 | 
					declare ingress_interface
 | 
				
			||||||
 | 
					declare ingress_port
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					mkdir -p /var/log/nginx
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					direct_port=$(bashio::addon.port 6052)
 | 
				
			||||||
 | 
					if bashio::var.has_value "${direct_port}"; then
 | 
				
			||||||
 | 
					    if bashio::config.true 'ssl'; then
 | 
				
			||||||
 | 
					        certfile=$(bashio::config 'certfile')
 | 
				
			||||||
 | 
					        keyfile=$(bashio::config 'keyfile')
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        mv /etc/nginx/servers/direct-ssl.disabled /etc/nginx/servers/direct.conf
 | 
				
			||||||
 | 
					        sed -i "s/%%certfile%%/${certfile}/g" /etc/nginx/servers/direct.conf
 | 
				
			||||||
 | 
					        sed -i "s/%%keyfile%%/${keyfile}/g" /etc/nginx/servers/direct.conf
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					        mv /etc/nginx/servers/direct.disabled /etc/nginx/servers/direct.conf
 | 
				
			||||||
 | 
					    fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    sed -i "s/%%port%%/${direct_port}/g" /etc/nginx/servers/direct.conf
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					ingress_port=$(bashio::addon.ingress_port)
 | 
				
			||||||
 | 
					ingress_interface=$(bashio::addon.ip_address)
 | 
				
			||||||
 | 
					sed -i "s/%%port%%/${ingress_port}/g" /etc/nginx/servers/ingress.conf
 | 
				
			||||||
 | 
					sed -i "s/%%interface%%/${ingress_interface}/g" /etc/nginx/servers/ingress.conf
 | 
				
			||||||
							
								
								
									
										9
									
								
								docker/ha-addon-rootfs/etc/cont-init.d/30-dirs.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										9
									
								
								docker/ha-addon-rootfs/etc/cont-init.d/30-dirs.sh
									
									
									
									
									
										Executable file
									
								
							@@ -0,0 +1,9 @@
 | 
				
			|||||||
 | 
					#!/usr/bin/with-contenv bashio
 | 
				
			||||||
 | 
					# ==============================================================================
 | 
				
			||||||
 | 
					# Community Hass.io Add-ons: ESPHome
 | 
				
			||||||
 | 
					# This files creates all directories used by esphome
 | 
				
			||||||
 | 
					# ==============================================================================
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					pio_cache_base=/data/cache/platformio
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					mkdir -p "${pio_cache_base}"
 | 
				
			||||||
@@ -1,9 +1,9 @@
 | 
				
			|||||||
proxy_http_version          1.1;
 | 
					proxy_http_version 1.1;
 | 
				
			||||||
proxy_ignore_client_abort   off;
 | 
					proxy_ignore_client_abort off;
 | 
				
			||||||
proxy_read_timeout          86400s;
 | 
					proxy_read_timeout 86400s;
 | 
				
			||||||
proxy_redirect              off;
 | 
					proxy_redirect off;
 | 
				
			||||||
proxy_send_timeout          86400s;
 | 
					proxy_send_timeout 86400s;
 | 
				
			||||||
proxy_max_temp_file_size    0;
 | 
					proxy_max_temp_file_size 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
proxy_set_header Accept-Encoding "";
 | 
					proxy_set_header Accept-Encoding "";
 | 
				
			||||||
proxy_set_header Connection $connection_upgrade;
 | 
					proxy_set_header Connection $connection_upgrade;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,7 +1,5 @@
 | 
				
			|||||||
root            /dev/null;
 | 
					root /dev/null;
 | 
				
			||||||
server_name     $hostname;
 | 
					server_name $hostname;
 | 
				
			||||||
 | 
					 | 
				
			||||||
client_max_body_size 512m;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
add_header X-Content-Type-Options nosniff;
 | 
					add_header X-Content-Type-Options nosniff;
 | 
				
			||||||
add_header X-XSS-Protection "1; mode=block";
 | 
					add_header X-XSS-Protection "1; mode=block";
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,6 +1,7 @@
 | 
				
			|||||||
ssl_protocols TLSv1.2 TLSv1.3;
 | 
					ssl_protocols TLSv1.2;
 | 
				
			||||||
ssl_prefer_server_ciphers off;
 | 
					ssl_prefer_server_ciphers on;
 | 
				
			||||||
ssl_ciphers ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-ECDSA-CHACHA20-POLY1305:ECDHE-RSA-CHACHA20-POLY1305:DHE-RSA-AES128-GCM-SHA256:DHE-RSA-AES256-GCM-SHA384;
 | 
					ssl_ciphers ECDHE-RSA-AES256-GCM-SHA512:DHE-RSA-AES256-GCM-SHA512:ECDHE-RSA-AES256-GCM-SHA384:DHE-RSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-SHA384:ECDHE-RSA-AES256-SHA:DHE-RSA-AES256-SHA;
 | 
				
			||||||
 | 
					ssl_ecdh_curve secp384r1;
 | 
				
			||||||
ssl_session_timeout  10m;
 | 
					ssl_session_timeout  10m;
 | 
				
			||||||
ssl_session_cache shared:SSL:10m;
 | 
					ssl_session_cache shared:SSL:10m;
 | 
				
			||||||
ssl_session_tickets off;
 | 
					ssl_session_tickets off;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,3 +0,0 @@
 | 
				
			|||||||
upstream esphome {
 | 
					 | 
				
			||||||
    server unix:/var/run/esphome.sock;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@@ -2,6 +2,7 @@ daemon off;
 | 
				
			|||||||
user root;
 | 
					user root;
 | 
				
			||||||
pid /var/run/nginx.pid;
 | 
					pid /var/run/nginx.pid;
 | 
				
			||||||
worker_processes 1;
 | 
					worker_processes 1;
 | 
				
			||||||
 | 
					# Hass.io addon log
 | 
				
			||||||
error_log /proc/1/fd/1 error;
 | 
					error_log /proc/1/fd/1 error;
 | 
				
			||||||
events {
 | 
					events {
 | 
				
			||||||
    worker_connections 1024;
 | 
					    worker_connections 1024;
 | 
				
			||||||
@@ -9,22 +10,24 @@ events {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
http {
 | 
					http {
 | 
				
			||||||
    include /etc/nginx/includes/mime.types;
 | 
					    include /etc/nginx/includes/mime.types;
 | 
				
			||||||
 | 
					    access_log stdout;
 | 
				
			||||||
    access_log              off;
 | 
					    default_type application/octet-stream;
 | 
				
			||||||
    default_type            application/octet-stream;
 | 
					    gzip on;
 | 
				
			||||||
    gzip                    on;
 | 
					    keepalive_timeout 65;
 | 
				
			||||||
    keepalive_timeout       65;
 | 
					    sendfile on;
 | 
				
			||||||
    sendfile                on;
 | 
					    server_tokens off;
 | 
				
			||||||
    server_tokens           off;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    tcp_nodelay             on;
 | 
					 | 
				
			||||||
    tcp_nopush              on;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    map $http_upgrade $connection_upgrade {
 | 
					    map $http_upgrade $connection_upgrade {
 | 
				
			||||||
        default upgrade;
 | 
					        default upgrade;
 | 
				
			||||||
        ''      close;
 | 
					        ''      close;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    include /etc/nginx/includes/upstream.conf;
 | 
					    # Use Hass.io supervisor as resolver
 | 
				
			||||||
 | 
					    resolver 172.30.32.2;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    upstream esphome {
 | 
				
			||||||
 | 
					        server unix:/var/run/esphome.sock;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    include /etc/nginx/servers/*.conf;
 | 
					    include /etc/nginx/servers/*.conf;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1 +0,0 @@
 | 
				
			|||||||
Without requirements or design, programming is the art of adding bugs to an empty text file. (Louis Srygley)
 | 
					 | 
				
			||||||
@@ -1,26 +1,20 @@
 | 
				
			|||||||
server {
 | 
					server {
 | 
				
			||||||
    {{ if not .ssl }}
 | 
					    listen %%port%% default_server ssl http2;
 | 
				
			||||||
    listen 6052 default_server;
 | 
					 | 
				
			||||||
    {{ else }}
 | 
					 | 
				
			||||||
    listen 6052 default_server ssl http2;
 | 
					 | 
				
			||||||
    {{ end }}
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    include /etc/nginx/includes/server_params.conf;
 | 
					    include /etc/nginx/includes/server_params.conf;
 | 
				
			||||||
    include /etc/nginx/includes/proxy_params.conf;
 | 
					    include /etc/nginx/includes/proxy_params.conf;
 | 
				
			||||||
 | 
					 | 
				
			||||||
    {{ if .ssl }}
 | 
					 | 
				
			||||||
    include /etc/nginx/includes/ssl_params.conf;
 | 
					    include /etc/nginx/includes/ssl_params.conf;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    ssl_certificate /ssl/{{ .certfile }};
 | 
					    ssl on;
 | 
				
			||||||
    ssl_certificate_key /ssl/{{ .keyfile }};
 | 
					    ssl_certificate /ssl/%%certfile%%;
 | 
				
			||||||
 | 
					    ssl_certificate_key /ssl/%%keyfile%%;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    # Clear Hass.io Ingress header
 | 
				
			||||||
 | 
					    proxy_set_header X-HA-Ingress "";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    # Redirect http requests to https on the same port.
 | 
					    # Redirect http requests to https on the same port.
 | 
				
			||||||
    # https://rageagainstshell.com/2016/11/redirect-http-to-https-on-the-same-port-in-nginx/
 | 
					    # https://rageagainstshell.com/2016/11/redirect-http-to-https-on-the-same-port-in-nginx/
 | 
				
			||||||
    error_page 497 https://$http_host$request_uri;
 | 
					    error_page 497 https://$http_host$request_uri;
 | 
				
			||||||
    {{ end }}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    # Clear Home Assistant Ingress header
 | 
					 | 
				
			||||||
    proxy_set_header X-HA-Ingress "";
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    location / {
 | 
					    location / {
 | 
				
			||||||
        proxy_pass http://esphome;
 | 
					        proxy_pass http://esphome;
 | 
				
			||||||
							
								
								
									
										12
									
								
								docker/ha-addon-rootfs/etc/nginx/servers/direct.disabled
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								docker/ha-addon-rootfs/etc/nginx/servers/direct.disabled
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,12 @@
 | 
				
			|||||||
 | 
					server {
 | 
				
			||||||
 | 
					    listen %%port%% default_server;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    include /etc/nginx/includes/server_params.conf;
 | 
				
			||||||
 | 
					    include /etc/nginx/includes/proxy_params.conf;
 | 
				
			||||||
 | 
					    # Clear Hass.io Ingress header
 | 
				
			||||||
 | 
					    proxy_set_header X-HA-Ingress "";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    location / {
 | 
				
			||||||
 | 
					        proxy_pass http://esphome;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -1,16 +1,14 @@
 | 
				
			|||||||
server {
 | 
					server {
 | 
				
			||||||
    listen 127.0.0.1:{{ .port }} default_server;
 | 
					    listen %%interface%%:%%port%% default_server;
 | 
				
			||||||
    listen {{ .interface }}:{{ .port }} default_server;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    include /etc/nginx/includes/server_params.conf;
 | 
					    include /etc/nginx/includes/server_params.conf;
 | 
				
			||||||
    include /etc/nginx/includes/proxy_params.conf;
 | 
					    include /etc/nginx/includes/proxy_params.conf;
 | 
				
			||||||
 | 
					 | 
				
			||||||
    # Set Home Assistant Ingress header
 | 
					    # Set Home Assistant Ingress header
 | 
				
			||||||
    proxy_set_header X-HA-Ingress "YES";
 | 
					    proxy_set_header X-HA-Ingress "YES";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    location / {
 | 
					    location / {
 | 
				
			||||||
 | 
					        # Only allow from Hass.io supervisor
 | 
				
			||||||
        allow   172.30.32.2;
 | 
					        allow   172.30.32.2;
 | 
				
			||||||
        allow   127.0.0.1;
 | 
					 | 
				
			||||||
        deny    all;
 | 
					        deny    all;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        proxy_pass http://esphome;
 | 
					        proxy_pass http://esphome;
 | 
				
			||||||
@@ -1,32 +0,0 @@
 | 
				
			|||||||
#!/command/with-contenv bashio
 | 
					 | 
				
			||||||
# shellcheck shell=bash
 | 
					 | 
				
			||||||
# ==============================================================================
 | 
					 | 
				
			||||||
# Home Assistant Add-on: ESPHome
 | 
					 | 
				
			||||||
# Sends discovery information to Home Assistant.
 | 
					 | 
				
			||||||
# ==============================================================================
 | 
					 | 
				
			||||||
declare config
 | 
					 | 
				
			||||||
declare port
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# We only disable it when disabled explicitly
 | 
					 | 
				
			||||||
if bashio::config.false 'home_assistant_dashboard_integration';
 | 
					 | 
				
			||||||
then
 | 
					 | 
				
			||||||
    bashio::log.info "Home Assistant discovery is disabled for this add-on."
 | 
					 | 
				
			||||||
    bashio::exit.ok
 | 
					 | 
				
			||||||
fi
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
port=$(bashio::addon.ingress_port)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Wait for NGINX to become available
 | 
					 | 
				
			||||||
bashio::net.wait_for "${port}" "127.0.0.1" 300
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
config=$(\
 | 
					 | 
				
			||||||
    bashio::var.json \
 | 
					 | 
				
			||||||
        host "127.0.0.1" \
 | 
					 | 
				
			||||||
        port "^${port}" \
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
if bashio::discovery "esphome" "${config}" > /dev/null; then
 | 
					 | 
				
			||||||
    bashio::log.info "Successfully send discovery information to Home Assistant."
 | 
					 | 
				
			||||||
else
 | 
					 | 
				
			||||||
    bashio::log.error "Discovery message to Home Assistant failed!"
 | 
					 | 
				
			||||||
fi
 | 
					 | 
				
			||||||
@@ -1 +0,0 @@
 | 
				
			|||||||
oneshot
 | 
					 | 
				
			||||||
@@ -1 +0,0 @@
 | 
				
			|||||||
/etc/s6-overlay/s6-rc.d/discovery/run
 | 
					 | 
				
			||||||
@@ -1,26 +0,0 @@
 | 
				
			|||||||
#!/command/with-contenv bashio
 | 
					 | 
				
			||||||
# shellcheck shell=bash
 | 
					 | 
				
			||||||
# ==============================================================================
 | 
					 | 
				
			||||||
# Home Assistant Community Add-on: ESPHome
 | 
					 | 
				
			||||||
# Take down the S6 supervision tree when ESPHome dashboard fails
 | 
					 | 
				
			||||||
# ==============================================================================
 | 
					 | 
				
			||||||
declare exit_code
 | 
					 | 
				
			||||||
readonly exit_code_container=$(</run/s6-linux-init-container-results/exitcode)
 | 
					 | 
				
			||||||
readonly exit_code_service="${1}"
 | 
					 | 
				
			||||||
readonly exit_code_signal="${2}"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bashio::log.info \
 | 
					 | 
				
			||||||
  "Service ESPHome dashboard exited with code ${exit_code_service}" \
 | 
					 | 
				
			||||||
  "(by signal ${exit_code_signal})"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
if [[ "${exit_code_service}" -eq 256 ]]; then
 | 
					 | 
				
			||||||
  if [[ "${exit_code_container}" -eq 0 ]]; then
 | 
					 | 
				
			||||||
    echo $((128 + $exit_code_signal)) > /run/s6-linux-init-container-results/exitcode
 | 
					 | 
				
			||||||
  fi
 | 
					 | 
				
			||||||
  [[ "${exit_code_signal}" -eq 15 ]] && exec /run/s6/basedir/bin/halt
 | 
					 | 
				
			||||||
elif [[ "${exit_code_service}" -ne 0 ]]; then
 | 
					 | 
				
			||||||
  if [[ "${exit_code_container}" -eq 0 ]]; then
 | 
					 | 
				
			||||||
    echo "${exit_code_service}" > /run/s6-linux-init-container-results/exitcode
 | 
					 | 
				
			||||||
  fi
 | 
					 | 
				
			||||||
  exec /run/s6/basedir/bin/halt
 | 
					 | 
				
			||||||
fi
 | 
					 | 
				
			||||||
@@ -1 +0,0 @@
 | 
				
			|||||||
longrun
 | 
					 | 
				
			||||||
@@ -1,27 +0,0 @@
 | 
				
			|||||||
#!/command/with-contenv bashio
 | 
					 | 
				
			||||||
# shellcheck shell=bash
 | 
					 | 
				
			||||||
# ==============================================================================
 | 
					 | 
				
			||||||
# Community Hass.io Add-ons: ESPHome
 | 
					 | 
				
			||||||
# Configures NGINX for use with ESPHome
 | 
					 | 
				
			||||||
# ==============================================================================
 | 
					 | 
				
			||||||
mkdir -p /var/log/nginx
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Generate Ingress configuration
 | 
					 | 
				
			||||||
bashio::var.json \
 | 
					 | 
				
			||||||
    interface "$(bashio::addon.ip_address)" \
 | 
					 | 
				
			||||||
    port "^$(bashio::addon.ingress_port)" \
 | 
					 | 
				
			||||||
    | tempio \
 | 
					 | 
				
			||||||
        -template /etc/nginx/templates/ingress.gtpl \
 | 
					 | 
				
			||||||
        -out /etc/nginx/servers/ingress.conf
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# Generate direct access configuration, if enabled.
 | 
					 | 
				
			||||||
if bashio::var.has_value "$(bashio::addon.port 6052)"; then
 | 
					 | 
				
			||||||
    bashio::config.require.ssl
 | 
					 | 
				
			||||||
    bashio::var.json \
 | 
					 | 
				
			||||||
        certfile "$(bashio::config 'certfile')" \
 | 
					 | 
				
			||||||
        keyfile "$(bashio::config 'keyfile')" \
 | 
					 | 
				
			||||||
        ssl "^$(bashio::config 'ssl')" \
 | 
					 | 
				
			||||||
        | tempio \
 | 
					 | 
				
			||||||
            -template /etc/nginx/templates/direct.gtpl \
 | 
					 | 
				
			||||||
            -out /etc/nginx/servers/direct.conf
 | 
					 | 
				
			||||||
fi
 | 
					 | 
				
			||||||
@@ -1 +0,0 @@
 | 
				
			|||||||
oneshot
 | 
					 | 
				
			||||||
@@ -1 +0,0 @@
 | 
				
			|||||||
/etc/s6-overlay/s6-rc.d/init-nginx/run
 | 
					 | 
				
			||||||
@@ -1,25 +0,0 @@
 | 
				
			|||||||
#!/command/with-contenv bashio
 | 
					 | 
				
			||||||
# ==============================================================================
 | 
					 | 
				
			||||||
# Community Hass.io Add-ons: ESPHome
 | 
					 | 
				
			||||||
# Take down the S6 supervision tree when NGINX fails
 | 
					 | 
				
			||||||
# ==============================================================================
 | 
					 | 
				
			||||||
declare exit_code
 | 
					 | 
				
			||||||
readonly exit_code_container=$(</run/s6-linux-init-container-results/exitcode)
 | 
					 | 
				
			||||||
readonly exit_code_service="${1}"
 | 
					 | 
				
			||||||
readonly exit_code_signal="${2}"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bashio::log.info \
 | 
					 | 
				
			||||||
  "Service NGINX exited with code ${exit_code_service}" \
 | 
					 | 
				
			||||||
  "(by signal ${exit_code_signal})"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
if [[ "${exit_code_service}" -eq 256 ]]; then
 | 
					 | 
				
			||||||
  if [[ "${exit_code_container}" -eq 0 ]]; then
 | 
					 | 
				
			||||||
    echo $((128 + $exit_code_signal)) > /run/s6-linux-init-container-results/exitcode
 | 
					 | 
				
			||||||
  fi
 | 
					 | 
				
			||||||
  [[ "${exit_code_signal}" -eq 15 ]] && exec /run/s6/basedir/bin/halt
 | 
					 | 
				
			||||||
elif [[ "${exit_code_service}" -ne 0 ]]; then
 | 
					 | 
				
			||||||
  if [[ "${exit_code_container}" -eq 0 ]]; then
 | 
					 | 
				
			||||||
    echo "${exit_code_service}" > /run/s6-linux-init-container-results/exitcode
 | 
					 | 
				
			||||||
  fi
 | 
					 | 
				
			||||||
  exec /run/s6/basedir/bin/halt
 | 
					 | 
				
			||||||
fi
 | 
					 | 
				
			||||||
@@ -1 +0,0 @@
 | 
				
			|||||||
longrun
 | 
					 | 
				
			||||||
							
								
								
									
										15
									
								
								docker/ha-addon-rootfs/etc/services.d/esphome/finish
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										15
									
								
								docker/ha-addon-rootfs/etc/services.d/esphome/finish
									
									
									
									
									
										Executable file
									
								
							@@ -0,0 +1,15 @@
 | 
				
			|||||||
 | 
					#!/usr/bin/execlineb -S0
 | 
				
			||||||
 | 
					# ==============================================================================
 | 
				
			||||||
 | 
					# Community Hass.io Add-ons: ESPHome
 | 
				
			||||||
 | 
					# Take down the S6 supervision tree when ESPHome fails
 | 
				
			||||||
 | 
					# ==============================================================================
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					declare APP_EXIT_CODE=${1}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if [[ "${APP_EXIT_CODE}" -ne 0 ]] && [[ "${APP_EXIT_CODE}" -ne 256 ]]; then
 | 
				
			||||||
 | 
					  bashio::log.warning "Halt add-on with exit code ${APP_EXIT_CODE}"
 | 
				
			||||||
 | 
					  echo "${APP_EXIT_CODE}" > /run/s6-linux-init-container-results/exitcode
 | 
				
			||||||
 | 
					  exec /run/s6/basedir/bin/halt
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bashio::log.info "Service restart after closing"
 | 
				
			||||||
@@ -1,19 +1,10 @@
 | 
				
			|||||||
#!/command/with-contenv bashio
 | 
					#!/usr/bin/with-contenv bashio
 | 
				
			||||||
# shellcheck shell=bash
 | 
					 | 
				
			||||||
# ==============================================================================
 | 
					# ==============================================================================
 | 
				
			||||||
# Community Hass.io Add-ons: ESPHome
 | 
					# Community Hass.io Add-ons: ESPHome
 | 
				
			||||||
# Runs the ESPHome dashboard
 | 
					# Runs the ESPHome dashboard
 | 
				
			||||||
# ==============================================================================
 | 
					# ==============================================================================
 | 
				
			||||||
readonly pio_cache_base=/data/cache/platformio
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
export ESPHOME_IS_HA_ADDON=true
 | 
					export ESPHOME_IS_HA_ADDON=true
 | 
				
			||||||
export PLATFORMIO_GLOBALLIB_DIR=/piolibs
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
# we can't set core_dir, because the settings file is stored in `core_dir/appstate.json`
 | 
					 | 
				
			||||||
# setting `core_dir` would therefore prevent pio from accessing
 | 
					 | 
				
			||||||
export PLATFORMIO_PLATFORMS_DIR="${pio_cache_base}/platforms"
 | 
					 | 
				
			||||||
export PLATFORMIO_PACKAGES_DIR="${pio_cache_base}/packages"
 | 
					 | 
				
			||||||
export PLATFORMIO_CACHE_DIR="${pio_cache_base}/cache"
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
if bashio::config.true 'leave_front_door_open'; then
 | 
					if bashio::config.true 'leave_front_door_open'; then
 | 
				
			||||||
    export DISABLE_HA_AUTHENTICATION=true
 | 
					    export DISABLE_HA_AUTHENTICATION=true
 | 
				
			||||||
@@ -39,7 +30,14 @@ else
 | 
				
			|||||||
    fi
 | 
					    fi
 | 
				
			||||||
fi
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
mkdir -p "${pio_cache_base}"
 | 
					pio_cache_base=/data/cache/platformio
 | 
				
			||||||
 | 
					# we can't set core_dir, because the settings file is stored in `core_dir/appstate.json`
 | 
				
			||||||
 | 
					# setting `core_dir` would therefore prevent pio from accessing
 | 
				
			||||||
 | 
					export PLATFORMIO_PLATFORMS_DIR="${pio_cache_base}/platforms"
 | 
				
			||||||
 | 
					export PLATFORMIO_PACKAGES_DIR="${pio_cache_base}/packages"
 | 
				
			||||||
 | 
					export PLATFORMIO_CACHE_DIR="${pio_cache_base}/cache"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export PLATFORMIO_GLOBALLIB_DIR=/piolibs
 | 
				
			||||||
 | 
					
 | 
				
			||||||
bashio::log.info "Starting ESPHome dashboard..."
 | 
					bashio::log.info "Starting ESPHome dashboard..."
 | 
				
			||||||
exec esphome dashboard /config/esphome --socket /var/run/esphome.sock --ha-addon
 | 
					exec esphome dashboard /config/esphome --socket /var/run/esphome.sock --ha-addon
 | 
				
			||||||
							
								
								
									
										15
									
								
								docker/ha-addon-rootfs/etc/services.d/nginx/finish
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										15
									
								
								docker/ha-addon-rootfs/etc/services.d/nginx/finish
									
									
									
									
									
										Executable file
									
								
							@@ -0,0 +1,15 @@
 | 
				
			|||||||
 | 
					#!/usr/bin/execlineb -S0
 | 
				
			||||||
 | 
					# ==============================================================================
 | 
				
			||||||
 | 
					# Community Hass.io Add-ons: ESPHome
 | 
				
			||||||
 | 
					# Take down the S6 supervision tree when NGINX fails
 | 
				
			||||||
 | 
					# ==============================================================================
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					declare APP_EXIT_CODE=${1}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					if [[ "${APP_EXIT_CODE}" -ne 0 ]] && [[ "${APP_EXIT_CODE}" -ne 256 ]]; then
 | 
				
			||||||
 | 
					  bashio::log.warning "Halt add-on with exit code ${APP_EXIT_CODE}"
 | 
				
			||||||
 | 
					  echo "${APP_EXIT_CODE}" > /run/s6-linux-init-container-results/exitcode
 | 
				
			||||||
 | 
					  exec /run/s6/basedir/bin/halt
 | 
				
			||||||
 | 
					fi
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					bashio::log.info "Service restart after closing"
 | 
				
			||||||
@@ -1,11 +1,10 @@
 | 
				
			|||||||
#!/command/with-contenv bashio
 | 
					#!/usr/bin/with-contenv bashio
 | 
				
			||||||
# shellcheck shell=bash
 | 
					 | 
				
			||||||
# ==============================================================================
 | 
					# ==============================================================================
 | 
				
			||||||
# Community Hass.io Add-ons: ESPHome
 | 
					# Community Hass.io Add-ons: ESPHome
 | 
				
			||||||
# Runs the NGINX proxy
 | 
					# Runs the NGINX proxy
 | 
				
			||||||
# ==============================================================================
 | 
					# ==============================================================================
 | 
				
			||||||
 | 
					
 | 
				
			||||||
bashio::log.info "Waiting for ESPHome dashboard to come up..."
 | 
					bashio::log.info "Waiting for dashboard to come up..."
 | 
				
			||||||
 | 
					
 | 
				
			||||||
while [[ ! -S /var/run/esphome.sock ]]; do
 | 
					while [[ ! -S /var/run/esphome.sock ]]; do
 | 
				
			||||||
  sleep 0.5
 | 
					  sleep 0.5
 | 
				
			||||||
@@ -152,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)
 | 
				
			||||||
@@ -341,7 +339,7 @@ def command_config(args, config):
 | 
				
			|||||||
    _LOGGER.info("Configuration is valid!")
 | 
					    _LOGGER.info("Configuration is valid!")
 | 
				
			||||||
    if not CORE.verbose:
 | 
					    if not CORE.verbose:
 | 
				
			||||||
        config = strip_default_ids(config)
 | 
					        config = strip_default_ids(config)
 | 
				
			||||||
    safe_print(yaml_util.dump(config, args.show_secrets))
 | 
					    safe_print(yaml_util.dump(config))
 | 
				
			||||||
    return 0
 | 
					    return 0
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -667,9 +665,6 @@ def parse_args(argv):
 | 
				
			|||||||
    parser_config.add_argument(
 | 
					    parser_config.add_argument(
 | 
				
			||||||
        "configuration", help="Your YAML configuration file(s).", nargs="+"
 | 
					        "configuration", help="Your YAML configuration file(s).", nargs="+"
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
    parser_config.add_argument(
 | 
					 | 
				
			||||||
        "--show-secrets", help="Show secrets in output.", action="store_true"
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    parser_compile = subparsers.add_parser(
 | 
					    parser_compile = subparsers.add_parser(
 | 
				
			||||||
        "compile", help="Read the configuration and compile a program."
 | 
					        "compile", help="Read the configuration and compile a program."
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -254,11 +254,7 @@ async def repeat_action_to_code(config, action_id, template_arg, args):
 | 
				
			|||||||
    var = cg.new_Pvariable(action_id, template_arg)
 | 
					    var = cg.new_Pvariable(action_id, template_arg)
 | 
				
			||||||
    count_template = await cg.templatable(config[CONF_COUNT], args, cg.uint32)
 | 
					    count_template = await cg.templatable(config[CONF_COUNT], args, cg.uint32)
 | 
				
			||||||
    cg.add(var.set_count(count_template))
 | 
					    cg.add(var.set_count(count_template))
 | 
				
			||||||
    actions = await build_action_list(
 | 
					    actions = await build_action_list(config[CONF_THEN], template_arg, args)
 | 
				
			||||||
        config[CONF_THEN],
 | 
					 | 
				
			||||||
        cg.TemplateArguments(cg.uint32, *template_arg.args),
 | 
					 | 
				
			||||||
        [(cg.uint32, "iteration"), *args],
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    cg.add(var.add_then(actions))
 | 
					    cg.add(var.add_then(actions))
 | 
				
			||||||
    return var
 | 
					    return var
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -47,7 +47,6 @@ from esphome.cpp_helpers import (  # noqa
 | 
				
			|||||||
    build_registry_list,
 | 
					    build_registry_list,
 | 
				
			||||||
    extract_registry_entry_config,
 | 
					    extract_registry_entry_config,
 | 
				
			||||||
    register_parented,
 | 
					    register_parented,
 | 
				
			||||||
    past_safe_mode,
 | 
					 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
from esphome.cpp_types import (  # noqa
 | 
					from esphome.cpp_types import (  # noqa
 | 
				
			||||||
    global_ns,
 | 
					    global_ns,
 | 
				
			||||||
@@ -64,7 +63,6 @@ from esphome.cpp_types import (  # noqa
 | 
				
			|||||||
    uint16,
 | 
					    uint16,
 | 
				
			||||||
    uint32,
 | 
					    uint32,
 | 
				
			||||||
    uint64,
 | 
					    uint64,
 | 
				
			||||||
    int16,
 | 
					 | 
				
			||||||
    int32,
 | 
					    int32,
 | 
				
			||||||
    int64,
 | 
					    int64,
 | 
				
			||||||
    size_t,
 | 
					    size_t,
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -46,7 +46,6 @@ void A4988::loop() {
 | 
				
			|||||||
    return;
 | 
					    return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  this->dir_pin_->digital_write(dir == 1);
 | 
					  this->dir_pin_->digital_write(dir == 1);
 | 
				
			||||||
  delayMicroseconds(50);
 | 
					 | 
				
			||||||
  this->step_pin_->digital_write(true);
 | 
					  this->step_pin_->digital_write(true);
 | 
				
			||||||
  delayMicroseconds(5);
 | 
					  delayMicroseconds(5);
 | 
				
			||||||
  this->step_pin_->digital_write(false);
 | 
					  this->step_pin_->digital_write(false);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1 +0,0 @@
 | 
				
			|||||||
CODEOWNERS = ["@DAVe3283"]
 | 
					 | 
				
			||||||
@@ -1,182 +0,0 @@
 | 
				
			|||||||
#include "esphome/core/log.h"
 | 
					 | 
				
			||||||
#include "absolute_humidity.h"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
namespace esphome {
 | 
					 | 
				
			||||||
namespace absolute_humidity {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static const char *const TAG = "absolute_humidity.sensor";
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void AbsoluteHumidityComponent::setup() {
 | 
					 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "Setting up absolute humidity '%s'...", this->get_name().c_str());
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  ESP_LOGD(TAG, "  Added callback for temperature '%s'", this->temperature_sensor_->get_name().c_str());
 | 
					 | 
				
			||||||
  this->temperature_sensor_->add_on_state_callback([this](float state) { this->temperature_callback_(state); });
 | 
					 | 
				
			||||||
  if (this->temperature_sensor_->has_state()) {
 | 
					 | 
				
			||||||
    this->temperature_callback_(this->temperature_sensor_->get_state());
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  ESP_LOGD(TAG, "  Added callback for relative humidity '%s'", this->humidity_sensor_->get_name().c_str());
 | 
					 | 
				
			||||||
  this->humidity_sensor_->add_on_state_callback([this](float state) { this->humidity_callback_(state); });
 | 
					 | 
				
			||||||
  if (this->humidity_sensor_->has_state()) {
 | 
					 | 
				
			||||||
    this->humidity_callback_(this->humidity_sensor_->get_state());
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void AbsoluteHumidityComponent::dump_config() {
 | 
					 | 
				
			||||||
  LOG_SENSOR("", "Absolute Humidity", this);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  switch (this->equation_) {
 | 
					 | 
				
			||||||
    case BUCK:
 | 
					 | 
				
			||||||
      ESP_LOGCONFIG(TAG, "Saturation Vapor Pressure Equation: Buck");
 | 
					 | 
				
			||||||
      break;
 | 
					 | 
				
			||||||
    case TETENS:
 | 
					 | 
				
			||||||
      ESP_LOGCONFIG(TAG, "Saturation Vapor Pressure Equation: Tetens");
 | 
					 | 
				
			||||||
      break;
 | 
					 | 
				
			||||||
    case WOBUS:
 | 
					 | 
				
			||||||
      ESP_LOGCONFIG(TAG, "Saturation Vapor Pressure Equation: Wobus");
 | 
					 | 
				
			||||||
      break;
 | 
					 | 
				
			||||||
    default:
 | 
					 | 
				
			||||||
      ESP_LOGE(TAG, "Invalid saturation vapor pressure equation selection!");
 | 
					 | 
				
			||||||
      break;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "Sources");
 | 
					 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "  Temperature: '%s'", this->temperature_sensor_->get_name().c_str());
 | 
					 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "  Relative Humidity: '%s'", this->humidity_sensor_->get_name().c_str());
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
float AbsoluteHumidityComponent::get_setup_priority() const { return setup_priority::DATA; }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void AbsoluteHumidityComponent::loop() {
 | 
					 | 
				
			||||||
  if (!this->next_update_) {
 | 
					 | 
				
			||||||
    return;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  this->next_update_ = false;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  // Ensure we have source data
 | 
					 | 
				
			||||||
  const bool no_temperature = std::isnan(this->temperature_);
 | 
					 | 
				
			||||||
  const bool no_humidity = std::isnan(this->humidity_);
 | 
					 | 
				
			||||||
  if (no_temperature || no_humidity) {
 | 
					 | 
				
			||||||
    if (no_temperature) {
 | 
					 | 
				
			||||||
      ESP_LOGW(TAG, "No valid state from temperature sensor!");
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    if (no_humidity) {
 | 
					 | 
				
			||||||
      ESP_LOGW(TAG, "No valid state from temperature sensor!");
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    ESP_LOGW(TAG, "Unable to calculate absolute humidity.");
 | 
					 | 
				
			||||||
    this->publish_state(NAN);
 | 
					 | 
				
			||||||
    this->status_set_warning();
 | 
					 | 
				
			||||||
    return;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  // Convert to desired units
 | 
					 | 
				
			||||||
  const float temperature_c = this->temperature_;
 | 
					 | 
				
			||||||
  const float temperature_k = temperature_c + 273.15;
 | 
					 | 
				
			||||||
  const float hr = this->humidity_ / 100;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  // Calculate saturation vapor pressure
 | 
					 | 
				
			||||||
  float es;
 | 
					 | 
				
			||||||
  switch (this->equation_) {
 | 
					 | 
				
			||||||
    case BUCK:
 | 
					 | 
				
			||||||
      es = es_buck(temperature_c);
 | 
					 | 
				
			||||||
      break;
 | 
					 | 
				
			||||||
    case TETENS:
 | 
					 | 
				
			||||||
      es = es_tetens(temperature_c);
 | 
					 | 
				
			||||||
      break;
 | 
					 | 
				
			||||||
    case WOBUS:
 | 
					 | 
				
			||||||
      es = es_wobus(temperature_c);
 | 
					 | 
				
			||||||
      break;
 | 
					 | 
				
			||||||
    default:
 | 
					 | 
				
			||||||
      ESP_LOGE(TAG, "Invalid saturation vapor pressure equation selection!");
 | 
					 | 
				
			||||||
      this->publish_state(NAN);
 | 
					 | 
				
			||||||
      this->status_set_error();
 | 
					 | 
				
			||||||
      return;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  ESP_LOGD(TAG, "Saturation vapor pressure %f kPa", es);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  // Calculate absolute humidity
 | 
					 | 
				
			||||||
  const float absolute_humidity = vapor_density(es, hr, temperature_k);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  // Publish absolute humidity
 | 
					 | 
				
			||||||
  ESP_LOGD(TAG, "Publishing absolute humidity %f g/m³", absolute_humidity);
 | 
					 | 
				
			||||||
  this->status_clear_warning();
 | 
					 | 
				
			||||||
  this->publish_state(absolute_humidity);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// Buck equation (https://en.wikipedia.org/wiki/Arden_Buck_equation)
 | 
					 | 
				
			||||||
// More accurate than Tetens in normal meteorologic conditions
 | 
					 | 
				
			||||||
float AbsoluteHumidityComponent::es_buck(float temperature_c) {
 | 
					 | 
				
			||||||
  float a, b, c, d;
 | 
					 | 
				
			||||||
  if (temperature_c >= 0) {
 | 
					 | 
				
			||||||
    a = 0.61121;
 | 
					 | 
				
			||||||
    b = 18.678;
 | 
					 | 
				
			||||||
    c = 234.5;
 | 
					 | 
				
			||||||
    d = 257.14;
 | 
					 | 
				
			||||||
  } else {
 | 
					 | 
				
			||||||
    a = 0.61115;
 | 
					 | 
				
			||||||
    b = 18.678;
 | 
					 | 
				
			||||||
    c = 233.7;
 | 
					 | 
				
			||||||
    d = 279.82;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  return a * expf((b - (temperature_c / c)) * (temperature_c / (d + temperature_c)));
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// Tetens equation (https://en.wikipedia.org/wiki/Tetens_equation)
 | 
					 | 
				
			||||||
float AbsoluteHumidityComponent::es_tetens(float temperature_c) {
 | 
					 | 
				
			||||||
  float a, b;
 | 
					 | 
				
			||||||
  if (temperature_c >= 0) {
 | 
					 | 
				
			||||||
    a = 17.27;
 | 
					 | 
				
			||||||
    b = 237.3;
 | 
					 | 
				
			||||||
  } else {
 | 
					 | 
				
			||||||
    a = 21.875;
 | 
					 | 
				
			||||||
    b = 265.5;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  return 0.61078 * expf((a * temperature_c) / (temperature_c + b));
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// Wobus equation
 | 
					 | 
				
			||||||
// https://wahiduddin.net/calc/density_altitude.htm
 | 
					 | 
				
			||||||
// https://wahiduddin.net/calc/density_algorithms.htm
 | 
					 | 
				
			||||||
// Calculate the saturation vapor pressure (kPa)
 | 
					 | 
				
			||||||
float AbsoluteHumidityComponent::es_wobus(float t) {
 | 
					 | 
				
			||||||
  // THIS FUNCTION RETURNS THE SATURATION VAPOR PRESSURE ESW (MILLIBARS)
 | 
					 | 
				
			||||||
  // OVER LIQUID WATER GIVEN THE TEMPERATURE T (CELSIUS). THE POLYNOMIAL
 | 
					 | 
				
			||||||
  // APPROXIMATION BELOW IS DUE TO HERMAN WOBUS, A MATHEMATICIAN WHO
 | 
					 | 
				
			||||||
  // WORKED AT THE NAVY WEATHER RESEARCH FACILITY, NORFOLK, VIRGINIA,
 | 
					 | 
				
			||||||
  // BUT WHO IS NOW RETIRED. THE COEFFICIENTS OF THE POLYNOMIAL WERE
 | 
					 | 
				
			||||||
  // CHOSEN TO FIT THE VALUES IN TABLE 94 ON PP. 351-353 OF THE SMITH-
 | 
					 | 
				
			||||||
  // SONIAN METEOROLOGICAL TABLES BY ROLAND LIST (6TH EDITION). THE
 | 
					 | 
				
			||||||
  // APPROXIMATION IS VALID FOR -50 < T < 100C.
 | 
					 | 
				
			||||||
  //
 | 
					 | 
				
			||||||
  //     Baker, Schlatter  17-MAY-1982     Original version.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  const float c0 = +0.99999683e00;
 | 
					 | 
				
			||||||
  const float c1 = -0.90826951e-02;
 | 
					 | 
				
			||||||
  const float c2 = +0.78736169e-04;
 | 
					 | 
				
			||||||
  const float c3 = -0.61117958e-06;
 | 
					 | 
				
			||||||
  const float c4 = +0.43884187e-08;
 | 
					 | 
				
			||||||
  const float c5 = -0.29883885e-10;
 | 
					 | 
				
			||||||
  const float c6 = +0.21874425e-12;
 | 
					 | 
				
			||||||
  const float c7 = -0.17892321e-14;
 | 
					 | 
				
			||||||
  const float c8 = +0.11112018e-16;
 | 
					 | 
				
			||||||
  const float c9 = -0.30994571e-19;
 | 
					 | 
				
			||||||
  const float p = c0 + t * (c1 + t * (c2 + t * (c3 + t * (c4 + t * (c5 + t * (c6 + t * (c7 + t * (c8 + t * (c9)))))))));
 | 
					 | 
				
			||||||
  return 0.61078 / pow(p, 8);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// From https://www.environmentalbiophysics.org/chalk-talk-how-to-calculate-absolute-humidity/
 | 
					 | 
				
			||||||
// H/T to https://esphome.io/cookbook/bme280_environment.html
 | 
					 | 
				
			||||||
// H/T to https://carnotcycle.wordpress.com/2012/08/04/how-to-convert-relative-humidity-to-absolute-humidity/
 | 
					 | 
				
			||||||
float AbsoluteHumidityComponent::vapor_density(float es, float hr, float ta) {
 | 
					 | 
				
			||||||
  // es = saturated vapor pressure (kPa)
 | 
					 | 
				
			||||||
  // hr = relative humidity [0-1]
 | 
					 | 
				
			||||||
  // ta = absolute temperature (K)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  const float ea = hr * es * 1000;   // vapor pressure of the air (Pa)
 | 
					 | 
				
			||||||
  const float mw = 18.01528;         // molar mass of water (g⋅mol⁻¹)
 | 
					 | 
				
			||||||
  const float r = 8.31446261815324;  // molar gas constant (J⋅K⁻¹)
 | 
					 | 
				
			||||||
  return (ea * mw) / (r * ta);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
}  // namespace absolute_humidity
 | 
					 | 
				
			||||||
}  // namespace esphome
 | 
					 | 
				
			||||||
@@ -1,76 +0,0 @@
 | 
				
			|||||||
#pragma once
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#include "esphome/core/component.h"
 | 
					 | 
				
			||||||
#include "esphome/components/sensor/sensor.h"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
namespace esphome {
 | 
					 | 
				
			||||||
namespace absolute_humidity {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
/// Enum listing all implemented saturation vapor pressure equations.
 | 
					 | 
				
			||||||
enum SaturationVaporPressureEquation {
 | 
					 | 
				
			||||||
  BUCK,
 | 
					 | 
				
			||||||
  TETENS,
 | 
					 | 
				
			||||||
  WOBUS,
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
/// This class implements calculation of absolute humidity from temperature and relative humidity.
 | 
					 | 
				
			||||||
class AbsoluteHumidityComponent : public sensor::Sensor, public Component {
 | 
					 | 
				
			||||||
 public:
 | 
					 | 
				
			||||||
  AbsoluteHumidityComponent() = default;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  void set_temperature_sensor(sensor::Sensor *temperature_sensor) { this->temperature_sensor_ = temperature_sensor; }
 | 
					 | 
				
			||||||
  void set_humidity_sensor(sensor::Sensor *humidity_sensor) { this->humidity_sensor_ = humidity_sensor; }
 | 
					 | 
				
			||||||
  void set_equation(SaturationVaporPressureEquation equation) { this->equation_ = equation; }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  void setup() override;
 | 
					 | 
				
			||||||
  void dump_config() override;
 | 
					 | 
				
			||||||
  float get_setup_priority() const override;
 | 
					 | 
				
			||||||
  void loop() override;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 protected:
 | 
					 | 
				
			||||||
  void temperature_callback_(float state) {
 | 
					 | 
				
			||||||
    this->next_update_ = true;
 | 
					 | 
				
			||||||
    this->temperature_ = state;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  void humidity_callback_(float state) {
 | 
					 | 
				
			||||||
    this->next_update_ = true;
 | 
					 | 
				
			||||||
    this->humidity_ = state;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /** Buck equation for saturation vapor pressure in kPa.
 | 
					 | 
				
			||||||
   *
 | 
					 | 
				
			||||||
   * @param temperature_c Air temperature in °C.
 | 
					 | 
				
			||||||
   */
 | 
					 | 
				
			||||||
  static float es_buck(float temperature_c);
 | 
					 | 
				
			||||||
  /** Tetens equation for saturation vapor pressure in kPa.
 | 
					 | 
				
			||||||
   *
 | 
					 | 
				
			||||||
   * @param temperature_c Air temperature in °C.
 | 
					 | 
				
			||||||
   */
 | 
					 | 
				
			||||||
  static float es_tetens(float temperature_c);
 | 
					 | 
				
			||||||
  /** Wobus equation for saturation vapor pressure in kPa.
 | 
					 | 
				
			||||||
   *
 | 
					 | 
				
			||||||
   * @param temperature_c Air temperature in °C.
 | 
					 | 
				
			||||||
   */
 | 
					 | 
				
			||||||
  static float es_wobus(float temperature_c);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /** Calculate vapor density (absolute humidity) in g/m³.
 | 
					 | 
				
			||||||
   *
 | 
					 | 
				
			||||||
   * @param es Saturation vapor pressure in kPa.
 | 
					 | 
				
			||||||
   * @param hr Relative humidity 0 to 1.
 | 
					 | 
				
			||||||
   * @param ta Absolute temperature in K.
 | 
					 | 
				
			||||||
   * @param heater_duration The duration in ms that the heater should turn on for when measuring.
 | 
					 | 
				
			||||||
   */
 | 
					 | 
				
			||||||
  static float vapor_density(float es, float hr, float ta);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  sensor::Sensor *temperature_sensor_{nullptr};
 | 
					 | 
				
			||||||
  sensor::Sensor *humidity_sensor_{nullptr};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  bool next_update_{false};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  float temperature_{NAN};
 | 
					 | 
				
			||||||
  float humidity_{NAN};
 | 
					 | 
				
			||||||
  SaturationVaporPressureEquation equation_;
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
}  // namespace absolute_humidity
 | 
					 | 
				
			||||||
}  // namespace esphome
 | 
					 | 
				
			||||||
@@ -1,56 +0,0 @@
 | 
				
			|||||||
import esphome.codegen as cg
 | 
					 | 
				
			||||||
import esphome.config_validation as cv
 | 
					 | 
				
			||||||
from esphome.components import sensor
 | 
					 | 
				
			||||||
from esphome.const import (
 | 
					 | 
				
			||||||
    CONF_HUMIDITY,
 | 
					 | 
				
			||||||
    CONF_TEMPERATURE,
 | 
					 | 
				
			||||||
    STATE_CLASS_MEASUREMENT,
 | 
					 | 
				
			||||||
    CONF_EQUATION,
 | 
					 | 
				
			||||||
    ICON_WATER,
 | 
					 | 
				
			||||||
    UNIT_GRAMS_PER_CUBIC_METER,
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
absolute_humidity_ns = cg.esphome_ns.namespace("absolute_humidity")
 | 
					 | 
				
			||||||
AbsoluteHumidityComponent = absolute_humidity_ns.class_(
 | 
					 | 
				
			||||||
    "AbsoluteHumidityComponent", sensor.Sensor, cg.Component
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
SaturationVaporPressureEquation = absolute_humidity_ns.enum(
 | 
					 | 
				
			||||||
    "SaturationVaporPressureEquation"
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
EQUATION = {
 | 
					 | 
				
			||||||
    "BUCK": SaturationVaporPressureEquation.BUCK,
 | 
					 | 
				
			||||||
    "TETENS": SaturationVaporPressureEquation.TETENS,
 | 
					 | 
				
			||||||
    "WOBUS": SaturationVaporPressureEquation.WOBUS,
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
CONFIG_SCHEMA = (
 | 
					 | 
				
			||||||
    sensor.sensor_schema(
 | 
					 | 
				
			||||||
        unit_of_measurement=UNIT_GRAMS_PER_CUBIC_METER,
 | 
					 | 
				
			||||||
        icon=ICON_WATER,
 | 
					 | 
				
			||||||
        accuracy_decimals=2,
 | 
					 | 
				
			||||||
        state_class=STATE_CLASS_MEASUREMENT,
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    .extend(
 | 
					 | 
				
			||||||
        {
 | 
					 | 
				
			||||||
            cv.GenerateID(): cv.declare_id(AbsoluteHumidityComponent),
 | 
					 | 
				
			||||||
            cv.Required(CONF_TEMPERATURE): cv.use_id(sensor.Sensor),
 | 
					 | 
				
			||||||
            cv.Required(CONF_HUMIDITY): cv.use_id(sensor.Sensor),
 | 
					 | 
				
			||||||
            cv.Optional(CONF_EQUATION, default="WOBUS"): cv.enum(EQUATION, upper=True),
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    .extend(cv.COMPONENT_SCHEMA)
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
async def to_code(config):
 | 
					 | 
				
			||||||
    var = await sensor.new_sensor(config)
 | 
					 | 
				
			||||||
    await cg.register_component(var, config)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    temperature_sensor = await cg.get_variable(config[CONF_TEMPERATURE])
 | 
					 | 
				
			||||||
    cg.add(var.set_temperature_sensor(temperature_sensor))
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    humidity_sensor = await cg.get_variable(config[CONF_HUMIDITY])
 | 
					 | 
				
			||||||
    cg.add(var.set_humidity_sensor(humidity_sensor))
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    cg.add(var.set_equation(config[CONF_EQUATION]))
 | 
					 | 
				
			||||||
@@ -16,16 +16,13 @@ ADC128S102Sensor = adc128s102_ns.class_(
 | 
				
			|||||||
)
 | 
					)
 | 
				
			||||||
CONF_ADC128S102_ID = "adc128s102_id"
 | 
					CONF_ADC128S102_ID = "adc128s102_id"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
CONFIG_SCHEMA = (
 | 
					CONFIG_SCHEMA = sensor.SENSOR_SCHEMA.extend(
 | 
				
			||||||
    sensor.sensor_schema(ADC128S102Sensor)
 | 
					    {
 | 
				
			||||||
    .extend(
 | 
					        cv.GenerateID(): cv.declare_id(ADC128S102Sensor),
 | 
				
			||||||
        {
 | 
					        cv.GenerateID(CONF_ADC128S102_ID): cv.use_id(ADC128S102),
 | 
				
			||||||
            cv.GenerateID(CONF_ADC128S102_ID): cv.use_id(ADC128S102),
 | 
					        cv.Required(CONF_CHANNEL): cv.int_range(min=0, max=7),
 | 
				
			||||||
            cv.Required(CONF_CHANNEL): cv.int_range(min=0, max=7),
 | 
					    }
 | 
				
			||||||
        }
 | 
					).extend(cv.polling_component_schema("60s"))
 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    .extend(cv.polling_component_schema("60s"))
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async def to_code(config):
 | 
					async def to_code(config):
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -9,7 +9,7 @@ static const char *const TAG = "ads1115";
 | 
				
			|||||||
static const uint8_t ADS1115_REGISTER_CONVERSION = 0x00;
 | 
					static const uint8_t ADS1115_REGISTER_CONVERSION = 0x00;
 | 
				
			||||||
static const uint8_t ADS1115_REGISTER_CONFIG = 0x01;
 | 
					static const uint8_t ADS1115_REGISTER_CONFIG = 0x01;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static const uint8_t ADS1115_DATA_RATE_860_SPS = 0b111;  // 3300_SPS for ADS1015
 | 
					static const uint8_t ADS1115_DATA_RATE_860_SPS = 0b111;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void ADS1115Component::setup() {
 | 
					void ADS1115Component::setup() {
 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "Setting up ADS1115...");
 | 
					  ESP_LOGCONFIG(TAG, "Setting up ADS1115...");
 | 
				
			||||||
@@ -18,9 +18,6 @@ void ADS1115Component::setup() {
 | 
				
			|||||||
    this->mark_failed();
 | 
					    this->mark_failed();
 | 
				
			||||||
    return;
 | 
					    return;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "Configuring ADS1115...");
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  uint16_t config = 0;
 | 
					  uint16_t config = 0;
 | 
				
			||||||
  // Clear single-shot bit
 | 
					  // Clear single-shot bit
 | 
				
			||||||
  //        0b0xxxxxxxxxxxxxxx
 | 
					  //        0b0xxxxxxxxxxxxxxx
 | 
				
			||||||
@@ -80,7 +77,6 @@ void ADS1115Component::dump_config() {
 | 
				
			|||||||
    LOG_SENSOR("  ", "Sensor", sensor);
 | 
					    LOG_SENSOR("  ", "Sensor", sensor);
 | 
				
			||||||
    ESP_LOGCONFIG(TAG, "    Multiplexer: %u", sensor->get_multiplexer());
 | 
					    ESP_LOGCONFIG(TAG, "    Multiplexer: %u", sensor->get_multiplexer());
 | 
				
			||||||
    ESP_LOGCONFIG(TAG, "    Gain: %u", sensor->get_gain());
 | 
					    ESP_LOGCONFIG(TAG, "    Gain: %u", sensor->get_gain());
 | 
				
			||||||
    ESP_LOGCONFIG(TAG, "    Resolution: %u", sensor->get_resolution());
 | 
					 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
float ADS1115Component::request_measurement(ADS1115Sensor *sensor) {
 | 
					float ADS1115Component::request_measurement(ADS1115Sensor *sensor) {
 | 
				
			||||||
@@ -131,45 +127,27 @@ float ADS1115Component::request_measurement(ADS1115Sensor *sensor) {
 | 
				
			|||||||
    this->status_set_warning();
 | 
					    this->status_set_warning();
 | 
				
			||||||
    return NAN;
 | 
					    return NAN;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					 | 
				
			||||||
  if (sensor->get_resolution() == ADS1015_12_BITS) {
 | 
					 | 
				
			||||||
    bool negative = (raw_conversion >> 15) == 1;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    // shift raw_conversion as it's only 12-bits, left justified
 | 
					 | 
				
			||||||
    raw_conversion = raw_conversion >> (16 - ADS1015_12_BITS);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    // check if number was negative in order to keep the sign
 | 
					 | 
				
			||||||
    if (negative) {
 | 
					 | 
				
			||||||
      // the number was negative
 | 
					 | 
				
			||||||
      // 1) set the negative bit back
 | 
					 | 
				
			||||||
      raw_conversion |= 0x8000;
 | 
					 | 
				
			||||||
      // 2) reset the former (shifted) negative bit
 | 
					 | 
				
			||||||
      raw_conversion &= 0xF7FF;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  auto signed_conversion = static_cast<int16_t>(raw_conversion);
 | 
					  auto signed_conversion = static_cast<int16_t>(raw_conversion);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  float millivolts;
 | 
					  float millivolts;
 | 
				
			||||||
  float divider = (sensor->get_resolution() == ADS1115_16_BITS) ? 32768.0f : 2048.0f;
 | 
					 | 
				
			||||||
  switch (sensor->get_gain()) {
 | 
					  switch (sensor->get_gain()) {
 | 
				
			||||||
    case ADS1115_GAIN_6P144:
 | 
					    case ADS1115_GAIN_6P144:
 | 
				
			||||||
      millivolts = (signed_conversion * 6144) / divider;
 | 
					      millivolts = signed_conversion * 0.187500f;
 | 
				
			||||||
      break;
 | 
					      break;
 | 
				
			||||||
    case ADS1115_GAIN_4P096:
 | 
					    case ADS1115_GAIN_4P096:
 | 
				
			||||||
      millivolts = (signed_conversion * 4096) / divider;
 | 
					      millivolts = signed_conversion * 0.125000f;
 | 
				
			||||||
      break;
 | 
					      break;
 | 
				
			||||||
    case ADS1115_GAIN_2P048:
 | 
					    case ADS1115_GAIN_2P048:
 | 
				
			||||||
      millivolts = (signed_conversion * 2048) / divider;
 | 
					      millivolts = signed_conversion * 0.062500f;
 | 
				
			||||||
      break;
 | 
					      break;
 | 
				
			||||||
    case ADS1115_GAIN_1P024:
 | 
					    case ADS1115_GAIN_1P024:
 | 
				
			||||||
      millivolts = (signed_conversion * 1024) / divider;
 | 
					      millivolts = signed_conversion * 0.031250f;
 | 
				
			||||||
      break;
 | 
					      break;
 | 
				
			||||||
    case ADS1115_GAIN_0P512:
 | 
					    case ADS1115_GAIN_0P512:
 | 
				
			||||||
      millivolts = (signed_conversion * 512) / divider;
 | 
					      millivolts = signed_conversion * 0.015625f;
 | 
				
			||||||
      break;
 | 
					      break;
 | 
				
			||||||
    case ADS1115_GAIN_0P256:
 | 
					    case ADS1115_GAIN_0P256:
 | 
				
			||||||
      millivolts = (signed_conversion * 256) / divider;
 | 
					      millivolts = signed_conversion * 0.007813f;
 | 
				
			||||||
      break;
 | 
					      break;
 | 
				
			||||||
    default:
 | 
					    default:
 | 
				
			||||||
      millivolts = NAN;
 | 
					      millivolts = NAN;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -30,11 +30,6 @@ enum ADS1115Gain {
 | 
				
			|||||||
  ADS1115_GAIN_0P256 = 0b101,
 | 
					  ADS1115_GAIN_0P256 = 0b101,
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
enum ADS1115Resolution {
 | 
					 | 
				
			||||||
  ADS1115_16_BITS = 16,
 | 
					 | 
				
			||||||
  ADS1015_12_BITS = 12,
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class ADS1115Sensor;
 | 
					class ADS1115Sensor;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class ADS1115Component : public Component, public i2c::I2CDevice {
 | 
					class ADS1115Component : public Component, public i2c::I2CDevice {
 | 
				
			||||||
@@ -63,17 +58,15 @@ class ADS1115Sensor : public sensor::Sensor, public PollingComponent, public vol
 | 
				
			|||||||
  void update() override;
 | 
					  void update() override;
 | 
				
			||||||
  void set_multiplexer(ADS1115Multiplexer multiplexer) { multiplexer_ = multiplexer; }
 | 
					  void set_multiplexer(ADS1115Multiplexer multiplexer) { multiplexer_ = multiplexer; }
 | 
				
			||||||
  void set_gain(ADS1115Gain gain) { gain_ = gain; }
 | 
					  void set_gain(ADS1115Gain gain) { gain_ = gain; }
 | 
				
			||||||
  void set_resolution(ADS1115Resolution resolution) { resolution_ = resolution; }
 | 
					
 | 
				
			||||||
  float sample() override;
 | 
					  float sample() override;
 | 
				
			||||||
  uint8_t get_multiplexer() const { return multiplexer_; }
 | 
					  uint8_t get_multiplexer() const { return multiplexer_; }
 | 
				
			||||||
  uint8_t get_gain() const { return gain_; }
 | 
					  uint8_t get_gain() const { return gain_; }
 | 
				
			||||||
  uint8_t get_resolution() const { return resolution_; }
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 protected:
 | 
					 protected:
 | 
				
			||||||
  ADS1115Component *parent_;
 | 
					  ADS1115Component *parent_;
 | 
				
			||||||
  ADS1115Multiplexer multiplexer_;
 | 
					  ADS1115Multiplexer multiplexer_;
 | 
				
			||||||
  ADS1115Gain gain_;
 | 
					  ADS1115Gain gain_;
 | 
				
			||||||
  ADS1115Resolution resolution_;
 | 
					 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
}  // namespace ads1115
 | 
					}  // namespace ads1115
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -4,7 +4,6 @@ from esphome.components import sensor, voltage_sampler
 | 
				
			|||||||
from esphome.const import (
 | 
					from esphome.const import (
 | 
				
			||||||
    CONF_GAIN,
 | 
					    CONF_GAIN,
 | 
				
			||||||
    CONF_MULTIPLEXER,
 | 
					    CONF_MULTIPLEXER,
 | 
				
			||||||
    CONF_RESOLUTION,
 | 
					 | 
				
			||||||
    DEVICE_CLASS_VOLTAGE,
 | 
					    DEVICE_CLASS_VOLTAGE,
 | 
				
			||||||
    STATE_CLASS_MEASUREMENT,
 | 
					    STATE_CLASS_MEASUREMENT,
 | 
				
			||||||
    UNIT_VOLT,
 | 
					    UNIT_VOLT,
 | 
				
			||||||
@@ -36,12 +35,6 @@ GAIN = {
 | 
				
			|||||||
    "0.256": ADS1115Gain.ADS1115_GAIN_0P256,
 | 
					    "0.256": ADS1115Gain.ADS1115_GAIN_0P256,
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
ADS1115Resolution = ads1115_ns.enum("ADS1115Resolution")
 | 
					 | 
				
			||||||
RESOLUTION = {
 | 
					 | 
				
			||||||
    "16_BITS": ADS1115Resolution.ADS1115_16_BITS,
 | 
					 | 
				
			||||||
    "12_BITS": ADS1115Resolution.ADS1015_12_BITS,
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
def validate_gain(value):
 | 
					def validate_gain(value):
 | 
				
			||||||
    if isinstance(value, float):
 | 
					    if isinstance(value, float):
 | 
				
			||||||
@@ -70,9 +63,6 @@ CONFIG_SCHEMA = (
 | 
				
			|||||||
            cv.GenerateID(CONF_ADS1115_ID): cv.use_id(ADS1115Component),
 | 
					            cv.GenerateID(CONF_ADS1115_ID): cv.use_id(ADS1115Component),
 | 
				
			||||||
            cv.Required(CONF_MULTIPLEXER): cv.enum(MUX, upper=True, space="_"),
 | 
					            cv.Required(CONF_MULTIPLEXER): cv.enum(MUX, upper=True, space="_"),
 | 
				
			||||||
            cv.Required(CONF_GAIN): validate_gain,
 | 
					            cv.Required(CONF_GAIN): validate_gain,
 | 
				
			||||||
            cv.Optional(CONF_RESOLUTION, default="16_BITS"): cv.enum(
 | 
					 | 
				
			||||||
                RESOLUTION, upper=True, space="_"
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
    .extend(cv.polling_component_schema("60s"))
 | 
					    .extend(cv.polling_component_schema("60s"))
 | 
				
			||||||
@@ -87,6 +77,5 @@ async def to_code(config):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    cg.add(var.set_multiplexer(config[CONF_MULTIPLEXER]))
 | 
					    cg.add(var.set_multiplexer(config[CONF_MULTIPLEXER]))
 | 
				
			||||||
    cg.add(var.set_gain(config[CONF_GAIN]))
 | 
					    cg.add(var.set_gain(config[CONF_GAIN]))
 | 
				
			||||||
    cg.add(var.set_resolution(config[CONF_RESOLUTION]))
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    cg.add(paren.register_sensor(var))
 | 
					    cg.add(paren.register_sensor(var))
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -65,7 +65,7 @@ void Am43Component::control(const CoverCall &call) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    if (this->invert_position_)
 | 
					    if (this->invert_position_)
 | 
				
			||||||
      pos = 1 - pos;
 | 
					      pos = 1 - pos;
 | 
				
			||||||
    auto *packet = this->encoder_->get_set_position_request(100 - (uint8_t) (pos * 100));
 | 
					    auto *packet = this->encoder_->get_set_position_request(100 - (uint8_t)(pos * 100));
 | 
				
			||||||
    auto status =
 | 
					    auto status =
 | 
				
			||||||
        esp_ble_gattc_write_char(this->parent_->get_gattc_if(), this->parent_->get_conn_id(), this->char_handle_,
 | 
					        esp_ble_gattc_write_char(this->parent_->get_gattc_if(), this->parent_->get_conn_id(), this->char_handle_,
 | 
				
			||||||
                                 packet->length, packet->data, ESP_GATT_WRITE_TYPE_NO_RSP, ESP_GATT_AUTH_REQ_NONE);
 | 
					                                 packet->length, packet->data, ESP_GATT_WRITE_TYPE_NO_RSP, ESP_GATT_AUTH_REQ_NONE);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -15,24 +15,18 @@ AnalogThresholdBinarySensor = analog_threshold_ns.class_(
 | 
				
			|||||||
CONF_UPPER = "upper"
 | 
					CONF_UPPER = "upper"
 | 
				
			||||||
CONF_LOWER = "lower"
 | 
					CONF_LOWER = "lower"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
CONFIG_SCHEMA = (
 | 
					CONFIG_SCHEMA = binary_sensor.BINARY_SENSOR_SCHEMA.extend(
 | 
				
			||||||
    binary_sensor.binary_sensor_schema(AnalogThresholdBinarySensor)
 | 
					    {
 | 
				
			||||||
    .extend(
 | 
					        cv.GenerateID(): cv.declare_id(AnalogThresholdBinarySensor),
 | 
				
			||||||
        {
 | 
					        cv.Required(CONF_SENSOR_ID): cv.use_id(sensor.Sensor),
 | 
				
			||||||
            cv.Required(CONF_SENSOR_ID): cv.use_id(sensor.Sensor),
 | 
					        cv.Required(CONF_THRESHOLD): cv.Any(
 | 
				
			||||||
            cv.Required(CONF_THRESHOLD): cv.Any(
 | 
					            cv.float_,
 | 
				
			||||||
                cv.float_,
 | 
					            cv.Schema(
 | 
				
			||||||
                cv.Schema(
 | 
					                {cv.Required(CONF_UPPER): cv.float_, cv.Required(CONF_LOWER): cv.float_}
 | 
				
			||||||
                    {
 | 
					 | 
				
			||||||
                        cv.Required(CONF_UPPER): cv.float_,
 | 
					 | 
				
			||||||
                        cv.Required(CONF_LOWER): cv.float_,
 | 
					 | 
				
			||||||
                    }
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
        }
 | 
					        ),
 | 
				
			||||||
    )
 | 
					    }
 | 
				
			||||||
    .extend(cv.COMPONENT_SCHEMA)
 | 
					).extend(cv.COMPONENT_SCHEMA)
 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async def to_code(config):
 | 
					async def to_code(config):
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -76,6 +76,8 @@ async def to_code(config):
 | 
				
			|||||||
        pos = 0
 | 
					        pos = 0
 | 
				
			||||||
        for frameIndex in range(frames):
 | 
					        for frameIndex in range(frames):
 | 
				
			||||||
            image.seek(frameIndex)
 | 
					            image.seek(frameIndex)
 | 
				
			||||||
 | 
					            if CONF_RESIZE in config:
 | 
				
			||||||
 | 
					                image.thumbnail(config[CONF_RESIZE])
 | 
				
			||||||
            frame = image.convert("RGB")
 | 
					            frame = image.convert("RGB")
 | 
				
			||||||
            if CONF_RESIZE in config:
 | 
					            if CONF_RESIZE in config:
 | 
				
			||||||
                frame = frame.resize([width, height])
 | 
					                frame = frame.resize([width, height])
 | 
				
			||||||
@@ -115,7 +117,7 @@ async def to_code(config):
 | 
				
			|||||||
                data[pos] = rgb & 255
 | 
					                data[pos] = rgb & 255
 | 
				
			||||||
                pos += 1
 | 
					                pos += 1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    elif config[CONF_TYPE] in ["BINARY", "TRANSPARENT_BINARY"]:
 | 
					    elif config[CONF_TYPE] == "BINARY":
 | 
				
			||||||
        width8 = ((width + 7) // 8) * 8
 | 
					        width8 = ((width + 7) // 8) * 8
 | 
				
			||||||
        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):
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -23,7 +23,7 @@ void APDS9960::setup() {
 | 
				
			|||||||
    return;
 | 
					    return;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  if (id != 0xAB && id != 0x9C && id != 0xA8) {  // APDS9960 all should have one of these IDs
 | 
					  if (id != 0xAB && id != 0x9C) {  // APDS9960 all should have one of these IDs
 | 
				
			||||||
    this->error_code_ = WRONG_ID;
 | 
					    this->error_code_ = WRONG_ID;
 | 
				
			||||||
    this->mark_failed();
 | 
					    this->mark_failed();
 | 
				
			||||||
    return;
 | 
					    return;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -53,9 +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) {}
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -209,10 +206,6 @@ message DeviceInfoResponse {
 | 
				
			|||||||
  uint32 bluetooth_proxy_version = 11;
 | 
					  uint32 bluetooth_proxy_version = 11;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  string manufacturer = 12;
 | 
					  string manufacturer = 12;
 | 
				
			||||||
 | 
					 | 
				
			||||||
  string friendly_name = 13;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  uint32 voice_assistant_version = 14;
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
message ListEntitiesRequest {
 | 
					message ListEntitiesRequest {
 | 
				
			||||||
@@ -792,7 +785,6 @@ enum ClimateFanMode {
 | 
				
			|||||||
  CLIMATE_FAN_MIDDLE = 6;
 | 
					  CLIMATE_FAN_MIDDLE = 6;
 | 
				
			||||||
  CLIMATE_FAN_FOCUS = 7;
 | 
					  CLIMATE_FAN_FOCUS = 7;
 | 
				
			||||||
  CLIMATE_FAN_DIFFUSE = 8;
 | 
					  CLIMATE_FAN_DIFFUSE = 8;
 | 
				
			||||||
  CLIMATE_FAN_QUIET = 9;
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
enum ClimateSwingMode {
 | 
					enum ClimateSwingMode {
 | 
				
			||||||
  CLIMATE_SWING_OFF = 0;
 | 
					  CLIMATE_SWING_OFF = 0;
 | 
				
			||||||
@@ -834,7 +826,7 @@ message ListEntitiesClimateResponse {
 | 
				
			|||||||
  repeated ClimateMode supported_modes = 7;
 | 
					  repeated ClimateMode supported_modes = 7;
 | 
				
			||||||
  float visual_min_temperature = 8;
 | 
					  float visual_min_temperature = 8;
 | 
				
			||||||
  float visual_max_temperature = 9;
 | 
					  float visual_max_temperature = 9;
 | 
				
			||||||
  float visual_target_temperature_step = 10;
 | 
					  float visual_temperature_step = 10;
 | 
				
			||||||
  // for older peer versions - in new system this
 | 
					  // for older peer versions - in new system this
 | 
				
			||||||
  // is if CLIMATE_PRESET_AWAY exists is supported_presets
 | 
					  // is if CLIMATE_PRESET_AWAY exists is supported_presets
 | 
				
			||||||
  bool legacy_supports_away = 11;
 | 
					  bool legacy_supports_away = 11;
 | 
				
			||||||
@@ -847,7 +839,6 @@ message ListEntitiesClimateResponse {
 | 
				
			|||||||
  bool disabled_by_default = 18;
 | 
					  bool disabled_by_default = 18;
 | 
				
			||||||
  string icon = 19;
 | 
					  string icon = 19;
 | 
				
			||||||
  EntityCategory entity_category = 20;
 | 
					  EntityCategory entity_category = 20;
 | 
				
			||||||
  float visual_current_temperature_step = 21;
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
message ClimateStateResponse {
 | 
					message ClimateStateResponse {
 | 
				
			||||||
  option (id) = 47;
 | 
					  option (id) = 47;
 | 
				
			||||||
@@ -1130,7 +1121,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";
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
message BluetoothServiceData {
 | 
					message BluetoothServiceData {
 | 
				
			||||||
@@ -1162,7 +1152,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 {
 | 
				
			||||||
@@ -1346,91 +1335,3 @@ message BluetoothGATTNotifyResponse {
 | 
				
			|||||||
  uint64 address = 1;
 | 
					  uint64 address = 1;
 | 
				
			||||||
  uint32 handle = 2;
 | 
					  uint32 handle = 2;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					 | 
				
			||||||
message BluetoothDevicePairingResponse {
 | 
					 | 
				
			||||||
  option (id) = 85;
 | 
					 | 
				
			||||||
  option (source) = SOURCE_SERVER;
 | 
					 | 
				
			||||||
  option (ifdef) = "USE_BLUETOOTH_PROXY";
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  uint64 address = 1;
 | 
					 | 
				
			||||||
  bool paired = 2;
 | 
					 | 
				
			||||||
  int32 error = 3;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
message BluetoothDeviceUnpairingResponse {
 | 
					 | 
				
			||||||
  option (id) = 86;
 | 
					 | 
				
			||||||
  option (source) = SOURCE_SERVER;
 | 
					 | 
				
			||||||
  option (ifdef) = "USE_BLUETOOTH_PROXY";
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  uint64 address = 1;
 | 
					 | 
				
			||||||
  bool success = 2;
 | 
					 | 
				
			||||||
  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;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
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;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 
 | 
				
			|||||||
@@ -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 {
 | 
				
			||||||
@@ -184,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();
 | 
				
			||||||
@@ -217,8 +212,7 @@ 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();
 | 
				
			||||||
@@ -281,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();
 | 
				
			||||||
@@ -344,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();
 | 
				
			||||||
@@ -426,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);
 | 
				
			||||||
@@ -457,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();
 | 
				
			||||||
@@ -543,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();
 | 
				
			||||||
@@ -559,9 +548,7 @@ bool APIConnection::send_climate_info(climate::Climate *climate) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  msg.visual_min_temperature = traits.get_visual_min_temperature();
 | 
					  msg.visual_min_temperature = traits.get_visual_min_temperature();
 | 
				
			||||||
  msg.visual_max_temperature = traits.get_visual_max_temperature();
 | 
					  msg.visual_max_temperature = traits.get_visual_max_temperature();
 | 
				
			||||||
  msg.visual_target_temperature_step = traits.get_visual_target_temperature_step();
 | 
					  msg.visual_temperature_step = traits.get_visual_temperature_step();
 | 
				
			||||||
  msg.visual_current_temperature_step = traits.get_visual_current_temperature_step();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  msg.legacy_supports_away = traits.supports_preset(climate::CLIMATE_PRESET_AWAY);
 | 
					  msg.legacy_supports_away = traits.supports_preset(climate::CLIMATE_PRESET_AWAY);
 | 
				
			||||||
  msg.supports_action = traits.get_supports_action();
 | 
					  msg.supports_action = traits.get_supports_action();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -622,8 +609,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();
 | 
				
			||||||
@@ -664,8 +650,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();
 | 
				
			||||||
@@ -692,8 +677,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();
 | 
				
			||||||
@@ -724,8 +708,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();
 | 
				
			||||||
@@ -770,8 +753,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();
 | 
				
			||||||
@@ -815,8 +797,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();
 | 
				
			||||||
@@ -896,30 +877,6 @@ BluetoothConnectionsFreeResponse APIConnection::subscribe_bluetooth_connections_
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#ifdef USE_VOICE_ASSISTANT
 | 
					 | 
				
			||||||
bool APIConnection::request_voice_assistant(bool start) {
 | 
					 | 
				
			||||||
  if (!this->voice_assistant_subscription_)
 | 
					 | 
				
			||||||
    return false;
 | 
					 | 
				
			||||||
  VoiceAssistantRequest msg;
 | 
					 | 
				
			||||||
  msg.start = start;
 | 
					 | 
				
			||||||
  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
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
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;
 | 
				
			||||||
@@ -939,7 +896,7 @@ 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;
 | 
				
			||||||
@@ -973,7 +930,6 @@ DeviceInfoResponse APIConnection::device_info(const DeviceInfoRequest &msg) {
 | 
				
			|||||||
  DeviceInfoResponse resp{};
 | 
					  DeviceInfoResponse resp{};
 | 
				
			||||||
  resp.uses_password = this->parent_->uses_password();
 | 
					  resp.uses_password = this->parent_->uses_password();
 | 
				
			||||||
  resp.name = App.get_name();
 | 
					  resp.name = App.get_name();
 | 
				
			||||||
  resp.friendly_name = App.get_friendly_name();
 | 
					 | 
				
			||||||
  resp.mac_address = get_mac_address_pretty();
 | 
					  resp.mac_address = get_mac_address_pretty();
 | 
				
			||||||
  resp.esphome_version = ESPHOME_VERSION;
 | 
					  resp.esphome_version = ESPHOME_VERSION;
 | 
				
			||||||
  resp.compilation_time = App.get_compilation_time();
 | 
					  resp.compilation_time = App.get_compilation_time();
 | 
				
			||||||
@@ -994,12 +950,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.bluetooth_proxy_version = bluetooth_proxy::global_bluetooth_proxy->has_active()
 | 
					  resp.bluetooth_proxy_version = bluetooth_proxy::global_bluetooth_proxy->has_active() ? 3 : 1;
 | 
				
			||||||
                                     ? bluetooth_proxy::ACTIVE_CONNECTIONS_VERSION
 | 
					 | 
				
			||||||
                                     : bluetooth_proxy::PASSIVE_ONLY_VERSION;
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
#ifdef USE_VOICE_ASSISTANT
 | 
					 | 
				
			||||||
  resp.voice_assistant_version = 1;
 | 
					 | 
				
			||||||
#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>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -98,12 +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 {
 | 
					 | 
				
			||||||
    this->bluetooth_le_advertisement_subscription_ = true;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  void unsubscribe_bluetooth_le_advertisements(const UnsubscribeBluetoothLEAdvertisementsRequest &msg) override {
 | 
					 | 
				
			||||||
    this->bluetooth_le_advertisement_subscription_ = false;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  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;
 | 
				
			||||||
@@ -124,15 +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);
 | 
					 | 
				
			||||||
  void on_voice_assistant_response(const VoiceAssistantResponse &msg) override;
 | 
					 | 
				
			||||||
  void on_voice_assistant_event_response(const VoiceAssistantEventResponse &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
 | 
				
			||||||
@@ -166,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();
 | 
				
			||||||
@@ -211,12 +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_BLUETOOTH_PROXY
 | 
					 | 
				
			||||||
  bool bluetooth_le_advertisement_subscription_{false};
 | 
					  bool bluetooth_le_advertisement_subscription_{false};
 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
#ifdef USE_VOICE_ASSISTANT
 | 
					 | 
				
			||||||
  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_;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -295,7 +295,7 @@ APIError APINoiseFrameHelper::state_action_() {
 | 
				
			|||||||
    if (aerr != APIError::OK)
 | 
					    if (aerr != APIError::OK)
 | 
				
			||||||
      return aerr;
 | 
					      return aerr;
 | 
				
			||||||
    // ignore contents, may be used in future for flags
 | 
					    // ignore contents, may be used in future for flags
 | 
				
			||||||
    prologue_.push_back((uint8_t) (frame.msg.size() >> 8));
 | 
					    prologue_.push_back((uint8_t)(frame.msg.size() >> 8));
 | 
				
			||||||
    prologue_.push_back((uint8_t) frame.msg.size());
 | 
					    prologue_.push_back((uint8_t) frame.msg.size());
 | 
				
			||||||
    prologue_.insert(prologue_.end(), frame.msg.begin(), frame.msg.end());
 | 
					    prologue_.insert(prologue_.end(), frame.msg.begin(), frame.msg.end());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -492,9 +492,9 @@ APIError APINoiseFrameHelper::write_packet(uint16_t type, const uint8_t *payload
 | 
				
			|||||||
  // tmpbuf[1], tmpbuf[2] to be set later
 | 
					  // tmpbuf[1], tmpbuf[2] to be set later
 | 
				
			||||||
  const uint8_t msg_offset = 3;
 | 
					  const uint8_t msg_offset = 3;
 | 
				
			||||||
  const uint8_t payload_offset = msg_offset + 4;
 | 
					  const uint8_t payload_offset = msg_offset + 4;
 | 
				
			||||||
  tmpbuf[msg_offset + 0] = (uint8_t) (type >> 8);  // type
 | 
					  tmpbuf[msg_offset + 0] = (uint8_t)(type >> 8);  // type
 | 
				
			||||||
  tmpbuf[msg_offset + 1] = (uint8_t) type;
 | 
					  tmpbuf[msg_offset + 1] = (uint8_t) type;
 | 
				
			||||||
  tmpbuf[msg_offset + 2] = (uint8_t) (payload_len >> 8);  // data_len
 | 
					  tmpbuf[msg_offset + 2] = (uint8_t)(payload_len >> 8);  // data_len
 | 
				
			||||||
  tmpbuf[msg_offset + 3] = (uint8_t) payload_len;
 | 
					  tmpbuf[msg_offset + 3] = (uint8_t) payload_len;
 | 
				
			||||||
  // copy data
 | 
					  // copy data
 | 
				
			||||||
  std::copy(payload, payload + payload_len, &tmpbuf[payload_offset]);
 | 
					  std::copy(payload, payload + payload_len, &tmpbuf[payload_offset]);
 | 
				
			||||||
@@ -512,7 +512,7 @@ APIError APINoiseFrameHelper::write_packet(uint16_t type, const uint8_t *payload
 | 
				
			|||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  size_t total_len = 3 + mbuf.size;
 | 
					  size_t total_len = 3 + mbuf.size;
 | 
				
			||||||
  tmpbuf[1] = (uint8_t) (mbuf.size >> 8);
 | 
					  tmpbuf[1] = (uint8_t)(mbuf.size >> 8);
 | 
				
			||||||
  tmpbuf[2] = (uint8_t) mbuf.size;
 | 
					  tmpbuf[2] = (uint8_t) mbuf.size;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  struct iovec iov;
 | 
					  struct iovec iov;
 | 
				
			||||||
@@ -610,15 +610,12 @@ APIError APINoiseFrameHelper::write_raw_(const struct iovec *iov, int iovcnt) {
 | 
				
			|||||||
APIError APINoiseFrameHelper::write_frame_(const uint8_t *data, size_t len) {
 | 
					APIError APINoiseFrameHelper::write_frame_(const uint8_t *data, size_t len) {
 | 
				
			||||||
  uint8_t header[3];
 | 
					  uint8_t header[3];
 | 
				
			||||||
  header[0] = 0x01;  // indicator
 | 
					  header[0] = 0x01;  // indicator
 | 
				
			||||||
  header[1] = (uint8_t) (len >> 8);
 | 
					  header[1] = (uint8_t)(len >> 8);
 | 
				
			||||||
  header[2] = (uint8_t) len;
 | 
					  header[2] = (uint8_t) len;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  struct iovec iov[2];
 | 
					  struct iovec iov[2];
 | 
				
			||||||
  iov[0].iov_base = header;
 | 
					  iov[0].iov_base = header;
 | 
				
			||||||
  iov[0].iov_len = 3;
 | 
					  iov[0].iov_len = 3;
 | 
				
			||||||
  if (len == 0) {
 | 
					 | 
				
			||||||
    return write_raw_(iov, 1);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  iov[1].iov_base = const_cast<uint8_t *>(data);
 | 
					  iov[1].iov_base = const_cast<uint8_t *>(data);
 | 
				
			||||||
  iov[1].iov_len = len;
 | 
					  iov[1].iov_len = len;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -916,9 +913,6 @@ APIError APIPlaintextFrameHelper::write_packet(uint16_t type, const uint8_t *pay
 | 
				
			|||||||
  struct iovec iov[2];
 | 
					  struct iovec iov[2];
 | 
				
			||||||
  iov[0].iov_base = &header[0];
 | 
					  iov[0].iov_base = &header[0];
 | 
				
			||||||
  iov[0].iov_len = header.size();
 | 
					  iov[0].iov_len = header.size();
 | 
				
			||||||
  if (payload_len == 0) {
 | 
					 | 
				
			||||||
    return write_raw_(iov, 1);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  iov[1].iov_base = const_cast<uint8_t *>(payload);
 | 
					  iov[1].iov_base = const_cast<uint8_t *>(payload);
 | 
				
			||||||
  iov[1].iov_len = payload_len;
 | 
					  iov[1].iov_len = payload_len;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -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
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -235,8 +235,6 @@ template<> const char *proto_enum_to_string<enums::ClimateFanMode>(enums::Climat
 | 
				
			|||||||
      return "CLIMATE_FAN_FOCUS";
 | 
					      return "CLIMATE_FAN_FOCUS";
 | 
				
			||||||
    case enums::CLIMATE_FAN_DIFFUSE:
 | 
					    case enums::CLIMATE_FAN_DIFFUSE:
 | 
				
			||||||
      return "CLIMATE_FAN_DIFFUSE";
 | 
					      return "CLIMATE_FAN_DIFFUSE";
 | 
				
			||||||
    case enums::CLIMATE_FAN_QUIET:
 | 
					 | 
				
			||||||
      return "CLIMATE_FAN_QUIET";
 | 
					 | 
				
			||||||
    default:
 | 
					    default:
 | 
				
			||||||
      return "UNKNOWN";
 | 
					      return "UNKNOWN";
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
@@ -400,34 +398,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:
 | 
					    default:
 | 
				
			||||||
      return "UNKNOWN";
 | 
					      return "UNKNOWN";
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
@@ -620,10 +590,6 @@ bool DeviceInfoResponse::decode_varint(uint32_t field_id, ProtoVarInt value) {
 | 
				
			|||||||
      this->bluetooth_proxy_version = value.as_uint32();
 | 
					      this->bluetooth_proxy_version = value.as_uint32();
 | 
				
			||||||
      return true;
 | 
					      return true;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    case 14: {
 | 
					 | 
				
			||||||
      this->voice_assistant_version = value.as_uint32();
 | 
					 | 
				
			||||||
      return true;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    default:
 | 
					    default:
 | 
				
			||||||
      return false;
 | 
					      return false;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
@@ -662,10 +628,6 @@ bool DeviceInfoResponse::decode_length(uint32_t field_id, ProtoLengthDelimited v
 | 
				
			|||||||
      this->manufacturer = value.as_string();
 | 
					      this->manufacturer = value.as_string();
 | 
				
			||||||
      return true;
 | 
					      return true;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    case 13: {
 | 
					 | 
				
			||||||
      this->friendly_name = value.as_string();
 | 
					 | 
				
			||||||
      return true;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    default:
 | 
					    default:
 | 
				
			||||||
      return false;
 | 
					      return false;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
@@ -683,8 +645,6 @@ void DeviceInfoResponse::encode(ProtoWriteBuffer buffer) const {
 | 
				
			|||||||
  buffer.encode_uint32(10, this->webserver_port);
 | 
					  buffer.encode_uint32(10, this->webserver_port);
 | 
				
			||||||
  buffer.encode_uint32(11, this->bluetooth_proxy_version);
 | 
					  buffer.encode_uint32(11, this->bluetooth_proxy_version);
 | 
				
			||||||
  buffer.encode_string(12, this->manufacturer);
 | 
					  buffer.encode_string(12, this->manufacturer);
 | 
				
			||||||
  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 {
 | 
				
			||||||
@@ -739,15 +699,6 @@ void DeviceInfoResponse::dump_to(std::string &out) const {
 | 
				
			|||||||
  out.append("  manufacturer: ");
 | 
					  out.append("  manufacturer: ");
 | 
				
			||||||
  out.append("'").append(this->manufacturer).append("'");
 | 
					  out.append("'").append(this->manufacturer).append("'");
 | 
				
			||||||
  out.append("\n");
 | 
					  out.append("\n");
 | 
				
			||||||
 | 
					 | 
				
			||||||
  out.append("  friendly_name: ");
 | 
					 | 
				
			||||||
  out.append("'").append(this->friendly_name).append("'");
 | 
					 | 
				
			||||||
  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
 | 
				
			||||||
@@ -3489,11 +3440,7 @@ bool ListEntitiesClimateResponse::decode_32bit(uint32_t field_id, Proto32Bit val
 | 
				
			|||||||
      return true;
 | 
					      return true;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    case 10: {
 | 
					    case 10: {
 | 
				
			||||||
      this->visual_target_temperature_step = value.as_float();
 | 
					      this->visual_temperature_step = value.as_float();
 | 
				
			||||||
      return true;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    case 21: {
 | 
					 | 
				
			||||||
      this->visual_current_temperature_step = value.as_float();
 | 
					 | 
				
			||||||
      return true;
 | 
					      return true;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    default:
 | 
					    default:
 | 
				
			||||||
@@ -3512,7 +3459,7 @@ void ListEntitiesClimateResponse::encode(ProtoWriteBuffer buffer) const {
 | 
				
			|||||||
  }
 | 
					  }
 | 
				
			||||||
  buffer.encode_float(8, this->visual_min_temperature);
 | 
					  buffer.encode_float(8, this->visual_min_temperature);
 | 
				
			||||||
  buffer.encode_float(9, this->visual_max_temperature);
 | 
					  buffer.encode_float(9, this->visual_max_temperature);
 | 
				
			||||||
  buffer.encode_float(10, this->visual_target_temperature_step);
 | 
					  buffer.encode_float(10, this->visual_temperature_step);
 | 
				
			||||||
  buffer.encode_bool(11, this->legacy_supports_away);
 | 
					  buffer.encode_bool(11, this->legacy_supports_away);
 | 
				
			||||||
  buffer.encode_bool(12, this->supports_action);
 | 
					  buffer.encode_bool(12, this->supports_action);
 | 
				
			||||||
  for (auto &it : this->supported_fan_modes) {
 | 
					  for (auto &it : this->supported_fan_modes) {
 | 
				
			||||||
@@ -3533,7 +3480,6 @@ void ListEntitiesClimateResponse::encode(ProtoWriteBuffer buffer) const {
 | 
				
			|||||||
  buffer.encode_bool(18, this->disabled_by_default);
 | 
					  buffer.encode_bool(18, this->disabled_by_default);
 | 
				
			||||||
  buffer.encode_string(19, this->icon);
 | 
					  buffer.encode_string(19, this->icon);
 | 
				
			||||||
  buffer.encode_enum<enums::EntityCategory>(20, this->entity_category);
 | 
					  buffer.encode_enum<enums::EntityCategory>(20, this->entity_category);
 | 
				
			||||||
  buffer.encode_float(21, this->visual_current_temperature_step);
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
 | 
					#ifdef HAS_PROTO_MESSAGE_DUMP
 | 
				
			||||||
void ListEntitiesClimateResponse::dump_to(std::string &out) const {
 | 
					void ListEntitiesClimateResponse::dump_to(std::string &out) const {
 | 
				
			||||||
@@ -3580,8 +3526,8 @@ void ListEntitiesClimateResponse::dump_to(std::string &out) const {
 | 
				
			|||||||
  out.append(buffer);
 | 
					  out.append(buffer);
 | 
				
			||||||
  out.append("\n");
 | 
					  out.append("\n");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  out.append("  visual_target_temperature_step: ");
 | 
					  out.append("  visual_temperature_step: ");
 | 
				
			||||||
  sprintf(buffer, "%g", this->visual_target_temperature_step);
 | 
					  sprintf(buffer, "%g", this->visual_temperature_step);
 | 
				
			||||||
  out.append(buffer);
 | 
					  out.append(buffer);
 | 
				
			||||||
  out.append("\n");
 | 
					  out.append("\n");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -3634,11 +3580,6 @@ void ListEntitiesClimateResponse::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("  visual_current_temperature_step: ");
 | 
					 | 
				
			||||||
  sprintf(buffer, "%g", this->visual_current_temperature_step);
 | 
					 | 
				
			||||||
  out.append(buffer);
 | 
					 | 
				
			||||||
  out.append("\n");
 | 
					 | 
				
			||||||
  out.append("}");
 | 
					  out.append("}");
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
@@ -6012,290 +5953,6 @@ void BluetoothGATTNotifyResponse::dump_to(std::string &out) const {
 | 
				
			|||||||
  out.append("}");
 | 
					  out.append("}");
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
bool BluetoothDevicePairingResponse::decode_varint(uint32_t field_id, ProtoVarInt value) {
 | 
					 | 
				
			||||||
  switch (field_id) {
 | 
					 | 
				
			||||||
    case 1: {
 | 
					 | 
				
			||||||
      this->address = value.as_uint64();
 | 
					 | 
				
			||||||
      return true;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    case 2: {
 | 
					 | 
				
			||||||
      this->paired = value.as_bool();
 | 
					 | 
				
			||||||
      return true;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    case 3: {
 | 
					 | 
				
			||||||
      this->error = value.as_int32();
 | 
					 | 
				
			||||||
      return true;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    default:
 | 
					 | 
				
			||||||
      return false;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
void BluetoothDevicePairingResponse::encode(ProtoWriteBuffer buffer) const {
 | 
					 | 
				
			||||||
  buffer.encode_uint64(1, this->address);
 | 
					 | 
				
			||||||
  buffer.encode_bool(2, this->paired);
 | 
					 | 
				
			||||||
  buffer.encode_int32(3, this->error);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
 | 
					 | 
				
			||||||
void BluetoothDevicePairingResponse::dump_to(std::string &out) const {
 | 
					 | 
				
			||||||
  __attribute__((unused)) char buffer[64];
 | 
					 | 
				
			||||||
  out.append("BluetoothDevicePairingResponse {\n");
 | 
					 | 
				
			||||||
  out.append("  address: ");
 | 
					 | 
				
			||||||
  sprintf(buffer, "%llu", this->address);
 | 
					 | 
				
			||||||
  out.append(buffer);
 | 
					 | 
				
			||||||
  out.append("\n");
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  out.append("  paired: ");
 | 
					 | 
				
			||||||
  out.append(YESNO(this->paired));
 | 
					 | 
				
			||||||
  out.append("\n");
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  out.append("  error: ");
 | 
					 | 
				
			||||||
  sprintf(buffer, "%d", this->error);
 | 
					 | 
				
			||||||
  out.append(buffer);
 | 
					 | 
				
			||||||
  out.append("\n");
 | 
					 | 
				
			||||||
  out.append("}");
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
bool BluetoothDeviceUnpairingResponse::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 BluetoothDeviceUnpairingResponse::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 BluetoothDeviceUnpairingResponse::dump_to(std::string &out) const {
 | 
					 | 
				
			||||||
  __attribute__((unused)) char buffer[64];
 | 
					 | 
				
			||||||
  out.append("BluetoothDeviceUnpairingResponse {\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
 | 
					 | 
				
			||||||
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;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
void VoiceAssistantRequest::encode(ProtoWriteBuffer buffer) const { buffer.encode_bool(1, this->start); }
 | 
					 | 
				
			||||||
#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("}");
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
#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
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
}  // namespace api
 | 
					}  // namespace api
 | 
				
			||||||
}  // namespace esphome
 | 
					}  // namespace esphome
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -99,7 +99,6 @@ enum ClimateFanMode : uint32_t {
 | 
				
			|||||||
  CLIMATE_FAN_MIDDLE = 6,
 | 
					  CLIMATE_FAN_MIDDLE = 6,
 | 
				
			||||||
  CLIMATE_FAN_FOCUS = 7,
 | 
					  CLIMATE_FAN_FOCUS = 7,
 | 
				
			||||||
  CLIMATE_FAN_DIFFUSE = 8,
 | 
					  CLIMATE_FAN_DIFFUSE = 8,
 | 
				
			||||||
  CLIMATE_FAN_QUIET = 9,
 | 
					 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
enum ClimateSwingMode : uint32_t {
 | 
					enum ClimateSwingMode : uint32_t {
 | 
				
			||||||
  CLIMATE_SWING_OFF = 0,
 | 
					  CLIMATE_SWING_OFF = 0,
 | 
				
			||||||
@@ -163,18 +162,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,
 | 
					 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
}  // namespace enums
 | 
					}  // namespace enums
 | 
				
			||||||
@@ -289,8 +276,6 @@ class DeviceInfoResponse : public ProtoMessage {
 | 
				
			|||||||
  uint32_t webserver_port{0};
 | 
					  uint32_t webserver_port{0};
 | 
				
			||||||
  uint32_t bluetooth_proxy_version{0};
 | 
					  uint32_t bluetooth_proxy_version{0};
 | 
				
			||||||
  std::string manufacturer{};
 | 
					  std::string manufacturer{};
 | 
				
			||||||
  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;
 | 
				
			||||||
@@ -928,7 +913,7 @@ class ListEntitiesClimateResponse : public ProtoMessage {
 | 
				
			|||||||
  std::vector<enums::ClimateMode> supported_modes{};
 | 
					  std::vector<enums::ClimateMode> supported_modes{};
 | 
				
			||||||
  float visual_min_temperature{0.0f};
 | 
					  float visual_min_temperature{0.0f};
 | 
				
			||||||
  float visual_max_temperature{0.0f};
 | 
					  float visual_max_temperature{0.0f};
 | 
				
			||||||
  float visual_target_temperature_step{0.0f};
 | 
					  float visual_temperature_step{0.0f};
 | 
				
			||||||
  bool legacy_supports_away{false};
 | 
					  bool legacy_supports_away{false};
 | 
				
			||||||
  bool supports_action{false};
 | 
					  bool supports_action{false};
 | 
				
			||||||
  std::vector<enums::ClimateFanMode> supported_fan_modes{};
 | 
					  std::vector<enums::ClimateFanMode> supported_fan_modes{};
 | 
				
			||||||
@@ -939,7 +924,6 @@ class ListEntitiesClimateResponse : 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{};
 | 
				
			||||||
  float visual_current_temperature_step{0.0f};
 | 
					 | 
				
			||||||
  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;
 | 
				
			||||||
@@ -1541,113 +1525,6 @@ class BluetoothGATTNotifyResponse : 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 BluetoothDevicePairingResponse : public ProtoMessage {
 | 
					 | 
				
			||||||
 public:
 | 
					 | 
				
			||||||
  uint64_t address{0};
 | 
					 | 
				
			||||||
  bool paired{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 BluetoothDeviceUnpairingResponse : 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 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};
 | 
					 | 
				
			||||||
  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 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;
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
}  // 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());
 | 
				
			||||||
@@ -427,46 +425,6 @@ bool APIServerConnectionBase::send_bluetooth_gatt_notify_response(const Bluetoot
 | 
				
			|||||||
  return this->send_message_<BluetoothGATTNotifyResponse>(msg, 84);
 | 
					  return this->send_message_<BluetoothGATTNotifyResponse>(msg, 84);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
#ifdef USE_BLUETOOTH_PROXY
 | 
					 | 
				
			||||||
bool APIServerConnectionBase::send_bluetooth_device_pairing_response(const BluetoothDevicePairingResponse &msg) {
 | 
					 | 
				
			||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
 | 
					 | 
				
			||||||
  ESP_LOGVV(TAG, "send_bluetooth_device_pairing_response: %s", msg.dump().c_str());
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
  return this->send_message_<BluetoothDevicePairingResponse>(msg, 85);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
#ifdef USE_BLUETOOTH_PROXY
 | 
					 | 
				
			||||||
bool APIServerConnectionBase::send_bluetooth_device_unpairing_response(const BluetoothDeviceUnpairingResponse &msg) {
 | 
					 | 
				
			||||||
#ifdef HAS_PROTO_MESSAGE_DUMP
 | 
					 | 
				
			||||||
  ESP_LOGVV(TAG, "send_bluetooth_device_unpairing_response: %s", msg.dump().c_str());
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
  return this->send_message_<BluetoothDeviceUnpairingResponse>(msg, 86);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
#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
 | 
					 | 
				
			||||||
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: {
 | 
				
			||||||
@@ -735,14 +693,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: {
 | 
				
			||||||
@@ -830,50 +786,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
 | 
					#endif
 | 
				
			||||||
      break;
 | 
					      break;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@@ -1137,7 +1049,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()) {
 | 
				
			||||||
@@ -1150,7 +1061,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()) {
 | 
				
			||||||
@@ -1259,33 +1169,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
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
}  // namespace api
 | 
					}  // namespace api
 | 
				
			||||||
}  // namespace esphome
 | 
					}  // namespace esphome
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -154,10 +154,8 @@ 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
 | 
				
			||||||
@@ -211,31 +209,6 @@ class APIServerConnectionBase : public ProtoService {
 | 
				
			|||||||
#endif
 | 
					#endif
 | 
				
			||||||
#ifdef USE_BLUETOOTH_PROXY
 | 
					#ifdef USE_BLUETOOTH_PROXY
 | 
				
			||||||
  bool send_bluetooth_gatt_notify_response(const BluetoothGATTNotifyResponse &msg);
 | 
					  bool send_bluetooth_gatt_notify_response(const BluetoothGATTNotifyResponse &msg);
 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
#ifdef USE_BLUETOOTH_PROXY
 | 
					 | 
				
			||||||
  bool send_bluetooth_device_pairing_response(const BluetoothDevicePairingResponse &msg);
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
#ifdef USE_BLUETOOTH_PROXY
 | 
					 | 
				
			||||||
  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
 | 
					#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;
 | 
				
			||||||
@@ -288,9 +261,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
 | 
				
			||||||
@@ -315,12 +286,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
 | 
					#endif
 | 
				
			||||||
 protected:
 | 
					 protected:
 | 
				
			||||||
  void on_hello_request(const HelloRequest &msg) override;
 | 
					  void on_hello_request(const HelloRequest &msg) override;
 | 
				
			||||||
@@ -368,9 +333,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
 | 
				
			||||||
@@ -395,13 +358,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
 | 
					 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
}  // 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();
 | 
				
			||||||
@@ -309,39 +309,6 @@ void APIServer::send_bluetooth_device_connection(uint64_t address, bool connecte
 | 
				
			|||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
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_device_clear_cache(uint64_t address, bool success, esp_err_t error) {
 | 
					 | 
				
			||||||
  BluetoothDeviceClearCacheResponse call;
 | 
					 | 
				
			||||||
  call.address = address;
 | 
					 | 
				
			||||||
  call.success = success;
 | 
					 | 
				
			||||||
  call.error = error;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  for (auto &client : this->clients_) {
 | 
					 | 
				
			||||||
    client->send_bluetooth_device_clear_cache_response(call);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void APIServer::send_bluetooth_connections_free(uint8_t free, uint8_t limit) {
 | 
					void APIServer::send_bluetooth_connections_free(uint8_t free, uint8_t limit) {
 | 
				
			||||||
  BluetoothConnectionsFreeResponse call;
 | 
					  BluetoothConnectionsFreeResponse call;
 | 
				
			||||||
  call.free = free;
 | 
					  call.free = free;
 | 
				
			||||||
@@ -427,18 +394,5 @@ void APIServer::on_shutdown() {
 | 
				
			|||||||
  delay(10);
 | 
					  delay(10);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#ifdef USE_VOICE_ASSISTANT
 | 
					 | 
				
			||||||
void APIServer::start_voice_assistant() {
 | 
					 | 
				
			||||||
  for (auto &c : this->clients_) {
 | 
					 | 
				
			||||||
    c->request_voice_assistant(true);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
void APIServer::stop_voice_assistant() {
 | 
					 | 
				
			||||||
  for (auto &c : this->clients_) {
 | 
					 | 
				
			||||||
    c->request_voice_assistant(false);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
}  // namespace api
 | 
					}  // namespace api
 | 
				
			||||||
}  // namespace esphome
 | 
					}  // namespace esphome
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -78,9 +78,6 @@ class APIServer : public Component, public Controller {
 | 
				
			|||||||
#ifdef USE_BLUETOOTH_PROXY
 | 
					#ifdef USE_BLUETOOTH_PROXY
 | 
				
			||||||
  void send_bluetooth_le_advertisement(const BluetoothLEAdvertisementResponse &call);
 | 
					  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_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_device_clear_cache(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_connections_free(uint8_t free, uint8_t limit);
 | 
				
			||||||
  void send_bluetooth_gatt_read_response(const BluetoothGATTReadResponse &call);
 | 
					  void send_bluetooth_gatt_read_response(const BluetoothGATTReadResponse &call);
 | 
				
			||||||
  void send_bluetooth_gatt_write_response(const BluetoothGATTWriteResponse &call);
 | 
					  void send_bluetooth_gatt_write_response(const BluetoothGATTWriteResponse &call);
 | 
				
			||||||
@@ -95,11 +92,6 @@ class APIServer : public Component, public Controller {
 | 
				
			|||||||
  void request_time();
 | 
					  void request_time();
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#ifdef USE_VOICE_ASSISTANT
 | 
					 | 
				
			||||||
  void start_voice_assistant();
 | 
					 | 
				
			||||||
  void stop_voice_assistant();
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  bool is_connected() const;
 | 
					  bool is_connected() const;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  struct HomeAssistantStateSubscription {
 | 
					  struct HomeAssistantStateSubscription {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -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;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -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_;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,271 +0,0 @@
 | 
				
			|||||||
#include "as7341.h"
 | 
					 | 
				
			||||||
#include "esphome/core/log.h"
 | 
					 | 
				
			||||||
#include "esphome/core/hal.h"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
namespace esphome {
 | 
					 | 
				
			||||||
namespace as7341 {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static const char *const TAG = "as7341";
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void AS7341Component::setup() {
 | 
					 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "Setting up AS7341...");
 | 
					 | 
				
			||||||
  LOG_I2C_DEVICE(this);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  // Verify device ID
 | 
					 | 
				
			||||||
  uint8_t id;
 | 
					 | 
				
			||||||
  this->read_byte(AS7341_ID, &id);
 | 
					 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "  Read ID: 0x%X", id);
 | 
					 | 
				
			||||||
  if ((id & 0xFC) != (AS7341_CHIP_ID << 2)) {
 | 
					 | 
				
			||||||
    this->mark_failed();
 | 
					 | 
				
			||||||
    return;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  // Power on (enter IDLE state)
 | 
					 | 
				
			||||||
  if (!this->enable_power(true)) {
 | 
					 | 
				
			||||||
    ESP_LOGE(TAG, "  Power on failed!");
 | 
					 | 
				
			||||||
    this->mark_failed();
 | 
					 | 
				
			||||||
    return;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  // Set configuration
 | 
					 | 
				
			||||||
  this->write_byte(AS7341_CONFIG, 0x00);
 | 
					 | 
				
			||||||
  this->setup_atime(this->atime_);
 | 
					 | 
				
			||||||
  this->setup_astep(this->astep_);
 | 
					 | 
				
			||||||
  this->setup_gain(this->gain_);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void AS7341Component::dump_config() {
 | 
					 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "AS7341:");
 | 
					 | 
				
			||||||
  LOG_I2C_DEVICE(this);
 | 
					 | 
				
			||||||
  if (this->is_failed()) {
 | 
					 | 
				
			||||||
    ESP_LOGE(TAG, "Communication with AS7341 failed!");
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  LOG_UPDATE_INTERVAL(this);
 | 
					 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "  Gain: %u", get_gain());
 | 
					 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "  ATIME: %u", get_atime());
 | 
					 | 
				
			||||||
  ESP_LOGCONFIG(TAG, "  ASTEP: %u", get_astep());
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  LOG_SENSOR("  ", "F1", this->f1_);
 | 
					 | 
				
			||||||
  LOG_SENSOR("  ", "F2", this->f2_);
 | 
					 | 
				
			||||||
  LOG_SENSOR("  ", "F3", this->f3_);
 | 
					 | 
				
			||||||
  LOG_SENSOR("  ", "F4", this->f4_);
 | 
					 | 
				
			||||||
  LOG_SENSOR("  ", "F5", this->f5_);
 | 
					 | 
				
			||||||
  LOG_SENSOR("  ", "F6", this->f6_);
 | 
					 | 
				
			||||||
  LOG_SENSOR("  ", "F7", this->f7_);
 | 
					 | 
				
			||||||
  LOG_SENSOR("  ", "F8", this->f8_);
 | 
					 | 
				
			||||||
  LOG_SENSOR("  ", "Clear", this->clear_);
 | 
					 | 
				
			||||||
  LOG_SENSOR("  ", "NIR", this->nir_);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
float AS7341Component::get_setup_priority() const { return setup_priority::DATA; }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void AS7341Component::update() {
 | 
					 | 
				
			||||||
  this->read_channels(this->channel_readings_);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  if (this->f1_ != nullptr) {
 | 
					 | 
				
			||||||
    this->f1_->publish_state(this->channel_readings_[0]);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  if (this->f2_ != nullptr) {
 | 
					 | 
				
			||||||
    this->f2_->publish_state(this->channel_readings_[1]);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  if (this->f3_ != nullptr) {
 | 
					 | 
				
			||||||
    this->f3_->publish_state(this->channel_readings_[2]);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  if (this->f4_ != nullptr) {
 | 
					 | 
				
			||||||
    this->f4_->publish_state(this->channel_readings_[3]);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  if (this->f5_ != nullptr) {
 | 
					 | 
				
			||||||
    this->f5_->publish_state(this->channel_readings_[6]);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  if (this->f6_ != nullptr) {
 | 
					 | 
				
			||||||
    this->f6_->publish_state(this->channel_readings_[7]);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  if (this->f7_ != nullptr) {
 | 
					 | 
				
			||||||
    this->f7_->publish_state(this->channel_readings_[8]);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  if (this->f8_ != nullptr) {
 | 
					 | 
				
			||||||
    this->f8_->publish_state(this->channel_readings_[9]);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  if (this->clear_ != nullptr) {
 | 
					 | 
				
			||||||
    this->clear_->publish_state(this->channel_readings_[10]);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  if (this->nir_ != nullptr) {
 | 
					 | 
				
			||||||
    this->nir_->publish_state(this->channel_readings_[11]);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
AS7341Gain AS7341Component::get_gain() {
 | 
					 | 
				
			||||||
  uint8_t data;
 | 
					 | 
				
			||||||
  this->read_byte(AS7341_CFG1, &data);
 | 
					 | 
				
			||||||
  return (AS7341Gain) data;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
uint8_t AS7341Component::get_atime() {
 | 
					 | 
				
			||||||
  uint8_t data;
 | 
					 | 
				
			||||||
  this->read_byte(AS7341_ATIME, &data);
 | 
					 | 
				
			||||||
  return data;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
uint16_t AS7341Component::get_astep() {
 | 
					 | 
				
			||||||
  uint16_t data;
 | 
					 | 
				
			||||||
  this->read_byte_16(AS7341_ASTEP, &data);
 | 
					 | 
				
			||||||
  return this->swap_bytes(data);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::setup_gain(AS7341Gain gain) { return this->write_byte(AS7341_CFG1, gain); }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::setup_atime(uint8_t atime) { return this->write_byte(AS7341_ATIME, atime); }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::setup_astep(uint16_t astep) { return this->write_byte_16(AS7341_ASTEP, swap_bytes(astep)); }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::read_channels(uint16_t *data) {
 | 
					 | 
				
			||||||
  this->set_smux_low_channels(true);
 | 
					 | 
				
			||||||
  this->enable_spectral_measurement(true);
 | 
					 | 
				
			||||||
  this->wait_for_data();
 | 
					 | 
				
			||||||
  bool low_success = this->read_bytes_16(AS7341_CH0_DATA_L, data, 6);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  this->set_smux_low_channels(false);
 | 
					 | 
				
			||||||
  this->enable_spectral_measurement(true);
 | 
					 | 
				
			||||||
  this->wait_for_data();
 | 
					 | 
				
			||||||
  bool high_sucess = this->read_bytes_16(AS7341_CH0_DATA_L, &data[6], 6);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  return low_success && high_sucess;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void AS7341Component::set_smux_low_channels(bool enable) {
 | 
					 | 
				
			||||||
  this->enable_spectral_measurement(false);
 | 
					 | 
				
			||||||
  this->set_smux_command(AS7341_SMUX_CMD_WRITE);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  if (enable) {
 | 
					 | 
				
			||||||
    this->configure_smux_low_channels();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  } else {
 | 
					 | 
				
			||||||
    this->configure_smux_high_channels();
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  this->enable_smux();
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::set_smux_command(AS7341SmuxCommand command) {
 | 
					 | 
				
			||||||
  uint8_t data = command << 3;  // Write to bits 4:3 of the register
 | 
					 | 
				
			||||||
  return this->write_byte(AS7341_CFG6, data);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void AS7341Component::configure_smux_low_channels() {
 | 
					 | 
				
			||||||
  // SMUX Config for F1,F2,F3,F4,NIR,Clear
 | 
					 | 
				
			||||||
  this->write_byte(0x00, 0x30);  // F3 left set to ADC2
 | 
					 | 
				
			||||||
  this->write_byte(0x01, 0x01);  // F1 left set to ADC0
 | 
					 | 
				
			||||||
  this->write_byte(0x02, 0x00);  // Reserved or disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x03, 0x00);  // F8 left disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x04, 0x00);  // F6 left disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x05, 0x42);  // F4 left connected to ADC3/f2 left connected to ADC1
 | 
					 | 
				
			||||||
  this->write_byte(0x06, 0x00);  // F5 left disbled
 | 
					 | 
				
			||||||
  this->write_byte(0x07, 0x00);  // F7 left disbled
 | 
					 | 
				
			||||||
  this->write_byte(0x08, 0x50);  // CLEAR connected to ADC4
 | 
					 | 
				
			||||||
  this->write_byte(0x09, 0x00);  // F5 right disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x0A, 0x00);  // F7 right disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x0B, 0x00);  // Reserved or disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x0C, 0x20);  // F2 right connected to ADC1
 | 
					 | 
				
			||||||
  this->write_byte(0x0D, 0x04);  // F4 right connected to ADC3
 | 
					 | 
				
			||||||
  this->write_byte(0x0E, 0x00);  // F6/F8 right disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x0F, 0x30);  // F3 right connected to AD2
 | 
					 | 
				
			||||||
  this->write_byte(0x10, 0x01);  // F1 right connected to AD0
 | 
					 | 
				
			||||||
  this->write_byte(0x11, 0x50);  // CLEAR right connected to AD4
 | 
					 | 
				
			||||||
  this->write_byte(0x12, 0x00);  // Reserved or disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x13, 0x06);  // NIR connected to ADC5
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void AS7341Component::configure_smux_high_channels() {
 | 
					 | 
				
			||||||
  // SMUX Config for F5,F6,F7,F8,NIR,Clear
 | 
					 | 
				
			||||||
  this->write_byte(0x00, 0x00);  // F3 left disable
 | 
					 | 
				
			||||||
  this->write_byte(0x01, 0x00);  // F1 left disable
 | 
					 | 
				
			||||||
  this->write_byte(0x02, 0x00);  // reserved/disable
 | 
					 | 
				
			||||||
  this->write_byte(0x03, 0x40);  // F8 left connected to ADC3
 | 
					 | 
				
			||||||
  this->write_byte(0x04, 0x02);  // F6 left connected to ADC1
 | 
					 | 
				
			||||||
  this->write_byte(0x05, 0x00);  // F4/ F2 disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x06, 0x10);  // F5 left connected to ADC0
 | 
					 | 
				
			||||||
  this->write_byte(0x07, 0x03);  // F7 left connected to ADC2
 | 
					 | 
				
			||||||
  this->write_byte(0x08, 0x50);  // CLEAR Connected to ADC4
 | 
					 | 
				
			||||||
  this->write_byte(0x09, 0x10);  // F5 right connected to ADC0
 | 
					 | 
				
			||||||
  this->write_byte(0x0A, 0x03);  // F7 right connected to ADC2
 | 
					 | 
				
			||||||
  this->write_byte(0x0B, 0x00);  // Reserved or disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x0C, 0x00);  // F2 right disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x0D, 0x00);  // F4 right disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x0E, 0x24);  // F8 right connected to ADC2/ F6 right connected to ADC1
 | 
					 | 
				
			||||||
  this->write_byte(0x0F, 0x00);  // F3 right disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x10, 0x00);  // F1 right disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x11, 0x50);  // CLEAR right connected to AD4
 | 
					 | 
				
			||||||
  this->write_byte(0x12, 0x00);  // Reserved or disabled
 | 
					 | 
				
			||||||
  this->write_byte(0x13, 0x06);  // NIR connected to ADC5
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::enable_smux() {
 | 
					 | 
				
			||||||
  this->set_register_bit(AS7341_ENABLE, 4);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  uint16_t timeout = 1000;
 | 
					 | 
				
			||||||
  for (uint16_t time = 0; time < timeout; time++) {
 | 
					 | 
				
			||||||
    // The SMUXEN bit is cleared once the SMUX operation is finished
 | 
					 | 
				
			||||||
    bool smuxen = this->read_register_bit(AS7341_ENABLE, 4);
 | 
					 | 
				
			||||||
    if (!smuxen) {
 | 
					 | 
				
			||||||
      return true;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    delay(1);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  return false;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::wait_for_data() {
 | 
					 | 
				
			||||||
  uint16_t timeout = 1000;
 | 
					 | 
				
			||||||
  for (uint16_t time = 0; time < timeout; time++) {
 | 
					 | 
				
			||||||
    if (this->is_data_ready()) {
 | 
					 | 
				
			||||||
      return true;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    delay(1);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  return false;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::is_data_ready() { return this->read_register_bit(AS7341_STATUS2, 6); }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::enable_power(bool enable) { return this->write_register_bit(AS7341_ENABLE, enable, 0); }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::enable_spectral_measurement(bool enable) {
 | 
					 | 
				
			||||||
  return this->write_register_bit(AS7341_ENABLE, enable, 1);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::read_register_bit(uint8_t address, uint8_t bit_position) {
 | 
					 | 
				
			||||||
  uint8_t data;
 | 
					 | 
				
			||||||
  this->read_byte(address, &data);
 | 
					 | 
				
			||||||
  bool bit = (data & (1 << bit_position)) > 0;
 | 
					 | 
				
			||||||
  return bit;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::write_register_bit(uint8_t address, bool value, uint8_t bit_position) {
 | 
					 | 
				
			||||||
  if (value) {
 | 
					 | 
				
			||||||
    return this->set_register_bit(address, bit_position);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  return this->clear_register_bit(address, bit_position);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::set_register_bit(uint8_t address, uint8_t bit_position) {
 | 
					 | 
				
			||||||
  uint8_t data;
 | 
					 | 
				
			||||||
  this->read_byte(address, &data);
 | 
					 | 
				
			||||||
  data |= (1 << bit_position);
 | 
					 | 
				
			||||||
  return this->write_byte(address, data);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
bool AS7341Component::clear_register_bit(uint8_t address, uint8_t bit_position) {
 | 
					 | 
				
			||||||
  uint8_t data;
 | 
					 | 
				
			||||||
  this->read_byte(address, &data);
 | 
					 | 
				
			||||||
  data &= ~(1 << bit_position);
 | 
					 | 
				
			||||||
  return this->write_byte(address, data);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
uint16_t AS7341Component::swap_bytes(uint16_t data) { return (data >> 8) | (data << 8); }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
}  // namespace as7341
 | 
					 | 
				
			||||||
}  // namespace esphome
 | 
					 | 
				
			||||||
@@ -1,144 +0,0 @@
 | 
				
			|||||||
#pragma once
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#include "esphome/core/component.h"
 | 
					 | 
				
			||||||
#include "esphome/components/sensor/sensor.h"
 | 
					 | 
				
			||||||
#include "esphome/components/i2c/i2c.h"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
namespace esphome {
 | 
					 | 
				
			||||||
namespace as7341 {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CHIP_ID = 0X09;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CONFIG = 0x70;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_LED = 0x74;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static const uint8_t AS7341_ENABLE = 0x80;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_ATIME = 0x81;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static const uint8_t AS7341_WTIME = 0x83;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static const uint8_t AS7341_AUXID = 0x90;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_REVID = 0x91;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_ID = 0x92;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_STATUS = 0x93;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CH0_DATA_L = 0x95;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CH0_DATA_H = 0x96;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CH1_DATA_L = 0x97;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CH1_DATA_H = 0x98;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CH2_DATA_L = 0x99;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CH2_DATA_H = 0x9A;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CH3_DATA_L = 0x9B;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CH3_DATA_H = 0x9C;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CH4_DATA_L = 0x9D;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CH4_DATA_H = 0x9E;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CH5_DATA_L = 0x9F;
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CH5_DATA_H = 0xA0;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static const uint8_t AS7341_STATUS2 = 0xA3;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CFG1 = 0xAA;  ///< Controls ADC Gain
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CFG6 = 0xAF;  // Stores SMUX command
 | 
					 | 
				
			||||||
static const uint8_t AS7341_CFG9 = 0xB2;  // Config for system interrupts (SMUX, Flicker detection)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static const uint8_t AS7341_ASTEP = 0xCA;      // LSB
 | 
					 | 
				
			||||||
static const uint8_t AS7341_ASTEP_MSB = 0xCB;  // MSB
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
enum AS7341AdcChannel {
 | 
					 | 
				
			||||||
  AS7341_ADC_CHANNEL_0,
 | 
					 | 
				
			||||||
  AS7341_ADC_CHANNEL_1,
 | 
					 | 
				
			||||||
  AS7341_ADC_CHANNEL_2,
 | 
					 | 
				
			||||||
  AS7341_ADC_CHANNEL_3,
 | 
					 | 
				
			||||||
  AS7341_ADC_CHANNEL_4,
 | 
					 | 
				
			||||||
  AS7341_ADC_CHANNEL_5,
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
enum AS7341SmuxCommand {
 | 
					 | 
				
			||||||
  AS7341_SMUX_CMD_ROM_RESET,  ///< ROM code initialization of SMUX
 | 
					 | 
				
			||||||
  AS7341_SMUX_CMD_READ,       ///< Read SMUX configuration to RAM from SMUX chain
 | 
					 | 
				
			||||||
  AS7341_SMUX_CMD_WRITE,      ///< Write SMUX configuration from RAM to SMUX chain
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
enum AS7341Gain {
 | 
					 | 
				
			||||||
  AS7341_GAIN_0_5X,
 | 
					 | 
				
			||||||
  AS7341_GAIN_1X,
 | 
					 | 
				
			||||||
  AS7341_GAIN_2X,
 | 
					 | 
				
			||||||
  AS7341_GAIN_4X,
 | 
					 | 
				
			||||||
  AS7341_GAIN_8X,
 | 
					 | 
				
			||||||
  AS7341_GAIN_16X,
 | 
					 | 
				
			||||||
  AS7341_GAIN_32X,
 | 
					 | 
				
			||||||
  AS7341_GAIN_64X,
 | 
					 | 
				
			||||||
  AS7341_GAIN_128X,
 | 
					 | 
				
			||||||
  AS7341_GAIN_256X,
 | 
					 | 
				
			||||||
  AS7341_GAIN_512X,
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class AS7341Component : public PollingComponent, public i2c::I2CDevice {
 | 
					 | 
				
			||||||
 public:
 | 
					 | 
				
			||||||
  void setup() override;
 | 
					 | 
				
			||||||
  void dump_config() override;
 | 
					 | 
				
			||||||
  float get_setup_priority() const override;
 | 
					 | 
				
			||||||
  void update() override;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  void set_f1_sensor(sensor::Sensor *f1_sensor) { this->f1_ = f1_sensor; }
 | 
					 | 
				
			||||||
  void set_f2_sensor(sensor::Sensor *f2_sensor) { f2_ = f2_sensor; }
 | 
					 | 
				
			||||||
  void set_f3_sensor(sensor::Sensor *f3_sensor) { f3_ = f3_sensor; }
 | 
					 | 
				
			||||||
  void set_f4_sensor(sensor::Sensor *f4_sensor) { f4_ = f4_sensor; }
 | 
					 | 
				
			||||||
  void set_f5_sensor(sensor::Sensor *f5_sensor) { f5_ = f5_sensor; }
 | 
					 | 
				
			||||||
  void set_f6_sensor(sensor::Sensor *f6_sensor) { f6_ = f6_sensor; }
 | 
					 | 
				
			||||||
  void set_f7_sensor(sensor::Sensor *f7_sensor) { f7_ = f7_sensor; }
 | 
					 | 
				
			||||||
  void set_f8_sensor(sensor::Sensor *f8_sensor) { f8_ = f8_sensor; }
 | 
					 | 
				
			||||||
  void set_clear_sensor(sensor::Sensor *clear_sensor) { clear_ = clear_sensor; }
 | 
					 | 
				
			||||||
  void set_nir_sensor(sensor::Sensor *nir_sensor) { nir_ = nir_sensor; }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  void set_gain(AS7341Gain gain) { gain_ = gain; }
 | 
					 | 
				
			||||||
  void set_atime(uint8_t atime) { atime_ = atime; }
 | 
					 | 
				
			||||||
  void set_astep(uint16_t astep) { astep_ = astep; }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  AS7341Gain get_gain();
 | 
					 | 
				
			||||||
  uint8_t get_atime();
 | 
					 | 
				
			||||||
  uint16_t get_astep();
 | 
					 | 
				
			||||||
  bool setup_gain(AS7341Gain gain);
 | 
					 | 
				
			||||||
  bool setup_atime(uint8_t atime);
 | 
					 | 
				
			||||||
  bool setup_astep(uint16_t astep);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  uint16_t read_channel(AS7341AdcChannel channel);
 | 
					 | 
				
			||||||
  bool read_channels(uint16_t *data);
 | 
					 | 
				
			||||||
  void set_smux_low_channels(bool enable);
 | 
					 | 
				
			||||||
  bool set_smux_command(AS7341SmuxCommand command);
 | 
					 | 
				
			||||||
  void configure_smux_low_channels();
 | 
					 | 
				
			||||||
  void configure_smux_high_channels();
 | 
					 | 
				
			||||||
  bool enable_smux();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  bool wait_for_data();
 | 
					 | 
				
			||||||
  bool is_data_ready();
 | 
					 | 
				
			||||||
  bool enable_power(bool enable);
 | 
					 | 
				
			||||||
  bool enable_spectral_measurement(bool enable);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  bool read_register_bit(uint8_t address, uint8_t bit_position);
 | 
					 | 
				
			||||||
  bool write_register_bit(uint8_t address, bool value, uint8_t bit_position);
 | 
					 | 
				
			||||||
  bool set_register_bit(uint8_t address, uint8_t bit_position);
 | 
					 | 
				
			||||||
  bool clear_register_bit(uint8_t address, uint8_t bit_position);
 | 
					 | 
				
			||||||
  uint16_t swap_bytes(uint16_t data);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 protected:
 | 
					 | 
				
			||||||
  sensor::Sensor *f1_{nullptr};
 | 
					 | 
				
			||||||
  sensor::Sensor *f2_{nullptr};
 | 
					 | 
				
			||||||
  sensor::Sensor *f3_{nullptr};
 | 
					 | 
				
			||||||
  sensor::Sensor *f4_{nullptr};
 | 
					 | 
				
			||||||
  sensor::Sensor *f5_{nullptr};
 | 
					 | 
				
			||||||
  sensor::Sensor *f6_{nullptr};
 | 
					 | 
				
			||||||
  sensor::Sensor *f7_{nullptr};
 | 
					 | 
				
			||||||
  sensor::Sensor *f8_{nullptr};
 | 
					 | 
				
			||||||
  sensor::Sensor *clear_{nullptr};
 | 
					 | 
				
			||||||
  sensor::Sensor *nir_{nullptr};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  uint16_t astep_;
 | 
					 | 
				
			||||||
  AS7341Gain gain_;
 | 
					 | 
				
			||||||
  uint8_t atime_;
 | 
					 | 
				
			||||||
  uint16_t channel_readings_[12];
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
}  // namespace as7341
 | 
					 | 
				
			||||||
}  // namespace esphome
 | 
					 | 
				
			||||||
@@ -1,112 +0,0 @@
 | 
				
			|||||||
import esphome.codegen as cg
 | 
					 | 
				
			||||||
import esphome.config_validation as cv
 | 
					 | 
				
			||||||
from esphome.components import i2c, sensor
 | 
					 | 
				
			||||||
from esphome.const import (
 | 
					 | 
				
			||||||
    CONF_GAIN,
 | 
					 | 
				
			||||||
    CONF_ID,
 | 
					 | 
				
			||||||
    DEVICE_CLASS_ILLUMINANCE,
 | 
					 | 
				
			||||||
    ICON_BRIGHTNESS_5,
 | 
					 | 
				
			||||||
    STATE_CLASS_MEASUREMENT,
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
CODEOWNERS = ["@mrgnr"]
 | 
					 | 
				
			||||||
DEPENDENCIES = ["i2c"]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
as7341_ns = cg.esphome_ns.namespace("as7341")
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
AS7341Component = as7341_ns.class_(
 | 
					 | 
				
			||||||
    "AS7341Component", cg.PollingComponent, i2c.I2CDevice
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
CONF_ATIME = "atime"
 | 
					 | 
				
			||||||
CONF_ASTEP = "astep"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
CONF_F1 = "f1"
 | 
					 | 
				
			||||||
CONF_F2 = "f2"
 | 
					 | 
				
			||||||
CONF_F3 = "f3"
 | 
					 | 
				
			||||||
CONF_F4 = "f4"
 | 
					 | 
				
			||||||
CONF_F5 = "f5"
 | 
					 | 
				
			||||||
CONF_F6 = "f6"
 | 
					 | 
				
			||||||
CONF_F7 = "f7"
 | 
					 | 
				
			||||||
CONF_F8 = "f8"
 | 
					 | 
				
			||||||
CONF_CLEAR = "clear"
 | 
					 | 
				
			||||||
CONF_NIR = "nir"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
UNIT_COUNTS = "#"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
AS7341_GAIN = as7341_ns.enum("AS7341Gain")
 | 
					 | 
				
			||||||
GAIN_OPTIONS = {
 | 
					 | 
				
			||||||
    "X0.5": AS7341_GAIN.AS7341_GAIN_0_5X,
 | 
					 | 
				
			||||||
    "X1": AS7341_GAIN.AS7341_GAIN_1X,
 | 
					 | 
				
			||||||
    "X2": AS7341_GAIN.AS7341_GAIN_2X,
 | 
					 | 
				
			||||||
    "X4": AS7341_GAIN.AS7341_GAIN_4X,
 | 
					 | 
				
			||||||
    "X8": AS7341_GAIN.AS7341_GAIN_8X,
 | 
					 | 
				
			||||||
    "X16": AS7341_GAIN.AS7341_GAIN_16X,
 | 
					 | 
				
			||||||
    "X32": AS7341_GAIN.AS7341_GAIN_32X,
 | 
					 | 
				
			||||||
    "X64": AS7341_GAIN.AS7341_GAIN_64X,
 | 
					 | 
				
			||||||
    "X128": AS7341_GAIN.AS7341_GAIN_128X,
 | 
					 | 
				
			||||||
    "X256": AS7341_GAIN.AS7341_GAIN_256X,
 | 
					 | 
				
			||||||
    "X512": AS7341_GAIN.AS7341_GAIN_512X,
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
SENSOR_SCHEMA = sensor.sensor_schema(
 | 
					 | 
				
			||||||
    unit_of_measurement=UNIT_COUNTS,
 | 
					 | 
				
			||||||
    icon=ICON_BRIGHTNESS_5,
 | 
					 | 
				
			||||||
    accuracy_decimals=0,
 | 
					 | 
				
			||||||
    device_class=DEVICE_CLASS_ILLUMINANCE,
 | 
					 | 
				
			||||||
    state_class=STATE_CLASS_MEASUREMENT,
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
CONFIG_SCHEMA = (
 | 
					 | 
				
			||||||
    cv.Schema(
 | 
					 | 
				
			||||||
        {
 | 
					 | 
				
			||||||
            cv.GenerateID(): cv.declare_id(AS7341Component),
 | 
					 | 
				
			||||||
            cv.Optional(CONF_F1): SENSOR_SCHEMA,
 | 
					 | 
				
			||||||
            cv.Optional(CONF_F2): SENSOR_SCHEMA,
 | 
					 | 
				
			||||||
            cv.Optional(CONF_F3): SENSOR_SCHEMA,
 | 
					 | 
				
			||||||
            cv.Optional(CONF_F4): SENSOR_SCHEMA,
 | 
					 | 
				
			||||||
            cv.Optional(CONF_F5): SENSOR_SCHEMA,
 | 
					 | 
				
			||||||
            cv.Optional(CONF_F6): SENSOR_SCHEMA,
 | 
					 | 
				
			||||||
            cv.Optional(CONF_F7): SENSOR_SCHEMA,
 | 
					 | 
				
			||||||
            cv.Optional(CONF_F8): SENSOR_SCHEMA,
 | 
					 | 
				
			||||||
            cv.Optional(CONF_CLEAR): SENSOR_SCHEMA,
 | 
					 | 
				
			||||||
            cv.Optional(CONF_NIR): SENSOR_SCHEMA,
 | 
					 | 
				
			||||||
            cv.Optional(CONF_GAIN, default="X8"): cv.enum(GAIN_OPTIONS),
 | 
					 | 
				
			||||||
            cv.Optional(CONF_ATIME, default=29): cv.int_range(min=0, max=255),
 | 
					 | 
				
			||||||
            cv.Optional(CONF_ASTEP, default=599): cv.int_range(min=0, max=65534),
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    .extend(cv.polling_component_schema("60s"))
 | 
					 | 
				
			||||||
    .extend(i2c.i2c_device_schema(0x39))
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
SENSORS = {
 | 
					 | 
				
			||||||
    CONF_F1: "set_f1_sensor",
 | 
					 | 
				
			||||||
    CONF_F2: "set_f2_sensor",
 | 
					 | 
				
			||||||
    CONF_F3: "set_f3_sensor",
 | 
					 | 
				
			||||||
    CONF_F4: "set_f4_sensor",
 | 
					 | 
				
			||||||
    CONF_F5: "set_f5_sensor",
 | 
					 | 
				
			||||||
    CONF_F6: "set_f6_sensor",
 | 
					 | 
				
			||||||
    CONF_F7: "set_f7_sensor",
 | 
					 | 
				
			||||||
    CONF_F8: "set_f8_sensor",
 | 
					 | 
				
			||||||
    CONF_CLEAR: "set_clear_sensor",
 | 
					 | 
				
			||||||
    CONF_NIR: "set_nir_sensor",
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
async def to_code(config):
 | 
					 | 
				
			||||||
    var = cg.new_Pvariable(config[CONF_ID])
 | 
					 | 
				
			||||||
    await cg.register_component(var, config)
 | 
					 | 
				
			||||||
    await i2c.register_i2c_device(var, config)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    cg.add(var.set_gain(config[CONF_GAIN]))
 | 
					 | 
				
			||||||
    cg.add(var.set_atime(config[CONF_ATIME]))
 | 
					 | 
				
			||||||
    cg.add(var.set_astep(config[CONF_ASTEP]))
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    for conf_id, set_sensor_func in SENSORS.items():
 | 
					 | 
				
			||||||
        if conf_id in config:
 | 
					 | 
				
			||||||
            sens = await sensor.new_sensor(config[conf_id])
 | 
					 | 
				
			||||||
            cg.add(getattr(var, set_sensor_func)(sens))
 | 
					 | 
				
			||||||
@@ -80,7 +80,7 @@ async def to_code(config):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    cg.add(var.set_address(config[CONF_MAC_ADDRESS].as_hex))
 | 
					    cg.add(var.set_address(config[CONF_MAC_ADDRESS].as_hex))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    for config_key, setter in [
 | 
					    for (config_key, setter) in [
 | 
				
			||||||
        (CONF_TEMPERATURE, var.set_temperature),
 | 
					        (CONF_TEMPERATURE, var.set_temperature),
 | 
				
			||||||
        (CONF_HUMIDITY, var.set_humidity),
 | 
					        (CONF_HUMIDITY, var.set_humidity),
 | 
				
			||||||
        (CONF_BATTERY_VOLTAGE, var.set_battery_voltage),
 | 
					        (CONF_BATTERY_VOLTAGE, var.set_battery_voltage),
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,5 +1,3 @@
 | 
				
			|||||||
#ifdef USE_ESP32
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#include "bedjet_hub.h"
 | 
					#include "bedjet_hub.h"
 | 
				
			||||||
#include "bedjet_child.h"
 | 
					#include "bedjet_child.h"
 | 
				
			||||||
#include "bedjet_const.h"
 | 
					#include "bedjet_const.h"
 | 
				
			||||||
@@ -543,5 +541,3 @@ void BedJetHub::register_child(BedJetClient *obj) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
}  // namespace bedjet
 | 
					}  // namespace bedjet
 | 
				
			||||||
}  // namespace esphome
 | 
					}  // namespace esphome
 | 
				
			||||||
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,5 +1,4 @@
 | 
				
			|||||||
#pragma once
 | 
					#pragma once
 | 
				
			||||||
#ifdef USE_ESP32
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include "esphome/components/ble_client/ble_client.h"
 | 
					#include "esphome/components/ble_client/ble_client.h"
 | 
				
			||||||
#include "esphome/components/esp32_ble_tracker/esp32_ble_tracker.h"
 | 
					#include "esphome/components/esp32_ble_tracker/esp32_ble_tracker.h"
 | 
				
			||||||
@@ -15,6 +14,8 @@
 | 
				
			|||||||
#include "esphome/components/time/real_time_clock.h"
 | 
					#include "esphome/components/time/real_time_clock.h"
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef USE_ESP32
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include <esp_gattc_api.h>
 | 
					#include <esp_gattc_api.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
namespace esphome {
 | 
					namespace esphome {
 | 
				
			||||||
@@ -116,7 +117,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() {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -37,13 +37,9 @@ void BedJetFan::control(const fan::FanCall &call) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  // ignore speed changes if not on or turning on
 | 
					  // ignore speed changes if not on or turning on
 | 
				
			||||||
  if (this->state && call.get_speed().has_value()) {
 | 
					  if (this->state && call.get_speed().has_value()) {
 | 
				
			||||||
    auto speed = *call.get_speed();
 | 
					    this->speed = *call.get_speed();
 | 
				
			||||||
    if (speed >= 1) {
 | 
					    this->parent_->set_fan_index(this->speed);
 | 
				
			||||||
      this->speed = speed;
 | 
					    did_change = true;
 | 
				
			||||||
      // Fan.speed is 1-20, but Bedjet expects 0-19, so subtract 1
 | 
					 | 
				
			||||||
      this->parent_->set_fan_index(this->speed - 1);
 | 
					 | 
				
			||||||
      did_change = true;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  if (did_change) {
 | 
					  if (did_change) {
 | 
				
			||||||
@@ -61,9 +57,8 @@ void BedJetFan::on_status(const BedjetStatusPacket *data) {
 | 
				
			|||||||
    did_change = true;
 | 
					    did_change = true;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // BedjetStatusPacket.fan_step is in range 0-19, but Fan.speed wants 1-20.
 | 
					  if (data->fan_step != this->speed) {
 | 
				
			||||||
  if (data->fan_step + 1 != this->speed) {
 | 
					    this->speed = data->fan_step;
 | 
				
			||||||
    this->speed = data->fan_step + 1;
 | 
					 | 
				
			||||||
    did_change = true;
 | 
					    did_change = true;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -27,13 +27,13 @@ from esphome.const import (
 | 
				
			|||||||
    CONF_TIMING,
 | 
					    CONF_TIMING,
 | 
				
			||||||
    CONF_TRIGGER_ID,
 | 
					    CONF_TRIGGER_ID,
 | 
				
			||||||
    CONF_MQTT_ID,
 | 
					    CONF_MQTT_ID,
 | 
				
			||||||
 | 
					    DEVICE_CLASS_EMPTY,
 | 
				
			||||||
    DEVICE_CLASS_BATTERY,
 | 
					    DEVICE_CLASS_BATTERY,
 | 
				
			||||||
    DEVICE_CLASS_BATTERY_CHARGING,
 | 
					    DEVICE_CLASS_BATTERY_CHARGING,
 | 
				
			||||||
    DEVICE_CLASS_CARBON_MONOXIDE,
 | 
					    DEVICE_CLASS_CARBON_MONOXIDE,
 | 
				
			||||||
    DEVICE_CLASS_COLD,
 | 
					    DEVICE_CLASS_COLD,
 | 
				
			||||||
    DEVICE_CLASS_CONNECTIVITY,
 | 
					    DEVICE_CLASS_CONNECTIVITY,
 | 
				
			||||||
    DEVICE_CLASS_DOOR,
 | 
					    DEVICE_CLASS_DOOR,
 | 
				
			||||||
    DEVICE_CLASS_EMPTY,
 | 
					 | 
				
			||||||
    DEVICE_CLASS_GARAGE_DOOR,
 | 
					    DEVICE_CLASS_GARAGE_DOOR,
 | 
				
			||||||
    DEVICE_CLASS_GAS,
 | 
					    DEVICE_CLASS_GAS,
 | 
				
			||||||
    DEVICE_CLASS_HEAT,
 | 
					    DEVICE_CLASS_HEAT,
 | 
				
			||||||
@@ -62,13 +62,13 @@ from esphome.util import Registry
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
CODEOWNERS = ["@esphome/core"]
 | 
					CODEOWNERS = ["@esphome/core"]
 | 
				
			||||||
DEVICE_CLASSES = [
 | 
					DEVICE_CLASSES = [
 | 
				
			||||||
 | 
					    DEVICE_CLASS_EMPTY,
 | 
				
			||||||
    DEVICE_CLASS_BATTERY,
 | 
					    DEVICE_CLASS_BATTERY,
 | 
				
			||||||
    DEVICE_CLASS_BATTERY_CHARGING,
 | 
					    DEVICE_CLASS_BATTERY_CHARGING,
 | 
				
			||||||
    DEVICE_CLASS_CARBON_MONOXIDE,
 | 
					    DEVICE_CLASS_CARBON_MONOXIDE,
 | 
				
			||||||
    DEVICE_CLASS_COLD,
 | 
					    DEVICE_CLASS_COLD,
 | 
				
			||||||
    DEVICE_CLASS_CONNECTIVITY,
 | 
					    DEVICE_CLASS_CONNECTIVITY,
 | 
				
			||||||
    DEVICE_CLASS_DOOR,
 | 
					    DEVICE_CLASS_DOOR,
 | 
				
			||||||
    DEVICE_CLASS_EMPTY,
 | 
					 | 
				
			||||||
    DEVICE_CLASS_GARAGE_DOOR,
 | 
					    DEVICE_CLASS_GARAGE_DOOR,
 | 
				
			||||||
    DEVICE_CLASS_GAS,
 | 
					    DEVICE_CLASS_GAS,
 | 
				
			||||||
    DEVICE_CLASS_HEAT,
 | 
					    DEVICE_CLASS_HEAT,
 | 
				
			||||||
@@ -393,21 +393,28 @@ def binary_sensor_schema(
 | 
				
			|||||||
    entity_category: str = _UNDEF,
 | 
					    entity_category: str = _UNDEF,
 | 
				
			||||||
    device_class: str = _UNDEF,
 | 
					    device_class: str = _UNDEF,
 | 
				
			||||||
) -> cv.Schema:
 | 
					) -> cv.Schema:
 | 
				
			||||||
    schema = {}
 | 
					    schema = BINARY_SENSOR_SCHEMA
 | 
				
			||||||
 | 
					 | 
				
			||||||
    if class_ is not _UNDEF:
 | 
					    if class_ is not _UNDEF:
 | 
				
			||||||
        # Not cv.optional
 | 
					        schema = schema.extend({cv.GenerateID(): cv.declare_id(class_)})
 | 
				
			||||||
        schema[cv.GenerateID()] = cv.declare_id(class_)
 | 
					    if icon is not _UNDEF:
 | 
				
			||||||
 | 
					        schema = schema.extend({cv.Optional(CONF_ICON, default=icon): cv.icon})
 | 
				
			||||||
    for key, default, validator in [
 | 
					    if entity_category is not _UNDEF:
 | 
				
			||||||
        (CONF_ICON, icon, cv.icon),
 | 
					        schema = schema.extend(
 | 
				
			||||||
        (CONF_ENTITY_CATEGORY, entity_category, cv.entity_category),
 | 
					            {
 | 
				
			||||||
        (CONF_DEVICE_CLASS, device_class, validate_device_class),
 | 
					                cv.Optional(
 | 
				
			||||||
    ]:
 | 
					                    CONF_ENTITY_CATEGORY, default=entity_category
 | 
				
			||||||
        if default is not _UNDEF:
 | 
					                ): cv.entity_category
 | 
				
			||||||
            schema[cv.Optional(key, default=default)] = validator
 | 
					            }
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
    return BINARY_SENSOR_SCHEMA.extend(schema)
 | 
					    if device_class is not _UNDEF:
 | 
				
			||||||
 | 
					        schema = schema.extend(
 | 
				
			||||||
 | 
					            {
 | 
				
			||||||
 | 
					                cv.Optional(
 | 
				
			||||||
 | 
					                    CONF_DEVICE_CLASS, default=device_class
 | 
				
			||||||
 | 
					                ): validate_device_class
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					    return schema
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async def setup_binary_sensor_core_(var, config):
 | 
					async def setup_binary_sensor_core_(var, config):
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -41,13 +41,16 @@ void BinarySensor::send_state_internal(bool state, bool is_initial) {
 | 
				
			|||||||
    this->state_callback_.call(state);
 | 
					    this->state_callback_.call(state);
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					std::string BinarySensor::device_class() { return ""; }
 | 
				
			||||||
BinarySensor::BinarySensor() : state(false) {}
 | 
					BinarySensor::BinarySensor() : state(false) {}
 | 
				
			||||||
void BinarySensor::set_device_class(const std::string &device_class) { this->device_class_ = device_class; }
 | 
					void BinarySensor::set_device_class(const std::string &device_class) { this->device_class_ = device_class; }
 | 
				
			||||||
std::string BinarySensor::get_device_class() {
 | 
					std::string BinarySensor::get_device_class() {
 | 
				
			||||||
  if (this->device_class_.has_value())
 | 
					  if (this->device_class_.has_value())
 | 
				
			||||||
    return *this->device_class_;
 | 
					    return *this->device_class_;
 | 
				
			||||||
  return "";
 | 
					#pragma GCC diagnostic push
 | 
				
			||||||
 | 
					#pragma GCC diagnostic ignored "-Wdeprecated-declarations"
 | 
				
			||||||
 | 
					  return this->device_class();
 | 
				
			||||||
 | 
					#pragma GCC diagnostic pop
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
void BinarySensor::add_filter(Filter *filter) {
 | 
					void BinarySensor::add_filter(Filter *filter) {
 | 
				
			||||||
  filter->parent_ = this;
 | 
					  filter->parent_ = this;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -19,15 +19,6 @@ namespace binary_sensor {
 | 
				
			|||||||
    } \
 | 
					    } \
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define SUB_BINARY_SENSOR(name) \
 | 
					 | 
				
			||||||
 protected: \
 | 
					 | 
				
			||||||
  binary_sensor::BinarySensor *name##_binary_sensor_{nullptr}; \
 | 
					 | 
				
			||||||
\
 | 
					 | 
				
			||||||
 public: \
 | 
					 | 
				
			||||||
  void set_##name##_binary_sensor(binary_sensor::BinarySensor *binary_sensor) { \
 | 
					 | 
				
			||||||
    this->name##_binary_sensor_ = binary_sensor; \
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
/** Base class for all binary_sensor-type classes.
 | 
					/** Base class for all binary_sensor-type classes.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * This class includes a callback that components such as MQTT can subscribe to for state changes.
 | 
					 * This class includes a callback that components such as MQTT can subscribe to for state changes.
 | 
				
			||||||
@@ -80,6 +71,14 @@ class BinarySensor : public EntityBase {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  virtual bool is_status_binary_sensor() const;
 | 
					  virtual bool is_status_binary_sensor() const;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  // ========== OVERRIDE METHODS ==========
 | 
				
			||||||
 | 
					  // (You'll only need this when creating your own custom binary sensor)
 | 
				
			||||||
 | 
					  /** Override this to set the default device class.
 | 
				
			||||||
 | 
					   *
 | 
				
			||||||
 | 
					   * @deprecated This method is deprecated, set the property during config validation instead. (2022.1)
 | 
				
			||||||
 | 
					   */
 | 
				
			||||||
 | 
					  virtual std::string device_class();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 protected:
 | 
					 protected:
 | 
				
			||||||
  CallbackManager<void(bool)> state_callback_{};
 | 
					  CallbackManager<void(bool)> state_callback_{};
 | 
				
			||||||
  optional<std::string> device_class_{};  ///< Stores the override of the device class
 | 
					  optional<std::string> device_class_{};  ///< Stores the override of the device class
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -30,7 +30,7 @@ void BinarySensorMap::process_group_() {
 | 
				
			|||||||
    if (bs.binary_sensor->state) {
 | 
					    if (bs.binary_sensor->state) {
 | 
				
			||||||
      num_active_sensors++;
 | 
					      num_active_sensors++;
 | 
				
			||||||
      total_current_value += bs.sensor_value;
 | 
					      total_current_value += bs.sensor_value;
 | 
				
			||||||
      mask |= 1ULL << i;
 | 
					      mask |= 1 << i;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
  // check if the sensor map was touched
 | 
					  // check if the sensor map was touched
 | 
				
			||||||
@@ -38,11 +38,12 @@ void BinarySensorMap::process_group_() {
 | 
				
			|||||||
    // did the bit_mask change or is it a new sensor touch
 | 
					    // 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) {
 | 
				
			||||||
    // is this a new sensor release
 | 
					    // 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;
 | 
				
			||||||
@@ -51,22 +52,28 @@ void BinarySensorMap::process_group_() {
 | 
				
			|||||||
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_sensor states
 | 
					  // check all binary_sensors for its state. when active add its value to total_current_value.
 | 
				
			||||||
  // - if active, add its value to total_current_value
 | 
					  // create a bitmask for the binary_sensor status on all channels
 | 
				
			||||||
  // - creates a bitmask for the binary_sensor status 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.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 the 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;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -39,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), cv.Length(min=1, max=64)
 | 
					                    cv.ensure_list(entry), cv.Length(min=1)
 | 
				
			||||||
                ),
 | 
					                ),
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
@@ -50,7 +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), cv.Length(min=1, max=64)
 | 
					                    cv.ensure_list(entry), cv.Length(min=1)
 | 
				
			||||||
                ),
 | 
					                ),
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -9,7 +9,6 @@ from esphome.const import (
 | 
				
			|||||||
    DEVICE_CLASS_POWER,
 | 
					    DEVICE_CLASS_POWER,
 | 
				
			||||||
    DEVICE_CLASS_VOLTAGE,
 | 
					    DEVICE_CLASS_VOLTAGE,
 | 
				
			||||||
    STATE_CLASS_MEASUREMENT,
 | 
					    STATE_CLASS_MEASUREMENT,
 | 
				
			||||||
    STATE_CLASS_TOTAL_INCREASING,
 | 
					 | 
				
			||||||
    UNIT_AMPERE,
 | 
					    UNIT_AMPERE,
 | 
				
			||||||
    UNIT_KILOWATT_HOURS,
 | 
					    UNIT_KILOWATT_HOURS,
 | 
				
			||||||
    UNIT_VOLT,
 | 
					    UNIT_VOLT,
 | 
				
			||||||
@@ -67,19 +66,16 @@ CONFIG_SCHEMA = (
 | 
				
			|||||||
                unit_of_measurement=UNIT_KILOWATT_HOURS,
 | 
					                unit_of_measurement=UNIT_KILOWATT_HOURS,
 | 
				
			||||||
                accuracy_decimals=3,
 | 
					                accuracy_decimals=3,
 | 
				
			||||||
                device_class=DEVICE_CLASS_ENERGY,
 | 
					                device_class=DEVICE_CLASS_ENERGY,
 | 
				
			||||||
                state_class=STATE_CLASS_TOTAL_INCREASING,
 | 
					 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
            cv.Optional(CONF_ENERGY_2): sensor.sensor_schema(
 | 
					            cv.Optional(CONF_ENERGY_2): sensor.sensor_schema(
 | 
				
			||||||
                unit_of_measurement=UNIT_KILOWATT_HOURS,
 | 
					                unit_of_measurement=UNIT_KILOWATT_HOURS,
 | 
				
			||||||
                accuracy_decimals=3,
 | 
					                accuracy_decimals=3,
 | 
				
			||||||
                device_class=DEVICE_CLASS_ENERGY,
 | 
					                device_class=DEVICE_CLASS_ENERGY,
 | 
				
			||||||
                state_class=STATE_CLASS_TOTAL_INCREASING,
 | 
					 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
            cv.Optional(CONF_ENERGY_TOTAL): sensor.sensor_schema(
 | 
					            cv.Optional(CONF_ENERGY_TOTAL): sensor.sensor_schema(
 | 
				
			||||||
                unit_of_measurement=UNIT_KILOWATT_HOURS,
 | 
					                unit_of_measurement=UNIT_KILOWATT_HOURS,
 | 
				
			||||||
                accuracy_decimals=3,
 | 
					                accuracy_decimals=3,
 | 
				
			||||||
                device_class=DEVICE_CLASS_ENERGY,
 | 
					                device_class=DEVICE_CLASS_ENERGY,
 | 
				
			||||||
                state_class=STATE_CLASS_TOTAL_INCREASING,
 | 
					 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,5 +1,3 @@
 | 
				
			|||||||
#ifdef USE_ESP32
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#include "automation.h"
 | 
					#include "automation.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include <esp_bt_defs.h>
 | 
					#include <esp_bt_defs.h>
 | 
				
			||||||
@@ -75,5 +73,3 @@ void BLEWriterClientNode::gattc_event_handler(esp_gattc_cb_event_t event, esp_ga
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
}  // namespace ble_client
 | 
					}  // namespace ble_client
 | 
				
			||||||
}  // namespace esphome
 | 
					}  // namespace esphome
 | 
				
			||||||
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 
 | 
				
			|||||||
@@ -1,13 +1,13 @@
 | 
				
			|||||||
#pragma once
 | 
					#pragma once
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#ifdef USE_ESP32
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#include <utility>
 | 
					#include <utility>
 | 
				
			||||||
#include <vector>
 | 
					#include <vector>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include "esphome/core/automation.h"
 | 
					#include "esphome/core/automation.h"
 | 
				
			||||||
#include "esphome/components/ble_client/ble_client.h"
 | 
					#include "esphome/components/ble_client/ble_client.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifdef USE_ESP32
 | 
				
			||||||
 | 
					
 | 
				
			||||||
namespace esphome {
 | 
					namespace esphome {
 | 
				
			||||||
namespace ble_client {
 | 
					namespace ble_client {
 | 
				
			||||||
class BLEClientConnectTrigger : public Trigger<>, public BLEClientNode {
 | 
					class BLEClientConnectTrigger : public Trigger<>, public BLEClientNode {
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -158,25 +158,6 @@ bool BluetoothConnection::gattc_event_handler(esp_gattc_cb_event_t event, esp_ga
 | 
				
			|||||||
  return true;
 | 
					  return true;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void BluetoothConnection::gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) {
 | 
					 | 
				
			||||||
  BLEClientBase::gap_event_handler(event, param);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  switch (event) {
 | 
					 | 
				
			||||||
    case ESP_GAP_BLE_AUTH_CMPL_EVT:
 | 
					 | 
				
			||||||
      if (memcmp(param->ble_security.auth_cmpl.bd_addr, this->remote_bda_, 6) != 0)
 | 
					 | 
				
			||||||
        break;
 | 
					 | 
				
			||||||
      if (param->ble_security.auth_cmpl.success) {
 | 
					 | 
				
			||||||
        api::global_api_server->send_bluetooth_device_pairing(this->address_, true);
 | 
					 | 
				
			||||||
      } else {
 | 
					 | 
				
			||||||
        api::global_api_server->send_bluetooth_device_pairing(this->address_, false,
 | 
					 | 
				
			||||||
                                                              param->ble_security.auth_cmpl.fail_reason);
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
      break;
 | 
					 | 
				
			||||||
    default:
 | 
					 | 
				
			||||||
      break;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
esp_err_t BluetoothConnection::read_characteristic(uint16_t handle) {
 | 
					esp_err_t BluetoothConnection::read_characteristic(uint16_t handle) {
 | 
				
			||||||
  if (!this->connected()) {
 | 
					  if (!this->connected()) {
 | 
				
			||||||
    ESP_LOGW(TAG, "[%d] [%s] Cannot read GATT characteristic, not connected.", this->connection_index_,
 | 
					    ESP_LOGW(TAG, "[%d] [%s] Cannot read GATT characteristic, not connected.", this->connection_index_,
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -13,7 +13,6 @@ class BluetoothConnection : public esp32_ble_client::BLEClientBase {
 | 
				
			|||||||
 public:
 | 
					 public:
 | 
				
			||||||
  bool gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
 | 
					  bool gattc_event_handler(esp_gattc_cb_event_t event, esp_gatt_if_t gattc_if,
 | 
				
			||||||
                           esp_ble_gattc_cb_param_t *param) override;
 | 
					                           esp_ble_gattc_cb_param_t *param) override;
 | 
				
			||||||
  void gap_event_handler(esp_gap_ble_cb_event_t event, esp_ble_gap_cb_param_t *param) override;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
  esp_err_t read_characteristic(uint16_t handle);
 | 
					  esp_err_t read_characteristic(uint16_t handle);
 | 
				
			||||||
  esp_err_t write_characteristic(uint16_t handle, const std::string &data, bool response);
 | 
					  esp_err_t write_characteristic(uint16_t handle, const std::string &data, bool response);
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -257,7 +257,12 @@ void BluetoothProxy::bluetooth_device_request(const api::BluetoothDeviceRequest
 | 
				
			|||||||
        ESP_LOGI(TAG, "[%d] [%s] Connecting v1", connection->get_connection_index(), connection->address_str().c_str());
 | 
					        ESP_LOGI(TAG, "[%d] [%s] Connecting v1", connection->get_connection_index(), connection->address_str().c_str());
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      if (msg.has_address_type) {
 | 
					      if (msg.has_address_type) {
 | 
				
			||||||
        uint64_to_bd_addr(msg.address, connection->remote_bda_);
 | 
					        connection->remote_bda_[0] = (msg.address >> 40) & 0xFF;
 | 
				
			||||||
 | 
					        connection->remote_bda_[1] = (msg.address >> 32) & 0xFF;
 | 
				
			||||||
 | 
					        connection->remote_bda_[2] = (msg.address >> 24) & 0xFF;
 | 
				
			||||||
 | 
					        connection->remote_bda_[3] = (msg.address >> 16) & 0xFF;
 | 
				
			||||||
 | 
					        connection->remote_bda_[4] = (msg.address >> 8) & 0xFF;
 | 
				
			||||||
 | 
					        connection->remote_bda_[5] = (msg.address >> 0) & 0xFF;
 | 
				
			||||||
        connection->set_remote_addr_type(static_cast<esp_ble_addr_type_t>(msg.address_type));
 | 
					        connection->set_remote_addr_type(static_cast<esp_ble_addr_type_t>(msg.address_type));
 | 
				
			||||||
        connection->set_state(espbt::ClientState::DISCOVERED);
 | 
					        connection->set_state(espbt::ClientState::DISCOVERED);
 | 
				
			||||||
      } else {
 | 
					      } else {
 | 
				
			||||||
@@ -285,34 +290,9 @@ void BluetoothProxy::bluetooth_device_request(const api::BluetoothDeviceRequest
 | 
				
			|||||||
      }
 | 
					      }
 | 
				
			||||||
      break;
 | 
					      break;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    case api::enums::BLUETOOTH_DEVICE_REQUEST_TYPE_PAIR: {
 | 
					    case api::enums::BLUETOOTH_DEVICE_REQUEST_TYPE_PAIR:
 | 
				
			||||||
      auto *connection = this->get_connection_(msg.address, false);
 | 
					    case api::enums::BLUETOOTH_DEVICE_REQUEST_TYPE_UNPAIR:
 | 
				
			||||||
      if (connection != nullptr) {
 | 
					 | 
				
			||||||
        if (!connection->is_paired()) {
 | 
					 | 
				
			||||||
          auto err = connection->pair();
 | 
					 | 
				
			||||||
          if (err != ESP_OK) {
 | 
					 | 
				
			||||||
            api::global_api_server->send_bluetooth_device_pairing(msg.address, false, err);
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
        } else {
 | 
					 | 
				
			||||||
          api::global_api_server->send_bluetooth_device_pairing(msg.address, true);
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
      break;
 | 
					      break;
 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    case api::enums::BLUETOOTH_DEVICE_REQUEST_TYPE_UNPAIR: {
 | 
					 | 
				
			||||||
      esp_bd_addr_t address;
 | 
					 | 
				
			||||||
      uint64_to_bd_addr(msg.address, address);
 | 
					 | 
				
			||||||
      esp_err_t ret = esp_ble_remove_bond_device(address);
 | 
					 | 
				
			||||||
      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::global_api_server->send_bluetooth_device_clear_cache(msg.address, ret == ESP_OK, ret);
 | 
					 | 
				
			||||||
      break;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user