Apple MCP Calnot
Provides tools for managing iCloud Notes (list, get, search, create, append, delete notes) through an authenticated iCloud session.
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@Apple MCP Calnotfind notes containing 'API design'"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
Apple MCP Calnot
Local MCP/WebUI bridge for iCloud Notes.
The app keeps an authenticated iCloud Notes browser session alive, syncs notes into MongoDB, and exposes note operations through MCP.
Commands
make start
make stop
make cleanmake startstarts Docker Compose without rebuilding and preserves volumes/session state.make stopstops containers and preserves volumes/session state.make cleanremoves containers and volumes. This wipes MongoDB and the browser profile, so iCloud login will be required again.
To apply code changes to the app container while preserving volumes/session:
docker compose up -d --build mcp-notesRelated MCP server: io.github.henilcalagiya/mcp-apple-notes
Login Flow
Open the WebUI at
http://localhost:3000.Click
Generate Code.Copy the generated code.
Log into iCloud in the embedded browser view.
Click
Start.After start, WebUI/API/MCP access requires the generated code.
The browser profile is persisted in the Docker volume mounted at /data, so normal make stop / make start should keep the iCloud session.
Current Architecture
WebUI / MCP
|
NotesProcessor
|
BrowserController
|
Playwright authenticated iCloud page
|
iCloud Notes iframe
|
window.NotesApp
|
NotesApp.dataManager.allNotes
|
note.getTopoText()
|
MongoDBPlaywright is still useful, but not for OCR or visual scraping. It is used to keep an authenticated iCloud page open and to evaluate JavaScript inside the iCloud Notes iframe.
iCloud Notes Discovery
We originally saw note content rendered through a canvas-like/custom editor surface. OCR was rejected because it is unreliable and loses structure. The important discovery is that the visible editor is only the presentation layer; the real Notes model is available in the running iCloud web app.
The path to discovery was:
The top-level iCloud HTML showed that
/notesbootstraps a child application iframe.The bootstrap script resolves
/notestonotes3.It creates an iframe with id
early-child.That iframe loads:
https://www.icloud.com/applications/notes3/current/en-us/index.html?rootDomain=www...Safari Apple Events inspection confirmed the top page has that iframe.
Inspecting
document.querySelector('iframe').contentWindowshowed these globals:
CloudKit
NotesAppDrilling into
window.NotesAppshowed:
NotesApp.dataManager
NotesApp.mainViewModel
NotesApp.rootViewControllerNotesApp.dataManager.allNotescontains the loaded note models.NotesApp.mainViewModel.selectedNotepoints to the selected note.Each note model exposes Notes-specific fields and helpers:
id
recordName
Title
Snippet
TopoTextString
getTopoText()
CreationDate
ModificationDate
zoneIDnote.getTopoText() loads/decodes the full note body using Apple's own app code. This avoids OCR, canvas parsing, and reimplementing Apple's TopoText decoder.
CloudKit vs NotesApp
CloudKit is Apple's generic iCloud database transport layer. It talks to endpoints such as:
ckdatabasews/.../database/1/com.apple.notes/production/private/records/query
ckdatabasews/.../database/1/com.apple.notes/production/private/records/lookup
ckdatabasews/.../database/1/com.apple.notes/production/private/changes/zoneNotesApp is the running iCloud Notes web application loaded inside the iframe. It wraps CloudKit, owns UI/application state, manages folders and notes, and decodes Notes-specific content.
For this project, NotesApp is the preferred first integration point because it already exposes decoded note models:
const notesWindow = document.querySelector('iframe').contentWindow;
const notes = notesWindow.NotesApp.dataManager.allNotes;
const body = String(await notes[0].getTopoText());Direct CloudKit access is still useful later for lower-level sync/write operations, but it requires handling raw record fields, assets, zipped protobuf TopoText, and write semantics.
Stable Note Identity
iCloud note URLs contain the CloudKit identity encoded as base64:
/notes/note/<base64>Decoding the URL path gives:
Private::Notes::currentUser::<recordName>Example observed from Safari:
Private::Notes::currentUser::ADAC358D-E303-4639-A5C2-192AE0726967The sync stores this metadata as cloudKit:
{
"recordId": "Private::Notes::currentUser::<recordName>",
"database": "Private",
"zoneName": "Notes",
"ownerName": "currentUser",
"recordName": "<recordName>"
}Sync Strategy
The current read path is:
Open or reuse the authenticated iCloud Notes page.
Find the Notes iframe.
Evaluate JavaScript inside the iframe.
Read
NotesApp.dataManager.allNotes.Filter deleted/trash notes.
Await
note.getTopoText()for each note.Store title, body, URL identity, and CloudKit metadata in MongoDB.
The older DOM/card scraper remains only as a fallback.
Write Strategy
Safari runtime testing confirmed that create, update, and delete can be driven through NotesApp directly.
Observed working methods:
const app = document.querySelector('iframe').contentWindow.NotesApp;
const dataManager = app.dataManager;
const Note = app.mainViewModel.selectedNote.constructor;Create:
const note = Note.createNoteWithTitleText(fullText, folder);
dataManager.userDidCreateNote(note);
await note.save(true);Update:
const replacement = Note.createInitialTopoTextString(nextText);
dataManager.topoTextManager.load(note.id, replacement);
note.userDidChangeTopoText();
await note.save(true);Delete:
await note.deleteOrMoveToRecentlyDeletedAsNeeded();The delete path moves normal private notes to Recently Deleted, matching the web app behavior.
The probe used a temporary note and verified:
create through
Note.createNoteWithTitleTextupdate through
topoTextManager.loadanduserDidChangeTopoTextdelete through
deleteOrMoveToRecentlyDeletedAsNeededstable CloudKit identity persisted as
Private::Notes::currentUser::<recordName>
MCP writes now use this runtime path first. UI keyboard fallback remains only as a backup for append/create.
What Not To Do
Do not use OCR/Tesseract for note bodies.
Do not treat canvas pixels as the source of truth.
Do not identify notes by title; titles are mutable and non-unique.
Do not use
make cleanunless you intentionally want to wipe browser and database persistence.
Validation
npm run checkMCP Endpoint
The MCP server is exposed at:
POST /mcpAuthentication accepts any of:
Authorization: Bearer <generated-code>
X-Auth-Token: <generated-code>
?token=<generated-code>
apple_mcp_token cookieThe server advertises these MCP tools:
listNotes
getNote
searchNotes
createNote
appendNote
deleteNoteFor ChatGPT testing, expose the app over HTTPS and configure the MCP URL as:
https://your-domain.example/mcp?token=<generated-code>Using the token in the URL is convenient for testing because the current server uses a generated static code, not OAuth. For a durable public deployment, prefer adding OAuth or a reverse proxy that injects the bearer token server-side, so the code is not stored in connector URLs or logs.
This server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/davidfesteban/apple-mcp-calnot'
If you have feedback or need assistance with the MCP directory API, please join our Discord server