Skip to main content
Glama

Kanban Lite

A lightweight kanban board stored as plain markdown files. Manage your tasks via a web UI, CLI, REST API, or MCP server for AI agents. Everything is human-readable, version-controllable, and lives right in your project.

npm License

Kanban Board Overview

Kanban Skill

Install the kanban skill via skills.sh to give your AI agent full context of your board and the ability to create, update, and move features directly from the terminal. Works with Claude Code, Codex, OpenCode, and any skills.sh-compatible agent.

npx skills add https://github.com/borgius/kanban-lite

See SKILL.md for the full skill reference covering MCP tools, CLI, and REST API.

This repo also ships a dedicated kanban-storage-plugin-author skill for generating third-party card.storage / attachment.storage npm packages that can be selected from .kanban.json:

npx skills add https://github.com/borgius/kanban-lite --skill kanban-storage-plugin-author

See .agents/skills/kanban-storage-plugin-author/SKILL.md for the workflow, bundled references, and package templates.

This workspace also includes a custom VS Code chat agent for building polished Slidev demo decks from repo materials such as the Chat SDK / IncidentMind demo guide:

Related MCP server: issue-tracker-mcp

Quick Start

# Install globally
npm install -g kanban-lite

# Initialize a board in your project
kl init

# Start the web UI
kl serve

# Or create your first card from the terminal
kl add --title "My first task" --priority high

Examples

Start with the stable docs hub at /docs/examples/ for the shipped walkthroughs, then jump into the matching runnable apps:

See examples/README.md for the canonical top-level example app slugs, local install/run expectations, and the placeholder-only env-file convention. The Chat SDK / Vercel AI example now also ships a self-hosted local stack (npm run dev / npm run start print both the Kanban and chat URLs), seeds demo cards with comments/forms/actions, and keeps the chat route honest with live integration coverage when OPENAI_API_KEY is available. The examples stay self-contained outside the root pnpm workspace by default so the main repo build/watch flow remains unchanged.

Features

Web UI

  • Multi-board support: Create multiple boards with independent columns and settings

  • 5-column workflow: Backlog, To Do, In Progress, Review, Done (fully customizable per board)

  • Drag-and-drop: Move cards between columns and reorder within columns

  • Split-view editor: Board on left, inline CodeMirror markdown editor on right with the existing toolbar shortcuts preserved

  • CodeMirror comments editor: Card comments now use the same CodeMirror-backed markdown composer as the main body editor, with markdown toolbar actions, preview, and consistent keyboard shortcuts in both standalone and VS Code views

  • Comment author defaults: The comment composer now pre-fills the author field from the logged-in username when available, and otherwise falls back to User

  • Dynamic form tabs: Every attached card form renders as its own tab in the card editor, alongside the built-in markdown, comments, and logs tabs; fields display with consistent spacing and theme-aware styling in both standalone and VS Code webview runtimes

  • Meta tab — inline YAML metadata editor: The card editor includes a dedicated Meta tab for editing frontmatter.metadata as raw YAML inside a CodeMirror editor. In edit mode, valid changes autosave via the existing debounce pipeline; invalid YAML stays as local draft text with an inline error and never reaches storage. In the create-card dialog, metadata is staged locally and included in the card payload on Save; the Save button is disabled while any YAML errors remain unresolved.

  • Layout toggle: Switch between horizontal and vertical board layouts

  • Event-driven pub/sub: SDK events are dispatched through an EventEmitter2-based event bus with wildcard routing, powering webhooks, auth events, and custom subscriptions

  • Explicit SDK unread/card-state APIs: Advanced SDK consumers can inspect side-effect-free actor-scoped unread/open state with getCardState() / getUnreadSummary() and acknowledge it intentionally via markCardOpened() / markCardRead() without coupling unread semantics to setActiveCard() or the UI's active-card selection

  • SQLite card.state auto-derived from storage: When using an external storage plugin (e.g. sqlite, mongodb, postgresql, mysql, redis), card-state is automatically derived from the same storage package — no separate card.state configuration or package installation required

  • Real-time updates: WebSocket-powered live sync on the Node standalone server, plus an event-driven Cloudflare Worker mode that uses Durable Object WebSocket invalidations with HTTP latest-state resync between tabs

  • Light & dark mode support

  • Tabbed settings panel: Settings are organized into General, Board, and Plugin Options tabs, with board-level subviews for Defaults, Title, Actions, Labels, and Meta

  • Board settings deep links: Standalone browser routes now include /settings/board/defaults, /settings/board/title, /settings/board/actions, /settings/board/labels, and /settings/board/meta, with legacy board-setting URLs such as /settings/defaults and /settings/labels redirecting to the matching board subview, and the shared settings panel opens the matching left-rail board sub-tab for each route

  • Broader shared settings coverage: The shared settings UI now exposes showBuildWithAI, markdownEditorMode, and drawer position alongside the existing layout and display controls

  • Board title/action editors: Manage boards.<id>.title metadata prefixes and boards.<id>.actions toolbar actions directly from Board → Title and Board → Actions

  • Simplified metadata field builder: The Board → Meta settings view now uses a quieter form-and-list layout with stacked edit fields, duplicate-name validation, inline editing, and straightforward controls for deciding which metadata appears on card previews

  • Plugin Options tab: Discover providers by capability, flip provider toggles on/off with in-flight loading feedback, edit schema-driven options in dedicated sections after the capability list even before a provider is enabled, keep same-package provider variants separate by capability + providerId, reuse provider-authored JSON Forms uiSchema layouts for grouped sections, inline array editors, and code-enabled controls such as callback inline source, reopen stored secrets as masked write-only fields, and install supported kl-* packages from the UI

  • Flexible panel layouts: Open card details and creation flows as a right-side drawer or a centered popup

  • Adjustable drawer width: Tune drawer mode between 20–80% of the viewport from the Layout settings

  • Polished card detail view: Card details now open with a calmer desktop-first split layout, tighter control density, cleaner attachment/comment presentation, and refined popup/drawer styling in both drawer and popup modes

  • Custom board backgrounds: Choose between simple plain canvases and a richer library of fancy colorful ambient presets from the settings panel, with multiple light/dark-friendly moods for different boards and preferences

  • Zoom controls: Scale the board view and card detail panel independently between 75–150% via settings sliders or keyboard shortcuts

  • Column sorting: Sort cards within a column by priority, due date, or creation date from the column menu

  • Column visibility controls: Minimize a column into a narrow rail with its name and card count from the column menu, or hide/show columns from Board options → Columns

  • Smooth scroll to selection: Board automatically scrolls to the selected card

  • URL-synced standalone navigation: In standalone mode, the active board, card, tab, filters, search query, and fuzzy mode persist in browser history and deep links

  • Keyboard shortcuts:

    • N - Create new card

    • Esc - Close dialogs

    • Cmd/Ctrl + Enter - Submit create dialog

    • Ctrl/Cmd + = / Ctrl/Cmd + - - Zoom board view in / out

    • Ctrl/Cmd + Shift + = / Ctrl/Cmd + Shift + - - Zoom card detail in / out

Cards

Editor View

  • Priority levels: Critical, High, Medium, Low (color-coded badges)

  • Assignees: Assign team members to cards

  • Due dates: Smart formatting (Overdue, Today, Tomorrow, "5d", etc.)

  • Labels: Tag cards with multiple labels

  • Attachments: Attach files to cards

  • Comments: Add discussion threads to cards (stored in the same markdown file)

  • Streaming comments: AI agents can stream a comment live — a blinking-cursor indicator and streaming badge are shown to all connected viewers while text is being written; the comment is persisted once the stream completes (see REST API and sdk.streamComment)

  • Logs: Append timestamped log entries to cards (stored as <cardId>.log text file, supports markdown, optional source labels and structured data objects)

  • Actions: Attach named triggers to a card (e.g. retry, deploy, notify) and fire them from the UI, CLI, API, or MCP server — calls a configured webhook with the card's full context

  • Reusable and inline forms: Attach named workspace forms from .kanban.json or define card-local inline forms directly on the card

  • Per-form saved data: Each attached form persists its submitted payload separately under formData[formId], so multiple forms on one card do not collide; stored entries may be partial — the full canonical form state is prepared at runtime by merging config defaults, attachment defaults, persisted data, and metadata

  • ${path} placeholder interpolation: String values in config defaults, attachment defaults, and persisted form data are resolved against full card context (${id}, ${status}, ${assignee}, ${metadata.key}, etc.) before the form tab opens; unresolved placeholders become empty strings

  • Auto-generated IDs: Based on title and timestamp (e.g., implement-dark-mode-2026-01-29)

  • Timestamps: Created and modified dates tracked automatically

  • Toolbar search with optional fuzzy mode: Exact search is the default; enable the Fuzzy toggle in the web UI to match near-misses across card text and metadata values

  • Metadata token search: Use meta.field: value tokens for field-scoped searches that behave consistently across the web UI, CLI, REST API, and MCP

  • Removable search chips: Mixed searches are split into separate toolbar chips for plain text and each meta.* token, so you can remove one constraint without clearing the whole query

  • Metadata filter buttons: Click the filter icon next to rendered metadata values in the card detail panel to inject the correct meta.field: value token into the shared search box

  • Clickable label filters: Click a label on a board card or in the card detail panel to immediately filter the board to that label

  • Priority filter: Show only critical, high, medium, or low items

  • Assignee filter: Filter by team member or show unassigned items

  • Label filter: Filter by specific labels

  • Due date filters: Overdue, due today, due this week, or no due date

File Organization

  • Plain markdown: Cards are standard markdown files with YAML frontmatter

  • Status subfolders: Automatically organized into subfolders by status

  • Git-friendly: Everything is version-controllable

Installation

npm install -g kanban-lite

CLI

Manage your kanban board from the terminal using kanban-lite or the shorthand kl:

# List all cards
kl list

# List with filters
kl list --status todo --priority high

# Exact and fuzzy search
kl list --search "release meta.team: backend"
kl list --search "meta.team: backnd api plumbng" --fuzzy

# Combine fuzzy search with structured metadata filters
kl list --search "release" --fuzzy --meta sprint=Q1 --meta links.jira=PROJ-123

# Create a card
kl add --title "Implement search" --priority high --label "frontend,search"

# Create a card with actions
kl add --title "Deploy service" --actions "retry,rollback,notify"

# Create a card with attached forms
kl add --title "Investigate outage" --forms '[{"name":"incident-report"}]'

# Update form attachments or persisted per-form data
kl edit investigate-outage --forms @forms.json --form-data @form-data.json

# Show card details
kl show implement-search

# Show the currently active/open card
kl active

# Move to a different column
kl move implement-search in-progress

# Update fields
kl edit implement-search --assignee alice --due 2026-03-01

# Add actions to an existing card
kl edit deploy-service --actions "retry,rollback,notify"

# Trigger an action
kl action trigger deploy-service retry

# Submit a card form payload
kl form submit investigate-outage incident-report --data '{"severity":"high","owner":"alice"}'

# Delete a card
kl delete implement-search

# Attachments
kl attach implement-search                              # List attachments
kl attach add implement-search ./screenshot.png         # Attach a file
kl attach remove implement-search screenshot.png        # Remove attachment

# Comments
kl comment implement-search                             # List comments
kl comment add implement-search --author alice \
  --body "Looks good, needs tests"                      # Add a comment
kl comment stream implement-search --author agent       # Stream comment from stdin
echo "Analysis complete" | kl comment stream 42 --author ci  # Pipe output as comment
kl comment edit implement-search c1 --body "Updated"    # Edit a comment
kl comment remove implement-search c1                   # Remove a comment

# Logs
kl log list implement-search                             # List log entries
kl log add implement-search --text "Build passed"         # Add a log entry
kl log add implement-search --text "Deployed" \
  --source ci --object '{"version":"1.0"}'               # With source and data
kl log clear implement-search                            # Clear all logs

# Board Logs
kl board-log list                                        # List board-level log entries
kl board-log add --text "Deployment complete"            # Add a board log entry
kl board-log add --text "Pipeline passed" \
  --source ci --object '{"build":"42"}'                    # With source and data
kl board-log clear                                       # Clear all board logs

# Board Actions
kl board-actions list --board default                    # List board-level actions
kl board-actions add --board default \
  --key deploy --title "Deploy to Production"            # Add/update a board action
kl board-actions fire --board default deploy             # Trigger a board action

# Boards
kl boards                                               # List boards
kl boards add --id bugs --name "Bug Tracker"            # Create a board
kl boards show bugs                                     # Show board details
kl boards remove bugs                                   # Remove an empty board
kl boards default bugs                                  # Set default board
kl transfer card-42 --from default --to bugs            # Transfer card between boards

# Target a specific board (works with most commands)
kl list --board bugs                                    # List cards in a board
kl add --title "Login bug" --board bugs                 # Create card in a board

# Manage columns
kl columns                                              # List columns
kl columns add --id testing --name Testing              # Add column
kl columns update testing --color "#ff9900"             # Update column
kl columns remove testing                               # Remove column

# Webhooks
kl webhooks                                             # List webhooks
kl webhooks add --url https://example.com/hook          # Register webhook
kl webhooks add --url https://example.com/hook \
  --events task.created,task.moved --secret mykey       # With event filter and secret
kl webhooks update wh_abc123 --active false             # Disable a webhook
kl webhooks update wh_abc123 --events task.created      # Change subscribed events
kl webhooks remove wh_abc123                            # Remove webhook

# Settings
kl settings                                             # Show current settings
kl settings update --compactMode true                   # Update a setting

# Plugin settings
kl plugin-settings list                                # List capability-grouped providers
kl plugin-settings show auth.identity local            # Read one provider's redacted state
kl plugin-settings select callback.runtime callbacks   # Enable the callback runtime provider
kl plugin-settings select auth.identity local          # Select one provider for a capability
kl plugin-settings select webhook.delivery none        # Explicitly disable webhook runtime delivery
kl plugin-settings update-options auth.identity local \
  --options '{"apiToken":"••••••"}'                  # Persist provider options (masked secrets keep existing values)
kl plugin-settings install kl-plugin-auth --scope workspace  # Safe install into the workspace runtime

# Workspace
kl events                                               # List built-in and plugin-declared events
kl events --type before --mask "task.*"                # Filter by phase + wildcard mask
kl pwd                                                  # Print workspace root path

# Storage providers
kl storage status                                       # Show current provider + capability status
kl storage migrate-to-sqlite --sqlite-path .kanban/kanban.db  # Migrate to SQLite
kl storage migrate-to-markdown                          # Migrate back to markdown

# Card state / unread
kl card-state status                                    # Show active card.state backend + default actor contract / availability
kl card-state status <card-id>                          # Show side-effect-free actor-scoped unread/open summary (not active-card UI state)
kl card-state read <card-id>                            # Explicitly acknowledge unread activity for the current actor
kl card-state open <card-id>                            # Acknowledge unread and persist explicit actor-scoped open-card state

# Start web server
kl serve                                                # Start on port 3000
kl serve --port 8080 --no-browser                       # Custom port, no auto-open

# Start MCP server (stdio transport for AI agent integrations)
kl mcp                                                  # Auto-detect directory
kl mcp --dir .kanban                                    # Explicit directory

# Initialize features directory
kl init

# Other
kl version                                              # Print version
kl help                                                 # Show help
kl help sdk                                             # Show SDK documentation
kl help api                                             # Show REST API documentation

Use --json for machine-readable output. Use --dir <path> to specify a custom features directory. Use --board <id> to target a specific board.

--forms accepts a JSON array of attached form descriptors, and --form-data accepts a JSON object keyed by resolved form id. Both flags also support @path/to/file.json.

Standalone Server

Run the kanban board as a web application with a full REST API:

# Using the CLI
kl serve

# Or directly
kanban-md

# With options
kanban-md --port 8080 --dir .kanban --no-browser

The server provides:

  • Web UI at http://localhost:3000 — a full React-based kanban board

  • REST API at http://localhost:3000/api — full programmatic access

  • WebSocket — real-time updates for connected clients

In standalone/browser mode, the app automatically retries same-page backend reconnects when possible after the connection drops. If reconnect cannot be restored, the UI shows an in-app connection-lost error with guidance to refresh or reopen the standalone page.

REST API

All responses follow the format { "ok": true, "data": ... } or { "ok": false, "error": "message" }. CORS is enabled for all origins.

The REST API source of truth is the standalone Swagger/OpenAPI spec. Browse the interactive docs at http://localhost:3000/api/docs, the raw OpenAPI JSON at http://localhost:3000/api/docs/json, or the generated repo copy at docs/api.md.

Boards

Method

Endpoint

Description

GET

/api/boards

List all boards

POST

/api/boards

Create a board

GET

/api/boards/:boardId

Get board configuration

PUT

/api/boards/:boardId

Update board configuration

DELETE

/api/boards/:boardId

Delete an empty board

Board Actions

Method

Endpoint

Description

GET

/api/boards/:boardId/actions

List named board actions

POST

/api/boards/:boardId/actions

Replace the board action map

PUT

/api/boards/:boardId/actions/:key

Add or update a single board action title

DELETE

/api/boards/:boardId/actions/:key

Remove a named board action

POST

/api/boards/:boardId/actions/:key/trigger

Trigger a board action webhook event

Tasks

Method

Endpoint

Description

GET

/api/tasks

List all tasks (query: ?q=&fuzzy=&meta.<field>=&status=&priority=&assignee=&label=)

GET

/api/tasks/active

Get the currently active/open task

GET

/api/tasks/:id

Get a single task

POST

/api/tasks

Create a task, including optional forms and formData

PUT

/api/tasks/:id

Update task properties, including forms and formData

POST

/api/tasks/:id/forms/:formId/submit

Validate and submit a card form payload

PATCH

/api/tasks/:id/move

Move task to column/position

DELETE

/api/tasks/:id

Delete a task

Board-scoped equivalents are available at /api/boards/:boardId/tasks/..., including POST /api/boards/:boardId/tasks/:id/forms/:formId/submit.

q is the free-text search input, fuzzy=true enables typo-tolerant matching, and meta.<field>=value keeps metadata filtering field-scoped. The same search semantics are shared with kl list --search ... --fuzzy and the MCP list_cards tool.

Transfer

Method

Endpoint

Description

POST

/api/boards/:boardId/tasks/:id/transfer

Transfer a task to another board

Columns

Method

Endpoint

Description

GET

/api/columns

List all columns

POST

/api/columns

Add a column

PUT

/api/columns/:id

Update a column

DELETE

/api/columns/:id

Delete a column

Settings

Method

Endpoint

Description

GET

/api/settings

Get board settings

PUT

/api/settings

Update board settings

Plugins

Method

Endpoint

Description

GET

/api/plugin-settings

List capability-grouped plugin providers and selected-provider state

GET

/api/plugin-settings/:capability/:providerId

Read one provider's redacted options/state

PUT

/api/plugin-settings/:capability/:providerId/select

Persist the selected provider for a capability

PUT

/api/plugin-settings/:capability/:providerId/options

Persist provider options and return the redacted read model

POST

/api/plugin-settings/install

Run the guarded installer for exact unscoped kl-* package names only

Webhooks

Method

Endpoint

Description

GET

/api/webhooks

List registered webhooks

POST

/api/webhooks

Register a webhook

PUT

/api/webhooks/:id

Update a webhook

DELETE

/api/webhooks/:id

Remove a webhook

Workspace

Method

Endpoint

Description

GET

/api/workspace

Get workspace root path plus active storage, auth, and webhook provider metadata

GET

/api/events

List built-in and plugin-declared events (`?type=before

GET

/api/storage

Get current card, attachment, and webhook provider status

POST

/api/storage/migrate-to-sqlite

Migrate cards to SQLite ({ sqlitePath? })

POST

/api/storage/migrate-to-markdown

Migrate cards back to markdown files

Comments

Method

Endpoint

Description

GET

/api/tasks/:id/comments

List comments on a task

POST

/api/tasks/:id/comments

Add a comment ({ author, content })

POST

/api/tasks/:id/comments/stream

Stream a comment live — request body is plain-text stream; query param ?author=<name> required. Chunks are broadcast in real-time to connected viewers via WebSocket.

PUT

/api/tasks/:id/comments/:commentId

Update a comment ({ content })

DELETE

/api/tasks/:id/comments/:commentId

Delete a comment

Logs

Method

Endpoint

Description

GET

/api/tasks/:id/logs

List log entries on a card

POST

/api/tasks/:id/logs

Add a log entry ({ text, source?, object?, timestamp? })

DELETE

/api/tasks/:id/logs

Clear all log entries

Board Logs

Method

Endpoint

Description

GET

/api/boards/:boardId/logs

List board-level log entries

POST

/api/boards/:boardId/logs

Add a board log entry ({ text, source?, object?, timestamp? })

DELETE

/api/boards/:boardId/logs

Clear all board log entries

Attachments

Method

Endpoint

Description

POST

/api/tasks/:id/attachments

Upload attachment(s)

GET

/api/tasks/:id/attachments/:filename

Download an attachment

DELETE

/api/tasks/:id/attachments/:filename

Remove an attachment

Local MinIO attachment plugin setup

For local development in this repo, the published kl-plugin-attachment-s3 package is installed and can be used as the default attachment.storage provider against MinIO.

  • local runtime settings live in a workspace .env (including AWS_ACCESS_KEY_ID, AWS_SECRET_ACCESS_KEY, and optional AWS_REGION)

  • local provider selection lives in the workspace .kanban.json

  • the default local card provider stays on localfs (markdown engine)

Local defaults used by the repo:

  • endpoint: http://127.0.0.1:9000

  • console: http://127.0.0.1:9001

  • bucket: kanban-local

  • credentials: minioadmin / minioadmin

Try it locally:

npm run minio:up
npm run test:integration:minio

Note: the S3 attachment plugin still does not expose a stable local attachment directory, but card log files now flow through the attachment capability itself, so MinIO-backed log attachments work too.

For providers that support efficient in-place appends, attachment.storage plugins may also expose an optional append hook. The SDK uses that hook for card logs when available and falls back to a safe read/modify/write update when it is not.

Example: Create a task via API

curl -X POST http://localhost:3000/api/tasks \
  -H "Content-Type: application/json" \
  -d '{
    "content": "# Investigate outage\n\nCollect incident details.",
    "status": "todo",
    "priority": "high",
    "forms": [{"name": "incident-report"}],
    "formData": {"incident-report": {"service": "billing"}}
  }'

Example: Update a task's attached forms via API

curl -X PUT http://localhost:3000/api/tasks/investigate-outage \
  -H "Content-Type: application/json" \
  -d '{
    "forms": [
      {"name": "incident-report"},
      {
        "schema": {
          "type": "object",
          "title": "Postmortem",
          "properties": {
            "summary": {"type": "string"}
          }
        },
        "data": {"summary": "Initial draft"}
      }
    ]
  }'

Example: Submit a task form via API

curl -X POST http://localhost:3000/api/tasks/investigate-outage/forms/incident-report/submit \
  -H "Content-Type: application/json" \
  -d '{
    "data": {
      "severity": "critical",
      "owner": "alice"
    }
  }'

Example: Search tasks via API

curl "http://localhost:3000/api/tasks?q=release&fuzzy=true&meta.team=backend"
curl "http://localhost:3000/api/boards/bugs/tasks?q=meta.team%3A%20backnd&fuzzy=true&meta.region=us-east"

Webhooks

Register webhooks to receive HTTP POST notifications when data changes. Webhooks fire from all interfaces — REST API, CLI, MCP server, and the web UI — ensuring consistent event delivery regardless of how a mutation is triggered.

See the full Webhooks documentation for detailed event payloads, signature verification, and delivery behavior.

Install and compatibility

Webhook runtime behavior is now owned by the external kl-plugin-webhook package wherever the host already exposes a plugin seam. That package owns the webhook.delivery provider, the listener-only runtime subscriber, the standalone /api/webhooks routes, the kl webhooks CLI command surface, and the webhook MCP tools registered through the narrow mcpPlugin seam.

The SDK owns the mutation lifecycle: it awaits before-event listeners before a write, emits after-events exactly once after commit, and webhook delivery listens only to that committed after-event phase.

For advanced SDK consumers, the same package also exports an additive SDK extension bag discoverable through sdk.getExtension('kl-plugin-webhook'). Core KanbanSDK webhook methods remain stable compatibility shims over that same provider-backed implementation.

  • Install it in the same environment that runs Kanban Lite (CLI, standalone server, MCP server, extension host, or SDK consumer).

  • Existing .kanban.json webhook registrations stay in the top-level webhooks array as a compatibility fallback; no migration is required.

  • A workspace that only sets plugins["webhook.delivery"] still activates webhook plugin discovery for the provider, standalone routes, CLI command loading, and MCP tool registration.

  • MCP now uses the same active-package discovery model as CLI and standalone. kl-plugin-webhook registers list_webhooks, add_webhook, update_webhook, and remove_webhook through the plugin seam while preserving their public names, schemas, auth wrapping, and secret redaction behavior.

  • Generated docs such as docs/webhooks.md are source-driven; update generator metadata and regenerate them instead of editing the generated markdown by hand.

npm install kl-plugin-webhook

For local sibling-repo development, a checkout at ../kl-plugin-webhook is resolved automatically. npm link ../kl-plugin-webhook is optional, but still useful when you want an explicit local package link.

{
  "plugins": {
    "webhook.delivery": {
      "provider": "webhooks",
      "options": {
        "webhooks": [
          {
            "id": "wh_a1b2c3d4e5f67890",
            "url": "https://example.com/hook",
            "events": ["task.created", "task.updated"],
            "active": true
          }
        ]
      }
    }
  }
}

Webhook management still converges on the same SDK methods, but ownership is now split by host surface:

  • SDK: additive extension path via sdk.getExtension('kl-plugin-webhook'), plus stable compatibility shims on sdk.listWebhooks(), sdk.createWebhook(), sdk.updateWebhook(), sdk.deleteWebhook(), and sdk.getWebhookStatus()

  • REST API: plugin-owned /api/webhooks routes via the standalone HTTP plugin seam

  • CLI: plugin-owned kl webhooks, kl webhooks add, kl webhooks update, kl webhooks remove

  • MCP: plugin-owned list_webhooks, add_webhook, update_webhook, remove_webhook via the mcpPlugin registration seam

Events

Event

Trigger

task.created

A new task is created

form.submit

A card form payload is validated, persisted, and submitted

task.updated

Task properties are changed

task.moved

Task is moved to a different column or transferred between boards

task.deleted

A task is deleted

comment.created

A comment is added to a task

comment.updated

A comment is edited

comment.deleted

A comment is removed

column.created

A new column is added

column.updated

Column name or color is changed

column.deleted

A column is removed

attachment.added

A file is attached to a task

attachment.removed

An attachment is removed from a task

settings.updated

Board display settings are changed

board.created

A new board is created

board.updated

Board configuration is changed

board.deleted

A board is deleted

board.action

A board-level action is triggered from the toolbar, CLI, REST API, or MCP

card.action.triggered

A card action is triggered

board.log.added

A board log entry is appended

board.log.cleared

Board log entries are cleared

log.added

A card log entry is appended

log.cleared

Card log entries are cleared

storage.migrated

Card storage is migrated between providers

form.submitted

A card form payload is validated, persisted, and submitted

Payload

{
  "event": "task.created",
  "timestamp": "2026-02-21T10:30:00.000Z",
  "data": { "id": "my-task", "status": "todo", "priority": "high", "..." : "..." }
}

Headers

  • Content-Type: application/json

  • X-Webhook-Event: task.created

  • X-Webhook-Signature: sha256=<hmac> (if a secret is configured)

  • Authorization: Bearer <token> (when KANBAN_LITE_TOKEN is set in the webhook runtime)

Manage via CLI or API

# Register
kl webhooks add --url https://example.com/hook --events task.created,task.moved --secret mykey

# Update
kl webhooks update wh_abc123 --active false
kl webhooks update wh_abc123 --events task.created,task.deleted

# API
curl -X POST http://localhost:3000/api/webhooks \
  -H "Content-Type: application/json" \
  -d '{"url": "https://example.com/hook", "events": ["task.created", "task.moved"], "secret": "mykey"}'

Webhook registrations are stored in .kanban.json at the workspace root and persist across server restarts.

Subscriptions support exact event names, *, and prefix wildcards ending in .* such as task.* or board.log.*. Only committed SDK after-events are delivered; before-events such as form.submit stay internal to the SDK lifecycle.

Card Forms

Forms let you attach structured JSON Schema-driven workflows directly to cards. Each attachment becomes its own tab in the webview card editor, is validated with the SDK-owned rules, and can be submitted from the UI, CLI, REST API, SDK, or MCP.

Define reusable workspace forms

Declare named reusable forms in .kanban.json under the top-level forms map:

{
  "forms": {
    "incident-report": {
      "schema": {
        "type": "object",
        "title": "Incident Report",
        "properties": {
          "severity": { "type": "string", "enum": ["low", "medium", "high", "critical"] },
          "owner": { "type": "string" },
          "service": { "type": "string" }
        },
        "required": ["severity", "owner"]
      },
      "ui": {
        "type": "VerticalLayout",
        "elements": [
          { "type": "Control", "scope": "#/properties/severity" },
          { "type": "Control", "scope": "#/properties/owner" },
          { "type": "Control", "scope": "#/properties/service" }
        ]
      },
      "data": {
        "severity": "medium"
      }
    }
  }
}

Attach forms to cards

Cards can attach reusable named forms or inline card-local definitions in frontmatter:

forms:
  - name: incident-report
  - schema:
      type: object
      title: Postmortem
      properties:
        summary:
          type: string
    data:
      summary: Initial draft
formData:
  incident-report:
    service: billing

Merge order and submit behavior

For every attached form, the resolved payload is built in this order, from lowest to highest priority:

  1. Workspace form defaults from .kanban.json (forms.<name>.data)

  2. Card-scoped attachment defaults, then previously persisted formData[formId]

  3. Card metadata fields whose keys exist in the form schema

  4. The submitted payload

The SDK validates that final payload before persistence and before any form.submit webhook fires. After a successful submission, Kanban Lite also appends a card log entry containing the submitted payload for traceability.

Webview behavior

  • Every attached form renders as an extra tab in the card editor.

  • Tabs are stable and deep-linkable in standalone mode using form:<resolved-id> tab ids.

  • Tab labels use the resolved display name and render as form: <Form Name>.

  • Shared config forms show a Shared badge in the tab content.

  • Shared config forms may define name (default: capitalized form key) and description (default: empty string) in .kanban.json.

  • Submit stays disabled until validation passes, and successful submissions update the card's saved formData.

  • Successful submissions also append a system card log whose JSON object includes formId, formName, and the submitted payload.

Programmatic submission surfaces

  • SDK: await sdk.submitForm({ cardId, formId, data, boardId? })

  • CLI: kl form submit <cardId> <formId> --data '<json|@file>'

  • REST API: POST /api/tasks/:id/forms/:formId/submit

  • Board-scoped REST API: POST /api/boards/:boardId/tasks/:id/forms/:formId/submit

  • MCP: submit_card_form

Card Actions

Actions let you attach named triggers to a card — things like retry, deploy, rollback, or notify. When triggered, the system sends an HTTP POST to a single global action webhook URL configured in .kanban.json, with the action name and the full card context. Your webhook handles the actual work.

How it works

  1. Configure the webhook URL in .kanban.json:

{
  "actionWebhookUrl": "https://example.com/kanban-actions"
}
  1. Add actions to a card — either as an array of action keys, or an object mapping action keys to display titles:

# Array form — action key is used as the button label
actions: ["retry", "rollback", "notify-slack"]

# Object form — keys are the action names sent to the webhook, values are the UI labels
actions:
  retry: "Retry deployment"
  rollback: "Roll back to v1"
  notify-slack: "Notify Slack"
  1. Trigger an action from any interface:

  • UI: Open a card in the editor — a "Run Action" dropdown appears in the header when the card has actions. Click to select and fire.

  • CLI: kl action trigger <cardId> <action>

  • REST API: POST /api/tasks/:id/actions/:action

  • SDK: await sdk.triggerAction(cardId, action)

  • MCP: trigger_action tool

Webhook payload

{
  "action": "notify-slack",
  "board": "default",
  "list": "in-progress",
  "card": {
    "id": "42-deploy-v2",
    "status": "in-progress",
    "priority": "high",
    "assignee": "alice",
    "labels": ["deploy"],
    "content": "# Deploy v2.0\n...",
    "actions": ["retry", "rollback", "notify-slack"]  // or {"retry": "Retry deployment", ...}
  }
}

The webhook receives the full card object (same shape as the SDK Card type, minus filePath). Your server responds with any 2xx status to acknowledge; non-2xx responses are treated as errors.

Managing actions

Actions are stored in the actions field of the card's YAML frontmatter — either as a plain string array or as a key→title object. Edit them directly in the markdown file, or use any interface:

# Add actions when creating a card
kl add --title "Deploy service" --priority high --actions "retry,rollback,notify-slack"

# Update actions on an existing card
kl edit 42 --actions "retry,rollback,notify-slack,promote"

# Trigger
kl action trigger 42 notify-slack
# Via REST API
curl -X POST http://localhost:3000/api/tasks/42/actions/notify-slack

# Create card with actions via API
curl -X POST http://localhost:3000/api/tasks \
  -H "Content-Type: application/json" \
  -d '{"content": "# Deploy v2.0", "status": "todo", "actions": ["retry", "rollback"]}'

Notes

  • There is one global webhook URL for all actions across all boards. The action field in the payload tells your server which action was triggered; the board and list fields provide context.

  • If actionWebhookUrl is not set, triggering an action returns an error.

  • Actions are fire-and-forget — no retry logic or delivery guarantees are built in. Implement idempotency and retries in your own webhook handler if needed.

  • Action strings have no special meaning to Kanban Lite; you define the vocabulary.

Board Actions

Board actions are the board-level sibling to card actions. Define them once per board, and they appear in the toolbar Actions dropdown for that board.

If you prefer not to edit raw JSON, the shared settings panel now exposes the same per-board action list under Board → Actions, including stable action keys and user-facing titles.

How it works

  1. Define actions on the board in .kanban.json:

{
  "boards": {
    "default": {
      "name": "Default Board",
      "actions": {
        "deploy": "Deploy to Production",
        "announce": "Post release update"
      }
    }
  }
}
  1. Trigger them from any interface:

  • UI: Use the toolbar Actions dropdown on a board that defines actions.

  • CLI: kl board-actions fire --board <boardId> <key>

  • REST API: POST /api/boards/:boardId/actions/:key/trigger

  • MCP: trigger_board_action

  1. Receive a webhook event: triggering a board action emits a board.action event containing the board ID, action key, and display title.

Managing board actions

# List existing actions
kl board-actions list --board default

# Add or update a named action
kl board-actions add --board default --key deploy --title "Deploy to Production"

# Remove an action
kl board-actions remove --board default deploy

# Trigger an action
kl board-actions fire --board default deploy

n8n Integration

Kanban Lite ships a first-party n8n package at packages/n8n-nodes-kanban-lite, published as n8n-nodes-kanban-lite. It adds two nodes:

  • Kanban Lite — app node for boards, cards, columns, comments, attachments, labels, settings, storage, forms, webhooks, workspace info, and auth status

  • Kanban Lite Trigger — trigger node for transport-aware Kanban Lite events

Both nodes support two connection modes:

Mode

Best for

Requirements

Remote API

Connecting n8n to a running standalone Kanban Lite server

Configure the Kanban Lite API credential with server URL and auth settings

Local SDK

Running n8n on the same machine as the workspace for direct local access

Configure the Kanban Lite SDK (Local) credential, make the workspace path accessible to n8n, and install kanban-lite in the n8n runtime so the node can load kanban-lite/sdk

Trigger parity is intentionally asymmetric:

  • Local SDK mode can subscribe to both before-events and after-events from the canonical SDK event catalog

  • Remote API mode receives after-events only through webhook delivery from the standalone server

  • Remote API trigger mode requires a reachable n8n webhook URL so Kanban Lite can POST event deliveries back into the workflow

In practice, that means SDK mode can observe pre-commit events such as card.create, column.reorder, storage.migrate, or form.submit, while both modes can observe committed events such as task.created, task.updated, board.created, comment.updated, attachment.added, form.submitted, storage.migrated, and auth.allowed / auth.denied.

See packages/n8n-nodes-kanban-lite/README.md for installation steps, credential setup, action coverage, and trigger examples.

CrewAI Integration

Kanban Lite ships a first-party Python package at packages/kl-adapter-crewai, published as kl-adapter-crewai. It wraps kanban-lite REST API operations as CrewAI BaseTool subclasses so specialized agents (PM, Dev, QA) can each manage their own board lane.

pip install kl-adapter-crewai
from crewai import Agent
from kl_adapter_crewai import KanbanLiteClient, KanbanLiteToolkit

client = KanbanLiteClient("http://localhost:3000")
toolkit = KanbanLiteToolkit(client=client)

pm_agent = Agent(
    role="Project Manager",
    goal="Keep the board organized",
    backstory="You are a seasoned PM.",
    tools=toolkit.get_tools(),
)

9 tools are available: list_cards, get_card, create_card, update_card, move_card, delete_card, list_columns, get_comments, add_comment. Use toolkit.get_tools(read_only=True) for report-only agents.

See packages/kl-adapter-crewai/README.md for full setup, multi-agent examples, and authentication configuration.

LangChain / LangGraph Integration

Kanban Lite ships a first-party LangChain adapter at packages/kl-adapter-langchain, published as kl-adapter-langchain. It exposes all kanban-lite features as 39 LangChain StructuredTool instances — including streaming comments, labels, actions, logs, and attachments.

npm install kl-adapter-langchain @langchain/core
# optional – for LangGraph state/nodes:
npm install @langchain/langgraph

Quick Start

import { KanbanSDK } from 'kanban-lite/sdk'
import { createKanbanToolkit } from 'kl-adapter-langchain'

const sdk = new KanbanSDK('/path/to/.kanban')
await sdk.init()

const tools = createKanbanToolkit(sdk)
// Pass `tools` to any LangChain agent or LangGraph ToolNode

LangGraph Support

Optional LangGraph helpers (getKanbanBoardState, createRefreshBoardNode, createKanbanToolNode) provide a board-state annotation and pre-built graph nodes for stateful agent workflows.

Streaming Comments

The streamCommentDirect helper enables true chunk-by-chunk comment streaming from an AsyncIterable<string> (e.g. an LLM textStream), with onStart / onChunk callbacks for live WebSocket broadcast.

See packages/kl-adapter-langchain/README.md for detailed usage, selective tool loading, streaming examples, and LangGraph integration guide.

MCP Server

Expose your kanban board to AI agents (Claude, Cursor, etc.) via the Model Context Protocol.

Setup with Claude Code

Add to your .claude/settings.json:

{
  "mcpServers": {
    "kanban": {
      "type": "stdio",
      "command": "npx",
      "args": ["kanban-lite", "mcp"],
      "env": {
        "KANBAN_DIR": "/path/to/your/project/.kanban"
      }
    }
  }
}

Or run directly:

kl mcp                          # Auto-detect directory
kl mcp --dir .kanban            # Explicit directory
kanban-mcp --dir .kanban        # Via dedicated binary

Available Tools

Tool

Description

list_boards

List all boards in the workspace

create_board

Create a new board with optional custom columns

get_board

Get board configuration and details

delete_board

Delete an empty board

list_board_actions

List all named actions defined on a board

add_board_action

Add or update a named board action

remove_board_action

Remove a named board action

trigger_board_action

Trigger a board action webhook event

transfer_card

Move a card from one board to another

list_cards

List/filter cards by status, priority, assignee, label, searchQuery, fuzzy, and metaFilter

get_card

Get full details of a card (supports partial ID matching)

get_card_state_status

Get the active card.state provider status for the workspace

get_card_state

Get the side-effect-free unread/open summary for one card

open_card

Explicitly acknowledge unread activity and persist actor-scoped open-card state

read_card

Explicitly acknowledge unread activity without changing open-card state

get_active_card

Get the currently active/open card, or null if none is active

create_card

Create a new card with title, body, status, priority, metadata, forms, and formData

update_card

Update fields of an existing card, including forms and formData

submit_card_form

Validate and submit a card form payload through the shared SDK workflow

move_card

Move a card to a different status column

delete_card

Permanently delete a card

trigger_action

Trigger a named action on a card, calling the configured action webhook

list_attachments

List attachments on a card

add_attachment

Attach a file to a card (copies to card directory)

remove_attachment

Remove an attachment reference from a card

list_comments

List comments on a card

add_comment

Add a comment to a card

stream_comment

Stream a comment to a card (content is delivered through the streaming path so connected viewers see it live)

update_comment

Edit a comment's content

delete_comment

Remove a comment from a card

list_logs

List log entries on a card

add_log

Add a log entry to a card

clear_logs

Clear all log entries from a card

list_board_logs

List board-level log entries

add_board_log

Add a board-level log entry

clear_board_logs

Clear all board-level log entries

list_columns

List all board columns

add_column

Add a new column to the board

update_column

Update a column's name or color

remove_column

Remove a column (must be empty)

list_available_events

List built-in and plugin-declared SDK events with optional phase/mask filters

get_settings

Get board display settings

update_settings

Update board display settings

list_webhooks

List registered webhooks

add_webhook

Register a new webhook

update_webhook

Update a webhook (url, events, secret, active)

remove_webhook

Remove a webhook

get_workspace_info

Get workspace root path, kanban directory, and active storage provider metadata

For agent-driven search, pass searchQuery for free text (including inline tokens like meta.team: backend), set fuzzy: true to widen matching across text and metadata values, or use metaFilter when you want structured dot-notation field filters. | get_storage_status | Get current card/attachment storage provider status | | migrate_to_sqlite | Migrate all card data from markdown to SQLite | | migrate_to_markdown | Migrate all card data from SQLite back to markdown files |

All card, column, comment, and attachment tools accept an optional boardId parameter to target a specific board.

SDK

Use the kanban SDK programmatically in your own tools. The KanbanSDK class is the single source of truth — the CLI, MCP server, VSCode extension, and standalone server all delegate to it.

import { KanbanSDK } from 'kanban-lite/sdk'

const sdk = new KanbanSDK('/path/to/.kanban')

// Boards
const boards = sdk.listBoards()
sdk.createBoard('bugs', 'Bug Tracker', { description: 'Track production bugs' })
await sdk.transferCard('42', 'default', 'bugs')
await sdk.deleteBoard('bugs')

// Cards (all accept optional boardId as last argument)
const cards = await sdk.listCards()
const activeCard = await sdk.getActiveCard()
const card = await sdk.createCard({ content: '# My Task', status: 'todo', priority: 'high' })
await sdk.moveCard(card.id, 'in-progress')
await sdk.updateCard(card.id, { assignee: 'alice' })
await sdk.deleteCard(card.id)

// Cards with actions
const deployCard = await sdk.createCard({
  content: '# Deploy v2.0',
  status: 'todo',
  priority: 'high',
  actions: ['retry', 'rollback', 'notify-slack']
})

// Cards with attached forms
const incidentCard = await sdk.createCard({
  content: '# Investigate outage',
  forms: [{ name: 'incident-report' }],
  formData: { 'incident-report': { service: 'billing' } },
})

// Submit a form payload with SDK-owned validation + persistence
await sdk.submitForm({
  cardId: incidentCard.id,
  formId: 'incident-report',
  data: { severity: 'critical', owner: 'alice' },
})

// Trigger an action (POSTs to the actionWebhookUrl in .kanban.json)
await sdk.triggerAction(deployCard.id, 'notify-slack')

// Discover built-in and plugin-declared events
const allEvents = sdk.listAvailableEvents()
const taskBeforeEvents = sdk.listAvailableEvents({ type: 'before', mask: 'task.*' })

// Add/replace actions on an existing card
await sdk.updateCard(deployCard.id, { actions: ['retry', 'rollback', 'notify-slack', 'promote'] })

// Comments
await sdk.addComment('card-id', 'alice', 'Looks good!')
await sdk.updateComment('card-id', 'c1', 'Updated')
await sdk.deleteComment('card-id', 'c1')

// Attachments
await sdk.addAttachment('card-id', '/path/to/file.png')
await sdk.removeAttachment('card-id', 'file.png')

// Columns
const columns = sdk.listColumns()
sdk.addColumn({ id: 'testing', name: 'Testing', color: '#ff9900' })
sdk.updateColumn('testing', { name: 'QA' })
await sdk.removeColumn('testing')

// Settings
const settings = sdk.getSettings()
sdk.updateSettings({ ...settings, compactMode: true })

Plugins can also contribute additive SDK methods without patching core. Call sdk.getExtension(id) to access a plugin's extension bag when that package is active. Webhooks are the first concrete example: kl-plugin-webhook contributes webhook CRUD through sdk.getExtension('kl-plugin-webhook'), while the direct sdk.listWebhooks() / createWebhook() / updateWebhook() / deleteWebhook() methods remain compatibility shims for existing callers.

Plugins may also declare discoverable event names through their sdkExtensionPlugin.events catalog. sdk.listAvailableEvents() merges those declarations with the built-in before/after event catalog and supports the same dotted wildcard masks used by the shared event bus.

When a host surface passes sdk into a plugin context, that value is now the full public KanbanSDK instance rather than a narrowed helper facade. Plugin code can call the same public methods core uses — for example sdk.getBoard(...), sdk.getExtension(...), and sdk.getConfigSnapshot() for a cloned read-only view of the current .kanban.json state. Prefer SDK methods and snapshot reads wherever an equivalent public API exists; keep direct plugin-owned writes only for flows that still lack a public SDK writer.

Event Bus

The SDK exposes a pub/sub event bus for custom subscriptions, and it now owns the full mutation lifecycle.

  • Before-events are awaited in registration order and may either return plain-object overrides (immutably deep-merged over a cloned input, with the original input preserved when no listener changes it) or throw to veto the write.

  • After-events fire exactly once after commit and stay non-blocking so side-effect listeners do not affect the caller.

  • The legacy onEvent callback still works as a compatibility subscription hook for SDK consumers, but runtime plugins should use the listener-only register() / unregister() contract.

You can subscribe either through KanbanSDK convenience proxies or directly through sdk.eventBus for advanced workflows. Both paths support wildcard matching and typed event envelopes:

// Subscribe to all task events directly on the SDK
const unsub = sdk.on('task.*', (event) => {
  console.log(event.type, event.data, event.timestamp)
})

// Wait for the next matching event once
const nextCreate = sdk.waitFor('task.created', { timeout: 1000 })

// Subscribe to auth events
sdk.once('auth.denied', (event) => {
  console.warn('Access denied:', event.actor, event.meta)
})

// Advanced access is still available on the shared EventBus instance
sdk.eventBus.onAny((name, event) => {
  console.log('saw event', name, event.timestamp)
})

// Clean up when done
unsub()
sdk.destroy()

Convenience methods available on KanbanSDK: on, once, many, onAny, off, offAny, waitFor, eventNames, listenerCount, hasListeners, and removeAllListeners.

See the full SDK documentation for detailed API reference, types, error handling, and file layout.

Data Storage

By default, cards are stored as markdown files with YAML frontmatter in .kanban/boards/<boardId>/ within your project:

---
id: "implement-dark-mode-toggle-2026-01-25"
status: "todo"
priority: "high"
assignee: "john"
dueDate: "2026-01-25"
created: "2026-01-25T10:30:00.000Z"
modified: "2026-01-25T14:20:00.000Z"
labels: ["feature", "ui"]
actions: ["retry", "notify-slack"]
order: 0
---

# Implement dark mode toggle

Add a toggle in settings to switch between light and dark themes...

---
comment: true
id: "c1"
author: "alice"
created: "2026-01-25T15:00:00.000Z"
---
Should we support system preference detection too?

---
comment: true
id: "c2"
author: "john"
created: "2026-01-25T15:30:00.000Z"
---
Yes, good idea. I'll add that as a follow-up.

Comments are stored as additional YAML documents in the same file, keeping everything in one place and version-controllable.

When you switch the card.storage provider to sqlite or mysql, card/comment persistence moves behind that provider while .kanban.json remains the source of truth for board config, columns, labels, settings, forms, and webhooks.

For first-party storage plugins, attachment.storage now follows the active card.storage provider automatically and reuses the same provider options. Configure attachment.storage explicitly only when you want a different attachment provider such as S3.

Plugin Settings

Kanban Lite now exposes one shared plugin-settings workflow across the Settings panel, CLI, REST API, and MCP surfaces.

  • Capability-grouped inventory: the Plugin Options tab groups providers by capability such as card.storage, attachment.storage, card.state, callback.runtime, auth.identity, auth.policy, and webhook.delivery.

  • Storage-backed attachment.storage reuse: when attachments come from the active storage plugin (sqlite, mongodb, postgresql, mysql, redis), Plugin Options reuses the same provider/database settings as card.storage instead of showing a second DB configuration form.

  • Storage-backed card.state reuse: when card.state comes from the active storage plugin (sqlite, mongodb, postgresql, mysql, redis), Plugin Options reuses the same provider/database settings as card.storage instead of showing a second DB configuration form.

  • Selected-provider semantics: enablement is represented only by the selected provider stored under plugins[capability] in .kanban.json; there is no separate enabled boolean. The UI now uses per-provider on/off toggles, and webhook.delivery may be explicitly disabled with provider: "none" while preserving stored options for later re-enable.

  • Discovery metadata: every provider row carries its package name and discovery source (builtin, workspace, dependency, global, or sibling) so you can tell why it is available in the current runtime.

  • Schema-driven configuration: when a provider exports optionsSchema(), the UI renders provider options in dedicated sections after the capability list through the same JSON Forms stack used elsewhere in the app instead of bespoke per-provider forms. Schema-backed providers remain editable even while toggled off; inactive-provider saves are cached under pluginOptions[capability][providerId] and restored into plugins[capability] when that provider is enabled later. Providers may also supply a matching uiSchema so nested arrays and object-heavy settings render with explicit groups, detail editors, and conditional rules instead of the generic fallback layout. Plugin settings discovery resolves sync/async schema metadata before it reaches JSON Forms, so provider authors may derive enum lists or other schema values from the active SDK runtime.

  • Callback runtime: the first-party kl-plugin-callback package uses the same schema-driven path at plugins["callback.runtime"]. Its options.handlers[] payload is one ordered mixed list for inline and process handlers, and the inline source field now renders in an embedded CodeMirror JavaScript editor inside the shared Plugin Options flow.

  • Masked secret behavior: read/list surfaces return redacted option payloads only. Persisted secret fields reopen as masked write-only placeholders (••••••); leave the masked value unchanged to keep the current secret, or type a new value to replace it.

  • Guarded installs: in-product installs accept only exact unscoped kl-* package names plus an explicit workspace or global scope. They always run with lifecycle scripts disabled, reject version specifiers / flags / URLs / paths / shell fragments, and surface only redacted diagnostics.

The same nouns are used everywhere:

  • CLI: kl plugin-settings <list|show|select|update-options|install>

  • REST API: /api/plugin-settings, /api/plugin-settings/:capability/:providerId, /select, /options, and /install

  • MCP: list_plugin_settings, select_plugin_settings_provider, update_plugin_settings_options, and install_plugin_settings_package

For the deeper runtime model, provider discovery rules, and plugin authoring details, see docs/plugins.md.

Storage Providers

Kanban Lite resolves storage by capability namespace. When no explicit config is present, it defaults to:

  • card.storagelocalfs

  • attachment.storage → follows card.storage (localfs by default)

Legacy .kanban.json fields still work and are normalized into the same runtime capability map:

{
  "storageEngine": "sqlite",
  "sqlitePath": ".kanban/kanban.db"
}

The equivalent capability-based config is:

{
  "plugins": {
    "card.storage": {
      "provider": "sqlite",
      "options": {
        "sqlitePath": ".kanban/kanban.db"
      }
    }
  }
}

If both forms are present, plugins[namespace] wins for that namespace and legacy fields remain as compatibility aliases.

Capability

Default

Core providers / compatibility ids

Notes

card.storage

localfs

localfs, sqlite, mysql, postgresql, mongodb, redis

Core owns localfs (markdown engine). Legacy markdown input is normalized to localfs. sqlite, mysql, postgresql, mongodb, and redis are compatibility ids that resolve to kl-plugin-storage-sqlite, kl-plugin-storage-mysql, kl-plugin-storage-postgresql, kl-plugin-storage-mongodb, and kl-plugin-storage-redis.

attachment.storage

follows card.storage

localfs, sqlite, mysql, postgresql, mongodb, redis

Core owns localfs. For first-party storage plugins, omitted or redundant matching attachment.storage config is auto-derived from the active card.storage provider and reuses its options. Configure this namespace only when you want a different provider such as kl-plugin-attachment-s3.

For first-party storage plugins, selecting card.storage is enough to activate the matching attachment handler with the same options:

{
  "plugins": {
    "card.storage": {
      "provider": "sqlite",
      "options": {
        "sqlitePath": ".kanban/kanban.db"
      }
    }
  }
}

If you want a different attachment backend, configure attachment.storage explicitly. For example, keep MySQL cards but move attachments to S3:

{
  "plugins": {
    "card.storage": {
      "provider": "mysql",
      "options": {
        "host": "localhost",
        "port": 3306,
        "user": "kanban",
        "password": "secret",
        "database": "kanban_db"
      }
    },
    "attachment.storage": {
      "provider": "kl-plugin-attachment-s3"
    }
  }
}

Installing and selecting providers

  • Core built-ins are localfs (card.storage) and localfs (attachment.storage).

  • Legacy card.storage: "markdown" values are normalized to localfs.

  • sqlite, mysql, postgresql, mongodb, and redis remain valid provider ids, but they resolve to the external packages kl-plugin-storage-sqlite, kl-plugin-storage-mysql, kl-plugin-storage-postgresql, kl-plugin-storage-mongodb, and kl-plugin-storage-redis.

  • External providers are resolved by npm package name at runtime from the environment running the CLI, standalone server, MCP server, extension host, or the published ESM SDK build. Install them in that environment before selecting them in .kanban.json.

  • Missing plugin packages fail with an actionable install hint (for example npm install <package>).

  • This repository also contains a developer-facing example/scaffold external attachment provider at tmp/kl-plugin-attachment-s3 for S3-compatible object stores. It is a separate package workspace, not a built-in kanban-lite provider.

Plugin package manifest

Every first-party plugin package exports a pluginManifest constant that declares its capabilities and optional integration surfaces. The engine uses this manifest for fast discovery:

export const pluginManifest = {
  id: 'kl-plugin-storage-sqlite',
  capabilities: {
    'card.storage': ['sqlite'],
    'attachment.storage': ['sqlite'],
    'card.state': ['sqlite'],
  },
} as const

The KLPluginPackageManifest and PluginIntegrationNamespace types are exported from kanban-lite/sdk for compile-time validation. Plugin packages should also import shared runtime contracts (for example CardStoragePlugin, AttachmentStoragePlugin, WebhookProviderPlugin, StandaloneHttpPlugin, KanbanCliPlugin, and plugin-settings schema metadata types) from kanban-lite/sdk instead of re-declaring local structural copies. Third-party plugins without pluginManifest still work via legacy probing.

Webhook delivery provider

Webhook delivery keeps its own top-level webhookPlugin config key. Selecting that key is enough to activate webhook package discovery for every supported plugin-owned surface: webhook.delivery, standalone /api/webhooks, the CLI webhooks command, and the MCP webhook tools registered by the package.

{
  "webhookPlugin": {
    "webhook.delivery": {
      "provider": "webhooks"
    }
  }
}
  • The default runtime provider id is webhooks.

  • The webhooks id resolves to the external package kl-plugin-webhook.

  • The persisted .kanban.json webhooks array is unchanged and remains the registry source of truth.

  • The package owns runtime delivery, standalone webhook routes, CLI webhook commands, and MCP webhook tool registration where those plugin seams are available.

  • Advanced SDK consumers can use sdk.getExtension('kl-plugin-webhook'); direct webhook SDK methods remain stable compatibility shims.

  • A sibling checkout at ../kl-plugin-webhook is resolved automatically for local development.

  • Generated webhook reference docs are emitted from scripts/generate-webhooks-docs.ts; regenerate them from source metadata instead of editing docs/webhooks.md directly.

Callback runtime provider

Same-runtime callback automation is owned by the external kl-plugin-callback package. Install it in the runtime environment, then select the callbacks provider through the shared plugin-settings flow at plugins["callback.runtime"].

{
  "plugins": {
    "callback.runtime": {
      "provider": "callbacks",
      "options": {
        "handlers": [
          {
            "name": "log task creation",
            "type": "inline",
            "events": ["task.created"],
            "enabled": true,
            "source": "async ({ event, sdk }) => { console.log(event.event, sdk.constructor.name) }"
          },
          {
            "name": "notify local worker",
            "type": "process",
            "events": ["task.created", "task.updated"],
            "enabled": true,
            "command": "node",
            "args": ["scripts/callback-worker.mjs"],
            "cwd": "."
          }
        ]
      }
    }
  }
}
  • handlers[] is one ordered mixed list; each matching row is either inline or process.

  • Inline handlers are trusted same-runtime JavaScript evaluated with new Function. They are not sandboxed, run with host process privileges, and receive exactly one argument shaped as ({ event, sdk }).

  • Process handlers are normal subprocesses, not sandboxed. They receive one serialized { event } JSON payload on stdin only and do not receive a live SDK object.

  • Matching handlers run in order. Failures are logged, then later matching handlers continue.

  • Inline JavaScript now uses the shared CodeMirror-backed JavaScript editor in Plugin Options; there is still no separate callback-specific management surface.

MySQL setup and runtime expectations

Use the MySQL compatibility provider id by selecting provider: "mysql" under plugins["card.storage"]:

{
  "plugins": {
    "card.storage": {
      "provider": "mysql",
      "options": {
        "host": "localhost",
        "port": 3306,
        "user": "kanban",
        "password": "secret",
        "database": "kanban_db"
      }
    }
  }
}

Notes:

  • database is required.

  • Install kl-plugin-storage-mysql in the host environment that loads the plugin.

  • The mysql2 driver is an optional runtime dependency of that external package and is loaded lazily. Install it only in environments that actually use the MySQL provider.

  • mysql stores cards/comments in MySQL, and the matching attachment handler is auto-derived from the same package/options unless you explicitly choose a different attachment.storage provider.

npm install kl-plugin-storage-mysql mysql2

PostgreSQL setup and runtime expectations

Use the PostgreSQL compatibility provider id by selecting provider: "postgresql" under plugins["card.storage"]:

{
  "plugins": {
    "card.storage": {
      "provider": "postgresql",
      "options": {
        "host": "localhost",
        "port": 5432,
        "user": "kanban",
        "password": "secret",
        "database": "kanban_db"
      }
    }
  }
}

Notes:

  • database is required.

  • Install kl-plugin-storage-postgresql in the host environment that loads the plugin.

  • The pg driver is an optional runtime dependency of that external package and is loaded lazily. Install it only in environments that actually use the PostgreSQL provider.

  • postgresql stores cards/comments in PostgreSQL, and the matching attachment handler is auto-derived from the same package/options unless you explicitly choose a different attachment.storage provider.

npm install kl-plugin-storage-postgresql pg

MongoDB

{
  "plugins": {
    "card.storage": {
      "provider": "mongodb",
      "options": {
        "uri": "mongodb://localhost:27017",
        "database": "kanban_db"
      }
    }
  }
}

Notes:

  • database is required.

  • Install kl-plugin-storage-mongodb in the host environment that loads the plugin.

  • The mongodb driver is an optional runtime dependency of that external package and is loaded lazily. Install it only in environments that actually use the MongoDB provider.

  • mongodb stores cards/comments in MongoDB, and the matching attachment handler is auto-derived from the same package/options unless you explicitly choose a different attachment.storage provider.

npm install kl-plugin-storage-mongodb mongodb

Redis

{
  "plugins": {
    "card.storage": {
      "provider": "redis",
      "options": {
        "host": "localhost",
        "port": 6379,
        "db": 0
      }
    }
  }
}

Notes:

  • Install kl-plugin-storage-redis in the host environment that loads the plugin.

  • The ioredis driver is an optional runtime dependency of that external package and is loaded lazily. Install it only in environments that actually use the Redis provider.

  • redis stores cards/comments in Redis hashes, and the matching attachment handler is auto-derived from the same package/options unless you explicitly choose a different attachment.storage provider.

npm install kl-plugin-storage-redis ioredis

Provider status surfaces

These commands/endpoints/tools expose provider ids and host-facing metadata without requiring callers to guess which compatibility shim or external package is active:

  • kl storage status

  • GET /api/storage

  • GET /api/workspace

  • MCP: get_storage_status, get_workspace_info

Core localfs (markdown engine) reports watchGlob: "boards/**/*.md". The sqlite, mysql, postgresql, mongodb, and redis compatibility providers report isFileBacked: false and watchGlob: null through their external plugin metadata, so host layers do not have to infer them from the storage engine name. Standalone GET /api/storage and GET /api/workspace also include providers["webhook.delivery"], and SDK consumers can call sdk.getWebhookStatus() to see whether kl-plugin-webhook is active. When the plugin is not installed, webhookProvider returns 'none' and webhook CRUD methods throw a deterministic install error.

For attachment.storage, the same first-party storage plugins (sqlite, mongodb, postgresql, mysql, redis) are auto-derived from the active card.storage provider and reuse the same option payload. There is no need to keep a matching plugins["attachment.storage"] block unless you want a different provider such as S3.

For card.state, card-state is automatically derived from the active card.storage plugin. When card.storage is localfs (default), the built-in file-backed provider is used. When card.storage is an external plugin (sqlite, mongodb, postgresql, mysql, redis), card-state is loaded from the same storage package. There is no need to install or configure a separate card.state package. In auth-absent mode, both surfaces share the same stable default actor contract. When a real auth.identity provider is configured but no actor can be resolved, status/read/open surfaces report identity-unavailable / ERR_CARD_STATE_IDENTITY_UNAVAILABLE instead of implying that the backend itself is missing. This actor-scoped unread/open state is separate from get_active_card, kl active, and /api/tasks/active, which describe UI-style active-card selection.

In the shared Plugin Options workflow, that means the storage-backed attachment.storage and card.state rows are informational only: they follow the selected card.storage provider automatically and do not require second connection-option blocks in .kanban.json.

Migrating between compatibility-backed providers

CLI:

# Check current engine
kl storage status

# Migrate to SQLite
kl storage migrate-to-sqlite --sqlite-path .kanban/kanban.db

# Migrate back to markdown
kl storage migrate-to-markdown

REST API:

curl -X POST http://localhost:3000/api/storage/migrate-to-sqlite \
  -H 'Content-Type: application/json' \
  -d '{"sqlitePath": ".kanban/kanban.db"}'

SDK:

const sdk = new KanbanSDK('.kanban')
await sdk.init()

// Migrate to SQLite
const count = await sdk.migrateToSqlite('.kanban/kanban.db')
console.log(`Migrated ${count} cards to SQLite`)

// Or migrate back
await sdk.migrateToMarkdown()

These migration helpers are compatibility aliases for the markdown ↔ sqlite flow. Existing files / the database are not deleted during migration — they serve as a manual backup until you remove them.

If a workspace was explicitly using the sqlite or mysql attachment compatibility provider, migrating back to markdown automatically drops that incompatible attachment override so the legacy localfs default continues to work without manual config cleanup.

MCP tools: get_storage_status, migrate_to_sqlite, migrate_to_markdown

Auth / Authz Plugin Contract

Kanban Lite ships auth/authz capability namespaces for auth.identity and auth.policy, enforced through listener-only before-events that run before a write is committed.

Auth listeners can veto a mutation by throwing AuthError, optionally return plain-object input overrides, and preserve the same clean denial behavior across standalone, CLI, MCP, and the extension host.

Host surfaces now install request-scoped auth with sdk.runWithAuth(authContext, fn) before calling SDK mutators. The first-party auth listener resolves identity/policy from that scoped carrier plus the before-event's actor/board hints; auth is no longer threaded through positional mutation args or a BeforeEventPayload.auth field.

Install the package in the environment that loads Kanban Lite:

npm install kl-plugin-auth

For local sibling-repo development, a checkout at ../kl-plugin-auth is resolved automatically. npm link ../kl-plugin-auth is optional, but useful when you want an explicit local package link.

The shipped provider ids behave as before:

  • auth.identitynoop: all callers are treated as anonymous (identity always resolves to null).

  • auth.policynoop: all actions are allowed regardless of identity (policy always returns true).

When non-noop auth providers are configured, the SDK now performs pre-action authorization for the privileged async mutation surface used by the Node-hosted adapters (standalone server, CLI, MCP, and the VS Code extension host). Workspaces without auth providers configured remain fully open-access.

Provider references for both namespaces are read from .kanban.json via the plugins key — the same namespace used for storage providers:

{
  "plugins": {
    "auth.identity": { "provider": "kl-plugin-auth" },
    "auth.policy": { "provider": "kl-plugin-auth", "options": { "strict": true } }
  }
}

Raw bearer tokens, token-to-role maps, and other live secrets must not be stored in .kanban.json. Token acquisition is host-specific (VS Code SecretStorage, env vars for CLI/MCP, in-memory or cookies for standalone). Password hashes for the local provider may be stored in config.

rbac provider

Kanban Lite ships a first-party Role-Based Access Control (RBAC) provider pair (rbac) in kl-plugin-auth. It enforces a fixed three-role action matrix without requiring a login flow or external identity service.

Enable it in .kanban.json:

{
  "plugins": {
    "auth.identity": { "provider": "kl-plugin-auth" },
    "auth.policy": { "provider": "kl-plugin-auth" }
  }
}

Token validation: Tokens are opaque strings validated against a runtime-owned principal registry supplied by the host at startup. Unknown or absent tokens resolve to null (deny). A Bearer prefix is stripped before the registry lookup. Token values, token-to-role maps, and role assignments must remain in host/runtime configuration only and must never appear in .kanban.json or diagnostics.

Role levels:

Role

Permitted actions

user

form.submit, comment.create, comment.update, comment.delete, attachment.add, attachment.remove, card.action.trigger, log.add

manager

All user actions plus card.create, card.update, card.move, card.transfer, card.delete, board.action.trigger, log.clear, board.log.add

admin

All manager actions plus all board/config mutations: board.create, board.update, board.delete, settings.update, webhook.*, label.*, column.*, storage.migrate, and more

Roles are cumulative upward: admin includes all manager and user actions.

When you enable auth.policy: rbac or auth.policy: kl-plugin-auth through the shared Plugin Options UI and no saved policy options exist yet, Kanban Lite now materializes the default permissions matrix from the built-in RBAC_ROLE_MATRIX into .kanban.json so you start from an editable baseline instead of an empty config block. The same backfill also runs when plugin-settings refresh sees a selected auth-policy provider with an empty options object.

Scope limits (v1):

  • Action-level only: access decisions are made per-action, not per-row or per-card.

  • No login flow: the provider has no interactive authentication UI.

  • No row filtering: a denied action blocks the write; no partial result filtering is performed.

  • Node-hosted only: the RBAC provider runs exclusively in the Node host layer; no browser execution is performed.

  • Tokens are resolved from the host token source at call time and are never persisted to .kanban.json, returned in API responses, or echoed in error bodies or logs.

local provider

Kanban Lite also ships a first-party local workspace auth provider pair (local) in kl-plugin-auth.

  • Standalone UI: unauthenticated browser requests redirect to a plugin-served /auth/login page.

  • Standalone API: every /api/* request requires either Authorization: Bearer <token> or an authenticated standalone session cookie.

  • CLI: use --token <value> for a one-off invocation, or the shared workspace token from KANBAN_LITE_TOKEN (KANBAN_TOKEN is still accepted as a compatibility alias).

  • MCP: use the shared workspace token from KANBAN_LITE_TOKEN (KANBAN_TOKEN is still accepted as a compatibility alias).

  • Workspace token bootstrap: when the standalone auth plugin starts and KANBAN_LITE_TOKEN is missing, it generates a kl-... token and saves it to <workspaceRoot>/.env.

Enable it in .kanban.json with bcrypt-hashed passwords:

{
  "plugins": {
    "auth.identity": {
      "provider": "kl-plugin-auth",
      "options": {
        "roles": ["user", "manager", "admin"],
        "users": [
          {
            "username": "alice",
            "password": "$2b$12$REPLACE_WITH_BCRYPT_HASH",
            "role": "user"
          }
        ]
      }
    },
    "auth.policy": { "provider": "kl-plugin-auth" }
  }
}

Use the CLI to add users without manually computing bcrypt hashes:

kl auth create-user --username alice --password s3cr3t
kl auth create-user --username admin --password s3cr3t --role reviewer

This command hashes the password, appends the user entry to plugins["auth.identity"].options.users, seeds the default user / manager / admin role catalog when missing, and appends any new custom role to plugins["auth.identity"].options.roles.

The shared settings UI now exposes that roles[] catalog directly beside users[], seeds it with user, manager, and admin by default, and uses it as the live enum source for both users[].role and auth.policy.permissions[].role. The local policy itself remains permissive for any authenticated identity unless you configure an explicit auth.policy.options.permissions[] matrix. Anonymous callers are still denied with auth.identity.missing.

To override the default role behavior, add a custom permission matrix on auth.policy. In the shared Plugin Options UI, each row now picks one role plus the before-events that role may run:

{
  "plugins": {
    "auth.policy": {
      "provider": "local",
      "options": {
        "permissions": [
          {
            "role": "admin",
            "actions": ["settings.update", "board.delete"]
          }
        ]
      }
    }
  }
}

Legacy options.matrix role maps remain supported for backward compatibility, and the shared Plugin Options UI now edits the simpler role-based permissions format while sourcing its action picker from the before-event catalog.

Host token sources

  • Standalone REST API: Authorization: Bearer <token> request header, or authenticated standalone cookie session

  • CLI: --token <value> for one-off calls, or KANBAN_LITE_TOKEN environment variable (KANBAN_TOKEN still accepted as an alias)

  • MCP: KANBAN_LITE_TOKEN environment variable (KANBAN_TOKEN still accepted as an alias)

  • VS Code extension host: secure SecretStorage (Kanban Lite: Set Auth Token / Clear Auth Token commands)

Raw tokens are treated as write-only input: they are never returned in REST responses, CLI/MCP output, logs, errors, or webview messages.

Diagnostics / status

  • SDK: sdk.getAuthStatus()

  • Standalone REST API: GET /api/auth and GET /api/workspace

  • CLI: kl auth status

  • MCP: get_auth_status

These status surfaces expose only safe metadata such as active provider ids, whether auth is configured, whether a host token source is currently present, and the transport/token-source label.

Configuration

Board configuration is stored in .kanban.json at your project root. It supports multiple boards, each with their own columns and settings:

{
  "version": 2,
  "defaultBoard": "default",
  "boards": {
    "default": {
      "name": "Default Board",
      "columns": [
        { "id": "backlog", "name": "Backlog", "color": "#6b7280" },
        { "id": "todo", "name": "To Do", "color": "#3b82f6" },
        { "id": "in-progress", "name": "In Progress", "color": "#f59e0b" },
        { "id": "review", "name": "Review", "color": "#8b5cf6" },
        { "id": "done", "name": "Done", "color": "#22c55e" }
      ],
      "actions": {
        "deploy": "Deploy to Production",
        "announce": "Post release update"
      },
      "metadata": ["customer", "owner"],
      "title": ["ticket", "customer"],
      "nextCardId": 1,
      "defaultStatus": "backlog",
      "defaultPriority": "medium"
    }
  },
  "showPriorityBadges": true,
  "showAssignee": true,
  "showDueDate": true,
  "showLabels": true,
  "compactMode": false,
  "boardZoom": 100,
  "cardZoom": 100,
  "panelMode": "drawer",
  "drawerWidth": 50,
  "forms": {
    "incident-report": {
      "schema": {
        "type": "object",
        "title": "Incident Report",
        "properties": {
          "severity": { "type": "string" },
          "owner": { "type": "string" }
        }
      },
      "data": {
        "severity": "medium"
      }
    }
  },
  "actionWebhookUrl": "https://example.com/kanban-actions",
  "customHeadHtml": "<script src='https://cdn.example.com/analytics.js'></script>",
  "customHeadHtmlFile": "head-extras.html",
  "basePath": "/kanban"
}

customHeadHtml injects raw HTML into the standalone board's <head> element — useful for analytics snippets, custom CSS, or guided-tour scripts. customHeadHtmlFile does the same but reads from a file path relative to the workspace root; when both are set, customHeadHtmlFile takes precedence.

basePath enables subfolder deployments behind a reverse proxy. Set it to the URL prefix under which the board is served (e.g. "/kanban") so that all asset URLs, the WebSocket endpoint, and API routes are correctly prefixed. Leave unset (or empty) for root-domain deployments.

When you deploy the standalone UI through the Cloudflare Worker entrypoint, the browser now switches to an event-driven hybrid live-sync transport: a Durable Object-backed /ws connection sends invalidation notices, while authoritative board/card refreshes continue over /api/webview-sync. That keeps cross-tab updates event-driven without periodic polling, and reconnects resync the latest state rather than replaying every missed event.

The generated Durable Object also persists workspace active-card selection, so /api/tasks/active, preview routes, and follow-up requests keep working even though the Worker runtime has no writable local filesystem. Non-Worker hosts still fall back to the local .active-card.json sidecar, and the Node standalone server remains the only host with full raw WebSocket payload parity for push-heavy flows.

The interactive scripts/deploy-cloudflare-worker.mjs flow also accepts repeatable --custom-domain <hostname> values (or KANBAN_CF_CUSTOM_DOMAIN / KANBAN_CF_CUSTOM_DOMAINS in .env.cloudflare) and emits Cloudflare Worker custom_domain route blocks in the generated wrangler.toml. Set KANBAN_CF_CUSTOM_DOMAIN_ZONE / --custom-domain-zone <zone> when you want to pin the target zone explicitly; otherwise the script infers it from the hostname. The generated config also keeps workers_dev = true, so env-driven deployments can attach hostnames like kl.incidentmind.com without giving up the default *.workers.dev URL.

Cloudflare deploys now also reuse already-existing R2 buckets when --create-resources is enabled, so repeat deploys stop tripping Wrangler's noisy “bucket already exists” error. The same helper auto-bundles the default webhook delivery provider (webhookskl-plugin-webhook) unless the config explicitly sets plugins["webhook.delivery"].provider to "none".

Columns are fully customizable per board — add, remove, rename, or recolor them from the web UI, CLI, or REST API.

boards.<id>.title is an optional ordered string array of metadata keys whose rendered values prefix user-visible card titles without changing stored markdown titles, slugs, or filenames. For example, with "title": ["ticket", "customer"], a card whose markdown title is # Investigate outage and metadata is { "ticket": "INC-42", "customer": "Acme" } renders as INC-42 Acme Investigate outage across the webview, VS Code sidebar, CLI, and MCP read surfaces. You can edit the same ordered list from Board → Title in the shared settings panel.

boardZoom and cardZoom set the default zoom percentage (75–150) for the board view and card detail panel respectively. They can also be adjusted live in the Settings panel or with Ctrl/Cmd + = / Ctrl/Cmd + - keyboard shortcuts.

panelMode controls whether card flows open as a centered popup or a drawer. When using drawer mode, drawerWidth sets the default width percentage (20–80) for card creation and detail panels, and drawerPosition chooses which edge (right, left, top, or bottom) the drawer uses.

forms defines reusable JSON Schema/JSON Forms descriptors that any card can attach by name. Card-local inline forms still live on the card frontmatter under forms, while submitted values persist per card under formData.

AI Agent Integration

  • Vercel AI Chat SDK adapter (kl-adapter-vercel-ai): Reusable npm package providing pre-built tool() definitions and a REST client for kanban-lite — covers cards CRUD, streaming comments, labels, actions, forms, columns, and boards. See packages/kl-adapter-vercel-ai/.

  • Claude Code: Default, Plan, Auto-edit, and Full Auto modes

  • Codex: Suggest, Auto-edit, and Full Auto modes

  • OpenCode: Agent integration support

  • AI receives card context (title, priority, labels, description) for informed assistance

Development

Prerequisites

  • Node.js 18+

  • pnpm

Setup

# Install dependencies
pnpm install

# Start development (watch mode)
pnpm dev

# Build for production
pnpm build

# Build individually
pnpm build:cli
pnpm build:mcp
pnpm build:standalone-server

# Run tests
pnpm test

# Type checking
pnpm typecheck

# Linting
pnpm lint

Documentation Site

A static product and documentation website lives in packages/docs-site (Eleventy-powered, private to the monorepo). It renders root docs as reference pages without relocating them.

# Build the static site
npm run site:build

# Develop with live reload
npm run site:dev

# Or use the workspace-scoped commands directly
pnpm --filter @kanban-lite/docs-site build
pnpm --filter @kanban-lite/docs-site dev

Release Workflow

Use the root release commands for the coordinated npm + GitHub release path:

# Patch release
npm run release

# Minor release
npm run release:minor

# Major release
npm run release:major

The coordinated release flow now:

  1. verifies the git working tree is clean

  2. checks npm and GitHub auth, pausing so you can refresh the npm release token if needed

  3. builds kanban-lite once and each publishable workspace package once

  4. bumps every public package version without creating intermediate git tags

  5. packages releases/kanban-lite-<version>.vsix

  6. publishes all public workspace packages to npm, including kanban-lite

  7. creates a single chore: release vX.Y.Z commit, tags it, and pushes the branch plus tag

  8. creates or updates the matching GitHub release artifact in place

If you also need VS Code Marketplace and Open VSX publishing for the extension package, run this separately after the release build is ready:

pnpm --filter kanban-lite run publish:all

Tech Stack

Web UI: React 18, Vite, Tailwind CSS, Zustand, Tiptap SDK/CLI/MCP: TypeScript, Node.js, @modelcontextprotocol/sdk

Architecture

src/
  sdk/           # Core SDK (no external dependencies)
  cli/           # CLI tool (built on SDK)
  mcp-server/    # MCP server (built on SDK)
  standalone/    # Standalone web server (uses SDK)
  webview/       # React frontend
  shared/        # Shared types

Acknowledgments

This project was originally created by LachyFS. Thank you for building the foundation that made Kanban Lite possible.

License

MIT

A
license - permissive license
-
quality - not tested
B
maintenance

Maintenance

Maintainers
Response time
2dRelease cycle
23Releases (12mo)
Commit activity

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/borgius/kanban-lite'

If you have feedback or need assistance with the MCP directory API, please join our Discord server