P4 Plan MCP Server
OfficialAllows interaction with Perforce's P4 Plan project management system, providing tools for managing projects, tasks, and retrieving project data via the P4 Plan GraphQL API.
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., "@P4 Plan MCP Serverlist my open tasks for the current sprint"
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.
Architecture
This service acts as a stateless protocol adapter between MCP clients (AI assistants) and the P4 Plan GraphQL API, using stdio transport (stdin/stdout).
┌─────────────────┐ stdio (stdin/stdout) ┌─────────────────┐ GraphQL ┌─────────────────┐
│ AI Client │ ────────────────────────▶ │ P4 Plan MCP │ ──────────────▶ │ P4 Plan GraphQL │
│(Claude, Copilot)│ Spawns as child process │ Server │ Port 4000 │ API │
│ │ P4PLAN_API_AUTH_TOKEN │ (stateless) │ Bearer token │ │
└─────────────────┘ env var └─────────────────┘ forwarded └─────────────────┘The client spawns the MCP server as a child process. Authentication is provided via the P4PLAN_API_AUTH_TOKEN environment variable, which the server validates at startup and forwards to the GraphQL API on every tool call.
Prerequisites
Requirement | Version | Notes |
Node.js | >= 20 (24+ recommended) | The server targets ES2023. Check with |
npm | >= 9 | Comes with Node.js. Check with |
P4 Plan API | >= 2026.1.002 | Required for all tool operations. Earlier versions are not supported. |
Tip: Use nvm to manage Node.js versions, or skip the Node.js requirement entirely by using Docker.
Quick Start (npx)
The fastest way to get started — no installation required. Just configure your MCP client to use npx:
npx -y @perforce/p4plan-mcpnpx automatically downloads and runs the latest version of the server. Your MCP client (VS Code, Claude Desktop, etc.) handles this for you — just add the config below and start chatting.
VS Code — add to .vscode/mcp.json:
{
"servers": {
"p4-plan": {
"type": "stdio",
"command": "npx",
"args": ["-y", "@perforce/p4plan-mcp"],
"env": {
"P4PLAN_API_AUTH_TOKEN": "YOUR_JWT_TOKEN",
"P4PLAN_API_URL": "http://localhost:4000"
}
}
}
}Claude Desktop — add to your config:
{
"mcpServers": {
"p4-plan": {
"command": "npx",
"args": ["-y", "@perforce/p4plan-mcp"],
"env": {
"P4PLAN_API_AUTH_TOKEN": "YOUR_JWT_TOKEN",
"P4PLAN_API_URL": "http://localhost:4000"
}
}
}
}Note: The
-yflag auto-confirms the npm install prompt so the server starts without user interaction.
See Client Configuration for more options including Docker, secure token prompts, and local builds.
MCP Registry
This server is published to the official MCP Registry as io.github.perforce/p4plan-mcp. Discover it via the registry API:
curl 'https://registry.modelcontextprotocol.io/v0/servers?search=io.github.perforce/p4plan-mcp'MCP clients that support registry discovery can install the server by name without needing to know the npm package identifier.
Installation
For development or when you want to run from a local clone:
npm ci
npm run build
# For using npx locally
npm linkRun the MCP server via Docker instead of installing Node.js locally. The MCP client (VS Code, Claude Desktop) spawns the container as a child process — same as npx, just using docker as the command.
The image is published to Docker Hub at perforce/p4plan-mcp, built multi-arch (linux/amd64 + linux/arm64) on every release, with SLSA provenance and SBOM attestations.
VS Code (.vscode/mcp.json):
{
"servers": {
"p4-plan": {
"type": "stdio",
"command": "docker",
"args": [
"run", "-i", "--rm",
"-e", "P4PLAN_API_AUTH_TOKEN=YOUR_JWT_TOKEN",
"-e", "P4PLAN_API_URL=http://host.docker.internal:4000",
"perforce/p4plan-mcp:latest"
]
}
}
}Claude Desktop:
{
"mcpServers": {
"p4-plan": {
"command": "docker",
"args": [
"run", "-i", "--rm",
"-e", "P4PLAN_API_AUTH_TOKEN=YOUR_JWT_TOKEN",
"-e", "P4PLAN_API_URL=http://host.docker.internal:4000",
"perforce/p4plan-mcp:latest"
]
}
}
}Note: Use
host.docker.internal(macOS/Windows) or172.17.0.1(Linux) to reach the P4 Plan GraphQL API running on the host machine.
Pin a specific version by replacing
:latestwith:2026.2.0(or whichever tag) for reproducible deployments.
Build locally (for development against unreleased changes):
docker build -t p4plan-mcp:dev .
# then swap "perforce/p4plan-mcp:latest" for "p4plan-mcp:dev" in the configs aboveConfiguration
Copy the example config and configure:
cp config-example.env .envEdit .env with your settings:
# JWT token for authenticating with P4 Plan GraphQL API
P4PLAN_API_AUTH_TOKEN=your-jwt-token
# P4 Plan GraphQL API URL
P4PLAN_API_URL=http://localhost:4000
# Logging level
LOG_LEVEL=debug
# Search results limit (default: 400)
# SEARCH_LIMIT=400
# Allow self-signed TLS certificates (for HTTPS APIs with untrusted certs)
# P4PLAN_ALLOW_SELF_SIGNED_CERTS=trueThe server communicates via stdin/stdout. It is not meant to be run interactively — MCP clients (VS Code, Claude Desktop) spawn it as a child process automatically.
Authentication
The MCP server requires a JWT token provided via the P4PLAN_API_AUTH_TOKEN environment variable. The token is validated at startup and forwarded to the P4 Plan GraphQL API on every tool call. No sessions or state are maintained.
Obtaining a JWT Token
Get a JWT token from the P4 Plan GraphQL API using curl. You can authenticate with either your password or a Personal Access Token (PAT):
curl -s -X POST http://localhost:4000/graphql \
-H "Content-Type: application/json" \
-d '{
"query": "mutation Login($loginUserInput: LoginUserInput!) { login(loginUserInput: $loginUserInput) { access_token } }",
"variables": { "loginUserInput": { "username": "YOUR_USERNAME", "password": "YOUR_PASSWORD" } }
}'A PAT can be used in place of your password in the same login mutation. This avoids exposing your actual password:
curl -s -X POST http://localhost:4000/graphql \
-H "Content-Type: application/json" \
-d '{
"query": "mutation Login($loginUserInput: LoginUserInput!) { login(loginUserInput: $loginUserInput) { access_token } }",
"variables": { "loginUserInput": { "username": "YOUR_USERNAME", "password": "YOUR_PERSONAL_ACCESS_TOKEN" } }
}'Both methods return the same response:
{
"data": {
"login": {
"access_token": "eyJhbGciOiJIUzI1NiIs..."
}
}
}Copy the access_token value and use it in your MCP client configuration.
Log in to P4 Plan
Go to User Settings → Personal Access Tokens
Click Generate New Token
Set an appropriate expiration date
Copy the token — use it in the login mutation above to obtain a JWT
Note: JWT tokens expire. When your token expires, the server will fail to start with an authentication error. Generate a new JWT using the same
curlcommand above.
Available Tools (28 total)
Projects
Use cases: Discover project IDs needed by other tools
Parameters:
projectIdUse cases: Retrieve section IDs (backlog, QA, planning) for other tools
Tasks
Parameters:
showCompleted,showOnlyNextFourWeeks,showHidden,showPipelineTasksThatCannotStartUse cases: View personal work queue across all projects
Parameters:
taskIds(array of strings, max 20)Use cases: Full item details, link inspection, batch retrieval of multiple items
Parameters:
findQuery,projectIdUses P4 Plan Find query syntax for all searches. Call
read_skillwithskillName="search-queries"first to get exact column names, operators, and value formats. For simple name search useItemname:Text("text"). Supports filtering by status, assignee, severity, item type, dates, boolean conditions, and combinations with AND/OR/NOT.Each project has three sections (Backlog, QA, Planning) with different IDs.
Use cases: Item discovery, filtering, reporting
Types:
backlog_task,bug,scheduled_task,sprint,release,sprint_taskParameters:
type,name,projectId,parentItemId,previousItemId, and type-specific fieldsUse cases: Task creation, sprint creation, bug filing
Types: BacklogTask, Bug, ScheduledTask, Sprint, Release
Parameters:
itemId, plus any updatable fields (name, status, assignedTo, points, etc.)Use cases: Status updates, assignments, estimation, sprint configuration
Sprint & Release Management
Parameters:
taskId,sprintIdUse cases: Sprint planning, backlog commitment
Parameters:
taskIdUse cases: Sprint scope adjustment
Custom Fields & Workflows
Parameters:
projectIdUse cases: Discover custom fields before reading/writing values
Parameters:
taskId,onlySetUse cases: Read project-specific metadata
Parameters:
taskId,columnId,valueUse cases: Update project-specific metadata
Parameters:
projectIdUse cases: Discover workflow statuses for status transitions
Task Actions
Parameters:
taskIdUse cases: Quick status update convenience method
Parameters:
taskIdUse cases: Quick status update convenience method
Comments & Attachments
Parameters:
taskIdUse cases: Read discussion history
Parameters:
taskId,textUse cases: Add discussion, acceptance criteria, notes
Parameters:
taskId,commentId,textUse cases: Correct or update existing comments
Parameters:
taskId,commentIdUse cases: Remove outdated or incorrect comments
Parameters:
taskIdUse cases: List attached files, discover paths for download
Parameters:
taskId,pathText files returned inline, images as base64
Use cases: Read attached documents, view screenshots
Parameters:
taskId,pathUse cases: Remove outdated attachments
Parameters:
taskId,imagePathUse cases: Set visual identity for cards/items
Links
Parameters:
fromItemId,toItemIdorurl,relation(blocks, duplicates, relatedTo)Use cases: Dependency tracking, cross-references, external URLs
Parameters:
fromItemId,toItemIdorurlUse cases: Clean up outdated dependencies
Users
Use cases: Identity verification, user context
Parameters:
projectIdUse cases: Find user IDs for assignments, sprint member management
Skills
Parameters:
skillNameReturns the full Markdown content of the requested skill document. The AI agent must call this with
skillName="search-queries"before composing anyfindQueryforsearch_tasks.Available skills:
project-navigation,search-queries,task-management,planning,backlog-refinement,bug-tracking,custom-fields,gantt-scheduling,workflowsUse cases: Learn correct query syntax, discover tool usage patterns, understand domain concepts
Client Configuration
Create .vscode/mcp.json in your workspace:
{
"servers": {
"p4-plan": {
"type": "stdio",
"command": "npx",
"args": ["-y", "@perforce/p4plan-mcp"],
"env": {
"P4PLAN_API_AUTH_TOKEN": "YOUR_JWT_TOKEN",
"P4PLAN_API_URL": "http://localhost:4000"
}
}
}
}Note: The
-yflag auto-confirms the npm install prompt so the server starts without user interaction.
If running from a local clone instead of npm:
{
"servers": {
"p4-plan": {
"type": "stdio",
"command": "node",
"args": ["/path/to/MCP/dist/main.js"],
"env": {
"P4PLAN_API_AUTH_TOKEN": "YOUR_JWT_TOKEN",
"P4PLAN_API_URL": "http://localhost:4000"
}
}
}
}For added security, you can use VS Code input prompts to avoid storing tokens in files:
{
"inputs": [
{
"type": "promptString",
"id": "p4-plan-jwt",
"description": "P4 Plan JWT Token",
"password": true
}
],
"servers": {
"p4-plan": {
"type": "stdio",
"command": "npx",
"args": ["-y", "@perforce/p4plan-mcp"],
"env": {
"P4PLAN_API_AUTH_TOKEN": "${input:p4-plan-jwt}",
"P4PLAN_API_URL": "http://localhost:4000"
}
}
}
}Add to ~/Library/Application Support/Claude/claude_desktop_config.json (macOS) or %APPDATA%\Claude\claude_desktop_config.json (Windows):
{
"mcpServers": {
"p4-plan": {
"command": "npx",
"args": ["-y", "@perforce/p4plan-mcp"],
"env": {
"P4PLAN_API_AUTH_TOKEN": "YOUR_JWT_TOKEN",
"P4PLAN_API_URL": "http://localhost:4000"
}
}
}
}Replace YOUR_JWT_TOKEN with a token obtained from the login mutation (see Obtaining a JWT Token).
Create .mcp.json in your project root:
{
"mcpServers": {
"p4-plan": {
"command": "npx",
"args": ["-y", "@perforce/p4plan-mcp"],
"env": {
"P4PLAN_API_AUTH_TOKEN": "YOUR_JWT_TOKEN",
"P4PLAN_API_URL": "http://localhost:4000"
}
}
}
}For a local build, replace "command" and "args" with:
{
"mcpServers": {
"p4-plan": {
"command": "node",
"args": ["/path/to/MCP/dist/main.js"],
"env": {
"P4PLAN_API_AUTH_TOKEN": "YOUR_JWT_TOKEN",
"P4PLAN_API_URL": "http://localhost:4000"
}
}
}
}Or add it via the CLI:
claude mcp add p4-plan \
-e P4PLAN_API_AUTH_TOKEN=YOUR_JWT_TOKEN \
-e P4PLAN_API_URL=http://localhost:4000 \
-- npx -y @perforce/p4plan-mcpNote: The server name must come before the
-eflags, otherwise the variadic-eparser consumes the name as an env value.
Verify it's running: type /mcp inside Claude Code to check server status.
Tip: Place
.mcp.jsonin your project root to share the config with your team (tokens excluded). For personal config, add the server to~/.claude.jsoninstead.
Verifying the Connection
Ensure the P4 Plan GraphQL API is running
Open VS Code with the workspace containing
.vscode/mcp.jsonLook for "MCP SERVERS" in the Extensions sidebar — you should see "p4-plan" listed
Start a new Copilot chat and ask "What tasks are assigned to me?"
Environment Variables
P4PLAN_API_AUTH_TOKEN- JWT token for authenticating with the P4 Plan GraphQL APIP4PLAN_API_URL- P4 Plan GraphQL API URL (default:http://localhost:4000)P4PLAN_ALLOW_SELF_SIGNED_CERTS- Set totrueto accept self-signed or untrusted TLS certificates when connecting to the API over HTTPS (default:false)LOG_LEVEL- Logging level:debug,info,warn,error(default:debug)SEARCH_LIMIT- Maximum number of results returned bysearch_tasks(default:400)
Skills & Resources
The server includes skill files — domain-specific guides that help AI agents construct correct tool calls. Skills are accessible in two ways:
read_skilltool — any MCP client can callread_skillwith askillNameto fetch skill content at runtime. This is the primary access method and works with all clients.MCP resources — skills are also registered as MCP resources (e.g.,
skill://p4-plan/search-queries) for clients that support native resource reading.
Skill | Purpose |
project-navigation | Finding projects, items, and getting started |
search-queries | P4 Plan Find query syntax (column names, values, operators) |
task-management | Task CRUD, status, assignments, comments, attachments |
planning | Sprints, releases, commitment, allocations |
backlog-refinement | Backlog items, estimation, prioritization |
bug-tracking | Bugs, severity, QA section |
custom-fields | Custom columns, project-specific metadata |
gantt-scheduling | Scheduled tasks, timeline, dependencies |
workflows | Workflows, pipelines, status state machines |
See skills/README.md for details on using skills with different AI clients.
Development
To test the npx experience locally without publishing to npm:
# Build and create a global symlink
npm run build
npm link
# Now test exactly as an end user would
P4PLAN_API_AUTH_TOKEN=your-jwt-token npx @perforce/p4plan-mcp
# Clean up when done
npm unlink -g @perforce/p4plan-mcpThe symlink persists across rebuilds — just run npm run build after code changes.
Create or edit a tools file in
src/tools/Define the tool with:
name: Unique tool identifierdescription: What the tool does (shown to AI)inputSchema: JSON Schema for parametershandler: Function that executes the tool
Register in
ToolsModule
# Unit tests
npm run test
# E2E tests (MCP protocol compliance via @modelcontextprotocol/sdk)
npm run test:e2eLogging
The server uses Winston with two transports:
Console (stderr): Only warnings and errors are written to stderr. In VS Code, these appear in the Output panel under the "p4-plan" dropdown. Only
warnanderrorlevel messages are shown to keep the output clean.File: Full debug logs are written to
logs/P4PlanMCP_<timestamp>.log. Use these for detailed troubleshooting.
Note: All console output goes to stderr (never stdout) because stdout is the MCP protocol channel. VS Code labels all stderr output as
[warning]— this is expected behavior and does not indicate a problem.
Protocol
This server uses the MCP stdio transport — communication happens over stdin/stdout using JSON-RPC 2.0 messages. The client spawns the server as a child process.
Transport: stdio (stdin/stdout)
Protocol: JSON-RPC 2.0
Authentication:
P4PLAN_API_AUTH_TOKENenvironment variable (validated at startup)SDK:
@modelcontextprotocol/sdkwithStdioServerTransport
Troubleshooting
Check P4PLAN_API_AUTH_TOKEN is set: The server requires a valid JWT token in the
P4PLAN_API_AUTH_TOKENenvironment variable. If missing, it exits immediately with an error.Check token is valid: If the token is expired or invalid, the server exits with "Authentication failed". Generate a new JWT using the login mutation.
Check GraphQL API is reachable: The P4 Plan GraphQL API must be accessible at the configured
P4PLAN_API_URL(default:http://localhost:4000).
Verify mcp.json syntax: Ensure your
.vscode/mcp.jsonis valid JSON. Check for trailing commas.Check the command is available: If using
npx, ensure Node.js is in VS Code's PATH. Ifnpxis not found, use the absolute path tonodeinstead (see local build config).Reload VS Code window: Press
Cmd+Shift+P→ "Developer: Reload Window"Start a new chat: MCP servers are connected when a new chat session starts.
Check GraphQL server is running: The P4 Plan GraphQL API must be accessible at the configured URL.
Check server logs: In VS Code, check the Output panel → select "p4-plan" from the dropdown to see warnings/errors. For full debug logs, check the
logs/directory.Verify the JWT hasn't expired: Generate a new JWT if needed and update your mcp.json config.
License
This project is licensed under the MIT License. See LICENSE for details.
This 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/perforce/p4plan-mcp'
If you have feedback or need assistance with the MCP directory API, please join our Discord server