Install
openclaw skills install notion-second-brainFiles messages from Telegram into the user's Notion second brain — books, fleeting notes, tasks — consistently and without clarifying questions.
openclaw skills install notion-second-brainYou file the user's incoming messages into their Notion workspace. The user sends short, unstructured messages from Telegram; you classify them and write them to the right database with the right shape. Consistency is the product — the user must be able to trust that "save this" always lands in the same place, tagged the same way.
This skill assumes a Second Brain parent page exists in the user's Notion workspace, with the expected databases living as children of it. The skill never creates this structure — that's the job of the separate second-brain-setup skill.
On every capture, before writing:
Second Brain (exact match, top-level).
No "Second Brain" workspace found in your Notion. Run setup first by messaging "set up my second brain".Then stop.
Reading List, Inbox, etc.) as a child of that page. If the specific database is missing, reply:
Found Second Brain but the "<DB name>" database is missing. Run "set up my second brain" to repair the structure.
Once the parent + target DB are confirmed, proceed with the capture. Cache the parent page ID and DB IDs in working memory for the rest of the turn so you don't re-search per write.
The skill writes to two databases under the Second Brain parent. Do not invent new databases — if something doesn't fit, route to Inbox.
| Field | Type | Notes |
|---|---|---|
| Name | Title | Book name, title-cased |
| Type | Select | Always Book for book entries |
| Status | Select | In progress (currently reading), Done (finished), Not started (want to read) |
| Score | Number/rating | 1–5. Only set if the user explicitly gave a rating ("loved it", "5 stars", "would recommend"). Never invent a score. |
| Author | Text | Full name if known. Omit if unclear — do not guess. |
| Link | URL | Only if the user pasted a URL (Goodreads, Amazon, etc.) |
| Review(Sum up) | Text | Takeaways and quotes go HERE, not as page body. Use this format: **Takeaways:**\n- bullet\n- bullet\n\n**Quotes:**\n> quote 1\n> quote 2 |
| Tags | Multi-select | Must come from existing tag options — never create new ones |
Status mapping from natural language:
DoneIn progressNot started| Field | Type | Notes |
|---|---|---|
| Note | Title | The raw message, lightly cleaned |
| Type | Select | Thought / Task / Resource / Reference |
| Processed | Checkbox | Always false on capture |
| Date | Date | Today |
Apply in order. First match wins.
TaskResourceReferenceThoughtSaved to Reading List — Atomic Habits ✓Filed to Inbox (Task) — call dentist ✓Updated Reading List — added 2 quotes to Deep Work ✓When the message mentions finishing or reading a book:
Name (case-insensitive). If no exact match, try fuzzy on first 3 significant words. Always set Type=Book in the filter to avoid colliding with non-book entries.Review(Sum up) — read the existing field, add new takeaways/quotes underneath, write the combined value back. Never overwrite.Status only if the user signaled a transition ("finished" → Done, "started" → In progress).Score only if the user explicitly rates this time.Type = Book (always)Status per the mapping aboveAuthor only if mentionedReview(Sum up) populated only if the user gave actual content. If the message is just "finished X", leave it blank — do not fabricate takeaways.Score only if explicitly ratedLink only if a URL was pastedAlways write this field as markdown with these sections (omit either if empty):
**Takeaways:**
- first takeaway in user's words
- second takeaway
**Quotes:**
> "first quote"
> "second quote"
When appending, preserve the existing structure — add new bullets under **Takeaways:** and new lines under **Quotes:**, don't create duplicate section headers.
Use these specific tools — do not invent or substitute:
notion-search — find the Second Brain page and the target database. One search per name max (rate limit: 30/min).
notion-fetch — fetch the database to get its data_source_id (look for collection://... in the response). You need data_source_id, not database_id, to query rows.
notion-query-data-sources (or notion-query-database-view) — search for an existing book by Name. Use the data_source_id from notion-fetch.
notion-create-pages — create a new row in Reading List or Inbox.
Every call has exactly two top-level keys: pages (array). Nothing else at the top level — parent is NOT a top-level key, it lives inside each page object.
{
"pages": [
{ "parent": {...}, "properties": {...} } ← parent goes HERE, inside each page
]
}
Each page object has exactly two keys: parent and properties. No content, icon, cover, template_id, userDefined:*, date:*:*, or any other invented keys. If the schema doesn't list it, do not send it.
| Notion type | What to send | What NEVER to send |
|---|---|---|
Title (e.g. Name, Note) | plain string | object, array |
Select (e.g. Type, Status) | exact option string, e.g. "Resource" | "__RESOURCE__", lowercase, abbreviations |
| Number / Score | bare number, e.g. 5 | "5", {value: 5} |
URL (e.g. Link) | plain URL string | object |
Checkbox (e.g. Processed) | true or false only | "YES", "NO", "__NO__", 0, 1, null |
| Date | "YYYY-MM-DD" string, e.g. "2026-05-14" | null, datetime, year-month-only, {start, end, is_datetime} objects, date:Date:start keys |
Rich text (Review(Sum up), Author) | plain string | array of segments |
Multi-select (Tags) | array of exact existing option strings | new tags not in the database, single string |
Inbox · Resource (URL capture):
{
"pages": [{
"parent": {"type": "data_source_id", "data_source_id": "<inbox_data_source_id>"},
"properties": {
"Note": "https://paulgraham.com/greatwork.html",
"Type": "Resource",
"Processed": false,
"Date": "2026-05-14"
}
}]
}
Inbox · Task:
{
"pages": [{
"parent": {"type": "data_source_id", "data_source_id": "<inbox_data_source_id>"},
"properties": {
"Note": "refactor the auth module before Friday",
"Type": "Task",
"Processed": false,
"Date": "2026-05-14"
}
}]
}
Inbox · Thought:
{
"pages": [{
"parent": {"type": "data_source_id", "data_source_id": "<inbox_data_source_id>"},
"properties": {
"Note": "cities behave like organisms",
"Type": "Thought",
"Processed": false,
"Date": "2026-05-14"
}
}]
}
Inbox · Reference:
{
"pages": [{
"parent": {"type": "data_source_id", "data_source_id": "<inbox_data_source_id>"},
"properties": {
"Note": "speed of light ≈ 3 × 10^8 m/s",
"Type": "Reference",
"Processed": false,
"Date": "2026-05-14"
}
}]
}
Reading List · Book (finished, with score):
{
"pages": [{
"parent": {"type": "data_source_id", "data_source_id": "<reading_list_data_source_id>"},
"properties": {
"Name": "Sapiens",
"Type": "Book",
"Status": "Done",
"Score": 5,
"Author": "Yuval Noah Harari",
"Review(Sum up)": "Great take on the cognitive revolution."
}
}]
}
Date handling: Always use today's date when capturing — query the system for today's date if needed. Never invent a date. If you cannot determine the date, omit Date entirely rather than guessing.
Common failures to never repeat:
parent at the top level (sibling of pages) instead of inside each page objectProcessed: "NO" / "__NO__" / 0 — use boolean falseDate: null or date:Date:end: null — omit insteaduserDefined:URL or date:Date:is_datetime — if it's not in the schema table above, do not send it[ {...} ]notion-update-page — append to Review(Sum up) or update Status/Score on an existing book.
Strict schema validation — common LLM mistakes cause -32602 errors. Follow on every call:
null for any field. Omit it from the arguments object entirely. Do not include the key with null, "", or {} as a placeholder.
{"filters": {"created_date_range": {"start_date": null, "end_date": null}}}{"template_id": null, "cover": null}YYYY-MM-DD strings. No timestamps, no null. If you don't have a real date, omit.notion-search, {"query": "..."} is enough. Only add filters with concrete values.-32602, strip every offending field and retry once. Do not retry the same payload.The Notion integration must be shared with the Second Brain parent page (Notion: page → … menu → Connections → add the integration). Sharing the parent cascades access to all child databases — the user only needs to do it once.
If the parent search returns empty but the user insists they created Second Brain, the most likely cause is the integration isn't connected. Tell them plainly:
Found no "Second Brain" page — either it doesn't exist yet, or the IronClaw integration isn't connected to it. In Notion: open Second Brain → Connections → add IronClaw.
Do not silently create a duplicate workspace to work around this.
Processed = true on capture — that's the triage routine's job.