From monday CRM
Turn meeting transcripts into deal updates: key points, commitments, and next steps posted to matching CRM deals. Auto-creates contacts for new attendees.
How this skill is triggered — by the user, by Claude, or both
Slash command
/monday-crm:meeting-to-deal [optional: time window — 'today', 'yesterday', 'last 7 days', or 'since 2026-05-20'][optional: time window — 'today', 'yesterday', 'last 7 days', or 'since 2026-05-20']This skill is limited to the following tools:
The summary Claude sees in its skill listing — used to decide when to auto-load this skill
Flow: **Trigger → Pull meetings → Match → Synthesize → Publish (α) → Auto-contact (β, opt-in) → Proactive next-step nudges (opt-in)**.
Flow: Trigger → Pull meetings → Match → Synthesize → Publish (α) → Auto-contact (β, opt-in) → Proactive next-step nudges (opt-in).
Meeting recap update per matched deal, summarizing key points + commitments + next steps. Body carries the Generated by Claude · <ISO timestamp> footer + <!-- claude-skill-id: meeting-to-deal --><!-- meeting-id: <id> --> comments. A Meeting sync — <Mon DD> doc lists what was matched, what wasn't, and unresolved attendees.Source = Claude on a status column.create_notification to deal owner for commitments due this week. Stage column edits and last-touch updates are also OK in Proactive (or in Default with batched confirm) when a clear transition phrase is present in the transcript ("we agreed to move to procurement review", "they signed today"). Surfaced as a single batched plan: " stage transitions detected. Apply all / review each / skip stage edits?". Amount-column writes still forbidden.get_user_context — user identity + email domain for filtering meetings and extracting internal_domain.get_notetaker_meetings — recent meetings + transcripts.search / list_workspaces / get_board_info — locate Deals + Contacts boards.get_board_items_page — fetch active deals to match against.get_column_type_info — resolve email/company columns by type.create_update — write the recap onto the matched deal.create_item — β auto-contact on the Contacts board.change_item_column_values — populate the new contact's email + company columns, write Source = Claude on contacts, and apply confirmed stage / last-touch edits on matched deals. Never for amount columns.create_notification — proactive next-step nudges.create_doc — sync summary doc.all_monday_api — escape hatch for connect-board column writes (linking contact ↔ deal). Note: connect-board writes via all_monday_api may fail silently — see Error handling reference.data-cleanup to clean contact data.Goal: Confirm the monday connector works and detect whether NoteTaker is available. Also extract internal_domain for use in Steps 4–5.
mcp__monday__get_user_context. On error → standard install prompt, stop.internal_domain from the user's email (e.g. "[email protected]" → "monday.com"). Store for Steps 4–5. If email is absent from the response, internal_domain = "monday.com" (safe default).mcp__monday__get_notetaker_meetings({ limit: 1 }).notetaker_available: true. Continue to Step 1.notetaker_available: false. Do not stop. The skill will collect meeting content from the user directly in Step 3.PAUSE: do not write any
create_*call before Step 6.
If undeclared, Default.
Hard safety rail regardless of mode: no deletes, no amount-column writes, no cross-workspace moves. Stage transitions are bundled into a batched plan — never written one-off, never silently.
AskUserQuestion (large windows produce noisy recaps).notetaker_available: true)mcp__monday__get_notetaker_meetings({ since: <ISO>, until: <ISO>, limit: 50 }). Cache transcripts + attendees + meeting metadata.
Edge cases:
<window>. You can paste meeting notes or describe what happened instead."notetaker_available: false, OR NoteTaker returned 0 meetings)Ask the user once via AskUserQuestion:
"No NoteTaker recordings available. How would you like to provide meeting notes?"
- (a) Paste the transcript or notes here — I'll read them from your next message
- (b) Point me to a file — provide the path and I'll read it
- (c) Describe what happened — tell me in free text (attendees, key points, commitments, next steps)
- (d) Skip — nothing to log right now
If (d) → stop with: "No meetings logged. Run this skill again when you have notes to add."
For (a) / (b) / (c): treat the user-provided content as a single meeting entry. Structure it as:
title: first line of the notes, or ask if blankdate: today (or extract from notes if mentioned)attendees: extract from notes, or ask oncetranscript_text: the full provided contentCache as one entry in meetings[] with source: "manual" and continue to Step 4. The matching and recap flow works identically — the transcript is just from the user's input instead of NoteTaker.
Resolve the boards directly — this skill is self-contained:
get_user_context → scan favorites + relevantBoards for names matching deals|opportunities|pipeline|sales. One candidate → use it; multiple → AskUserQuestion; zero → list_workspaces → search("deal", BOARD). Still zero → degrade per step 4 below.contacts|people|leads (only if β auto-contact may run). If Contacts board is missing and mode = Proactive and β was approved, surface a chat-level message before writing the sync doc: "Auto-contact was approved but I couldn't find a Contacts board. Contacts not created. Run /monday-crm:workspace-builder to set one up." Then continue with α recap only.get_board_info on each. Resolve columns by type:
email (if present), text company-name, people owner, status stage, numbers value, date last-touch.email, text (or company-link board_relation), phone, people owner./monday-crm:workspace-builder.Goal: For each meeting, find the single best matching deal — or flag it as unmatched.
internal_domain was extracted in Step 0. Use it throughout this step in place of hard-coded @monday.com checks.
For each meeting:
@<internal_domain> addresses), company names from email domains, meeting title keywords.limit: 500.@<internal_domain> (i.e. purely internal meeting). This signal is intentionally narrow: a deal owner being on an external sales call is expected and not a reliable match signal; it only carries weight for internal deal-review meetings where ownership is the strongest available signal.Edge cases:
@<internal_domain>) and no deal-name title match: skip (likely standup), log to summary doc as "skipped: internal-only".For each matched meeting, compose a structured recap from the transcript:
? or signaled by "we need to figure out", "TBD").confidence: high|low based on (a) explicit phrase ("we agreed to ", "signed today") = high, (b) inferred ("seems like they're ready for legal review") = low. Only high transitions enter the batched plan; low ones surface as text recommendations.Deal-side metadata lookup: pull current stage + value + close date for context (single get_board_items_page filtered to the matched item ID).
For each matched meeting + deal pair:
Idempotency check: Use all_monday_api with the following query pattern to scan existing updates on the deal item:
{ items_by_id(ids: [<dealId>]) { updates(limit: 50) { body } } }
Walk the body of each returned update and check for both <!-- claude-skill-id: meeting-to-deal --> and <!-- meeting-id: <id> -->. If a match is found → use all_monday_api to update that existing update's body; do not post a duplicate.
Note:
mcp__monday__searchcannot search update bodies — always useall_monday_apiitems_by_id for this check.
create_update({ itemId: <dealId>, body: <recap markdown> }).
Update body shape:
# Meeting recap — <meeting title>
*<meeting date> · <duration>m · <attendee count> attendees*
## Key points
- ...
## Commitments
- <person> → <commitment> (due <date or "open">)
## Open questions
- ...
## Stage signal
<"applied: <old> → <new>" if the batched plan included it> OR <text recommendation if low-confidence> OR "no stage change">
---
<!-- claude-skill-id: meeting-to-deal -->
<!-- meeting-id: <noteTakerMeetingId> -->
Generated by Claude · <ISO timestamp> · run `/monday-crm:meeting-to-deal` to refresh.
Before any write, print a single plan:
"I'll write: recaps · stage edits (high-confidence only) · last-touch updates · new contacts. meetings unmatched (will list separately). low-confidence stage signals will surface as text only. Proceed? (yes / show plan in detail / no)."
If user picks "show plan in detail":
<deal name> · <old stage> → <new stage> for every planned stage edit.Never per-deal confirms beyond that.
Auto-contact cap warning: If unmatched-attendee count > 25, add to the confirm message: " new contacts detected (> 25 cap). I'll create 25 and add the rest to a review-list doc. Proceed?" (e.g. "31 new contacts detected (> 25 cap). I'll create 25 and add the rest to a review-list doc. Proceed?")
Goal: Don't make the user manually add every meeting attendee to Contacts.
Only runs if mode = Proactive (option a) OR Default-mode user opts in at Step 7 confirm (added prompt: "+ create N missing contacts on Contacts board?").
For each unique attendee email not found on the Contacts board:
create_item({ boardId: <contactsBoard>, itemName: "<First Last>", groupId: "new" }).change_item_column_values to populate email, company (from email domain), and Source = Claude on the board's source/status column. If no source/status column exists, prompt the user once: "Add a Source status column on Contacts so we can flag plugin-created records? (yes / skip)". If skipped, contacts are still created but source isn't tracked.board_relation column using all_monday_api. If this call fails (connect-board writes can fail silently), log the contact as "contact created, deal link pending — set manually" in the sync doc rather than stopping. Do not retry silently.Cap at 25 contacts per run. Above that, batch into a review-list doc rather than creating silently. (Flag this at Step 7 confirm time — see auto-contact cap warning above.)
Only runs if mode = Proactive AND session-level approval was given.
For commitments parsed in Step 6 with a parseable due-date within next 7 days:
create_notification({ userId: <deal owner>, text: "Commitment due <date> on <deal>: <commitment>" }).create_notification requires an itemId: use the sync doc's item ID (from Step 10's create_doc return value). Do not pin the notification to an arbitrary deal item when the doc covers cross-deal commitments.Hard rail: never write commitments to amount columns. Notifications only for proactive nudges (stage edits already handled in Step 7's batched plan, not here).
If >20 nudges queued, batch into one This week's commitments doc (with <!-- claude-skill-id: meeting-to-deal --> in body) and notify once with a link to the doc.
create_doc (idempotent — same-day update-in-place):
# Meeting sync — <Mon DD>
## Matched (<N>)
- <meeting title> → <deal name> (<deal url>)
## Multi-match (<N>)
- <meeting title> → <deal A> + <deal B>
## Ambiguous (<N>)
- <meeting title> — candidates: <deal A>, <deal B> — review and re-run with explicit deal ID
## Unmatched (<N>)
- <meeting title> — attendees: <emails> — no matching deal. Suggest creating one.
## Skipped (<N>)
- <meeting title> — internal-only, no CRM relevance
## Contacts created (<N>)
- <name> (<email>) → linked to <deal>
## Contacts pending manual link (<N>)
- <name> (<email>) — contact created, deal link pending — set manually
---
Generated by Claude · <ISO timestamp>
One-line chat summary: Synced <N> meetings to <M> deals. <K> unmatched, <L> contacts created. Doc: <url>.
Generated by Claude footer + <!-- claude-skill-id: meeting-to-deal --> comment in update/doc bodies; Source = Claude on contact items.internal_domain from Step 0 used throughout for internal/external attendee classification — not hard-coded to @monday.com.meeting-to-opportunity — kept for backward compatibility with existing updates/docs carrying <!-- claude-skill-id: meeting-to-opportunity -->. The skill's invocation name is /monday-crm:meeting-to-deal.# Meeting recap — Quarterly Review — Acme Corp
*Jun 14, 2026 · 45m · 4 attendees*
## Key points
- Discussed Q3 expansion scope — 3 additional seats requested
- Budget approved internally at Acme, pending procurement sign-off
- Timeline: targeting signed contract by end of June
## Commitments
- Sarah (Acme) → send updated SOW (due Jun 18)
- Kevin (us) → schedule legal review call (due Jun 20)
## Open questions
- Will Acme need SSO configured before go-live?
- Multi-year discount structure not yet agreed
## Stage signal
Applied: Discovery → Proposal (high confidence — "we agreed to move to procurement review")
---
<!-- claude-skill-id: meeting-to-opportunity -->
<!-- meeting-id: ntm_abc123def456 -->
Generated by Claude · 2026-06-15T09:00:00Z · run `/monday-crm:meeting-to-deal` to refresh.
# Meeting sync — Jun 15
## Matched (2)
- Quarterly Review — Acme Corp → Acme Corp — Q3 expansion (https://monday.com/boards/...)
- Demo follow-up — Beta Industries → Beta Industries — pilot (https://monday.com/boards/...)
## Unmatched (1)
- Internal team standup — attendees: all @monday.com — skipped (internal-only)
## Contacts created (1)
- Mike Chen ([email protected]) → linked to Acme Corp — Q3 expansion
---
Generated by Claude · 2026-06-15T09:00:00Z
Chat one-liner:
Synced 2 meetings to 2 deals. 1 skipped (internal). 1 contact created. Doc: [url].
| Failure | Behavior |
|---|---|
| Connector missing | Step 0 stops; print install link. |
| NoteTaker unavailable | Step 3b: ask user for manual input — paste, file, or free-text description. |
| 0 meetings in window (NoteTaker worked) | Step 3b fallback — prompt for manual input. |
| No Deals board | Step 4 degrades — sync doc only, suggest workspace-builder. |
| Ambiguous match (≥2 candidates within 2 points) | Skip the write; log to sync doc as "ambiguous". |
| Idempotency hit (same meeting-id in update) | Update existing content, don't duplicate. |
429 on get_notetaker_meetings or create_update | Backoff 3x; halt on third with retry message. |
| Contacts board missing (β approved, Proactive mode) | Surface chat message before sync doc: "Auto-contact was approved but I couldn't find a Contacts board. Contacts not created. Run /monday-crm:workspace-builder to set one up." Continue with α recap. |
| Contacts board missing (β not triggered) | Skip β; log "no Contacts board" once in sync doc. |
connect-boards write via all_monday_api fails | Skip link; log in sync doc as "contact created, deal link pending — set manually". |
| Transcript empty / NoteTaker still processing | Skip that meeting; surface in sync doc as "transcript pending". |
| Cross-board permission error | Skip just that deal/contact; continue with the rest. |
internal_domain extracted.all_monday_api items_by_id update-body scan).<!-- claude-skill-id: meeting-to-deal --> + <!-- meeting-id: <id> --> + Generated by Claude footer.Source = Claude; failed deal links logged in sync doc as "pending — set manually".npx claudepluginhub mondaycom/mcp --plugin monday-crmProduces a daily prioritized briefing over monday CRM data and publishes it as a monday artifact. Handles pipelines, contacts, accounts, leads, or no board.
Enhances Granola meeting notes post-transcript with AI templates, shares to Slack/Notion/CRM, extracts action items, and supports follow-ups.
Reviews CRM pipeline stages by pulling prospects, analyzing email histories and meeting transcripts, and sending concise Slack summaries with actions and hygiene recommendations.