name: google-calendar
description: "Google Calendar via gws: list events, create, accept, find free time."
license: MIT
compatibility: "macOS and Linux. Requires the gws CLI authenticated against a Google account with Calendar scopes."
metadata:
gini:
version: 1.1.2
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 Calendar
Use gws calendar to list events, create and update events, look up free/busy windows, and manage calendar ACLs. The CLI wraps the Calendar v3 API and produces structured JSON.
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 calendar ...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. - OAuth scopes the user picked at login must cover the verbs the agent will use:
- Read-only agenda / free-busy:
calendar.readonly - Create, update, delete events:
calendar.events(or fullcalendar) - Manage calendar lists, ACLs, and secondary calendars: full
calendar
- Read-only agenda / free-busy:
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 calendar events list
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 look at, summarize, or modify their calendar.
- Scheduling a one-off meeting or a recurring event.
- Finding a free slot across the user's own calendars or comparing free/busy with another attendee.
- Adding a Google Meet link to an invite.
- Pulling today's, tomorrow's, or this-week's agenda.
When NOT to Use
- Personal to-dos the user wants to see on their iPhone — use
apple-remindersfor time-bound to-dos, not Calendar events. - One-off agent alerts ("ping me in an hour") — use the cronjob tool, not a self-invite.
- Cross-device personal note-taking — use
apple-notesorobsidian. - Meeting content (agenda doc, shared notes) — use
google-docsfor the doc and attach it to the event with a Drive link. - Project task tracking — use the project's issue tracker; Calendar is for time-bound events, not durable to-dos.
Quick Reference
The Calendar surface is the auto-generated v3 API (gws calendar events list, gws calendar events insert, gws calendar freebusy query, …) plus two curated helpers: +agenda for digests and +insert for creating events without hand-rolling the JSON body.
Agenda
gws calendar +agenda # next ~7 days, all calendars
gws calendar +agenda --today
gws calendar +agenda --tomorrow
gws calendar +agenda --week --format table
gws calendar +agenda --days 14
gws calendar +agenda --today --calendar 'Work'
gws calendar +agenda --today --timezone America/New_York
+agenda is read-only and uses the user's Google account timezone by default. Override with --timezone <IANA> when the user is travelling.
Create an event
gws calendar +insert \
--summary 'Standup' \
--start '2026-06-17T09:00:00-07:00' \
--end '2026-06-17T09:30:00-07:00'
# With attendees + Meet link
gws calendar +insert \
--summary 'Design review' \
--start '2026-06-17T14:00:00-07:00' \
--end '2026-06-17T15:00:00-07:00' \
--attendee alice@example.com \
--attendee bob@example.com \
--meet \
--location 'Room 4' \
--description 'Walk through the v2 mocks.'
Times are RFC 3339 / ISO 8601. Always include the offset (-07:00, +02:00, Z) — naked local times round-trip incorrectly across DST.
For natural-language event creation, the API surface also offers events.quickAdd:
gws calendar events quickAdd \
--params '{"calendarId":"primary","text":"Coffee with Pat tomorrow 3pm-3:30pm"}'
List, get, update, delete events
gws calendar events list \
--params '{"calendarId":"primary","timeMin":"2026-06-17T00:00:00Z","timeMax":"2026-06-18T00:00:00Z","singleEvents":true,"orderBy":"startTime"}'
gws calendar events get \
--params '{"calendarId":"primary","eventId":"<EVENT_ID>"}'
gws calendar events patch \
--params '{"calendarId":"primary","eventId":"<EVENT_ID>"}' \
--json '{"location":"Room 5"}'
gws calendar events delete \
--params '{"calendarId":"primary","eventId":"<EVENT_ID>"}'
Respond to an invite (RSVP)
Google does not expose a dedicated RSVP method — you respond by patching the event and updating the current user's entry in the attendees array. That means the agent needs three things first: the eventId, the user's own email address (so it can find its attendee entry), and the full existing attendees array (so the patch can preserve everyone else).
# 1. Look up the signed-in user's email — `gws auth status` returns
# structured JSON; the email lives at the top-level `.user` key.
gws auth status | jq -r '.user'
# 2. Fetch the event to read the existing attendees array.
gws calendar events get \
--params '{"calendarId":"primary","eventId":"<EVENT_ID>"}'
# 3. Patch the event with the full attendees array, flipping just the
# current user's responseStatus to accepted | declined | tentative.
gws calendar events patch \
--params '{"calendarId":"primary","eventId":"<EVENT_ID>","sendUpdates":"all"}' \
--json '{
"attendees": [
{"email":"alice@example.com"},
{"email":"bob@example.com"},
{"email":"me@example.com","responseStatus":"accepted"}
]
}'
Set sendUpdates to "all" (notify everyone), "externalOnly", or "none" based on what the user wants. RSVP changes always notify the organizer when sendUpdates is "all".
Find free time
gws calendar freebusy query --json '{
"timeMin": "2026-06-17T09:00:00-07:00",
"timeMax": "2026-06-17T18:00:00-07:00",
"items": [{"id":"primary"},{"id":"alice@example.com"}]
}'
The response lists busy windows per calendar; free time is the complement within [timeMin, timeMax].
Calendar list and ACLs
gws calendar calendarList list
gws calendar calendars insert --json '{"summary":"Side project"}'
gws calendar acl list --params '{"calendarId":"primary"}'
gws calendar acl insert --params '{"calendarId":"primary"}' \
--json '{"role":"reader","scope":{"type":"user","value":"alice@example.com"}}'
Rules
- Calendar events are time-bound things on a schedule — meetings, appointments, focus blocks. For personal to-dos that should appear on the user's iPhone, prefer
apple-reminders. For agent-internal alerts ("remind me in 2 hours"), use the cronjob tool. Confirm intent before deciding which. - Don't add a redundant text confirmation before
events.insert,events.patch,events.delete, or+insert. The runtime'sterminal_execapproval gate is the user's safety net — ifgws *isn't auto-approved they'll see the gate per-command; if it is, they've opted in to "just do it." When the user's command is clear ("remove dinner," "decline the 2pm"), execute. Do ask one clarifying question when the command is ambiguous — multiple events match the description, the user didn't specify which calendar, orsendUpdateswould email attendees the user might not want notified. - When the user says "accept" / "decline" / "tentative" on a meeting, confirm which event and which response status before patching. Resolve the user's own email via
gws auth status | jq -r '.user'(or read it from the event'sattendeesarray) so the patch updates the right attendee, and re-send the fullattendeesarray so other invitees aren't dropped. RSVP changes email-notify the organizer whensendUpdatesis"all". - Use RFC 3339 times with an explicit offset on
+insertandevents.insert. Naked local times silently drift across DST boundaries. - Prefer
+insertover hand-rollingevents.insert --jsonwhen the user just wants a normal event. The helper sets sane defaults (visibility, reminders) without forcing the agent to learn the full body schema. - Adding
--meetto+insertcreates a Google Meet space attached to the event — this is the simplest way to give the user a join link. Do not separately callgoogle-meetto create a standalone space when an event will exist anyway. - When listing events, set
singleEvents:trueandorderBy:"startTime"so recurring events expand into individual instances and arrive in chronological order. Without those, the response is grouped by recurrence master and confusing to summarize. - Free/busy queries are read-only and cheap — prefer them over scraping multiple
events.listcalls when the user only needs availability, not event content. - Respect the calendar's timezone, not just the host's.
+agendadefaults to the Google account timezone; only override with--timezonewhen the user is travelling.
For flags not shown here, run gws calendar --help or gws calendar <verb> --help (e.g. gws calendar +insert --help).