Send notifications through the unified notification router
assistant skills install notificationsCall this when something happened that the user would want to know about ā a completed task with a notable outcome, an interesting observation, a positive trend you noticed in monitored data, useful research worth surfacing, a workflow that got blocked, a credential or token failure, etc. Do not call it for routine task completions where nothing notable happened. When in doubt and you have a real observation to share, share it.
Always pass --title. Skipping it triggers a fallback that just truncates --message to 60 chars and shows it as the title ā the user sees the same text twice with no scannability gained.
assistant notifications send \
--title "Short headline" \
--message "Your verbatim observation in your own words"
For time-sensitive items:
assistant notifications send --title "..." --message "..." --urgent
| Flag | Required | Description |
|---|---|---|
--message <message> | Yes | Notification body. Markdown (GFM) renders in the detail panel; the OS banner shows plain text. |
--title <title> | Yes in practice | Short headline (⤠8 words). Omitting it triggers a body-truncation fallback that shows up as a duplicate of --message ā always write a real title. |
--urgent | No | Mark as needing attention now/soon |
--json | No | Output machine-readable JSON |
Write a --title for every notification. It's the only line the user sees in the lock-screen popup and the collapsed row of the notification list, so a short noun phrase (⤠8 words) is what makes the notification scannable. If you omit --title, the system falls back to the first sentence of --message (truncated at 60 chars) ā that's almost always worse than what you'd write, because it duplicates body text the user is already going to read.
Avoid restating the first sentence of --message verbatim ā the title should add scannability, not duplicate.
The body renders as markdown (GFM) in the home feed detail panel ā where the user actually opens the notification on web, iOS, and macOS. Light markdown makes multi-fact bodies scannable. The OS lock-screen banner shows the body as plain text, so prefer inline emphasis over heavy structure that looks ugly unrendered.
Supported: **bold**, *italic*, `inline code`, fenced code blocks, links, bulleted and numbered lists, blockquotes, headings, GFM tables, ~~strikethrough~~.
Use it like this:
code for identifiers, paths, commands, and short snippets.Avoid large headings (#, ##) and wide tables ā they render fine in the panel but look noisy in the banner preview.
Use --urgent for items needing attention now/soon (blocked work, broken auth, time-sensitive issues). Skip for items the user should see when they have time.
# Plain notification ā bold the headline fact
assistant notifications send \
--title "Backup complete" \
--message "Nightly backup finished ā **12.4 GB** archived to cold storage across **3** datasets."
# Urgent notification ā inline code for the identifier
assistant notifications send \
--title "Auth token expired" \
--message "Sync is paused until you reauthenticate the \`GitHub\` integration." \
--urgent
{ "ok": true, "signalId": "...", "dispatched": true }
assistant notifications list --json
Reads from the user's home feed (~/.vellum/workspace/data/home-feed.json) ā the inbox that mirrors background and async notifications surfaced via the unified pipeline. Real-time chat pushes that did not mirror to the feed (direct Telegram/Slack/Vellum-chat sends without --is-async-background) will not appear here.
--conversation-id or --after to dedupe).--conversation-id or date range.| Flag | Purpose |
|---|---|
--all | Include dismissed items (default: excluded ā assistant cares about outstanding work) |
--status <s> | Filter by status (new / seen / acted_on / dismissed); repeatable. Overrides the --all default. |
--before <iso> / --after <iso> | ISO-8601 createdAt bounds (strict; = is excluded). |
--urgency <u> | Filter by urgency (low / medium / high / critical); repeatable. |
--category <c> | Filter by category (security / scheduling / background / email / system); repeatable. |
--conversation-id <id> | Only items tied to this conversation. |
--from-assistant | Only items the assistant herself emitted. |
--noteworthy | Only items flagged as noteworthy. |
--limit <n> | Default 20, max 200. |
--offset <n> | Pagination offset. Combine with --limit to walk older pages. |
# What's outstanding right now (defaults: skip dismissed, newest first)
assistant notifications list --json
# Everything you've shown the user today
assistant notifications list --after 2026-05-28T00:00:00Z --all --json
# Only high-stakes items
assistant notifications list --urgency high --urgency critical --json
# Pre-send dedupe: anything you already surfaced for this conversation
assistant notifications list --conversation-id 7fab234c --after 2026-05-28T00:00:00Z --json
# Walk older pages
assistant notifications list --limit 20 --offset 20 --json
{
"ok": true,
"items": [ /* FeedItem records: id, title?, summary, status, urgency?, category?, conversationId?, createdAt, ... */ ],
"total": 12,
"returned": 3,
"hasMore": true,
"updatedAt": "2026-05-28T10:30:00.000Z"
}
Use edit when an already-sent notification needs revising ā a typo in the body, a status update on something you previously surfaced (e.g. "in progress" ā "done"), or de-escalating the urgency of a stale alert. Prefer editing over re-sending: a fresh notification with the corrected text creates duplicate noise in the user's inbox and pings them twice.
assistant notifications edit --id <notif:uuid> --message "Corrected body"
The id field is the full notif:<uuid> printed by notifications list --json under items[].id. Bare uuids (without the notif: prefix) are also accepted.
assistant notifications list --json | jq '.items[] | {id, title, summary}'
| Flag | Required | Description |
|---|---|---|
--id <id> | Yes | Feed item id (notif:<uuid>) or bare uuid |
--message <text> | No* | New body ā updates the home-feed summary AND the delivered channel message where supported |
--title <text> | No* | New short headline (⤠8 words) |
--urgency <level> | No* | Change urgency (low/medium/high/critical). Feed-only ā does not re-push channel messages |
--status <state> | No* | Lifecycle transition (new/seen/acted_on/dismissed). Feed-only |
--json | No | Machine-readable JSON |
*At least one of --message, --title, --urgency, or --status must be supplied.
| Channel | Edit behavior |
|---|---|
| Home feed (macOS/iOS inbox) | Always updated when the item exists. |
| Slack | Updated in-place via chat.update when the original delivery captured a Slack ts. Deliveries older than this feature returned messageId: null and report outcome: "unsupported". |
| Push, email, SMS | Cannot be edited ā reported as outcome: "unsupported" in the result. |
{
"ok": true,
"feedItem": { "id": "notif:...", "title": "...", "summary": "...", "status": "new", "urgency": "low" },
"channels": [
{ "channel": "slack", "deliveryId": "...", "outcome": "updated" },
{ "channel": "platform", "deliveryId": "...", "outcome": "unsupported", "reason": "platform adapter does not support in-place edits" }
]
}
outcome values: "updated" (channel message edited successfully), "unsupported" (channel cannot edit at all), "skipped" (delivery wasn't in sent status), "failed" (channel-side error ā see reason).
# Fix a typo in the body
assistant notifications edit \
--id notif:abc12345-... \
--message "Backup completed ā 12.4 GB archived to cold storage."
# De-escalate an urgent alert that resolved itself
assistant notifications edit --id notif:abc12345-... --urgency low
# Dismiss a notification you previously surfaced
assistant notifications edit --id notif:abc12345-... --status dismissed
display notification or other OS-level notification commands for assistant-managed alerts. Always use assistant notifications send.messaging_send. For Slack, use the Slack Web API directly (see the slack skill).schedule_create with fire_at. For recurring alerts, use schedule_create with an expression (cron/RRULE).