loading…
Search for a command to run...
loading…
Provides access to Google Meet API for retrieving meeting data, transcripts, and recordings. Enables polling for new transcripts, background watcher notificatio
Provides access to Google Meet API for retrieving meeting data, transcripts, and recordings. Enables polling for new transcripts, background watcher notifications, and integration with Google Calendar events.
Google Meet API client exposed as both an MCP server and a CLI. Built with silkweave.
Authentication is exclusively through a Google Workspace service account with domain-wide delegation (DWD). There is no OAuth flow, no per-user token registry, no interactive login. One service-account JSON on disk and a list of Workspace user emails in a config file is the entire setup.
eventPullTranscripts — on-demand polling with a persisted cursor (idempotent)transcriptWatch*) — streams from Pub/Sub and writes Markdown files, with optional per-save shell commandtranscriptBackfill — one-shot catch-up across all users for the past N daysIn your Google Workspace Admin Console:
client_id).https://www.googleapis.com/auth/userinfo.emailhttps://www.googleapis.com/auth/calendarhttps://www.googleapis.com/auth/drive.readonlyhttps://www.googleapis.com/auth/meetings.space.readonlyhttps://www.googleapis.com/auth/meetings.space.createdmkdir -p ~/.silkweave-meet
cp /path/to/sa-key.json ~/.silkweave-meet/service-account.json
chmod 600 ~/.silkweave-meet/service-account.json
That's all the auth configuration. No env vars, no OAuth flow.
{
"mcpServers": {
"meet": {
"command": "npx",
"args": ["-y", "-p", "@silkweave/meet", "meet-mcp"]
}
}
}
The package ships two bins: meet-mcp (stdio MCP server) and meet-cli (same action set as direct commands).
npx -p @silkweave/meet meet-cli calendar-event-list [email protected]
Everything non-secret lives in ~/.silkweave-meet/config.json:
{
"users": ["[email protected]", "[email protected]"],
"watcher": {
"pubsubSubscriptions": ["projects/my-project/subscriptions/meet-transcripts-sub"],
"transcriptDir": "/Users/alice/meet-transcripts",
"onTranscriptCommand": "osascript -e 'display notification ...'",
"autoStart": true
},
"cursors": {
"[email protected]": "2026-04-22T14:30:00Z"
},
"openai": {
"apiKey": "sk-...",
"embeddingModel": "text-embedding-3-small"
}
}
users — Workspace user emails the tool is allowed to impersonate. Populated by setupSubscribeAll --users=a,b,c or edited manually.watcher — persisted watcher config, written by transcriptWatchStart.cursors — per-user polling cursors for eventPullTranscripts.openai — optional. Enables vector/hybrid search over the persisted transcript archive. OPENAI_API_KEY and OPENAI_EMBEDDING_MODEL env vars are used as fallbacks.<transcriptDir>/<organizerEmail>/YYYY-MM-DD_{meetCodeOrConferenceId}_{transcriptId}.md. Because each saved file is fetched via the organizer's own subscription, the top-level folder makes it easy to browse "meetings I ran".~/.silkweave-meet/transcripts.msp holds the searchable metadata (subject, description, attendees, date range, file path, embedding). It is the source of truth for transcriptList / transcriptGet / transcriptSearch.Every action that reads Google data takes a required userEmail — the Workspace user the service account impersonates for that call. Permissions match exactly what that user can see.
The MCP surface is intentionally narrow: only the read-only transcript tools plus transcriptBackfill and mcpStatus are exposed over MCP. Everything that manages configuration, subscriptions, or the background watcher is CLI-only (via meet-cli).
MCP-exposed tools:
meetTranscriptList / meetTranscriptGet — live lookup against the Google Meet API.transcriptList / transcriptGet / transcriptSearch — read from the persisted local archive (no Google round-trip; works offline for previously-ingested transcripts).transcriptBackfill — catch-up ingest across all configured users (default: last 30 days). Dedupes by transcriptId.mcpStatus — health & status.Calendar* (CLI-only)| Tool | Purpose |
|---|---|
calendarEventList |
List Calendar events on the user's primary calendar (optionally filtered to Meet-enabled ones). |
calendarEventGet |
Get one event with Meet join info. |
Meet* (live Google API)meetTranscriptList and meetTranscriptGet are available on both MCP and CLI. The rest are CLI-only.
| Tool | Surface | Purpose |
|---|---|---|
meetTranscriptList |
MCP + CLI | List transcripts for a conference. |
meetTranscriptGet |
MCP + CLI | Fetch a full transcript; returns Markdown by default, format=json for raw entries. |
meetConferenceList |
CLI | List past conferenceRecords the user participated in (optional EBNF filter). |
meetConferenceGet |
CLI | Fetch a single conference record. |
meetParticipantList |
CLI | List participants of a conference. |
meetRecordingList |
CLI | List recording artifacts (Drive links). |
meetSpaceGet |
CLI | Resolve a space by spaces/{id} or meeting code. |
Transcript* (local Orama DB)Reads from ~/.silkweave-meet/transcripts.msp; populated by the background watcher (live) and transcriptBackfill (historical). Every record is enriched with its Calendar event (subject, description, attendees) — matching is deterministic via conferenceData.conferenceId → Meet space meetingCode, never time-guessing.
| Tool | Surface | Purpose |
|---|---|---|
transcriptList |
MCP + CLI | List persisted transcripts, newest first, filterable by organizer, attendee, and date range. |
transcriptGet |
MCP + CLI | Fetch a single persisted transcript by id (or full resource name); returns metadata plus the rendered markdown body read from disk. |
transcriptSearch |
MCP + CLI | Keyword search over subject / description / full transcript body. With mode=vector or mode=hybrid and an OpenAI key configured, runs semantic or hybrid search. |
transcriptBackfill |
MCP + CLI | Iterate every configured user, list conferences since startTime (default: 30 days ago), ingest any transcripts not already in the database. Dedupes by transcriptId. |
Event* (CLI-only)| Tool | Purpose |
|---|---|
eventPullTranscripts |
Polling. Returns transcripts generated since the user's stored cursor, advances it. No Pub/Sub needed. |
eventSubscriptionCreate |
Create a Workspace Events subscription for a specific Meet space (requires [email protected] to have Pub/Sub Publisher on the topic). |
eventSubscriptionCreateForUser |
Create a user-level Workspace Events subscription for the impersonated user. Covers meetings they own or attend. |
eventSubscriptionList |
List existing subscriptions owned by the impersonated user. |
eventSubscriptionDelete |
Delete a subscription. |
TranscriptWatch* (CLI-only)Singleton consumer that streams events from a Pub/Sub subscription (Pub/Sub auth from the same service-account key), and for each message impersonates (via DWD) the user whose Workspace Events subscription produced it — identified by the message's ce-source attribute — to fetch the transcript via the Meet API. One watcher covers every user listed in the config.
| Tool | Purpose |
|---|---|
transcriptWatchStart |
Start the watcher (and persist config). Pass pubsubSubscriptions, transcriptDir, onTranscriptCommand, autoStart. |
transcriptWatchStop |
Stop the watcher. Pass disableAutoStart=true to also disable boot-time auto-start. |
transcriptWatchStatus |
Current status: running flag, per-subscription counters, known subscription owners, recent saved files. |
When autoStart: true is persisted, the MCP server resumes the watcher on every boot. Each saved file is written to <transcriptDir>/<organizerEmail>/YYYY-MM-DD_{meetCodeOrConferenceId}_{transcriptId}.md and simultaneously inserted into the local search index.
Setup* (CLI-only)| Tool | Purpose |
|---|---|
setupStatus |
Report, for each user in the config, whether DWD impersonation succeeds and which Workspace Events subscriptions they own. With --pubsub-topic, flag which users are subscribed to it. |
setupSubscribeAll |
Create a user-level subscription for every user in the config. Optional --users=a,b,c also appends those emails to the config. Idempotent — re-running skips users already subscribed. |
These are registered only in src/cli.ts (not the MCP action list), because they orchestrate the whole config rather than a single user.
Mcp*| Tool | Surface | Purpose |
|---|---|---|
mcpStatus |
MCP + CLI | Lightweight status: version/uptime/pid, service-account key presence, watcher running state, configured users with per-user subscription coverage, and the 10 most recent saved transcripts. |
# 1. Create the shared Pub/Sub topic and let Meet publish to it.
gcloud pubsub topics create meet-transcripts
gcloud pubsub topics add-iam-policy-binding meet-transcripts \
--member=serviceAccount:[email protected] \
--role=roles/pubsub.publisher
# 2. Register the users and create one user-level subscription per person
# (idempotent). The --users flag appends to ~/.silkweave-meet/config.json.
npx -p @silkweave/meet meet-cli setup-subscribe-all \
--pubsub-topic=projects/<project>/topics/meet-transcripts \
[email protected],[email protected],[email protected],[email protected]
# 3. Confirm every user is subscribed.
npx -p @silkweave/meet meet-cli setup-status \
--pubsub-topic=projects/<project>/topics/meet-transcripts
# 4. Create the pull subscription the watcher will stream from, and grant the
# service account Subscriber on it.
gcloud pubsub subscriptions create meet-transcripts-sub --topic=meet-transcripts
SA_EMAIL=$(jq -r .client_email ~/.silkweave-meet/service-account.json)
gcloud pubsub subscriptions add-iam-policy-binding meet-transcripts-sub \
--member="serviceAccount:${SA_EMAIL}" --role=roles/pubsub.subscriber
# 5. Start the watcher. Routes each incoming event to the right user's
# impersonation context based on the message's ce-source.
npx -p @silkweave/meet meet-cli transcript-watch-start \
--pubsub-subscriptions=projects/<project>/subscriptions/meet-transcripts-sub \
--transcript-dir=~/meet-transcripts \
--auto-start=true
User-level subscriptions expire (Google's max TTL applies). Re-run setup-subscribe-all on a schedule (cron / launchd) to refresh any that have expired; existing valid ones are skipped.
User-level subscriptions capture transcript.v2.fileGenerated events for meetings the user owns or is merely invited to — that is the only non-owner event the Workspace Events API delivers, and it's exactly the one we want. Expect duplicates when multiple team members are in the same meeting; the watcher dedupes by transcriptId in memory per run.
The onTranscriptCommand runs via spawn(..., { shell: true }) with these environment variables exposed:
| Var | Meaning |
|---|---|
$TRANSCRIPT_PATH |
Absolute path to the saved Markdown file |
$TRANSCRIPT_RAW |
Full rendered Markdown as a string (subject to shell env size limits) |
$TRANSCRIPT_NAME |
conferenceRecords/{c}/transcripts/{t} resource name |
$CONFERENCE_RECORD |
conferenceRecords/{c} resource name |
$MEET_CODE |
Meeting code (e.g. abc-mnop-xyz), if resolvable |
$START_TIME / $END_TIME |
Transcript start/end (RFC3339) |
$ENTRY_COUNT |
Number of transcript entries |
$DATE |
YYYY-MM-DD prefix used in the filename |
$SUBJECT |
Calendar event subject if matched, else empty |
$CALENDAR_EVENT_ID |
Calendar event id if matched, else empty |
For very long transcripts, prefer reading from $TRANSCRIPT_PATH — environment size is bounded (~256KB on macOS).
pnpm add @silkweave/meet
import { MeetClient } from '@silkweave/meet'
import { google } from 'googleapis'
const conferences = await MeetClient.withAuth('[email protected]', async (auth) => {
const { data } = await google.meet({ version: 'v2', auth }).conferenceRecords.list()
return data.conferenceRecords ?? []
})
MeetClient.withAuth creates a DWD-impersonating JWT for that user using the key at ~/.silkweave-meet/service-account.json.
pnpm install
pnpm tsx src/mcp.ts # MCP server in dev
pnpm tsx src/cli.ts # CLI in dev
pnpm build # build to build/
pnpm lint # eslint
pnpm typecheck # tsc --noEmit
pnpm clean # rm -rf build/
When iterating through the MCP in Claude Code, the server is a child process — after code changes, restart the MCP connection (or kill the process) so changes are picked up. Claude Code caches the tool list at connection time, so any change to the MCP action set also requires a full reconnect.
MIT
Добавь это в claude_desktop_config.json и перезапусти Claude Desktop.
{
"mcpServers": {
"meet-mcp-server": {
"command": "npx",
"args": []
}
}
}