Agent skill

cache-notes

Fetch & embed AI transcripts as Obsidian callouts. Args: <path>, all, refresh <path>. Prompts for URLs if empty.

Stars 152
Forks 10

Install this agent skill to your Project

npx add-skill https://github.com/benoror/obsidianos_work/tree/main/.agents/skills/cache-notes

SKILL.md

Cache Notes

Usage

  • /cache-notes all — Scan all meetings with Notes: links that haven't been cached yet, fetch and embed them.
  • /cache-notes <path> — Cache notes for a specific meeting file.
  • /cache-notes refresh <path> — Re-fetch and overwrite existing cached content for a file.

If the target file has an empty Notes: property (no URLs yet), prompt the user to paste the external resource URLs (Google Docs, Otter.ai, etc.). Add them to the Notes: frontmatter, then proceed with fetching and caching as normal.

How It Works

External meeting resources (Google Docs, Otter.ai) linked in frontmatter Notes: are fetched, parsed by provider, and embedded as collapsible Obsidian callouts directly in the meeting note. A NotesCached: frontmatter timestamp marks the file as cached.

Frontmatter Convention

Before caching:

yaml
Notes:
  - https://docs.google.com/document/d/{id}/edit?tab=t.{tab}
  - https://otter.ai/u/{id}

After caching:

yaml
Notes:
  - https://docs.google.com/document/d/{id}/edit?tab=t.{tab}
  - https://otter.ai/u/{id}
NotesCached: 2026-02-25T15:00:00-06:00

Provider: Gemini (Google Docs)

Fetching

  1. Extract document ID from the Google Docs URL in Notes:.
  2. Run gws in the terminal to fetch the document (read-only). See google-workspace-cli for exact commands.
  3. Prefer gws docs documents get with includeTabsContent: true and parse the JSON into text (optionally inserting --- TAB: … --- separators from tab titles), or gws drive files export with mimeType: text/plain for a simpler single stream. Parsed content should mirror the old tab layout where needed: --- TAB: Notes ---, --- TAB: Transcript ---.

Parsing the Notes Tab

Gemini notes have a consistent structure. Parse into three sections:

Section Starts after Ends before Callout
Summary + Details Summary heading Suggested next steps heading [!gemini_notes]
Suggested next steps Suggested next steps heading Gemini disclaimer footer [!gemini_todos]
Transcript --- TAB: Transcript --- End of content [!gemini_transcript]

Strip the Gemini disclaimer footer — lines starting with "You should review Gemini's" or "Please provide feedback" are noise. Do not include them.

Output Format

All cached content goes under a top-level ## 🤖 AI Notes section appended after frontmatter. Each provider gets its own ### {Provider} subsection. Callouts are nested under the provider heading.

markdown
## 🤖 AI Notes

### Gemini

> [!gemini_notes]- Summary & Details
> ### Summary
> {summary text}
>
> ### Details
> {details as bullet points}

> [!gemini_todos]- Suggested Next Steps
> - {next step 1}
> - {next step 2}

> [!gemini_transcript]- Transcript
> {transcript content, if available}

### Otter

> [!otter_notes]- Summary & Details
> ...

Formatting rules for callout content:

  • Every line inside a callout must start with > .
  • Preserve markdown formatting (bold, bullet points, headings) inside callouts.
  • Keep the - after the callout type (e.g. [!gemini_notes]-) — this makes it collapsed by default in Obsidian.
  • Blank lines between callouts (no > prefix) to separate them.
  • If transcript tab is not available, omit the [!gemini_transcript] callout entirely.
  • The ## AI Transcripts heading is created once. Provider subsections are added beneath it.
  • If the file already has user-written content between frontmatter --- and ## AI Transcripts, preserve it. Always append ## AI Transcripts at the end of the file.

Provider: Otter.ai

Status: TBD — Otter.ai has no API/MCP integration. For now, skip Otter links silently. When available, the structure will mirror Gemini with [!otter_notes]-, [!otter_todos]-, [!otter_transcript]-.

Workflow

Mode: Specific file (/cache-notes <path>)

  1. Read the file. Check frontmatter for Notes: links and NotesCached:.
  2. If Notes: is empty or absent, prompt the user to paste the external resource URL(s). Add them to the Notes: frontmatter property using StrReplace, then continue.
  3. If NotesCached: exists and this is not a refresh, inform the user and stop.
  4. Verify each URL before caching (see URL Verification below).
  5. For each URL in Notes:, fetch and parse by provider:
    • Google Docs: Extract document ID, fetch via gws (see google-workspace-cli), parse into sections per Provider: Gemini.
    • Otter.ai: Per Provider: Otter.ai — skip silently until API integration is available.
  6. Build the callout blocks from all fetched content (Docs only until Otter is supported).
  7. If the file already has callout blocks (refresh mode), replace them. Otherwise append under ## 🤖 AI Notes after frontmatter --- (see Output Format).
  8. Set NotesCached: in frontmatter to current timestamp.

Mode: All (/cache-notes all)

  1. Run /note-status pending --step=cache to discover notes that have cacheable URLs but no NotesCached: yet, present them, and let the user select which to process.
  2. Process each selected file per the specific-file workflow above.
  3. Report results: successes, failures, skipped (Otter-only).

URL Verification

Before caching, verify that each external resource actually corresponds to the meeting note. This prevents accidentally caching the wrong document when a user pastes an incorrect URL.

How it works

  1. Extract the meeting date from the note. Try these sources in order until one yields a date:
    • Filename date pattern (any format — see Date Parsing below)
    • created: frontmatter timestamp
  2. Fetch the document title from Drive metadata via gws drive files get (name field), e.g. "Daily Standup - 2026/02/24 09:43 EST - Notes by Gemini" — same string you would have seen in older MCP File: lines.
  3. Extract the date from the doc title (any format — see Date Parsing below).
  4. Compare: normalize both dates to YYYY-MM-DD and check equality.
    • If the dates match, proceed silently.
    • If the dates don't match, warn the user:
⚠️ Date mismatch for Meetings/PAM/Scrum/2026-02-24.md:
   Note date:     2026-02-24
   Doc title:     "Daily Standup - 2026/02/26 09:41 EST - Notes by Gemini"
   Doc date:      2026-02-26

Wrong document? [Continue anyway / Skip this URL / Replace URL]
  1. If no date is found in the doc title, fall back to a title similarity check — warn if the doc title has no obvious overlap with the meeting note's title (e.g. a "Sprint Retro" doc linked to a "Daily Standup" note).
  2. In batch mode (/cache-notes all or /meeting wrap pending), collect all mismatches and present them together before proceeding, rather than interrupting one by one.

Date Parsing

See date-filter § "Date Parsing" for the full list of recognized formats and ambiguity rules.

Skip verification

Verification is skipped for:

  • refresh mode (the URL was already verified on first cache)
  • Non-Google Docs URLs (Otter.ai, etc.)

Detecting Uncached Files

Delegate to /note-status pending --step=cache. That skill handles discovery, frontmatter checks, and the dependency chain (only notes with Notes: containing a supported provider URL are considered actionable for caching). See /note-status § Dependency Chain for the full rules.

Refresh Mode

/cache-notes refresh <path>:

  • Delete the entire ## AI Transcripts section and everything below it.
  • Re-fetch all providers and re-build the section.
  • Update NotesCached: timestamp.

Offer to Commit

See /commit. Skip when called as part of a sequence (e.g. /meeting wrap).

Important Notes

  • Always read a file before editing — frontmatter may have changed since last read.
  • Use StrReplace for edits, targeting the closing --- of frontmatter as the anchor.
  • The Notes: URLs should NOT be modified — they remain as the canonical source links.
  • NotesCached: is the single indicator that caching was done. No other markers.
  • For all mode, process in batches and report progress to the user.
  • Gemini anonymizes some speakers as "someone in [location]" — this is expected; /fill-participants handles name resolution separately.

Expand your agent's capabilities with these related and highly-rated skills.

benoror/obsidianos_work

followup-todos

Extract action items as plain markdown bullets (with confirmation). Args: <path>. No args = run [/note-status pending --step=todos](../note-status/SKILL.md#pending-mode).

152 10
Explore
benoror/obsidianos_work

commit

Stage and commit with flexible intent parsing. Accepts file/folder scope, free-text description, amend, or any combination.

152 10
Explore
benoror/obsidianos_work

obsidian-cli

Interact with Obsidian vaults using the Obsidian CLI to read, create, search, and manage notes, tasks, properties, and more. Also supports plugin and theme development with commands to reload plugins, run JavaScript, capture errors, take screenshots, and inspect the DOM. Use when the user asks to interact with their Obsidian vault, manage notes, search vault content, perform vault operations from the command line, or develop and debug Obsidian plugins and themes.

152 10
Explore
benoror/obsidianos_work

defuddle

Extract clean markdown content from web pages using Defuddle CLI, removing clutter and navigation to save tokens. Use instead of WebFetch when the user provides a URL to read or analyze, for online documentation, articles, blog posts, or any standard web page.

152 10
Explore
benoror/obsidianos_work

obsidian-markdown

Create and edit Obsidian Flavored Markdown with wikilinks, embeds, callouts, properties, and other Obsidian-specific syntax. Use when working with .md files in Obsidian, or when the user mentions wikilinks, callouts, frontmatter, tags, embeds, or Obsidian notes.

152 10
Explore
benoror/obsidianos_work

obsidian-bases

Create and edit Obsidian Bases (.base files) with views, filters, formulas, and summaries. Use when working with .base files, creating database-like views of notes, or when the user mentions Bases, table views, card views, filters, or formulas in Obsidian.

152 10
Explore

Didn't find tool you were looking for?

Be as detailed as possible for better results