# LED Controller API This document covers: 1. **HTTP and WebSocket** exposed by the Raspberry Pi app (`src/main.py`) — profiles, zones, presets, transport send, pattern OTA helpers, and related resources. 2. **LED driver JSON** — the compact **v1** message format. It is sent over the **serial → ESP-NOW bridge** to ESP32 peers and as **newline-delimited JSON over TCP** to **Wi-Fi** drivers (same logical fields). Default HTTP listen address: `0.0.0.0`. Port defaults to **80**; override with the **`PORT`** environment variable (see `pipenv run run`). **Serial:** UART path and baud come from settings (defaults include `serial_port` such as `/dev/ttyS0` and `serial_baudrate`). **Wi-Fi drivers:** the Pi accepts TCP connections on **`tcp_port`** in settings (default **8765**). **UDP discovery** listens on **8766** so drivers can find the controller IP on the LAN. All JSON APIs use `Content-Type: application/json` for bodies and responses unless noted. --- ## UI behavior notes The main UI has two modes controlled by the mode toggle: - **Run mode**: optimized for operation (zone/preset selection and profile apply). - **Edit mode**: shows editing/management controls (zones, presets, patterns, colour palette, send presets, profile management actions, **Devices** registry for LED driver names/MACs, and related tools). Profiles are available in both modes, but behavior differs: - **Run mode**: profile **apply** only. - **Edit mode**: profile **create/clone/delete/apply**. `POST /presets/send` is wired to the **Send Presets** UI action, which is exposed in Edit mode. --- ## Session and scoping Several routes use **`@with_session`**: the server stores a **current profile** in the session (cookie). Endpoints that scope data to “the current profile” (notably **`/presets`**) only return or mutate presets whose `profile_id` matches that session value. Profiles are selected with **`POST /profiles//apply`**, which sets `current_profile` in the session. --- ## Static pages and assets | Method | Path | Description | |--------|------|-------------| | GET | `/` | Main UI (`templates/index.html`) | | GET | `/settings` | Settings page (`templates/settings.html`) | | GET | `/favicon.ico` | Empty response (204) | | GET | `/static/` | Static files under `src/static/` | --- ## WebSocket: `/ws` Connect to **`ws://:/ws`**. - Send **JSON**: the object is forwarded through the **serial sender** (6-byte MAC prefix + payload to the ESP-NOW bridge). Optional key **`to`**: 12-character hex MAC address; if present it is removed from the object and the payload is sent to that peer; otherwise the default destination from settings is used. - Send **non-JSON text**: forwarded as raw bytes with the default address. - On send failure, the server may reply with `{"error": "Send failed"}`. Wi-Fi devices are not targeted by `/ws` directly; use **`POST /presets/send`**, device routes, or **`POST /patterns//send`** as appropriate. --- ## HTTP API by resource Below, `` values are string identifiers used by the JSON stores (numeric strings in practice). ### Settings — `/settings` | Method | Path | Description | |--------|------|-------------| | GET | `/settings` | Full settings object (from `settings.json` / `Settings` model). | | PUT | `/settings/settings` | Merge keys into settings and save. Returns `{"message": "Settings updated successfully"}`. | | GET | `/settings/wifi/ap` | Saved Wi‑Fi AP fields: `saved_ssid`, `saved_password`, `saved_channel`, `active` (Pi: `active` is always false). | | POST | `/settings/wifi/ap` | Body: `ssid` (required), `password`, `channel` (1–11). Persists AP-related settings. | | GET | `/settings/page` | Serves `templates/settings.html` (same page as `GET /settings` from the root app, for convenience). | ### Devices — `/devices` Registry in `db/device.json`: storage key **``** (string, e.g. `"1"`) maps to an object that always includes: | Field | Description | |-------|-------------| | **`id`** | Same as the storage key (stable handle for URLs). | | **`name`** | Shown in the UI and used in `select` keys. | | **`type`** | `led` (only value today; extensible). | | **`transport`** | `espnow` or `wifi`. | | **`address`** | For **`espnow`**: optional 12-character lowercase hex MAC. For **`wifi`**: optional IP or hostname string. | | **`default_pattern`**, **`zones`** | Optional. Legacy **`tabs`** may still appear in old files and is migrated away on load. | Existing records without `type` / `transport` / `id` are backfilled on load (`led`, `espnow`, and `id` = key). | Method | Path | Description | |--------|------|-------------| | GET | `/devices` | Map of device id → device object. | | GET | `/devices/` | One device, 404 if missing. | | POST | `/devices` | Create. Body: **`name`** (required), **`type`** (default `led`), **`transport`** (default `espnow`), optional **`address`**, **`default_pattern`**, **`zones`**. Returns `{ "": { ... } }`, 201. | | PUT | `/devices/` | Partial update. **`name`** cannot be cleared. **`id`** in the body is ignored. **`type`** / **`transport`** validated; **`address`** normalised for the resulting transport. | | DELETE | `/devices/` | Remove device. | ### Profiles — `/profiles` | Method | Path | Description | |--------|------|-------------| | GET | `/profiles` | `{"profiles": {...}, "current_profile_id": ""}`. Ensures a default current profile when possible. | | GET | `/profiles/current` | `{"id": "...", "profile": {...}}` | | GET | `/profiles/` | Single profile. If `` is `current`, same as `/profiles/current`. | | POST | `/profiles` | Create profile. Body may include `name` and other fields. Optional `seed_dj_zone` (request-only) seeds a DJ zone + presets. New profiles always get a populated `default` zone. Returns `{ "": { ... } }` with status 201. | | POST | `/profiles//apply` | Sets session current profile to ``. | | POST | `/profiles//clone` | Clone profile (zones, palettes, presets). Body may include `name`. | | PUT | `/profiles/current` | Update the current profile (from session). | | PUT | `/profiles/` | Update profile by id. | | DELETE | `/profiles/` | Delete profile. | ### Presets — `/presets` Scoped to **current profile** in session (see above). | Method | Path | Description | |--------|------|-------------| | GET | `/presets` | Map of preset id → preset object for the current profile only. | | GET | `/presets/` | One preset, 404 if missing or wrong profile. | | POST | `/presets` | Create preset; server assigns id and sets `profile_id`. Body fields stored on the preset. Returns `{ "": { ... } }`, 201. | | PUT | `/presets/` | Update preset (must belong to current profile). | | DELETE | `/presets/` | Delete preset. | | POST | `/presets/send` | Push presets to the LED driver over the configured transport (see below). | **`POST /presets/send` body:** ```json { "preset_ids": ["1", "2"], "save": true, "default": "1", "destination_mac": "aabbccddeeff" } ``` - **`preset_ids`** (or **`ids`**): non-empty list of preset ids to include. - **`save`**: if true, the outgoing message includes `"save": true` so the driver may persist presets (default true). - **`default`**: optional preset id string; forwarded as top-level `"default"` in the driver message (startup selection on device). - **`destination_mac`** (or **`to`**): optional 12-character hex MAC for unicast; omitted uses the transport default (e.g. broadcast). Response on success includes `presets_sent`, `messages_sent` (chunking splits payloads so each JSON string stays ≤ 240 bytes). Stored preset records can include: - `colors`: resolved hex colours for editor/display. - `palette_refs`: optional array of palette indexes parallel to `colors`. If a slot contains an integer index, the colour is linked to the current profile palette at that index. ### Zones — `/zones` | Method | Path | Description | |--------|------|-------------| | GET | `/zones` | `zones` (map of zone id → zone object), `zone_order`, `current_zone_id`, `profile_id` for the session-backed profile. | | GET | `/zones/current` | Current zone from cookie/session. | | POST | `/zones` | Create zone; optional JSON `name`, `names`, `presets`; can append to current profile’s zone list. | | GET | `/zones/` | Zone JSON. | | PUT | `/zones/` | Update zone. | | DELETE | `/zones/` | Delete zone; can delete `current` to remove the active zone; updates profile zone list. | | POST | `/zones//set-current` | Sets `current_zone` cookie. | | POST | `/zones//clone` | Clone zone into current profile. | ### Palettes — `/palettes` | Method | Path | Description | |--------|------|-------------| | GET | `/palettes` | Map of id → colour list. | | GET | `/palettes/` | `{"colors": [...], "id": ""}` | | POST | `/palettes` | Body may include `colors`. Returns palette object with `id`, 201. | | PUT | `/palettes/` | Update colours (`name` ignored). | | DELETE | `/palettes/` | Delete palette. | ### Groups — `/groups` | Method | Path | Description | |--------|------|-------------| | GET | `/groups` | All groups. | | GET | `/groups/` | One group. | | POST | `/groups` | Create; optional `name` and fields. | | PUT | `/groups/` | Update. | | DELETE | `/groups/` | Delete. | ### Scenes — `/scenes` | Method | Path | Description | |--------|------|-------------| | GET | `/scenes` | All scenes. | | GET | `/scenes/` | One scene. | | POST | `/scenes` | Create (body JSON stored on scene). | | PUT | `/scenes/` | Update. | | DELETE | `/scenes/` | Delete. | ### Sequences — `/sequences` | Method | Path | Description | |--------|------|-------------| | GET | `/sequences` | All sequences. | | GET | `/sequences/` | One sequence. | | POST | `/sequences` | Create; may use `group_name`, `presets` in body. | | PUT | `/sequences/` | Update. | | DELETE | `/sequences/` | Delete. | ### Patterns — `/patterns` Pattern metadata lives in **`db/pattern.json`**; driver source files live under **`led-driver/src/patterns/`**. Several routes expose a **runtime map** (metadata merged with on-disk `.py` names so new files appear in menus). | Method | Path | Description | |--------|------|-------------| | GET | `/patterns` | Runtime pattern map (object keyed by pattern id). | | GET | `/patterns/definitions` | Same runtime map (intended for UI “definitions” clients). | | GET | `/patterns/ota/manifest` | JSON `{"files":[{"name":"blink.py","url":"http:///patterns/ota/file/blink.py"},...]}` for OTA pulls. Requires **`Host`** header. | | GET | `/patterns/ota/file/` | Raw **`.py`** source for one driver pattern (`name` must be a safe filename, e.g. `rainbow.py`). | | POST | `/patterns//send` | Push a **manifest** JSON line to **Wi-Fi** devices so they pull one pattern file over HTTP. Body may include **`device_id`** to target one device; otherwise all Wi-Fi devices with an **`address`** are tried. **``** may be with or without `.py`. | | POST | `/patterns/upload` | Body JSON: **`name`**, **`code`**, optional **`overwrite`** (default true). Writes **`led-driver/src/patterns/.py`**. | | POST | `/patterns/driver` | Body JSON: **`name`** (identifier), **`code`**, optional metadata (`min_delay`, `max_delay`, `max_colors`, `n1`…`n8`, **`overwrite`**). Creates/updates both the **`.py`** file and **`db/pattern.json`** via the Pattern model. | | GET | `/patterns/` | One pattern record from the Pattern model (metadata only). | | POST | `/patterns` | Create (`name`, optional `data`). | | PUT | `/patterns/` | Update. | | DELETE | `/patterns/` | Delete. | **Devices — pattern OTA push** | Method | Path | Description | |--------|------|-------------| | POST | `/devices//patterns/push` | Wi-Fi only. Asks the driver at **`address`** to pull pattern files from this server. Optional body **`manifest`**: either a **URL string** pointing at a manifest JSON document, or a **manifest object** (same shape as in driver messages). If omitted, a default manifest is built from the request **`Host`** header. | --- ## LED driver message format (transport / ESP-NOW / Wi-Fi) Messages are JSON objects. The Pi **`build_message()`** helper (`src/util/espnow_message.py`) produces the same shape sent over serial and forwarded by the ESP32 bridge, and the same logical object can be sent as a **single JSON line** to a Wi-Fi driver over TCP. ### Top-level fields ```json { "v": "1", "presets": { }, "select": { }, "save": true, "default": "preset_id", "b": 255 } ``` - **`v`** (required): Must be `"1"` or the driver ignores the message. - **`presets`**: Map of **preset id** (string) → preset object (see below). Optional **`name`** field on each value is accepted for display; the driver keys presets by map key. - **`select`**: Map of **device name** (as in device settings) → `[ "preset_id" ]` or `[ "preset_id", step ]`. - **`save`**: If present (e.g. true), the driver may persist presets to flash after applying. - **`default`**: Preset id string to use as startup default on the device. - **`b`**: Optional **global** brightness 0–255 (driver applies this in addition to per-preset brightness). ### Preset object (wire / driver keys) On the wire, presets use **short keys** (saves space in the ≤240-byte chunks): | Key | Meaning | Notes | |-----|---------|--------| | `p` | Pattern id | `off`, `on`, `blink`, `rainbow`, `pulse`, `transition`, `chase`, `circle` | | `c` | Colours | Array of `"#RRGGBB"` hex strings; converted to RGB on device | | `d` | Delay ms | Default 100 | | `b` | Preset brightness | 0–255; combined with global `b` on the device | | `a` | Auto | `true`: run continuously; `false`: one step/cycle per “beat” | | `n1`–`n6` | Pattern parameters | See below | The HTTP app’s **`POST /presets/send`** path builds this from stored presets via **`build_preset_dict()`** (long names like `pattern` / `colors` in the DB are translated to `p` / `c` / …). ### Pattern-specific parameters (`n1`–`n6`) #### Rainbow - **`n1`**: Step increment on the colour wheel per update (default 1). #### Pulse - **`n1`**: Attack (fade in) ms - **`n2`**: Hold ms - **`n3`**: Decay (fade out) ms - **`d`**: Off time between pulses ms #### Transition - **`d`**: Transition duration ms #### Chase - **`n1`**: LEDs with first colour - **`n2`**: LEDs with second colour - **`n3`**: Movement on even steps (may be negative) - **`n4`**: Movement on odd steps (may be negative) #### Circle - **`n1`**: Head speed (LEDs/s) - **`n2`**: Max length - **`n3`**: Tail speed (LEDs/s) - **`n4`**: Min length ### Select messages ```json { "select": { "device_name": ["preset_id"], "other_device": ["preset_id", 10] } } ``` - One element: select preset; step behavior follows driver rules (reset on `off`, etc.). - Two elements: explicit **step** for sync. ### Beat and sync behavior - Sending **`select`** again with the **same** preset name acts as a **beat** (advances manual patterns / restarts generators per driver logic). - Choosing **`off`** resets step as a sync point; then selecting a pattern aligns step 0 across devices unless a step is passed explicitly. ### Example (compact preset map) ```json { "v": "1", "save": true, "presets": { "1": { "name": "Red blink", "p": "blink", "c": ["#FF0000"], "d": 200, "b": 255, "a": true, "n1": 0, "n2": 0, "n3": 0, "n4": 0, "n5": 0, "n6": 0 } }, "select": { "living-room": ["1"] } } ``` --- ## Processing summary (driver) 1. Reject if `v != "1"`. 2. Apply optional top-level **`b`** (global brightness). 3. For each entry in **`presets`**, normalize colours and upsert preset by id. 4. If this device’s **`name`** appears in **`select`**, run selection (optional step). 5. If **`default`** is set, store startup preset id. 6. If **`save`** is set, persist presets. --- ## Error handling (HTTP) Controllers typically return JSON with an **`error`** string and 4xx/5xx status codes. Invalid JSON bodies often yield `{"error": "Invalid JSON"}`. --- ## Notes - **Human-readable preset fields** (`pattern`, `colors`, `delay`, …) are fine in the **web app / database**; the **send path** converts them to **`p` / `c` / `d`** for the driver. - For a copy of the older long-key reference, see **`led-driver/docs/API.md`** in this repo (conceptually the same behavior; wire format prefers short keys).