mirror of
https://github.com/esphome/esphome.git
synced 2026-06-24 13:27:14 +00:00
Co-authored-by: Copilot Autofix powered by AI <175728472+Copilot@users.noreply.github.com>
105 lines
5.0 KiB
Markdown
105 lines
5.0 KiB
Markdown
# 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:
|
|
|
|
1. **The configuration.** Anyone who can supply or edit a YAML config is trusted
|
|
(see below).
|
|
2. **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).
|
|
- The legacy bundled dashboard in this repo (`esphome/dashboard/`) — it is
|
|
deprecated and being replaced by Device Builder; report dashboard issues there.
|
|
- 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).
|