TaskFrame MCP Server
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@TaskFrame MCP Serverlist my projects"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
TaskFrame MCP Server
HTTP MCP server for TaskFrame — lets AI agents work with projects, tasks, comments and docs via the TaskFrame external API.
Each request is stateless. The Bearer token from the client is forwarded directly to TaskFrame — no credentials are stored on the server.
For humans
Requirements
Node.js 22+
TaskFrame API key (
tf_live_...) — create one in TaskFrame → Settings → API KeysTASKFRAME_BASE_URL— your TaskFrame instance URL
Run locally
cp .env.example .env
# edit .env: set TASKFRAME_BASE_URL
npm install
npm run devRun with Docker (HTTP only — for local testing)
docker build -t taskframe-mcp .
docker run -e TASKFRAME_BASE_URL=https://your-taskframe-domain.com -p 3000:3000 taskframe-mcpDeploy with HTTPS (docker-compose + Caddy) — recommended for production
This runs the MCP server behind Caddy, which obtains
and auto-renews a free Let's Encrypt TLS certificate for you. The Node server is
never exposed directly — only Caddy's 443 faces the internet.
Point a hostname at your server's IP. Options:
a real domain:
mcp.example.com→ your IP via an A recordno domain? use a free sslip.io name that encodes your IP, e.g. IP
203.0.113.10→203-0-113-10.sslip.io(resolves automatically)
Open ports 80 and 443 to the internet (80 is needed for certificate issuance).
Configure and launch:
cp .env.example .env
# edit .env: set TASKFRAME_BASE_URL and SITE_ADDRESS (your hostname)
docker compose up -dCaddy fetches the certificate on the first request. Your endpoint is then
https://<SITE_ADDRESS>/mcp — give that URL to clients.
Environment variables
Variable | Required | Description |
| yes | Base URL of your TaskFrame instance |
| compose | Public hostname Caddy serves HTTPS on (e.g. |
| no | Internal HTTP port, default |
| no | Comma-separated Host header allowlist; enables DNS-rebinding protection. In compose it is set to |
Health check
curl http://localhost:3000/health
# → {"ok":true}Related MCP server: plane-mcp-server
Security & deployment
Always deploy behind TLS. The API key is sent as a plaintext Authorization: Bearer header on every request. Terminate HTTPS at a reverse proxy (nginx, Caddy, Traefik, cloud LB) and point clients at the https:// URL. Never expose the raw http:// port to a network you don't fully control.
Built-in protections:
Stateless — no token, workspace id, or session is stored between requests. Each call is authenticated solely by its own Bearer header.
ID validation — all UUID/key arguments are schema-validated and URL-encoded, so a tool argument can never alter the upstream request path (no path traversal / SSRF into other TaskFrame endpoints).
Upstream timeout — requests to TaskFrame abort after 15s, returning
504instead of hanging.Body limit — JSON bodies over 1 MB are rejected with
413.Scoped by the key — the server never grants more than the API key's own permissions; it cannot escalate access.
Recommended for production: run the reverse proxy with rate limiting, set ALLOWED_HOSTS to your public hostname, and keep the container non-root (the node base image already runs as a non-root node user when configured).
Connect to Claude Code
Add to your ~/.claude/settings.json (global) or .claude/settings.json (project):
{
"mcpServers": {
"taskframe": {
"type": "http",
"url": "https://mcp.example.com/mcp",
"headers": {
"Authorization": "Bearer tf_live_..."
}
}
}
}Each user sets their own API key in the Authorization header. The server is shared — the key is not.
For agents
The server exposes a resource taskframe://guide that Claude Code loads automatically into context. If your client does not load resources automatically, call get_guide first.
How auth works
Pass your TaskFrame API key as Authorization: Bearer <key> in the HTTP request to this MCP server. The server resolves your workspaceId automatically via /me — never pass workspaceId as a tool argument.
Recommended flow
list_projects
→ pick projectId
get_project_meta(projectId)
→ columns[].id → use as statusId
→ sprints[].id → use as sprintId
→ members[].id → use as assigneeId
list_tasks(projectId) — read existing tasks
create_task(projectId, …) — create new task
update_task(taskId, …) — update fieldsKey shortcuts
Use
get_task_by_keywith keys likeAPP-43when you have them — it's faster thanget_taskby UUIDdescription(tasks) andcontent(docs/comments) accept plain text — the server wraps it in TipTap JSON automaticallysprintId: omit → active sprint,null→ backlog, UUID → specific sprint
Tools
Context
Tool | Description |
| Get usage instructions (fallback if resource not loaded) |
| Get workspace id, user, and effective permissions |
Projects
Tool | Arguments | Description |
|
| List projects |
|
| Columns, sprints, labels, members |
Tasks
Tool | Arguments | Description |
|
| All tasks in a project |
|
| Get task by UUID |
|
| Get task by key, e.g. |
|
| Create task |
|
| Update task (partial) |
|
| Delete task permanently |
create_task and update_task accept: description, type, priority, statusId, assigneeId, parentTaskId, estimateMinutes, storyPoints, dueDate, startDate, sprintId, labelIds.
Comments
Tool | Arguments | Description |
|
| List comments on a task |
|
| Add comment |
|
| Edit own comment |
|
| Delete comment |
Docs
Tool | Arguments | Description |
|
| Flat list of pages; use |
|
| Get page with full content |
|
| Create page |
|
| Update page; use |
|
| Delete page and all children permanently |
Permissions
The server uses whatever permissions your API key has. If a tool returns a 403 error, the key needs the corresponding permission added in TaskFrame settings.
Operation | Required permission |
Read projects |
|
Read tasks and comments |
|
Create / update / delete tasks |
|
Create / edit / delete comments |
|
Read docs |
|
Create / update / delete docs |
|
Columns in project meta |
|
Sprints in project meta |
|
Labels in project meta |
|
Members in project meta |
|
Error format
All tools return errors as text content with isError: true. The message includes the HTTP status and TaskFrame error description:
Error: 403: API key does not allow this operation
Error: 404: Task not foundDevelopment
npm run dev # watch mode with tsx
npm run build # compile to dist/
npm start # run compiled outputThis server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/DozZzeR/vertical-unit-mcp'
If you have feedback or need assistance with the MCP directory API, please join our Discord server