c06e6d6b4c
Notes server rewritten to use Joplin CLI Data API (joplin-cli sync clients on host, ports 41184/41185). Clean fast search, proper resource handling. New tools: list_note_resources, read_resource (attachments as inline image/document/text). Mail server: read_mail now lists attachments; new read_attachment tool returns images inline, PDFs/docs as EmbeddedResource, text directly. Tests: 54 total. Notes now real (notebooks, list, create+read). Mail attachment listing + fetch tested. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
220 lines
8.2 KiB
Python
220 lines
8.2 KiB
Python
"""MCP Notes Server — reads/writes Joplin notes via Joplin CLI Data API."""
|
|
|
|
import os, sys, contextlib, base64
|
|
from typing import Annotated
|
|
|
|
import httpx
|
|
from pydantic import Field
|
|
from mcp.server.fastmcp import FastMCP
|
|
from mcp.types import TextContent, ImageContent, EmbeddedResource, BlobResourceContents
|
|
from starlette.applications import Starlette
|
|
from starlette.routing import Mount
|
|
|
|
sys.path.insert(0, os.path.dirname(os.path.dirname(__file__)))
|
|
from common import get_current_user, OAUTH_ROUTES, BearerAuthMiddleware, load_config
|
|
|
|
_cfg = load_config()
|
|
DATA_API = _cfg["joplin_data_api"] # {user: {url, token}}
|
|
|
|
mcp = FastMCP("Notes", stateless_http=True,
|
|
transport_security={"enable_dns_rebinding_protection": False})
|
|
|
|
IMAGE_TYPES = {"image/jpeg", "image/png", "image/gif", "image/webp"}
|
|
|
|
|
|
def _api(user):
|
|
return DATA_API.get(user)
|
|
|
|
|
|
def _get(user, path, params=None):
|
|
api = _api(user)
|
|
if not api:
|
|
return None, "Kein Joplin-Zugang fuer diesen User"
|
|
p = dict(params or {})
|
|
p["token"] = api["token"]
|
|
try:
|
|
r = httpx.get(f"{api['url']}{path}", params=p, timeout=30)
|
|
if r.status_code >= 400:
|
|
return None, f"Joplin HTTP {r.status_code}"
|
|
return r, None
|
|
except Exception as e:
|
|
return None, str(e)
|
|
|
|
|
|
def _post(user, path, json_data):
|
|
api = _api(user)
|
|
if not api:
|
|
return None, "Kein Joplin-Zugang"
|
|
try:
|
|
r = httpx.post(f"{api['url']}{path}?token={api['token']}", json=json_data, timeout=30)
|
|
if r.status_code >= 400:
|
|
return None, f"Joplin HTTP {r.status_code}"
|
|
return r.json(), None
|
|
except Exception as e:
|
|
return None, str(e)
|
|
|
|
|
|
def _all_items(user, path, params=None):
|
|
"""Fetch all pages from a Data API list endpoint."""
|
|
items = []
|
|
page = 1
|
|
while True:
|
|
p = dict(params or {})
|
|
p["page"] = page
|
|
r, err = _get(user, path, p)
|
|
if err:
|
|
return [], err
|
|
d = r.json()
|
|
items.extend(d.get("items", []))
|
|
if not d.get("has_more"):
|
|
break
|
|
page += 1
|
|
if page > 50:
|
|
break
|
|
return items, None
|
|
|
|
|
|
@mcp.tool()
|
|
def list_notebooks() -> str:
|
|
"""List all Joplin notebooks (folders). Call this first to see notebook names and IDs."""
|
|
user = get_current_user()
|
|
if not user: return "Error: not authenticated"
|
|
items, err = _all_items(user, "/folders")
|
|
if err: return f"Fehler: {err}"
|
|
return "\n".join(f"{i['title']} (id: {i['id']})" for i in items) if items else "Keine Notizbuecher"
|
|
|
|
|
|
@mcp.tool()
|
|
def list_notes(
|
|
notebook: Annotated[str, Field(description="Notebook name or ID from list_notebooks. Leave empty for all notes.")] = "",
|
|
limit: Annotated[int, Field(description="Maximum number of notes")] = 50,
|
|
) -> str:
|
|
"""List notes, optionally filtered by notebook. Shows title and ID."""
|
|
user = get_current_user()
|
|
if not user: return "Error: not authenticated"
|
|
nb_id = ""
|
|
if notebook:
|
|
folders, err = _all_items(user, "/folders")
|
|
if err: return f"Fehler: {err}"
|
|
for f in folders:
|
|
if notebook.lower() in f["title"].lower() or notebook == f["id"]:
|
|
nb_id = f["id"]
|
|
break
|
|
if not nb_id:
|
|
return f"Notizbuch nicht gefunden: {notebook}"
|
|
path = f"/folders/{nb_id}/notes"
|
|
else:
|
|
path = "/notes"
|
|
items, err = _all_items(user, path, {"fields": "id,title", "order_by": "updated_time", "order_dir": "DESC"})
|
|
if err: return f"Fehler: {err}"
|
|
return "\n".join(f"{i['title']} (id: {i['id']})" for i in items[:limit]) if items else "Keine Notizen"
|
|
|
|
|
|
@mcp.tool()
|
|
def search_notes(
|
|
query: Annotated[str, Field(description="Search term — matches note titles and content. Example: 'Einkaufsliste', 'Rezept', 'Passwort'")],
|
|
) -> str:
|
|
"""Search notes by keyword (full text). Returns matching note titles and IDs."""
|
|
user = get_current_user()
|
|
if not user: return "Error: not authenticated"
|
|
items, err = _all_items(user, "/search", {"query": query, "type": "note", "fields": "id,title"})
|
|
if err: return f"Fehler: {err}"
|
|
return "\n".join(f"{i['title']} (id: {i['id']})" for i in items[:30]) if items else "Keine Treffer"
|
|
|
|
|
|
@mcp.tool()
|
|
def read_note(
|
|
note_id: Annotated[str, Field(description="Note ID from list_notes or search_notes")],
|
|
) -> str:
|
|
"""Read a note's full content (Markdown). Resource links appear as :/resourceId — use list_note_resources + read_resource for attachments."""
|
|
user = get_current_user()
|
|
if not user: return "Error: not authenticated"
|
|
r, err = _get(user, f"/notes/{note_id}", {"fields": "id,title,body"})
|
|
if err: return f"Fehler: {err}"
|
|
d = r.json()
|
|
return f"# {d.get('title','?')}\n\n{d.get('body','')}"
|
|
|
|
|
|
@mcp.tool()
|
|
def list_note_resources(
|
|
note_id: Annotated[str, Field(description="Note ID from list_notes or search_notes")],
|
|
) -> str:
|
|
"""List attachments (images, PDFs, files) embedded in a note. Use read_resource with the resource ID to fetch one."""
|
|
user = get_current_user()
|
|
if not user: return "Error: not authenticated"
|
|
items, err = _all_items(user, f"/notes/{note_id}/resources", {"fields": "id,title,mime,size"})
|
|
if err: return f"Fehler: {err}"
|
|
if not items:
|
|
return "Keine Anhaenge"
|
|
lines = []
|
|
for i in items:
|
|
size = i.get("size", 0)
|
|
lines.append(f"{i.get('title','?')} ({i.get('mime','?')}, {size:,} bytes) id: {i['id']}")
|
|
return "\n".join(lines)
|
|
|
|
|
|
@mcp.tool()
|
|
def read_resource(
|
|
resource_id: Annotated[str, Field(description="Resource/attachment ID from list_note_resources")],
|
|
) -> list[TextContent | ImageContent | EmbeddedResource]:
|
|
"""Read an attachment. Images shown inline, documents (PDF/docx) as binary, text directly."""
|
|
user = get_current_user()
|
|
if not user: return [TextContent(type="text", text="Error: not authenticated")]
|
|
# Get metadata
|
|
rmeta, err = _get(user, f"/resources/{resource_id}", {"fields": "id,title,mime,size"})
|
|
if err: return [TextContent(type="text", text=f"Fehler: {err}")]
|
|
meta = rmeta.json()
|
|
mime = meta.get("mime", "application/octet-stream")
|
|
title = meta.get("title", "attachment")
|
|
# Get blob
|
|
rfile, err = _get(user, f"/resources/{resource_id}/file")
|
|
if err: return [TextContent(type="text", text=f"Fehler: {err}")]
|
|
content = rfile.content
|
|
if mime in IMAGE_TYPES and len(content) < 10_000_000:
|
|
return [ImageContent(type="image", data=base64.b64encode(content).decode(), mimeType=mime)]
|
|
if mime.startswith("text/"):
|
|
return [TextContent(type="text", text=content.decode("utf-8", errors="replace")[:100000])]
|
|
return [EmbeddedResource(type="resource", resource=BlobResourceContents(
|
|
uri=f"joplin://resource/{resource_id}/{title}",
|
|
blob=base64.b64encode(content).decode(), mimeType=mime))]
|
|
|
|
|
|
@mcp.tool()
|
|
def create_note(
|
|
notebook: Annotated[str, Field(description="Notebook name or ID where the note is created")],
|
|
title: Annotated[str, Field(description="Note title")],
|
|
body: Annotated[str, Field(description="Note content in Markdown")],
|
|
) -> str:
|
|
"""Create a new note in the specified notebook."""
|
|
user = get_current_user()
|
|
if not user: return "Error: not authenticated"
|
|
folders, err = _all_items(user, "/folders")
|
|
if err: return f"Fehler: {err}"
|
|
nb_id = None
|
|
for f in folders:
|
|
if notebook.lower() in f["title"].lower() or notebook == f["id"]:
|
|
nb_id = f["id"]
|
|
break
|
|
if not nb_id: return f"Notizbuch nicht gefunden: {notebook}"
|
|
d, err = _post(user, "/notes", {"title": title, "body": body, "parent_id": nb_id})
|
|
if err: return f"Fehler: {err}"
|
|
return f"Notiz erstellt: {title} (id: {d.get('id','?')})"
|
|
|
|
|
|
def create_app():
|
|
from contextlib import asynccontextmanager
|
|
mcp_app = mcp.streamable_http_app()
|
|
@asynccontextmanager
|
|
async def lifespan(app):
|
|
async with contextlib.AsyncExitStack() as stack:
|
|
await stack.enter_async_context(mcp_app.router.lifespan_context(mcp_app))
|
|
yield
|
|
routes = list(OAUTH_ROUTES) + [Mount("/", app=mcp_app)]
|
|
app = Starlette(routes=routes, lifespan=lifespan)
|
|
app.add_middleware(BearerAuthMiddleware)
|
|
return app
|
|
|
|
if __name__ == "__main__":
|
|
import uvicorn
|
|
uvicorn.run(create_app(), host="127.0.0.1", port=5104)
|