Capabilities live on the read response
There is no separate/capabilities endpoint. The declaration ships inside the device read: GET /battery/{deviceId}, GET /ev-charger/{deviceId}, GET /hvac/{deviceId}, and their list endpoints all carry the commands block. One read returns state and commands, plus settings on the device types that declare a settings surface (battery and EV charger; a thermostat has commands but no settings). Settings ride the single-device read; list views carry commands, not settings:
commands.charge.parameters.power on the read maps directly to action.parameters.power on the write. See canonical actions.
conflictStrategies advertises the onConflict values the device accepts when a new push collides with an action already in flight. Read it before sending onConflict: a strategy outside this list is rejected with 422 STRATEGY_NOT_SUPPORTED. It is present on every commandable device type and absent on read-only types, which have no push to conflict. See conflict resolution.
The same shape across device types
Thecommands block is one contract, not a battery-specific one. The EV charger declares its commands the same way. Its two commands carry no parameters:
heat and cool commands take a target temperature; auto takes a heatSetpoint and coolSetpoint comfort band; idle and follow_schedule take nothing:
parameters map of units and bounds, and an execution array. Read the block, find the parameter bounds, build a valid push. The loop is the same whichever device answers.
Commands are keyed by their canonical name; the read returns only canonical vocabulary. Conflict resolution is device-level, not per-command: one action is active per device at a time. Any new command conflicts with the one in flight, whether it is a different mode on the same device or the same mode again, and onConflict resolves the overlap. There is no per-command grouping to reason about; the capability block tells you the commands and their bounds, the conflict model tells you how concurrent requests resolve.
Coming soon. Live control for EV chargers and thermostats. The sandbox environment serves the full
commands and settings surface so you can build the whole integration against it. A live EV charger or thermostat push returns 422 COMMAND_NOT_SUPPORTED until the live path opens; battery control is live.Some device types are read-only
Not every device type accepts commands. A solar inverter and a vehicle expose state, sync status, and metadata, and nocommands block at all:
Presence-based support
A capability is supported iff its key is present. Absent means rejected.discharge against this device returns 422 UNSUPPORTED_MODE. A push for charge with a reserve parameter (when parameters.reserve is absent on the read) returns 422 UNSUPPORTED_PARAMETER. There is no supported: boolean field, and no “parameter exists but is disabled” middle state. Either the device declares it and honours it, or the API rejects with structured detail you can act on.
An empty parameters: {} is a valid declaration. It means the command is supported and takes no parameters. auto.* commands typically declare empty parameters because their behaviour is intent-only; strategy parameters belong to the underlying optimisation, not to the canonical request.
Per-mode execution shapes
Each command declares anexecution array drawn from { "immediate", "scheduled", "windowed" }:
| Token | Request shape | Validation |
|---|---|---|
immediate | No start | Allowed only if "immediate" is in the array. |
scheduled | start only | Allowed only if "scheduled" is in the array. |
windowed | start and end | Allowed only if "windowed" is in the array. |
EXECUTION_NOT_SUPPORTED with details: { requestedExecution, supportedExecution }. Validation is per-mode, not per-device: a device may accept auto.balanced immediately but require charge to be windowed, and the capability response expresses that directly.
Units and bounds
Numeric parameters carry a unit and optional bounds:| Field | Behaviour |
|---|---|
unit | Required. Sending a different unit returns 422 UNSUPPORTED_UNIT. |
min | Optional. Open-ended on the lower side if absent. |
max | Optional. Open-ended on the upper side if absent. |
PARAMETER_OUT_OF_RANGE. The bounds you act on come from the device read, not from a fixed request schema. The thermostat’s target and heatSetpoint follow the same rule as the battery’s power: read the bound off commands, build a value inside it, post. Because the bound lives on the read, it is per-device, so a value valid for one device can be rejected by another of the same type. Bounds reflect the device’s documented operating range merged with Amps-side safety. They do not change the canonical name or unit; they only constrain the value.
Settings
settings declares persistent device configuration with current values and bounds. It sits beside commands on the read, and like commands it is a per-device-type surface: settings are how a device operates, not what it does. The battery’s canonical writable surface:
| Setting | Type | Unit | Meaning |
|---|---|---|---|
safety_reserve | number | percent | Lowest SOC the battery will reach, even during a power cut. |
discharge_floor | number | percent | Lowest SOC during normal operation. |
charge_ceiling | number | percent | Highest SOC the battery will charge to. |
export_limit | number | watts | Maximum power the battery sends to the grid. |
max_charge_rate | number | amps | Maximum charge current. |
max_discharge_rate | number | amps | Maximum discharge current. |
max_charge_rate (in kw, 0 to 50), the ceiling the charger draws under. It rides the same {value, unit} shape and the same POST /ev-charger/{deviceId}/settings write path as the battery’s settings. A ceiling is configuration, not an intent, so it lives here rather than as a push parameter, which keeps the charger’s two commands (charge, idle) clean. See canonical actions on the actions-versus-settings boundary.
Read-only settings (e.g., scheduler_enabled) appear in the read response but reject writes with 422 READ_ONLY_SETTING. Unknown setting keys reject with 422 UNSUPPORTED_SETTING. Writes use a separate endpoint and a sparse-map body. See What to expect on read-write parity.
Computed at read time
The capability declaration is computed at read time, not cached separately; an updated declaration takes effect on the next read. Only canonical names appear in the response.Availability tiers
Devices graduate from sandbox-only simulations to general availability. Sandbox devices are visible only in the sandbox environment. Generally-available devices are visible to everyone in live. Tiers in between gate visibility while an integration is firming up; promotion to general requires no per-customer changes.How agents use this
A read-then-build-then-write loop is what makes agentic device control mechanical. The agent does not need to know which OEM is on the other end; the device tells it what is acceptable and the agent picks a valid request shape. This is the “build once, control any device” property in code./ev-charger/{id} or /hvac/{id} and the loop is unchanged: read commands, pick a command the device declares, build parameters inside the declared bounds, post. The device tells the agent what is acceptable, the agent builds and posts. Build once, control any device.
Frequently asked questions
How do I check if a battery supports auto.balanced?
Read the device withGET /battery/{deviceId} and inspect commands. If auto.balanced is a key on the commands map, the device supports it. If absent, a push for auto.balanced returns 422 UNSUPPORTED_MODE. There is no supported: boolean field; capability is declared by presence. The same rule holds for parameters and units: present means supported, absent means rejected.
Where is the capability response defined?
Capabilities live on the device read response, not at a separate/capabilities endpoint. GET /battery/{deviceId}, GET /ev-charger/{deviceId}, and GET /hvac/{deviceId} all return commands in one body alongside state. The shape is one contract across device types: each command is keyed by its canonical name and carries a parameters map of units and bounds and an execution array. One read is enough to construct a valid command without external documentation.
Do all device types support commands?
No, and the read response tells you which do. A battery, EV charger, and thermostat return acommands block declaring the actions they accept. A solar inverter and a vehicle return state with no commands block, because the canonical surface for those device types is read-only. The absence is the contract: the device declares everything it offers, and for these types that is reads.
Can capabilities change at runtime?
Yes. The capability declaration is computed at read time, not cached separately, so an updated declaration takes effect on the next read. State values change continuously; the capability declaration changes when the platform ships an update. New capabilities surface automatically; clients that already poll the read pick them up without code changes.What are availability tiers?
Devices graduate from sandbox-only simulations to general availability. Sandbox devices are visible only in the sandbox environment. Generally-available devices are visible to everyone in live. Tiers in between gate visibility while an integration is firming up. Promotion does not require per-customer changes.Related concepts
Canonical Actions
The action shape that mirrors the capability response.
Device State
Where the capability response lives on the API.
What to expect
Why presence-based support and self-describing parameters.
Error Envelope
UNSUPPORTED_MODE, UNSUPPORTED_PARAMETER, READ_ONLY_SETTING.