name: google-gmail
description: "Gmail via gws: send, read, search, label, draft, reply, forward."
license: MIT
compatibility: "macOS and Linux. Requires the gws CLI authenticated against a Google account with Gmail scopes."
metadata:
gini:
version: 1.2.0
author: Gini
platforms: [macos, linux]
prerequisites:
commands: [gws]
env:
- GOOGLE_WORKSPACE_CLI_CLIENT_ID
- GOOGLE_WORKSPACE_CLI_CLIENT_SECRET
requires:
credentials: [google-workspace-oauth]
Google Gmail
Use gws gmail to read, search, send, reply, forward, draft, label, and triage Gmail directly from the terminal. The CLI wraps the Gmail v1 API and produces structured JSON, so it composes cleanly with jq and other shell tooling.
Prerequisites
gwsinstalled and authenticated. Ifgwsis not on PATH ORgws auth statusreports no authenticated user, do NOT silently call setup. Instead, in a single short reply to the user:- State plainly what's missing — e.g. "Google Workspace access isn't set up on this machine yet" or "your Google sign-in has expired."
- Ask one sentence: "Want me to walk you through setting it up?" Wait for the user's answer.
- If they say yes, call
read_skillwith namegoogle-workspace-setupand run that skill's onboarding flow turn-by-turn. If they say no or ask to defer, acknowledge briefly and stop — do not retry the original request.
- Apply the same flow when any
gws gmail ...call fails mid-task withcommand not found/ ENOENT, HTTP 401, "no credentials", or "scope required". Don't report the failure as a dead end — surface the missing prerequisite and ask if the user wants to set it up before moving on. - The OAuth scopes the user picked at login must cover the verbs the agent will use:
- Read-only triage:
gmail.readonly - Send a new message:
gmail.send - Reply, reply-all, forward:
gmail.modify— upstream helpers fetch the original message to threadIn-Reply-To/Referencesheaders, whichgmail.sendalone cannot do - Drafts and labels:
gmail.modify(orhttps://mail.google.com/for full access including permanent delete) - Watch for new mail (
+watch):gmail.modifyANDhttps://www.googleapis.com/auth/pubsub— Cloud Pub/Sub is a separate Google API and its scope must be granted alongside the Gmail scope
- Read-only triage:
Selecting a Google account
The connected Google accounts (each with its tag, email, and config dir) are listed in your system context under "Connected Google accounts". To target a specific account, prefix the command with its config dir:
GOOGLE_WORKSPACE_CLI_CONFIG_DIR="<configDir>" gws gmail +triage
Selection rule: one account connected → just use it. Two or more:
- The user named or clearly implied one account (a tag, an email, or unambiguous context) → use only that account.
- A read/lookup/search the user didn't tie to an account (e.g. listing events, searching mail, finding a doc) → run it against every connected account (one
gwscall per config dir) and aggregate, labeling each result by its tag and email. Don't pick just one, and don't ask — the user wants the whole picture across accounts. - A write (send, create, edit, delete) with no account named → ASK which account first; never guess.
If no accounts are connected yet, fall back to the setup flow in Prerequisites (read_skill with google-workspace-setup).
When to Use
- The user asks Gini to send, draft, read, search, label, reply to, or forward email.
- Summarizing or triaging the inbox (latest unread, by-sender, by-label digests).
- Saving an attachment from a thread, or pulling a message body into another workflow.
- Watching for new messages and streaming them as NDJSON (
gws gmail +watch).
When NOT to Use
- Agent-internal scratch notes or transient state — use the
memorytool, not email-to-self. - Personal to-dos that should appear on the user's iPhone — use
apple-reminders. - Cross-device personal note-taking — use
apple-notesorobsidian. - Calendar invites and meeting scheduling — use
google-calendar(a Gmail invite is still a Calendar event). - Bulk outbound mail (newsletters, marketing) — personal Gmail has aggressive sending limits and Google will throttle or suspend the account. Tell the user to use a transactional provider.
Quick Reference
The Gmail surface in gws is split into auto-generated API methods (gws gmail users messages list, gws gmail users labels create, …) plus a small set of curated helpers (+send, +reply, +read, +triage, …) that handle MIME encoding, threading, and base64 for you. Prefer the helpers for everyday tasks. The raw API is rooted at the users resource — every --params JSON must include "userId": "me" (or another delegated address).
Send
gws gmail +send --to alice@example.com --subject 'Hello' --body 'Hi Alice!'
# CC, BCC, alias send-from
gws gmail +send --to alice@example.com --cc bob@example.com \
--subject 'Status' --body 'See below.' --from alias@example.com
# Attachments (repeatable, 25 MB total)
gws gmail +send --to alice@example.com --subject 'Report' \
--body 'See attached.' -a report.pdf -a notes.txt
# HTML body
gws gmail +send --to alice@example.com --subject 'Update' \
--body '<b>Bold</b> text' --html
# Save as draft instead of sending
gws gmail +send --to alice@example.com --subject 'Draft' --body 'WIP' --draft
Show a saved draft to the user
After you save a draft (--draft), show it to the user inline so they can read it right in the chat — never tell them to open Gmail and search for it. Lead with one short sentence, then render the draft as a fenced email-draft block: optional To: / Cc: / Subject: header lines, a blank line, then the exact body you saved.
I drafted this reply for you:
```email-draft
To: support@plaud.ai
Subject: Follow-up on your Request #527545
Hi there,
I still haven't received the package, and the delivery photo shows it was left
inside a publicly accessible gate. Could you reopen the case and coordinate a
replacement or refund?
Thanks
```
Use the same recipient, subject, and body you passed to gws gmail +send … --draft so the card matches the saved draft. The app renders the email-draft block as a draft card; any non-rendering client degrades it to a readable code block.
Read
gws gmail +read --id <MESSAGE_ID> # plain-text body
gws gmail +read --id <MESSAGE_ID> --headers # include From/To/Subject/Date
gws gmail +read --id <MESSAGE_ID> --format json | jq '.body'
gws gmail +read --id <MESSAGE_ID> --html # HTML body instead of text
Search and list
gws gmail users messages list accepts standard Gmail search operators via the q param (from:, to:, subject:, label:, is:unread, has:attachment, newer_than:7d, etc.).
gws gmail users messages list --params '{"userId":"me","q":"from:alice@example.com is:unread","maxResults":20}'
gws gmail users messages list --params '{"userId":"me","q":"label:invoices newer_than:30d"}' --page-all
gws gmail +triage # curated unread inbox digest
Reply and forward
gws gmail +reply --message-id <MESSAGE_ID> --body 'Thanks — will follow up.'
gws gmail +reply-all --message-id <MESSAGE_ID> --body 'Looping in the team.'
gws gmail +forward --message-id <MESSAGE_ID> --to charlie@example.com \
--body 'FYI from the thread below.'
The helpers preserve In-Reply-To and References headers so the reply lands inside the original thread.
Labels and drafts
gws gmail users labels list --params '{"userId":"me"}'
gws gmail users labels create --params '{"userId":"me"}' \
--json '{"name":"Receipts","labelListVisibility":"labelShow"}'
gws gmail users messages modify --params '{"userId":"me","id":"<MESSAGE_ID>"}' \
--json '{"addLabelIds":["Label_123"],"removeLabelIds":["INBOX"]}'
gws gmail users drafts list --params '{"userId":"me"}'
gws gmail users drafts get --params '{"userId":"me","id":"<DRAFT_ID>"}'
Watch for new mail
gws gmail +watch # streams new messages as NDJSON (one JSON object per line)
Rules
- Don't add a redundant text confirmation before
gws gmail +send,+reply,+reply-all,+forward,messages.send, ordrafts.send. The runtime'sterminal_execapproval gate is the user's safety net. When the user's command is clear ("email alice@acme.com that I'll be 10 min late"), execute. Do ask one clarifying question when the command is ambiguous — multiple "alice" matches in the address book, more than one thread could be the reply target, or the user named a verb but no recipient. Keep generated bodies short and aligned with the user's exact intent — don't elaborate beyond what they asked for. - Prefer the curated helpers (
+send,+reply,+read,+triage) over the rawgws gmail <resource> <method>surface — they handle MIME, base64, threading, and HTML-to-text conversion automatically. - When replying, use
+reply/+reply-allso the thread stays intact. Building a new message with+sendand pasting in the prior subject does not thread correctly. - Treat the Gmail scopes as four separate trust boundaries:
gmail.readonlycovers+read/+triageand anymessages.list/getcall;gmail.sendcovers a brand-new+sendonly;gmail.modifyis required for+reply,+reply-all,+forward, labels, and drafts because those helpers must fetch the original message or mutate its state;+watchrequiresgmail.modifyANDhttps://www.googleapis.com/auth/pubsubbecause the upstream helper requests both tokens (Pub/Sub is a separate Google API). If the user only granted a narrower scope at setup, never silently call a verb that needs a wider one — direct them back togoogle-workspace-setupto widen scopes. - Do not bulk-send from a personal
@gmail.comaccount. Google throttles or suspends accounts that look like bulk senders. Use a transactional provider for newsletters or anything addressed to more than a handful of recipients. - Attachment cap is 25 MB total. For larger files, upload via
google-driveand send the share link instead. - Never paste raw message bodies that contain secrets (API keys, passwords, MFA codes) back into the chat transcript. Summarize, redact, or write to a file the user controls.
- When you save a draft, surface it to the user with an
email-draftfenced block (see "Show a saved draft to the user") instead of pointing them at Gmail. The user should be able to read the draft without leaving the app.
For flags not shown here, run gws gmail --help or gws gmail <verb> --help (e.g. gws gmail +send --help).