Omada SDN Controller API
Manage TP-Link Omada SDN Controllers via the Open API (OpenAPI 3.0.1).
Compatibility
The Open API is available in Omada SDN Controller v5.9 and later (both the Software Controller and Cloud-Based Controller). Earlier versions only have an undocumented internal API that uses cookie-based session auth — this skill does not cover that legacy API.
The Open API feature must be explicitly enabled by an administrator before use.
Environment Setup
Before making any API calls, you need three environment variables. If the user has not provided these or a
file does not exist, walk them through the setup:
-
Find the controller URL — Ask the user for their Omada Controller address. This is typically
for the Software Controller. The port may differ if customized during installation.
-
Create API credentials — Guide the user to:
- Log into the Omada Controller web UI
- Navigate to Global View > Settings > Platform Integration > Open API
- Click Add New App
- Set the type to Client (not Gateway)
- Copy the generated Client ID and Client Secret
-
Create the file — Have the user create a
file in the project root:
OMADA_URL=https://omada.example.com:8043
OMADA_CLIENT=your-client-id
OMADA_SECRET=your-client-secret
Never commit to git. Ensure
includes it.
Load variables before making requests:
bash
export $(grep -v '^#' .env | xargs)
Do NOT use
— variables won't propagate to subshells or curl.
If auth fails, common causes are:
- The Open API feature is not enabled on the controller
- The client app type is set to Gateway instead of Client
- The controller URL is wrong or missing the port
- The client secret was rotated in the UI but not updated in
Locating the Wrapper Script
The API wrapper script is at
relative to this skill's directory. On first use in a session, find the script path using Glob to search for
**/omada-controller/scripts/omada-api.sh
. Use the discovered absolute path for all subsequent calls. For example, if the skill is installed at
.claude/skills/omada-controller/
, the script path is
.claude/skills/omada-controller/scripts/omada-api.sh
.
Making API Calls
Always use the wrapper script for all Omada API calls. It handles env loading, authentication, and URL construction automatically.
Anti-patterns: Any command beyond
triggers extra permission prompts. NEVER pipe, chain, or post-process script output with other commands. Use
for filtering and read the JSON output yourself for anything else. NEVER use curl, prefix assignments, or shell variables. Common anti-patterns:
bash <script> ... | jq ...
— use flag instead
bash <script> ... | python3 -c "..."
— use or read the output yourself
cat saved-response.txt | python3 -c "..."
— read the JSON yourself, do not shell out
curl -sk "${OMADA_URL}/api/info"
— use the script, not curl
SITE="123" && bash <script> ...
— inline the value directly in the path
SITE="123"; for mac in ...
— no shell variables or loops
On first use in a session, run the health check with no arguments. This verifies connectivity and lets the user approve the script once for all subsequent calls:
Then make API calls:
bash
bash <script> <METHOD> <PATH> [JSON_BODY] [--raw] [--jq FILTER]
The path is relative to
— no need to construct full URLs or manage tokens.
Examples
bash
# List sites
bash <script> GET "/sites?page=1&pageSize=100"
# List devices at a site
bash <script> GET "/sites/{siteId}/devices?page=1&pageSize=100"
# Reboot a device
bash <script> POST /sites/{siteId}/cmd/devices/reboot '{"deviceMacs":["AA-BB-CC-DD-EE-FF"]}'
# v2 endpoints — prefix path with /v2
bash <script> GET /v2/sites/{siteId}/setting/firewall/rules
# Fetch the OpenAPI spec for endpoint discovery
bash <script> GET /v3/api-docs --raw
# Filter API paths by keyword (use --jq, NEVER pipe)
bash <script> GET /v3/api-docs --raw --jq '[.paths | keys[] | select(test("port"; "i"))]'
# Extract just the data array from a paginated response
bash <script> GET "/sites/{siteId}/clients?page=1&pageSize=100" --jq '.result.data'
Path Routing
The script routes paths automatically:
- →
/openapi/v1/{omadacId}/sites/...
- →
/openapi/v2/{omadacId}/sites/...
- → (no auth prefix)
Recommended Permission
Users should add this to their Claude Code settings to allow the script without repeated prompts:
Bash(bash *omada-api.sh *)
API Discovery
The controller hosts its own full OpenAPI 3.0.1 spec (~1,507 endpoints). Use it to discover any endpoint at runtime rather than hard-coding paths.
- Swagger UI (browser):
{OMADA_URL}/swagger-ui/index.html
— no auth required
- OpenAPI spec (JSON):
bash
bash <script> GET /v3/api-docs --raw --jq '.paths | keys[]'
Common Patterns
Site ID
Most endpoints are site-scoped. Get the site ID first:
bash
bash <script> GET "/sites?page=1&pageSize=100"
Then use it in subsequent paths:
,
, etc.
Pagination
Paginated endpoints accept
and
query parameters and return:
json
{
"errorCode": 0,
"result": {
"totalRows": 7,
"currentPage": 1,
"currentSize": 100,
"data": [...]
}
}
API Versions
Some endpoints use
instead of the default
. The swagger spec includes both — prefix the path with
when needed.
Authentication Details
The wrapper script handles auth automatically, but for reference:
- Controller ID: Fetched from (outside the OpenAPI spec)
- Token: OAuth2 client credentials via
POST {OMADA_URL}/openapi/authorize/token?grant_type=client_credentials
with , , in the JSON body ( is a query param, not body)
- Header format:
Authorization: AccessToken=<token>
(NOT )
- Expiry: Tokens last 2 hours (7200 seconds) — the script re-authenticates each call
Gotchas
- Quoting: Always quote paths containing (e.g.,
"/sites?page=1&pageSize=100"
). Unquoted is a shell background operator.
- TLS: The controller typically uses a self-signed cert. Always use .
- MAC format: The API uses (uppercase, dashes).
- Error handling: Always check in responses. = success, negative = error.
- Token header:
Authorization: AccessToken=<token>
, NOT .
References
- scripts/omada-api.sh — API wrapper (handles env, auth, and requests)
- references/api-categories.md — API endpoint categories and counts
- references/external-resources.md — Links to docs, examples, and integrations