Install
openclaw skills install surface-cliUse the Surface mail CLI to read and act on Gmail, Outlook, and generic IMAP/SMTP mail through one JSON-first contract. Prefer this skill when you need Outlook access for school or work accounts that do not expose IMAP, or generic IMAP for providers such as GMX, plus stable refs for unread fetch, sent-message lookup, structured search, thread refresh, message read, attachments, send or draft, archive, mark read or unread, and provider-supported RSVP.
openclaw skills install surface-cliSurface is a local-first mail CLI for Gmail, Outlook, and generic IMAP/SMTP. It is especially
useful for Outlook school or work accounts that only work through the web UI, and for mail
providers such as GMX that expose standard IMAP and SMTP settings. Surface prints
machine-readable JSON to stdout and stores local state in ~/.surface-cli.
thread_ref / message_ref values for follow-up commands or thread watchingsurface --help should work)Check setup:
surface account list
surface auth status
If the user asks to install the Surface skill for another agent on the same machine, use:
surface skill install codex
surface skill install claude-code
surface skill install all
Add an account:
surface account add personal_2 --provider gmail --email you@example.com
surface account add uni --provider outlook --email you@example.com
surface account add gmx --provider imap --email you@gmx.com
For reliable summary.needs_action, Surface should know who the account owner is. Gmail auth can
verify the mailbox email automatically; Outlook may need explicit human identifiers:
surface account identity set uni --email you@example.com --name "Your Name" --name-alias "FirstName"
surface account identity show uni
Log in:
surface auth login personal_2
surface auth login uni
surface auth login gmx \
--username you@gmx.com \
--password "$SURFACE_GMX_PASSWORD"
For GMX and similar providers, make sure IMAP/POP3 access is enabled in the
provider web settings before logging in. Generic IMAP login does not need a
Google Cloud project, OAuth client JSON, Microsoft Graph app registration, or a
browser session. Surface can infer server settings for supported domains such as
gmx.com and gmx.net; custom IMAP providers still require the explicit
--imap-* and --smtp-* flags. Generic IMAP does not run browser 2FA or OAuth
consent. If 2FA is enabled, use the provider's app-specific password when one is
available. --password <password> is supported and treats the flag value as the
password directly, but prefer --password-env, --password-file, or
--password-command because direct CLI passwords can leak through shell
history, process listings, terminal logs, or agent transcripts. Do not ask the
user to paste mailbox passwords into chat or store them in the repo.
Local policy lives in:
~/.surface-cli/config.toml
Important local knobs:
summarizer_backendsummarizer_modelwrites_enabledsend_modetest_recipientstest_account_allowlistSummarization is opt-in and controlled by the user's local config. Do not change
summarizer_backend, summarizer_model, or related environment variables unless the user
explicitly asks. If an external summarizer backend is enabled, email thread content may be sent to
the configured model provider; confirm the user accepts that privacy tradeoff before enabling or
changing summarization.
surface account list
surface auth status
surface auth status personal_2
surface mail fetch-unread --account uni --limit 10
surface mail fetch-unread --account personal_2 --limit 20
surface mail fetch-unread --account uni --session sess_01... --limit 10
surface mail search --account uni --text "invoice" --limit 10
surface mail search --account uni --from registrar@school.edu --subject "waitlist" --limit 10
surface mail search --account uni --session sess_01... --from registrar@school.edu --limit 10
surface mail search --account personal_2 --mailbox inbox --label unread --text "sale" --limit 10
surface mail search --account personal_2 --text "has:attachment newer_than:30d" --limit 5
surface mail search --account gmx --mailbox inbox --limit 10
surface mail sent --account uni
surface mail sent --account uni --recipient person@example.com --limit 10
surface mail sent --account uni --thread thr_01... --limit 10
surface mail sent --account uni --session sess_01... --recipient person@example.com --limit 10
surface mail sent --account personal_2 --limit 10
sent is message-first. Its default limit is the last 10 sent messages, not threads. Each returned
message includes message_ref and thread_ref; use surface mail thread get <thread_ref> --refresh when you need the full conversation around a sent message.
Use --thread <thread_ref> when you already know the conversation and need only the user's sent
messages in that thread for style or consistency. --thread may be combined with --recipient.
Surface is the polling primitive, not the scheduler or delivery transport. If the user asks to watch mail, use the surrounding automation system to rerun Surface commands and surface updates to the user-requested destination.
account, thread_ref, and the newest known
message/timestamp. On each check, rerun surface mail thread get <thread_ref> --refresh and
notify only when the newest message state changes.search, then use periodic fetch-unread checks
to catch new inbox arrivals and targeted search checks when the topic has clear --from,
--subject, --mailbox, --label, or --text filters.surface session start --account uni
surface session list
surface session stop sess_01...
Read-only commands may be run in parallel. Live probes passed for:
For Outlook, keep concurrency modest because each cold command or warm session uses browser
resources. If planning multiple concurrent Outlook operations, prefer one warm session per
parallel worker. Reusing the same --session concurrently works in the tested case but can be
slower due to contention.
surface mail thread get thr_01...
surface mail thread get thr_01... --refresh
surface mail thread get thr_01... --refresh --session sess_01...
surface mail read msg_01...
surface mail read msg_01... --refresh
surface mail read msg_01... --refresh --session sess_01...
surface mail read msg_01... --mark-read
surface attachment list msg_01...
surface attachment download msg_01... att_01...
surface mail send --account personal_2 --to recipient@example.com --subject "Hello" --body "Test"
surface mail send --account personal_2 --to recipient@example.com --subject "Hello" --body "Test" --attach ./briefing.txt
surface mail send --account personal_2 --to recipient@example.com --subject "Hello" --body "Test" --draft
surface mail reply msg_01... --body "Thanks"
surface mail reply msg_01... --body "Thanks" --draft
surface mail reply-all msg_01... --body "Thanks everyone"
surface mail forward msg_01... --to recipient@example.com --body "FYI"
surface mail archive msg_01... # IMAP requires an Archive/All Mail mailbox
surface mail mark-read msg_01...
surface mail mark-unread msg_01...
surface mail rsvp msg_01... --response accept # Gmail/Outlook only; IMAP returns unsupported
surface account list if the target account is unclear.surface auth status before assuming a provider is ready.surface account identity show <account> if summary.needs_action looks wrong; add
--name-alias or --email-alias with surface account identity set when the mailbox address
alone is not enough to identify the user in message bodies.fetch-unread or search and inspect the returned thread/message refs.surface mail sent --account <account> --recipient <email> --limit 3 and use the returned sent
messages as tone/context. When replying in an existing thread, prefer
surface mail sent --account <account> --thread <thread_ref> --limit 3; use recipient matching
as fallback if the thread has no sent examples.session_id.surface mail thread get <thread_ref> --refresh and compare the newest
message state against the stored prior observation before notifying.search to set the baseline, then use fetch-unread for new
inbox arrivals plus targeted search when the watch has narrow filters.surface mail read <message_ref>.message_ref and thread_ref for follow-up commands.search accepts structured filters for sender, subject, mailbox, and labels in addition to raw --text.session start is the explicit opt-in path for warm Outlook read sessions. In v1, --session is supported on search, fetch-unread, thread get --refresh, and read.thread get --refresh is the thread-level live refresh path for automations that watch a specific conversation.read is cache-first by default. Use --refresh when you need live provider state.read does not download attachments. Use surface attachment download.fetch-unread and search do not mutate mailbox state.mark-read or read --mark-read
is acceptable only after reporting and only when local write safety allows it--draft is the safe compose path when you do not need to send immediately.mail send accepts repeatable --attach <path> flags; result JSON exposes attachment
metadata only, not local paths or file bytes--attach,
reply/reply-all/forward,
archive, mark-read, mark-unread, RSVP, and --draft.--attach, reply/reply-all/forward, drafts, mark-read, and mark-unread. Archive works
only when the account exposes an Archive or All Mail style mailbox. RSVP is not supported for
generic IMAP.in_reply_to_message_ref; use sent --recipient or sent --thread for sent-message lookup.surface auth login <account>.~/.surface-cli/config.toml and any SURFACE_* env vars.--draft unless the task explicitly requires a live send.surface account list
surface auth status
surface auth status gmx
surface session start --account uni
surface mail fetch-unread --account uni --limit 10
surface mail search --account gmx --mailbox inbox --limit 5
surface mail fetch-unread --account uni --session sess_01... --limit 10
surface mail search --account personal_2 --from alerts@example.com --subject 'discount' --mailbox inbox --label unread --limit 5
surface mail thread get thr_01... --refresh --session sess_01...
surface mail read msg_01... --refresh --session sess_01...
surface mail read msg_01... --mark-read
surface attachment list msg_01...
surface attachment download msg_01... att_01...
surface mail reply msg_01... --body 'Thanks' --draft
surface mail archive msg_01...