notion-enhanced-markdown-integration
v1.0.0Notion API for creating and managing pages, databases, blocks and enhanced markdown.
Like a lobster shell, security has layers — review code before you run it.
License
Runtime requirements
SKILL.md
notion
Use the Notion API to create/read/update pages, data sources (databases), and blocks.
Setup
- Create an integration at https://notion.so/my-integrations
- Copy the API key (starts with
ntn_orsecret_) - Store it:
mkdir -p ~/.config/notion
echo "ntn_your_key_here" > ~/.config/notion/api_key
- Share target pages/databases with your integration (click "..." → "Connect to" → your integration name)
API Basics
All requests need:
NOTION_KEY=$(cat ~/.config/notion/api_key)
curl -X GET "https://api.notion.com/v1/..." \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11" \
-H "Content-Type: application/json"
Note: The
Notion-Versionheader is required. This skill uses2026-03-11(latest). In this version, databases are called "data sources" in the API.
Common Operations
Search for pages and data sources:
curl -X POST "https://api.notion.com/v1/search" \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11" \
-H "Content-Type: application/json" \
-d '{"query": "page title"}'
Get page:
curl "https://api.notion.com/v1/pages/{page_id}" \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11"
Get page content (blocks):
For simple content access, prefer the Markdown API (
GET /v1/pages/{page_id}/markdown). Use the blocks API only when you need low-level block data or unsupported block types.
curl "https://api.notion.com/v1/blocks/{page_id}/children" \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11"
Create page in a data source:
curl -X POST "https://api.notion.com/v1/pages" \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11" \
-H "Content-Type: application/json" \
-d '{
"parent": {"database_id": "xxx"},
"properties": {
"Name": {"title": [{"text": {"content": "New Item"}}]},
"Status": {"select": {"name": "Todo"}}
}
}'
Query a data source (database):
curl -X POST "https://api.notion.com/v1/data_sources/{data_source_id}/query" \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11" \
-H "Content-Type: application/json" \
-d '{
"filter": {"property": "Status", "select": {"equals": "Active"}},
"sorts": [{"property": "Date", "direction": "descending"}]
}'
Create a data source (database):
curl -X POST "https://api.notion.com/v1/data_sources" \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11" \
-H "Content-Type: application/json" \
-d '{
"parent": {"page_id": "xxx"},
"title": [{"text": {"content": "My Database"}}],
"properties": {
"Name": {"title": {}},
"Status": {"select": {"options": [{"name": "Todo"}, {"name": "Done"}]}},
"Date": {"date": {}}
}
}'
Update page properties:
curl -X PATCH "https://api.notion.com/v1/pages/{page_id}" \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11" \
-H "Content-Type: application/json" \
-d '{"properties": {"Status": {"select": {"name": "Done"}}}}'
Add blocks to page:
curl -X PATCH "https://api.notion.com/v1/blocks/{page_id}/children" \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11" \
-H "Content-Type: application/json" \
-d '{
"children": [
{"object": "block", "type": "paragraph", "paragraph": {"rich_text": [{"text": {"content": "Hello"}}]}}
]
}'
Markdown Page Operations
Use the Markdown API to read and write page content as plain Markdown — no need to construct block JSON.
Create page with markdown:
curl -X POST "https://api.notion.com/v1/pages" \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11" \
-H "Content-Type: application/json" \
-d '{
"parent": {"page_id": "xxx"},
"markdown": "# Meeting Notes\n\nContent here.\n\n- item 1\n- item 2"
}'
Constraints:
markdownandchildrenare mutually exclusive in the same request- If
properties.titleis omitted, the first# h1heading becomes the page title - Requires
insert_contentcapability on the parent - Only works for pages under a parent page (not a database)
Read page as markdown:
curl "https://api.notion.com/v1/pages/{page_id}/markdown" \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11"
Response: { "object": "page_markdown", "id": "...", "markdown": "...", "truncated": false, "unknown_block_ids": [] }
- Add
?include_transcript=trueto include meeting transcripts - If
truncated: true, fetch remaining blocks using the IDs inunknown_block_ids
Update page markdown:
# Search-and-replace (recommended)
curl -X PATCH "https://api.notion.com/v1/pages/{page_id}/markdown" \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11" \
-H "Content-Type: application/json" \
-d '{
"command": {
"type": "update_content",
"old_str": "old text",
"new_str": "new text"
}
}'
# Full replacement
curl -X PATCH "https://api.notion.com/v1/pages/{page_id}/markdown" \
-H "Authorization: Bearer $NOTION_KEY" \
-H "Notion-Version: 2026-03-11" \
-H "Content-Type: application/json" \
-d '{
"command": {
"type": "replace_content",
"new_str": "# New Title\n\nReplaced content."
}
}'
Two command types:
update_content— search-and-replace (old_str→new_str); fails withvalidation_errorifold_strappears multiple times (add"replace_all_matches": trueto override)replace_content— replaces all page content withnew_str
Both commands return the full updated page markdown. Constraints:
- Matching is case-sensitive
- Operations that would delete child pages/databases are rejected unless
"allow_deleting_content": trueis set - Synced pages cannot be updated
Supported Block Types (Markdown)
| Block Type | Markdown Syntax |
|---|---|
| Paragraph | plain text |
| Heading 1–4 | # through #### |
| Bulleted list item | - item |
| Numbered list item | 1. item |
| To-do | - [ ] / - [x] |
| Toggle | <details><summary>…</summary>…</details> |
| Quote | > quote |
| Callout | <callout>…</callout> |
| Divider | --- |
| Code block | ```language … ``` |
| Equation | $$ equation $$ |
| Table | <table><tr><td>…</td></tr></table> |
| Image |  |
| File / Video / Audio / PDF | <file src="url">caption</file> |
| Child page | <page url="…">title</page> |
| Columns | <columns><column>…</column></columns> |
Unsupported block types render as <unknown url="..." alt="block_type"/> placeholder tags.
Property Types
Common property formats for database items:
- Title:
{"title": [{"text": {"content": "..."}}]} - Rich text:
{"rich_text": [{"text": {"content": "..."}}]} - Select:
{"select": {"name": "Option"}} - Multi-select:
{"multi_select": [{"name": "A"}, {"name": "B"}]} - Date:
{"date": {"start": "2024-01-15", "end": "2024-01-16"}} - Checkbox:
{"checkbox": true} - Number:
{"number": 42} - URL:
{"url": "https://..."} - Email:
{"email": "a@b.com"} - Relation:
{"relation": [{"id": "page_id"}]}
Key Differences in 2026-03-11
- Databases → Data Sources: Use
/data_sources/endpoints for queries and retrieval - Two IDs: Each database now has both a
database_idand adata_source_id- Use
database_idwhen creating pages (parent: {"database_id": "..."}) - Use
data_source_idwhen querying (POST /v1/data_sources/{id}/query)
- Use
- Search results: Databases return as
"object": "data_source"with theirdata_source_id - Parent in responses: Pages show
parent.data_source_idalongsideparent.database_id - Finding the data_source_id: Search for the database, or call
GET /v1/data_sources/{data_source_id}
Notes
- Page/database IDs are UUIDs (with or without dashes)
- The API cannot set database view filters — that's UI-only
- Rate limit: ~3 requests/second average, with
429 rate_limitedresponses usingRetry-After - Append block children: up to 100 children per request, up to two levels of nesting in a single append request
- Payload size limits: up to 1000 block elements and 500KB overall
- Use
is_inline: truewhen creating data sources to embed them in pages markdownfield in create/update is mutually exclusive withchildren- Markdown pages exceeding ~20,000 blocks will have
truncated: true; fetch remaining content using block IDs inunknown_block_ids - All string matching in
update_contentis case-sensitive - In-page
\nmust be actual newlines in JSON; use single quotes in cURL
Files
1 totalComments
Loading comments…
