4.9 KiB
ESPHome Threat Model
This document defines the trust boundary for the ESPHome repository — the Python compiler/CLI and the device firmware it generates — so that real security bugs can be told apart from defense-in-depth improvements. It gives contributors, reviewers, and security researchers a clear answer to one question: does this issue let an unauthenticated attacker do something they shouldn't?
Related documents:
- Deployment guidance for operators: https://esphome.io/guides/security_best_practices/
- The Device Builder dashboard (the web UI, its authentication, ingress, Origin/Host gates, and peer-link pairing) lives in a separate repository and has its own threat model. If your report concerns any of that, please read and report there instead: https://github.com/esphome/device-builder/blob/main/docs/THREAT_MODEL.md
The trust boundary
For this repository there are two trusted inputs by design:
- The configuration. Anyone who can supply or edit a YAML config is trusted (see below).
- Authenticated peers of a running device — clients holding the device's API encryption key / password, OTA password, or web server credentials.
The security boundary is therefore unauthenticated network traffic vs. those trusted inputs. A bug that lets an unauthenticated attacker cross it is a security bug.
Config authors are host-equivalent by design
Anyone who can supply or edit a configuration is trusted with full code
execution on the host that runs esphome, on purpose. This is what the product
does, not a flaw. A config author can already, through fully supported features:
- Run arbitrary Python at validation/compile time via
external_components:(and other component-import mechanisms) — ESPHome imports those packages as ordinary Python. - Run arbitrary shell commands through the compile/validate/flash toolchain that ESPHome invokes as subprocesses.
- Read and write arbitrary files reachable by the process (e.g. via
!include,packages:,dashboard_import:, and generated build output).
Because of this, a malicious config author is equivalent to shell access on the host running the build.
What is not a security vulnerability
If exploiting an issue requires the ability to supply or edit configuration, it is not a vulnerability in ESPHome, because that ability already grants host code execution. This explicitly includes, among others:
- Template / expression injection in substitutions or any YAML string value
(e.g. Jinja
${...}evaluation reaching Python internals). This grants no capability a config author lacks. !include/packages:/dashboard_import:reading or fetching content from surprising or remote locations.- The validator or compiler crashing or behaving unexpectedly on adversarial YAML.
- ESPHome running as root in the official container — that is the documented deployment posture, reachable by the same caller through the features above.
These do not warrant a CVE or coordinated disclosure. Hardening in these areas (for example, sandboxing template evaluation as least-surprise defense-in-depth) is welcome as a normal enhancement PR, framed as cleanliness rather than a security fix — not as a vulnerability remediation.
What we do defend
These are security bugs in this repo, and we want to hear about them privately:
- Memory-safety or protocol bugs in the generated device firmware that are remotely triggerable over the network (native API, web server, OTA, BLE, captive portal, etc.) without valid credentials.
- Authentication or encryption bypass on the device — reaching API calls, OTA updates, or the web server without the configured key/password.
- Flaws that weaken the device's API encryption (Noise), OTA, or web server auth below their documented guarantees.
Explicitly out of scope
- Local attackers who already have shell access on the host that runs
esphome. - Supply-chain attacks against ESPHome or its dependencies.
- Operator-supplied hostile YAML (covered above — config authoring is trusted).
- Attacks that require an already-authenticated device peer (someone who already holds the API key / OTA / web credentials).
- Anything in the dashboard / device-builder — report that in its own repository (linked at the top).
- Deployments where the operator removed protections or exposed credentials. See the security best practices guide: https://esphome.io/guides/security_best_practices/
Reporting a vulnerability
If you believe you've found an issue that crosses the unauthenticated boundary above, please report it privately via GitHub Security Advisories rather than a public issue. For issues that require config-write access, please review this document first — they are very likely out of scope by design. For dashboard / device-builder issues, report against that repository and consult its threat model (linked at the top).