skills/uptimerobot/SKILL.md
--- name: uptimerobot description: Manage UptimeRobot monitors — create, list, pause, delete HTTP/keyword/port/ping monitors; set up alert contacts; fetch uptime stats. Use when user wants to add uptime monitoring, create a status check, pause a monitor, see uptime %, or wire a new deployed app into external monitoring. category: observability argument-hint: [monitor <list|create|pause|resume|delete>] [contact <list|create>] [account] [--readonly] allowed-tools: Bash(curl *) Bash(jq *) Read Writ
npx skillsauth add RonanCodes/ronan-skills skills/uptimerobotInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
CLI-first UptimeRobot ops via the v2 API. Covers monitors, alert contacts, and account queries. Uses UPTIMEROBOT_API_KEY by default; --readonly switches to the read-only key.
/ro:uptimerobot account # account + monitor limit + usage
/ro:uptimerobot monitor list # all monitors with status + uptime ratio
/ro:uptimerobot monitor create https://app.com --name "My App"
/ro:uptimerobot monitor create https://api.com --name "API" --keyword "ok"
/ro:uptimerobot monitor pause <id>
/ro:uptimerobot monitor resume <id>
/ro:uptimerobot monitor delete <id> # irreversible — asks confirm
/ro:uptimerobot contact list
/ro:uptimerobot contact create --email [email protected]
/ro:uptimerobot monitor list --readonly # uses read-only key
~/.claude/.env:
UPTIMEROBOT_API_KEY — main key, full CRUDUPTIMEROBOT_READONLY_API_KEY — read-only (safe for dashboards, status pages)Both verified working on 2026-04-19 against api.uptimerobot.com/v2.
UptimeRobot v2 API uses POST for every call (GET is not supported). Responses are JSON when format=json is passed.
Base URL: https://api.uptimerobot.com/v2
curl -s -X POST "https://api.uptimerobot.com/v2/getAccountDetails" \
-H "Content-Type: application/x-www-form-urlencoded" \
-d "api_key=${UPTIMEROBOT_API_KEY}&format=json" \
| jq '.account | {email, monitor_limit, up_monitors, down_monitors, paused_monitors, total_monitors_count, registered_at}'
Ronan's account (2026-04-19): free tier, 50-monitor limit.
curl -s -X POST "https://api.uptimerobot.com/v2/getMonitors" \
-H "Content-Type: application/x-www-form-urlencoded" \
-d "api_key=${UPTIMEROBOT_API_KEY}&format=json&custom_uptime_ratios=1-7-30" \
| jq '.monitors[] | {id, friendly_name, url, type, status, uptime_7d: (.custom_uptime_ratio // "n/a")}'
Status values:
0 — paused1 — not checked yet2 — up8 — seems down9 — downcurl -s -X POST "https://api.uptimerobot.com/v2/newMonitor" \
-H "Content-Type: application/x-www-form-urlencoded" \
--data-urlencode "api_key=${UPTIMEROBOT_API_KEY}" \
--data-urlencode "format=json" \
--data-urlencode "friendly_name=My App" \
--data-urlencode "url=https://app.example.com" \
--data-urlencode "type=1" \
--data-urlencode "interval=300"
type values:
1 — HTTP(s)2 — keyword (needs keyword_type=1|2 and keyword_value=<str>)3 — ping4 — port (needs sub_type + port)5 — heartbeat (passive — call the monitor URL from your app)interval is seconds (min 60 on paid plans, 300 / 5min on free).
curl -s -X POST "https://api.uptimerobot.com/v2/newMonitor" \
-H "Content-Type: application/x-www-form-urlencoded" \
--data-urlencode "api_key=${UPTIMEROBOT_API_KEY}" \
--data-urlencode "format=json" \
--data-urlencode "friendly_name=API Health" \
--data-urlencode "url=https://api.example.com/health" \
--data-urlencode "type=2" \
--data-urlencode "keyword_type=2" \
--data-urlencode "keyword_value=ok"
keyword_type: 1 = exists, 2 = not exists (alert when absent).
# Pause
curl -s -X POST "https://api.uptimerobot.com/v2/editMonitor" \
-d "api_key=${UPTIMEROBOT_API_KEY}&format=json&id=${ID}&status=0"
# Resume
curl -s -X POST "https://api.uptimerobot.com/v2/editMonitor" \
-d "api_key=${UPTIMEROBOT_API_KEY}&format=json&id=${ID}&status=1"
curl -s -X POST "https://api.uptimerobot.com/v2/deleteMonitor" \
-d "api_key=${UPTIMEROBOT_API_KEY}&format=json&id=${ID}"
Skill always prompts for confirmation before delete — irreversible.
# List
curl -s -X POST "https://api.uptimerobot.com/v2/getAlertContacts" \
-d "api_key=${UPTIMEROBOT_API_KEY}&format=json" \
| jq '.alert_contacts[] | {id, friendly_name, type, value, status}'
# Create email contact
curl -s -X POST "https://api.uptimerobot.com/v2/newAlertContact" \
-d "api_key=${UPTIMEROBOT_API_KEY}&format=json&type=2&[email protected]&friendly_name=Me"
type: 2 = email, 3 = webhook, 9 = Slack, 11 = Discord, etc.
Pass alert_contacts=<contact_id>_<threshold>_<recurrence>-<contact_id>_0_0-...:
contact_id — from getAlertContactsthreshold — minutes before alerting (0 for immediate)recurrence — minutes between re-alerts (0 for one-shot)Example — alert contact 1234 immediately and never re-alert:
alert_contacts=1234_0_0
Pass --readonly to swap UPTIMEROBOT_API_KEY for UPTIMEROBOT_READONLY_API_KEY. Any mutation (newMonitor, editMonitor, deleteMonitor, newAlertContact) returns:
{"stat":"fail","error":{"type":"not_authorized","message":"You are not allowed to perform this request"}}
Use for status-page rendering or CI dashboards where write access is unwanted.
stat: fail + error.type: "invalid_parameter" — usually missing format=json or a required param. Add --data-urlencode to every value to avoid breakage on URLs with & or =.type=3 (ping) for non-HTTP endpoints.~/.claude/.env.UPTIMEROBOT_READONLY_API_KEY — it's scoped to reads only./ro:cf-ship — pair a fresh deploy with a new monitor/ro:sentry — for in-app error tracking (complements UR's external HTTP checks)development
--- name: worktree description: Coordinate multiple agents on one repo via a worktree-lock pool, so two agents never clobber each other's working tree. Acquire the first free slot (main, then beta/gamma… worktrees, created on demand), work there on your own branch, release when you've pushed. Use before modifying any repo that might be in use by another agent (factory, dataforce, etc.), or whenever you're told a repo is being worked on. Backed by `ro worktree`. category: development argument-hin
testing
--- name: ship description: Ship a feature branch the local-CI-first way — run the full local gate, push, open a PR, squash-merge, then deploy, without waiting on GitHub Actions. Use when a branch is ready for main and you want it merged and deployed now. Reads CI policy from `ro ci` (default skips remote CI because GitHub Actions billing keeps hitting limits). Sibling to /ro:gh-ship (waits on GitHub checks) and /ro:cf-ship (the deploy half). Triggers on "ship it", "ship this", "merge and deploy
testing
--- name: setup-logging description: Set up (or audit) the observability stack in a TanStack Start + Cloudflare Workers app so it is "diagnosable by default" — structured logging (logtape) with a request context carrying trace_id + userId + tenant/orgId, a trace_id propagated FE→BE→logs→Sentry→PostHog, Cloudflare Workers observability enabled, and Sentry + PostHog wired. Two modes: `setup` (wire it into an app) and `audit` (check an existing app + report gaps). Use when scaffolding a new app, wh
development
Manage credentials INSIDE the active ~/.claude/.env file — read which token/account to use for a given app (Simplicity vs Dataforce vs Ronan-personal), add or update a secret WITHOUT it passing through the chat (an interactive Terminal window prompts for it), and track secrets that were exposed in a transcript so they get rotated. Sibling to /ro:context (which switches WHICH env file is active). Use when the user wants to add an API key/token/secret, asks "which credential do I use for X", needs the env organized/labelled, or a secret was pasted into the chat and should be rotated.