Appflowy MCP Server

MCP for self-hosted Appflowy Cloud instance with HTTP interface, prepared in Docker

Documentation

appflowy-mcp

Docker Image Version Docker Pulls Docker Image Size Architectures MCP Coverage License: MIT

🐳 m2n2/appflowy-mcp:latest on Docker Hub

A self-hosted, token-scoped Model Context Protocol server for AppFlowy. It gives AI agents (Claude, or any MCP client) tools to read and edit your AppFlowy workspaces β€” list workspaces, walk the page tree, create/update/read pages, and edit individual blocks in place β€” while bounding each client to exactly the pages you allow via per-token tree-shaped scopes.

  • πŸ”’ Token-scoped access. The server logs into AppFlowy once as a service account. Clients never see those credentials β€” they present an opaque token, and each token is restricted to a set of workspaces / page subtrees.
  • 🌳 Tree-shaped scopes. Grant a whole workspace, a top-level page and everything under it, or a page four levels deep and its descendants. Mix and match several grants per token.
  • 🐳 Runs anywhere. Streamable-HTTP transport, small multi-arch image (m2n2/appflowy-mcp, amd64 + arm64) on Docker Hub, ready for Docker Compose, Kubernetes, or a Helm chart.
  • ✏️ Real editing. Append blocks, insert blocks at any position, edit block text (rich formatting preserved), and delete blocks β€” via the same Yjs/CRDT path the official web client uses.

How access works

            β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”   token: scopes               β”Œβ”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”
 MCP client β”‚  Authorization: Bearer <token>  ──────────▢ β”‚  appflowy-mcp β”‚
 (Claude)   β””β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”€β”˜                               β”‚  enforces     β”‚
                                                          β”‚  scope, then  β”‚
                                                          β”‚  acts as the  β”‚
            service account (email+password / JWT) ◀──────│  service acct β”‚
                                                          β””β”€β”€β”€β”€β”€β”€β”¬β”€β”€β”€β”€β”€β”€β”€β”˜
                                                                 β–Ό
                                                          AppFlowy Cloud REST

Two layers of auth, kept separate:

  1. Backend auth (one service account). APPFLOWY_BASE_URL + APPFLOWY_EMAIL/APPFLOWY_PASSWORD (or a pre-minted APPFLOWY_ACCESS_TOKEN). The server logs in once and refreshes automatically on expiry.
  2. Client auth (many tokens). Each MCP client presents a token. The token decides what it can touch β€” the backend credentials are never exposed.

Scopes

A scope is a path of AppFlowy ids:

ScopeGrants
(empty list)everything the service account can see
WORKSPACEthe whole workspace
WORKSPACE/VIEWthat page and everything nested under it
WORKSPACE/VIEW_L1/VIEW_L2/VIEW_L3a page several levels deep and its subtree

The last id is the root of the allowed subtree; earlier ids only help locate it (AppFlowy view ids are globally unique, so intermediate ids are optional). A token may list several scopes to grant multiple disjoint subtrees at once.

Enforcement is by ancestry: for any page a tool touches, the server walks up the folder tree; if it reaches one of the token's allowed roots, the call proceeds, otherwise it's rejected. Get workspace list and Get workspace folder are pruned to what the token may see.

Configuration

Everything is configurable by environment variables (ideal for Docker / Helm) and/or a YAML/JSON file. Env wins over the file.

Environment variables

VariableDescription
APPFLOWY_BASE_URLAppFlowy Cloud base URL, e.g. https://appflowy.example.com
APPFLOWY_EMAIL / APPFLOWY_PASSWORDService-account login (GoTrue password grant)
APPFLOWY_ACCESS_TOKENPre-minted JWT instead of email/password (takes precedence)
APPFLOWY_MCP_CONFIGOptional path to a YAML/JSON config file
APPFLOWY_MCP_HOST / APPFLOWY_MCP_PORT / APPFLOWY_MCP_PATHListen address (default 0.0.0.0:8000/mcp)
APPFLOWY_MCP_REQUIRE_AUTHtrue (default) rejects unauthenticated requests; false + no tokens = open mode
APPFLOWY_MCP_FOLDER_CACHE_TTLSeconds to cache folder trees for scope checks (default 15)
APPFLOWY_MCP_LOG_LEVELINFO (default), DEBUG, …

Tokens via env β€” two equivalent forms.

JSON blob (best as a single Helm/Docker secret):

APPFLOWY_MCP_TOKENS='[
  {"token":"sk-full",    "name":"full",    "scopes":[]},
  {"token":"sk-teamws",  "name":"team",    "scopes":["WORKSPACE_ID"]},
  {"token":"sk-project", "name":"project", "scopes":["WORKSPACE_ID/ROOT_VIEW_ID",
                                                     "WORKSPACE_ID/A/B/DEEP_VIEW_ID"]}
]'

Indexed (no embedded JSON):

APPFLOWY_MCP_TOKEN_0=sk-full
APPFLOWY_MCP_TOKEN_0_NAME=full
APPFLOWY_MCP_TOKEN_0_SCOPES=                       # empty => all workspaces

APPFLOWY_MCP_TOKEN_1=sk-project
APPFLOWY_MCP_TOKEN_1_NAME=project
APPFLOWY_MCP_TOKEN_1_SCOPES=WORKSPACE_ID/ROOT_VIEW_ID,WORKSPACE_ID/A/B/DEEP_VIEW_ID

Config file

appflowy:
  base_url: https://appflowy.example.com
  email: [email protected]
  password: ${APPFLOWY_PASSWORD}   # plain string; env is not interpolated β€” set real value
server:
  host: 0.0.0.0
  port: 8000
  path: /mcp
  require_auth: true
tokens:
  - token: sk-full
    name: full
    scopes: []                     # all workspaces
  - token: sk-project
    name: project
    scopes:
      - WORKSPACE_ID/ROOT_VIEW_ID            # a page + its whole subtree
      - WORKSPACE_ID/A/B/DEEP_VIEW_ID        # a deep page + its subtree

See config.example.yaml and .env.example.

Running

Docker

docker run --rm -p 8000:8000 \
  -e APPFLOWY_BASE_URL=https://appflowy.example.com \
  -e [email protected] \
  -e APPFLOWY_PASSWORD=secret \
  -e APPFLOWY_MCP_TOKENS='[{"token":"sk-full","scopes":[]}]' \
  m2n2/appflowy-mcp:latest

Docker Compose

cp .env.example .env   # fill in values
docker compose up -d

Kubernetes / Helm

A minimal chart lives in deploy/helm:

helm install appflowy-mcp ./deploy/helm \
  --set appflowy.baseUrl=https://appflowy.example.com \
  --set [email protected] \
  --set appflowy.password=secret \
  --set-json 'tokens=[{"token":"sk-full","scopes":[]}]'

From source

uv run appflowy-mcp

Connecting a client

The server speaks streamable HTTP at http://HOST:PORT/mcp. Point your MCP client at it and send the token as a bearer header. For Claude Code:

{
  "mcpServers": {
    "appflowy": {
      "type": "http",
      "url": "https://appflowy-mcp.example.com/mcp",
      "headers": { "Authorization": "Bearer sk-full" }
    }
  }
}

Health check: GET /healthz β†’ {"status":"ok"}.

Tools

ToolPurpose
Get workspace listList workspaces visible to the token
Get workspace folderPage tree of a workspace, pruned to scope
Create new pageCreate a page under an allowed parent
Update pageRename / set icon / lock
Get page detailsFull page metadata + content
Append content to pageAppend blocks to the end
Get page blocksList a page's blocks in order (ids + text)
Insert blockInsert a new block at any position
Edit block textReplace a block's text/rich content in place
Delete blockDelete a leaf block
Move page to trash / Restore page from trash / Delete page from trashTrash lifecycle
Get trash / Get favorite pagesListings, scoped
Toggle favorite page(Un)favorite a page

Notes & limits

  • Block-editing tools require pycrdt (bundled). They mirror the web client's CRDT web-update; there is no official per-block REST endpoint.
  • Scope checks rely on the workspace folder tree, cached for APPFLOWY_MCP_FOLDER_CACHE_TTL seconds. Newly created pages invalidate the cache for their workspace.
  • Open mode (APPFLOWY_MCP_REQUIRE_AUTH=false with no tokens) grants full access to anyone who can reach the port β€” only use on a trusted network.

Development

uv sync            # install runtime + dev dependencies
uv run pytest      # run the test suite with the 100% coverage gate
uv run ruff check  # lint

The suite enforces 100% line and branch coverage (--cov-fail-under=100 in pyproject.toml). CI runs it as the test job in .github/workflows/docker.yml; the Docker image build needs: test, so a failing test or a coverage drop blocks the image from ever being built. See AGENTS.md for the testing definition of done.

License

MIT β€” see LICENSE.

This project began as a self-hosting-focused rework of LucasXu0/appflowy_mcp.