bnomei/kirby-mcp

A CLI-first MCP server for Kirby CMS

Installs: 3

Dependents: 0

Suggesters: 0

Security: 0

Stars: 1

Watchers: 0

Forks: 0

Open Issues: 1

pkg:composer/bnomei/kirby-mcp

1.0.2 2025-12-21 18:47 UTC

This package is auto-updated.

Last update: 2025-12-21 18:48:22 UTC


README

Kirby 5 PHP 8.2 Release Downloads Unittests PHPStan Discord Buymecoffee

CLI-first MCP server for Composer-based Kirby CMS projects. It lets an IDE or agent inspect your Kirby project (blueprints, templates, plugins, docs) and interact with a real Kirby runtime. It ships with a local knowledge base of Kirby concepts and tasks.

Warning

Prompt injection is a serious security threat, especially when used with documents retrieved from the internet. You might not see it happen when observing the conversation with the agent!

Quickstart

From your Kirby project root:

composer require bnomei/kirby-mcp --dev
vendor/bin/kirby-mcp install
vendor/bin/kirby-mcp

Then configure your MCP client (Cursor/Claude Code/Codex CLI) using the examples in Client setup.

Copy-paste prompt examples

Use these once your MCP client is connected to the server.

Planning & content

Tip

"Use the Kirby MCP to make a plan to..." is a reliable way to get your agent to use the tools and resources this MCP server provides. If you prompt it to take action, it tends to edit files directly based on its training and skills.

Use the Kirby MCP to make a plan to... build a contact form page.

kirby_roots, kirby_blueprint_read, kirby_templates_index, kirby_snippets_index, kirby_controllers_index, kirby_render_page, kirby_dump_log_tail, kirby://config/{option}

Agent response example:
Here’s a Kirby MCP–driven plan for a new contact form page in this project:

- Confirm requirements: fields (name/email/message), spam protection (honeypot + csrf()), success UX (inline vs redirect), and whether to store submissions.
- Panel setup: reuse existing blueprint site/blueprints/pages/contact.yml; update site/blueprints/site.yml so the site β€œpages” section allows creating template contact.
- Create the page: add content/4_contact/contact.txt (or create via Panel once allowed) and fill recipient, subject, success, text.
- Frontend: add site/templates/contact.php + a snippet like site/snippets/contact-form.php (escape all echoed user input; include hidden CSRF field + honeypot).
- Controller: add site/controllers/contact.php to handle POST: check csrf(get('csrf')), validate with invalid(), then send mail via $kirby->email() using replyTo (never use user input as from) and an email template like site/templates/emails/contact.text.php (optional contact.html.php).
- Email config: add site/config/config.php email.transport/email.presets (current kirby://config/email is null).
- Verify with MCP: render the page via kirby_render_page(id='contact', noCache=true); debug with mcp_dump() + kirby_dump_log_tail(traceId=...); inspect blueprint/roots via kirby_blueprint_read(id='pages/contact') and kirby://roots.
Show me the fields available on the home page blueprint and what they do using the MCP.

kirby_blueprint_read, kirby_blueprints_index

Show me the current content of the about page.

kirby_read_page_content

Append " with AI" to the title of the home page with Kirby MCP.

kirby_read_page_content, kirby_update_page_content, kirby://field/text/update-schema

Resource shortcuts

Tip

Either on its own or with a prompt, the resources can be used to quickly bring knowledge and runtime information into the current context of your agent.

kirby://glossary/collection

kirby://glossary/{term}

What is the kirby://config/debug for production?

kirby://config/{option}

Search & docs

Tip

The MCP server ships with a local knowledge base about Kirby. It consists of a glossary, common tasks, and update guides for content fields. This reduces the need to rely on external resources and is very fast.

kirby search for collection filtering

kirby_search

Tip

But sometimes you or your agent needs to dig deeper. That is why the MCP server also provides a fallback to the official Kirby search and docs (not including the forum). You can trigger it by mentioning search online in your prompt.

kirby search online for panel permissions

kirby_online

Tip

Your agent will use the next tool under the hood itself, but you can use it as well to quickly check what the MCP server knows about a given topic.

What mcp tool should I use to... list plugins?

kirby_tool_suggest

Inventory (runtime + filesystem)

list blueprints, templates, snippets, collections, controllers, models, plugins, routes, roots

kirby_blueprints_loaded, kirby_blueprints_index, kirby_templates_index, kirby_snippets_index, kirby_collections_index, kirby_controllers_index, kirby_models_index, kirby_plugins_index, kirby_routes_index, kirby_roots

Debug, tinker/eval and running commands

Important

The kirby_eval tool is disabled by default and CLI commands are protected by an allowlist/denylist, see config and security below.

kirby MCP tinker $site->index()->count()

kirby_eval

run kirby cli command uuid:populate

kirby_run_cli_command

My home page renders incorrectly. Help me debug it with mcp_dump() to return the current $page object.

kirby_render_page, kirby_dump_log_tail, kirby_templates_index, kirby_snippets_index, kirby_controllers_index, kirby_models_index

Capabilities

[!INFO] kirby_init is required once per session before calling any other tool or resource but the agent should figure this out automatically. Some capabilities require the runtime wrappers because they query Kirby at runtime. Installing/updating them should happen automatically as well.

At initialization, the server tells the agent which tools/resources to use. The knowledge base cross-references them so the agent can find the next step.

πŸ› οΈ Tools
  • kirby_blueprint_read β€” read a single blueprint by id
  • kirby_blueprints_index β€” index blueprints, includes plugin-registered ones when runtime is installed
  • kirby_blueprints_loaded β€” list blueprint ids loaded at runtime
  • kirby_cache_clear β€” clear in-memory caches for this MCP session (StaticCache, config, composer, roots, tool index)
  • kirby_cli_version β€” run kirby version and return stdout, stderr and exit code
  • kirby_composer_audit β€” parse composer.json for scripts and quality tools
  • kirby_collections_index β€” index named collections, includes plugin-registered ones when runtime is installed
  • kirby_controllers_index β€” index controllers, includes plugin-registered ones when runtime is installed
  • kirby_online β€” search official Kirby docs (online fallback) and optionally fetch markdown pages
  • kirby_dump_log_tail β€” tail .kirby-mcp/dumps.jsonl written by mcp_dump()
  • kirby_eval β€” execute PHP in Kirby runtime for quick inspection, requires enable plus confirm
  • kirby_generate_ide_helpers β€” generate regeneratable IDE helper files into .kirby-mcp/
  • kirby_ide_helpers_status β€” report missing template/snippet PHPDoc @var hints + helper file freshness (mtime-based)
  • kirby_info β€” project runtime info, composer audit and local environment detection
  • kirby_init β€” session guidance plus project-specific audit, call once per session
  • kirby_search β€” search the bundled local Kirby knowledge base markdown files (preferred)
  • kirby_models_index β€” index registered page models with class and file path info
  • kirby_plugins_index β€” index loaded plugins, prefers runtime truth when installed
  • kirby_read_page_content β€” read page content by id or uuid
  • kirby_render_page β€” render a page by id or uuid and return HTML plus errors
  • kirby_roots β€” resolved Kirby roots via kirby roots
  • kirby_routes_index β€” list registered routes with best-effort source location (config/plugin)
  • kirby_run_cli_command β€” run a Kirby CLI command, guarded by an allowlist
  • kirby_runtime_install β€” install project-local Kirby MCP runtime CLI commands into the project
  • kirby_runtime_status β€” check whether runtime command wrappers are installed
  • kirby_snippets_index β€” index snippets, includes plugin-registered ones when runtime is installed
  • kirby_templates_index β€” index templates, includes plugin-registered ones when runtime is installed
  • kirby_tool_suggest β€” suggest the best next Kirby MCP tool/resource for a task
  • kirby_update_page_content β€” update page content, plus confirm (see kirby://field/{type}/update-schema for payload shapes)
πŸ“š Resources

[!TIP] Call a resource to bring condensed knowledge into the current context of your agent.

Resources (read-only):

  • kirby://commands β€” Kirby CLI command list, parsed from kirby help
  • kirby://composer β€” composer audit, scripts and quality tooling
  • kirby://extensions β€” Kirby plugin extensions list (links to kirby://extension/{name})
  • kirby://fields β€” Kirby Panel field types list (links to kirby://field/{type})
  • kirby://fields/update-schema β€” Kirby content field guides list (links to kirby://field/{type}/update-schema)
  • kirby://glossary β€” Kirby glossary terms list (links to kirby://glossary/{term})
  • kirby://hooks β€” Kirby hook names list (links to kirby://hook/{name})
  • kirby://info β€” project runtime info, composer audit and local environment detection
  • kirby://prompts β€” MCP prompts with args/meta (fallback for clients without prompt support)
  • kirby://roots β€” Kirby roots discovered via CLI, respects configured host
  • kirby://sections β€” Kirby Panel section types list (links to kirby://section/{type})
  • kirby://tools β€” weighted keyword index for Kirby MCP tools/resources/templates

Resource templates (dynamic):

  • kirby://blueprint/{encodedId} β€” read a blueprint by URL-encoded id, e.g. pages%2Fhome
  • kirby://cli/command/{command} β€” parsed kirby <command> --help output, e.g. backup or uuid:generate
  • kirby://config/{option} β€” read a Kirby config option by dot path
  • kirby://extension/{name} β€” Kirby extension reference markdown from getkirby.com, e.g. commands or darkroom-drivers
  • kirby://field/{type} β€” Kirby Panel field reference markdown from getkirby.com, e.g. blocks or email
  • kirby://field/{type}/update-schema β€” bundled content field guide from kb/kirby/update-schema/{type}.md
  • kirby://glossary/{term} β€” read a bundled Kirby glossary entry by term, e.g. api or kql
  • kirby://hook/{name} β€” Kirby hook reference markdown from getkirby.com, e.g. file.changeName:after or file-changename-after
  • kirby://page/content/{encodedIdOrUuid} β€” read page content by URL-encoded id or uuid
  • kirby://prompt/{name} β€” prompt details + rendered default messages (fallback for clients without prompt support)
  • kirby://section/{type} β€” Kirby Panel section reference markdown from getkirby.com, e.g. fields or files
  • kirby://susie/{phase}/{step} β€” easter egg resource template
πŸ’¬ Prompts & completions

Prompts:

  • kirby_project_tour β€” map the project (roots + inventory) and suggest next steps
  • kirby_debug_render_trace β€” debug via kirby_render_page + mcp_dump traces (kirby_dump_log_tail)
  • kirby_scaffold_page_type β€” scaffold a new page type (blueprint + template + optional controller/model)
  • kirby_content_migration_assistant β€” plan/apply safe content migrations (read/update page content)
  • kirby_ide_support_boost β€” improve IDE support + optional helper generation (.kirby-mcp/)
  • kirby_upgrade_kirby β€” upgrade Kirby safely (docs + composer + verification)
  • kirby_performance_audit β€” guide an agent through a Kirby performance audit, cache and query pitfalls

Prompt fallback resources (for clients without MCP prompt support):

  • kirby://prompts and kirby://prompt/{name}

Completions:

  • Prompts and resource templates provide parameter completions (e.g. blueprint ids + config hosts).

Client setup

Note

The --project flag is optional when you run the server from the Kirby project root. Use it (or KIRBY_MCP_PROJECT_ROOT) when running from elsewhere or from a global MCP config.

Cursor

Add to .cursor/mcp.json (project) or ~/.cursor/mcp.json (global):

{
  "mcpServers": {
    "kirby": {
      "command": "vendor/bin/kirby-mcp",
      "args": ["--project=/absolute/path/to/kirby-project"]
    }
  }
}

If you use the global config, set "command" to an absolute path to the project’s vendor/bin/kirby-mcp (or create a wrapper script).

Claude Code

From the Kirby project directory:

claude mcp add kirby -- vendor/bin/kirby-mcp

Or explicitly:

claude mcp add kirby -- vendor/bin/kirby-mcp --project=/absolute/path/to/kirby-project

Codex CLI

From the Kirby project directory:

codex mcp add kirby -- vendor/bin/kirby-mcp

Or configure in config.toml under [mcp_servers.kirby] (see Codex CLI docs).

Manual

Start the server (point it at a composer-based Kirby project):

  • From the Kirby project root: vendor/bin/kirby-mcp
  • Or explicitly: vendor/bin/kirby-mcp --project=/absolute/path/to/kirby-project

IDE helpers (optional, for humans)

The agent can both check and generate IDE helpers for your project: kirby_ide_helpers_status and kirby_generate_ide_helpers. You can also use the CLI commands yourself.

  • Check baseline + freshness: vendor/bin/kirby-mcp ide:status (use --details and --limit=N for more output)
  • Generate regeneratable helper files: vendor/bin/kirby-mcp ide:generate (default is --dry-run; add --write to create files)
  • JSON output: --json (MCP markers) or --raw-json (plain JSON)

What the MCP server does (and doesn’t)

  • Provides MCP tools/resources for project inspection (blueprints, templates/snippets/collections, controllers/models, plugins, routes, roots).
  • Fetches official Kirby reference docs and ships a local Markdown knowledge base (kb/) for fast lookups.
  • Doesn’t modify your content by default; write-capable actions run by the MCP are guarded and require explicit opt-in/confirmation. But your agent still can do whatever you allow it to!
  • Only supports composer-based Kirby projects (Kirby CLI is used for many capabilities).

Security model

  • kirby_run_cli_command is guarded by an allowlist; extend it via .kirby-mcp/mcp.json (cli.allow, cli.allowWrite) and block via cli.deny.
  • Write-capable actions require explicit opt-in (e.g. allowWrite=true or confirm=true, depending on the tool).
  • kirby_eval is disabled by default; enable via KIRBY_MCP_ENABLE_EVAL=1 or .kirby-mcp/mcp.json ({"eval":{"enabled":true}}) and still confirm per call.

What install / update change in your project

vendor/bin/kirby-mcp install:

  • Creates .kirby-mcp/mcp.json if neither .kirby-mcp/mcp.json nor .kirby-mcp/config.json exist.
  • Copies runtime command wrappers into the project’s Kirby commands root (usually site/commands/mcp/).
  • Use --force to overwrite existing wrapper files.

vendor/bin/kirby-mcp update:

  • Overwrites the runtime wrappers (use after upgrading this package).
  • Creates .kirby-mcp/mcp.json only if missing; it won’t overwrite an existing config.

To remove everything:

  • Delete the runtime wrappers folder (site/commands/mcp/ in most projects).
  • Optionally delete .kirby-mcp/ (config + caches + optional helper files).

Debug dumps (mcp_dump)

This package provides a lightweight mcp_dump() helper that appends JSONL to .kirby-mcp/dumps.jsonl in the project root.

Secret redaction: By default, dump output is scanned for sensitive data (API keys, tokens, passwords, IPs) and redacted before writing. This protects against accidentally leaking secrets. Configure via dumps.secretPatterns in .kirby-mcp/mcp.json:

{
  "dumps": {
    "secretPatterns": []
  }
}
  • Omit secretPatterns β†’ use built-in patterns (OpenAI/Anthropic/GitHub/Stripe/AWS keys, JWTs, Bearer tokens, IPs, etc.)
  • Set to [] β†’ disable redaction entirely
  • Set to ["/pattern1/", "/pattern2/"] β†’ use only your custom regex patterns

Typical workflow for your coding agent:

  • Add mcp_dump($anything) (optionally chain ->green(), ->label('...'), ->caller(), ->trace(), ->pass($value)) anywhere in templates/snippets/controllers.
  • Call kirby_render_page (it returns a traceId).
  • Call kirby_dump_log_tail(traceId=...) to retrieve the captured dump events for that render.

Configuration

Project config lives in .kirby-mcp/mcp.json (or .kirby-mcp/config.json) in the Kirby project root. It is created by vendor/bin/kirby-mcp install if missing.

Kirby host selection:

  • By default, Kirby CLI runs with no KIRBY_HOST override.
  • To use host-specific Kirby config, set KIRBY_MCP_HOST (or KIRBY_HOST) when starting the MCP server, or set kirby.host in .kirby-mcp/mcp.json:
    • {"kirby":{"host":"localhost"}}
Option Type Default Description
cache.ttlSeconds int 60 In-memory cache TTL (seconds) for read-only resources like kirby://commands and kirby://cli/command/{command} plus a few internal caches (roots inspection, completions); set to 0 to disable caching.
docs.ttlSeconds int 86400 In-memory cache TTL (seconds) for fetched getkirby.com markdown docs (e.g. kirby://field/{type} and kirby://section/{type}); set to 0 to disable caching.
cli.allow string[] [] Additional allowlist patterns for kirby_run_cli_command (supports * wildcard, e.g. plugin:*).
cli.allowWrite string[] [] Additional allowlist patterns for write-capable commands; requires allowWrite=true when calling kirby_run_cli_command (supports *).
cli.deny string[] [] Deny patterns that always block commands, even if allowlisted (supports *).
dumps.enabled bool true Enable/disable mcp_dump() writes to .kirby-mcp/dumps.jsonl.
dumps.maxBytes int 2097152 Max size for .kirby-mcp/dumps.jsonl written by mcp_dump(). When the next write would exceed it, the log is compacted by keeping the newest half of lines, then the new entry is appended.
dumps.secretPatterns string[] (defaults) Regex patterns for secret redaction in dump logs. Omit to use defaults (API keys, tokens, IPs, etc.), set to [] to disable masking, or provide custom patterns.
ide.typeHintScanBytes int 16384 Max bytes to read from controller/model files when detecting Kirby IDE baseline type hints (see kirby_ide_helpers_status).
kirby.host string null Default Kirby host to pass as KIRBY_HOST to the Kirby CLI (affects host-specific config like config.{host}.php).
eval.enabled bool false Enable kirby_eval / kirby mcp:eval (still requires explicit confirmation per call).

Environment variables:

Env var Description
KIRBY_MCP_PROJECT_ROOT Project root (overrides auto-detection).
KIRBY_MCP_KIRBY_BIN Path to vendor/bin/kirby (overrides binary resolution).
KIRBY_MCP_HOST / KIRBY_HOST Kirby host override (takes precedence over config).
KIRBY_MCP_DUMPS_ENABLED Override dumps.enabled (1/0, true/false, on/off).
KIRBY_MCP_ENABLE_EVAL Enable eval override (takes precedence over config; still needs confirmation).

Troubleshooting

  • β€œUnable to determine Kirby project root”: run from the Kirby project root or pass --project=/absolute/path (or set KIRBY_MCP_PROJECT_ROOT).
  • Runtime-only tools fail: run vendor/bin/kirby-mcp install and check kirby_runtime_status.
  • CLI command blocked: add patterns to .kirby-mcp/mcp.json (cli.allow / cli.allowWrite) or block with cli.deny.
  • Host-specific config not applied: set KIRBY_MCP_HOST/KIRBY_HOST or configure {"kirby":{"host":"..."}}.
  • Docs resources are slow/failing: confirm network access or adjust docs.ttlSeconds (set to 0 to disable caching).
  • No dump output: ensure dumps.enabled=true, a .kirby-mcp/dumps.jsonl exists, and use the correct traceId with kirby_dump_log_tail.

Development

  • Install deps: composer install
  • Run tests: composer test
  • Run static analysis: composer analyse

Disclaimer

This MCP server is provided "as is" with no guarantee. Use it at your own risk and always test it yourself before using it in a production environment. If you find any issues, please create a new issue.

License

MIT

It is discouraged to use this MCP server in any project that promotes racism, sexism, homophobia, animal abuse, violence or any other form of hate speech.