linear-mcp-server

anthonylimo90/linear-mcp-server

3.2

If you are the rightful owner of linear-mcp-server and would like to certify it and/or have it hosted online, please leave a comment on the right or send an email to dayong@mcphub.com.

The Linear MCP Server is a Model Context Protocol server that facilitates interaction with the Linear issue tracking system, allowing AI agents to manage issues through a standardized interface.

Tools
5
Resources
0
Prompts
0

Linear MCP Server

A Model Context Protocol (MCP) server that enables AI agents like Claude to interact with Linear issue tracking. Search, create, update issues, and more - all through natural language.


⚡ 30-Second Quick Start

git clone https://github.com/yourusername/linear-mcp-server.git
cd linear-mcp-server
npm install
npm run setup:claude YOUR_LINEAR_API_KEY

Done! Restart Claude Desktop and start using Linear:

  • "Show me all issues assigned to me"
  • "Create a bug report for the login page"
  • "What teams do I have access to?"

Get your API key from Linear Settings


What Can You Do?

Once set up, you can ask Claude to:

📋 Search & Browse

  • "Show me all high-priority bugs assigned to me"
  • "Find issues related to authentication in the Engineering team"
  • "What are my current tasks?"

✨ Create & Update

  • "Create a new feature request for dark mode in the Design team"
  • "Update issue ENG-123 to high priority and assign it to me"
  • "Add a comment to issue PROJ-456 with a status update"

🔍 Get Information

  • "What teams do I have in Linear?"
  • "Show me the workflow states for the Engineering team"
  • "Get the details of issue ENG-789"

All 9 Available Tools

ToolDescription
search_issuesSearch and filter Linear issues with powerful queries
create_issueCreate new issues with priority, assignee, and more
update_issueModify existing issues (title, description, assignee, priority, state)
get_issueRetrieve detailed information about a specific issue
get_my_issuesGet all issues assigned to you
get_teamsList all teams in your workspace
get_workflow_statesView workflow states/statuses for a team
add_commentAdd comments to issues (supports Markdown)
health_checkVerify API connectivity and authentication

Features

Core Functionality

  • Search Issues: Search for issues in Linear with filters for team, status, and assignee
  • Create Issues: Create new issues in Linear with title, description, assignee, and priority
  • Update Issues: Modify existing issues including title, description, assignee, priority, and workflow state
  • Get Issue by ID: Retrieve a specific issue with full details
  • List Teams: Get a list of all teams in your Linear workspace
  • Get My Issues: Retrieve issues assigned to the current user with enhanced filtering
  • Workflow States: List all workflow states/statuses for a team
  • Add Comments: Add comments to existing issues
  • Health Check: Verify API connectivity and authentication status

Performance & Reliability

  • Rate Limiting: Built-in throttling (10 req/sec) to prevent API abuse
  • Viewer Caching: 5-minute cache for user info reduces API calls by ~95%
  • Input Validation: Zod-based schema validation for all tool inputs
  • Error Context: Detailed error messages with relevant parameters for debugging
  • Enhanced Issue Data: Issues include comprehensive team and project information

Code Quality

  • Type Safety: Fully typed interfaces with no any types
  • JSDoc Documentation: Comprehensive API documentation with examples
  • Test Coverage: 64.59% overall coverage with 55 passing tests
  • Clean Architecture: Service layer with singleton pattern
  • Logging: Advanced logging with Pino (stderr-only, no stdout pollution)
  • MCP Protocol Compliance: Full adherence to Model Context Protocol standards

Recent Improvements ✨

Version 0.4.0 brings significant enhancements:

  • New Tools: Added get_issue, get_workflow_states, add_comment, and health_check
  • Rate Limiting: Prevents API abuse with 10 req/sec throttling
  • Performance: Viewer caching reduces API calls by ~95%
  • Validation: Zod schemas for all inputs with helpful error messages
  • Documentation: Comprehensive JSDoc with examples for all methods
  • Type Safety: Eliminated all any types for better TypeScript support
  • Testing: 55 passing tests with 64.59% code coverage
  • Error Context: Detailed error messages with relevant parameters

Architecture

src/
├── index.ts              # Main MCP server entry point
├── types.ts              # TypeScript interfaces and type definitions
├── utils/
│   ├── config.ts         # Configuration management with validation
│   ├── error.ts          # Error handling utilities and custom error classes
│   ├── logger.ts         # Enhanced logging with Pino integration
│   └── validation.ts     # Zod schemas for input validation
└── services/
    └── linearService.ts  # Linear API service with rate limiting and caching

Installation & Setup

Automatic Setup (Recommended)

The fastest way to get started with Claude Desktop:

# 1. Install dependencies and build
npm install

# 2. Run setup with your Linear API key
npm run setup:claude YOUR_LINEAR_API_KEY

# 3. Restart Claude Desktop - you're done!

The setup script will:

  • ✅ Build the project automatically if needed
  • ✅ Verify your API key works
  • ✅ Configure Claude Desktop for your OS
  • ✅ Show personalized examples using your teams

Configuration

Create a .env file in the root directory with the following variables:

LINEAR_API_KEY=your_linear_api_key
LOG_LEVEL=info
NODE_ENV=development

Environment Variables

VariableDescriptionDefaultRequired
LINEAR_API_KEYYour Linear API key-✅ Yes
LOG_LEVELLogging level (trace, debug, info, warn, error)infoNo
NODE_ENVEnvironment (development, production)developmentNo
PORTPort for HTTP server (if used)3000No
ENABLE_RATE_LIMITEnable rate limitingfalseNo
RATE_LIMIT_MAXMaximum requests per window100No
RATE_LIMIT_WINDOW_MSRate limit window in milliseconds60000No

Getting Your Linear API Key

  1. Go to Linear Settings
  2. Click "Create API Key"
  3. Give it a descriptive name and select appropriate scopes
  4. Copy the generated key to your .env file

Usage

Run as MCP Server

The primary use case is as an MCP server that connects via stdio:

npm start

Development Mode

For development with watch mode:

npm run watch

Integration with MCP Clients

This server is designed to work with MCP-compatible clients. The server communicates via stdio transport and provides nine main tools:

  • search_issues - Search and filter Linear issues with enhanced team/project information
  • create_issue - Create new issues in Linear
  • update_issue - Modify existing issues including title, description, assignee, priority, and state
  • get_issue - Retrieve a specific issue by ID with full details
  • get_my_issues - Get issues assigned to the current user with comprehensive filtering
  • get_teams - List all teams in the workspace
  • get_workflow_states - List all workflow states for a team
  • add_comment - Add a comment to an issue
  • health_check - Verify API connectivity and authentication

Client Integration

Claude Desktop

Automatic Setup (see Installation section above) is recommended.

Manual Setup - If you prefer to configure manually:

macOS/Linux

Edit ~/Library/Application Support/Claude/claude_desktop_config.json:

{
  "mcpServers": {
    "linear": {
      "command": "node",
      "args": ["/path/to/your/linear-mcp-server/build/index.js"],
      "env": {
        "LINEAR_API_KEY": "your_linear_api_key_here"
      }
    }
  }
}
Windows

Edit %APPDATA%/Claude/claude_desktop_config.json:

{
  "mcpServers": {
    "linear": {
      "command": "node",
      "args": ["C:\\path\\to\\your\\linear-mcp-server\\build\\index.js"],
      "env": {
        "LINEAR_API_KEY": "your_linear_api_key_here"
      }
    }
  }
}

VS Code Integration

For VS Code with MCP extension support:

  1. Install an MCP-compatible extension (e.g., "MCP Client")
  2. Add server configuration to VS Code settings:
{
  "mcp.servers": {
    "linear": {
      "command": "node",
      "args": ["/path/to/your/linear-mcp-server/build/index.js"],
      "env": {
        "LINEAR_API_KEY": "your_linear_api_key_here"
      }
    }
  }
}

Cursor IDE Integration

For Cursor IDE with MCP support:

  1. Open Cursor Settings (Cmd/Ctrl + ,)
  2. Search for "MCP" or "Model Context Protocol"
  3. Add server configuration:
{
  "mcp.servers": {
    "linear": {
      "command": "node",
      "args": ["/path/to/your/linear-mcp-server/build/index.js"],
      "env": {
        "LINEAR_API_KEY": "your_linear_api_key_here"
      }
    }
  }
}

Or use the Cursor command palette:

  1. Press Cmd/Ctrl + Shift + P
  2. Type "MCP: Add Server"
  3. Enter the server details when prompted

Continue.dev Integration

For Continue.dev (VS Code/JetBrains extension):

Add to your ~/.continue/config.json:

{
  "mcpServers": {
    "linear": {
      "command": "node",
      "args": ["/path/to/your/linear-mcp-server/build/index.js"],
      "env": {
        "LINEAR_API_KEY": "your_linear_api_key_here"
      }
    }
  }
}

Other MCP Clients

For any MCP-compatible client, use these connection details:

  • Transport: stdio
  • Command: node /path/to/linear-mcp-server/build/index.js
  • Environment: Set LINEAR_API_KEY to your Linear API key

Using with NPX (Alternative)

You can also install and run globally using npx:

# Install globally
npm install -g linear-mcp-server

# Use with Claude Desktop
{
  "mcpServers": {
    "linear": {
      "command": "npx",
      "args": ["linear-mcp-server"],
      "env": {
        "LINEAR_API_KEY": "your_linear_api_key_here"
      }
    }
  }
}

Docker Integration

Run the server in a Docker container:

# Dockerfile
FROM node:18-alpine
WORKDIR /app
COPY package*.json ./
RUN npm ci --only=production
COPY build/ ./build/
EXPOSE 3000
CMD ["node", "build/index.js"]
# Build and run
docker build -t linear-mcp-server .
docker run -e LINEAR_API_KEY=your_key linear-mcp-server

Verification

After setting up the integration, verify it's working:

  1. Claude Desktop: Look for Linear tools in the available tools list
  2. VS Code: Check the MCP extension status for successful connection
  3. Command Line: Test the server directly:
echo '{"jsonrpc": "2.0", "id": 1, "method": "tools/list"}' | npm start

You should see a response listing all nine available tools: search_issues, create_issue, update_issue, get_issue, get_my_issues, get_teams, get_workflow_states, add_comment, and health_check.

API Reference

Tool: search_issues

Search for issues in Linear with advanced filtering options.

Input Schema:

{
  "type": "object",
  "properties": {
    "query": {
      "type": "string",
      "description": "Search query"
    },
    "teamId": {
      "type": "string", 
      "description": "Team ID to filter by"
    },
    "status": {
      "type": "string",
      "description": "Status to filter by"
    },
    "assigneeId": {
      "type": "string",
      "description": "Assignee ID to filter by. Use 'me' to filter by the current user"
    },
    "limit": {
      "type": "number",
      "description": "Maximum number of issues to return"
    }
  },
  "required": ["query"]
}

Response:

{
  "issues": [
    {
      "id": "issue-id",
      "identifier": "TEAM-123", 
      "title": "Issue title",
      "description": "Issue description",
      "status": "In Progress",
      "url": "https://linear.app/team/issue/TEAM-123",
      "assignee": "John Doe",
      "createdAt": "2023-01-01T00:00:00.000Z",
      "team": {
        "id": "team-id",
        "name": "Team Name",
        "key": "TEAM"
      },
      "project": {
        "id": "project-id",
        "name": "Project Name",
        "url": "https://linear.app/project/project-id",
        "status": "Active"
      }
    }
  ]
}

Tool: create_issue

Create a new issue in Linear with full customization options.

Input Schema:

{
  "type": "object",
  "properties": {
    "teamId": {
      "type": "string",
      "description": "Team ID"
    },
    "title": {
      "type": "string", 
      "description": "Issue title"
    },
    "description": {
      "type": "string",
      "description": "Issue description"
    },
    "assigneeId": {
      "type": "string",
      "description": "Assignee ID. Use 'me' to assign to the current user"
    },
    "priority": {
      "type": "number",
      "description": "Issue priority (0-4)"
    }
  },
  "required": ["teamId", "title"]
}

Response:

{
  "id": "issue-id",
  "identifier": "TEAM-123",
  "title": "Issue title", 
  "url": "https://linear.app/team/issue/TEAM-123"
}

Tool: update_issue

Update an existing issue in Linear with comprehensive modification options.

Input Schema:

{
  "type": "object",
  "properties": {
    "issueId": {
      "type": "string",
      "description": "Issue ID to update"
    },
    "title": {
      "type": "string",
      "description": "New issue title"
    },
    "description": {
      "type": "string",
      "description": "New issue description"
    },
    "assigneeId": {
      "type": "string",
      "description": "New assignee ID. Use 'me' to assign to current user, empty string to unassign"
    },
    "priority": {
      "type": "number",
      "description": "New issue priority (0-4)"
    },
    "stateId": {
      "type": "string",
      "description": "New workflow state ID"
    }
  },
  "required": ["issueId"]
}

Response:

{
  "id": "issue-id",
  "identifier": "TEAM-123",
  "title": "Updated issue title",
  "url": "https://linear.app/team/issue/TEAM-123",
  "status": "In Progress"
}

Tool: get_my_issues

Get issues assigned to the current user with enhanced filtering and information.

Input Schema:

{
  "type": "object",
  "properties": {
    "limit": {
      "type": "number",
      "description": "Maximum number of issues to return (default: 50, max: 250)"
    }
  }
}

Response:

{
  "issues": [
    {
      "id": "issue-id",
      "identifier": "TEAM-123",
      "title": "Issue title",
      "description": "Issue description",
      "status": "In Progress",
      "url": "https://linear.app/team/issue/TEAM-123",
      "assignee": "John Doe",
      "createdAt": "2023-01-01T00:00:00.000Z",
      "team": {
        "id": "team-id",
        "name": "Team Name",
        "key": "TEAM"
      },
      "project": {
        "id": "project-id",
        "name": "Project Name",
        "url": "https://linear.app/project/project-id",
        "status": "Active"
      }
    }
  ]
}

Tool: get_teams

Get all teams in the workspace.

Input Schema:

{
  "type": "object",
  "properties": {}
}

Response:

{
  "teams": [
    {
      "id": "team-id",
      "name": "Team Name",
      "key": "TEAM",
      "description": "Team description"
    }
  ]
}

Tool: get_issue

Get a specific issue by ID with full details.

Input Schema:

{
  "type": "object",
  "properties": {
    "issueId": {
      "type": "string",
      "description": "The ID of the issue to retrieve"
    }
  },
  "required": ["issueId"]
}

Response:

{
  "id": "issue-id",
  "identifier": "TEAM-123",
  "title": "Issue title",
  "description": "Issue description",
  "status": "In Progress",
  "url": "https://linear.app/team/issue/TEAM-123",
  "assignee": "John Doe",
  "createdAt": "2023-01-01T00:00:00.000Z",
  "team": {
    "id": "team-id",
    "name": "Team Name",
    "key": "TEAM"
  },
  "project": {
    "id": "project-id",
    "name": "Project Name",
    "url": "https://linear.app/project/project-id",
    "status": "Active"
  }
}

Tool: get_workflow_states

Get all workflow states (statuses) for a specific team.

Input Schema:

{
  "type": "object",
  "properties": {
    "teamId": {
      "type": "string",
      "description": "The ID of the team"
    }
  },
  "required": ["teamId"]
}

Response:

{
  "states": [
    {
      "id": "state-id",
      "name": "In Progress",
      "type": "started",
      "description": "Work is in progress",
      "position": 1
    },
    {
      "id": "state-id-2",
      "name": "Done",
      "type": "completed",
      "description": "Work is completed",
      "position": 2
    }
  ]
}

Tool: add_comment

Add a comment to an existing issue.

Input Schema:

{
  "type": "object",
  "properties": {
    "issueId": {
      "type": "string",
      "description": "The ID of the issue to comment on"
    },
    "body": {
      "type": "string",
      "description": "The comment text (supports markdown)"
    }
  },
  "required": ["issueId", "body"]
}

Response:

{
  "id": "comment-id",
  "body": "This is my comment",
  "createdAt": "2023-01-01T00:00:00.000Z",
  "issueId": "issue-id"
}

Tool: health_check

Check the health and connectivity of the Linear API.

Input Schema:

{
  "type": "object",
  "properties": {}
}

Response:

{
  "status": "healthy",
  "apiConnected": true,
  "userId": "user-id",
  "userName": "John Doe"
}

Or if unhealthy:

{
  "status": "unhealthy",
  "apiConnected": false,
  "error": "Error message"
}

Development

Project Structure

  • src/index.ts: Main server entry point with MCP protocol handling
  • src/types.ts: TypeScript type definitions for all interfaces
  • src/utils/: Utility modules for configuration, logging, and error handling
  • src/services/: Business logic layer for Linear API interactions

Key Design Patterns

  • Singleton Pattern: Used for services to ensure single instance management
  • Error Boundary Pattern: Comprehensive error handling at all levels
  • Configuration Pattern: Centralized configuration with validation
  • Service Layer Pattern: Clean separation between API logic and MCP handling

Available Scripts

npm run build         # Build TypeScript to JavaScript
npm run watch         # Watch mode for development
npm start             # Start the MCP server
npm test              # Run all tests with coverage
npm run setup:claude  # Auto-configure Claude Desktop
npm run repair:claude # Fix malformed Claude Desktop config

Testing

The project includes comprehensive test coverage:

# Run all tests
npm test

# Watch mode for test development
npm test -- --watch

# Run specific test file
npm test -- tests/unit/services/linearService.test.ts

Current Test Coverage:

  • Overall: 64.59% statement coverage
  • linearService.ts: 70.21% coverage
  • 55 passing tests across unit and integration suites

Tests include:

  • Unit tests for LinearService methods
  • Integration tests for MCP tool handlers
  • Configuration validation tests
  • Error handling scenarios

Troubleshooting

Common Issues

  1. "Missing LINEAR_API_KEY": Ensure your .env file exists and contains a valid Linear API key
  2. "Server does not support tools": This indicates an MCP client compatibility issue
  3. "Failed to fetch teams": Check your Linear API key permissions
  4. JSON syntax errors in Claude config: Use the repair script to fix malformed configuration files

MCP Communication Issues

If you encounter JSON parsing errors or communication issues with MCP clients (Cursor, Claude Desktop, etc.):

Symptoms
  • "Expected ',' or ']' after array element in JSON at position X"
  • "Unexpected token 'X' is not valid JSON"
  • Client shows "Client error for command" repeatedly
  • MCP tools not appearing or failing to execute
Root Cause

These issues typically occur when log output interferes with JSON-RPC communication. The MCP protocol requires:

  • stdout: JSON-RPC messages ONLY
  • stderr: All logging output
Solutions

✅ Server-Side (Already Fixed in v0.3.1+) The server now properly separates logging from JSON-RPC communication:

# Logs go to stderr, JSON-RPC to stdout
echo '{"jsonrpc": "2.0", "id": 1, "method": "tools/list"}' | node build/index.js

✅ Client-Side Verification

  1. Restart your MCP client (Cursor, Claude Desktop, etc.) after updating
  2. Check client logs for successful connection messages
  3. Test basic functionality: Try listing tools or calling a simple tool

✅ Manual Testing

# Test tools list (should return clean JSON)
echo '{"jsonrpc": "2.0", "id": 1, "method": "tools/list"}' | node build/index.js

# Test simple tool call
echo '{"jsonrpc": "2.0", "id": 1, "method": "tools/call", "params": {"name": "get_teams", "arguments": {}}}' | node build/index.js

✅ Debug Mode For troubleshooting, enable debug logging (outputs to stderr):

LOG_LEVEL=debug
NODE_ENV=development

If issues persist after updating to v0.3.1+, please check that you're using the latest build:

npm run build  # Rebuild with latest fixes

JSON Configuration Errors

If you see JSON parsing errors like "Unexpected token" or "is not valid JSON" in Claude Desktop configuration:

Quick Fix
# Run the repair script to fix JSON syntax issues
npm run repair:claude

# Then try setting up again
npm run setup:claude
Manual Fix
  1. Close Claude Desktop completely
  2. Navigate to the config file location:
    • macOS: ~/Library/Application Support/Claude/claude_desktop_config.json
    • Windows: %APPDATA%/Claude/claude_desktop_config.json
    • Linux: ~/.config/Claude/claude_desktop_config.json
  3. Delete the config file (a backup will be created automatically)
  4. Run the setup script again: npm run setup:claude

Debugging

Enable debug logging by setting LOG_LEVEL=debug in your .env file:

LOG_LEVEL=debug

This will provide detailed logs of all API calls and server operations.

Performance

The server is optimized for performance with:

  • Rate Limiting: Throttles API calls to 10 requests/second to prevent API abuse
  • Viewer Caching: 5-minute TTL cache for user information reduces API calls by ~95%
  • Parallel Processing: Concurrent API calls using Promise.all()
  • Singleton Pattern: Efficient resource management
  • Structured Logging: Minimal performance overhead (stderr-only)
  • TypeScript: Compile-time optimizations
  • Input Validation: Fast Zod-based schema validation

License

MIT

Contributing

Contributions are welcome! Please feel free to submit a Pull Request.

Development Setup

  1. Fork the repository
  2. Clone your fork: git clone https://github.com/yourusername/linear-mcp-server.git
  3. Install dependencies: npm install
  4. Create a .env file with your Linear API key
  5. Make your changes
  6. Run tests: npm test
  7. Submit a pull request

Code Style

  • Use TypeScript for all new code
  • Follow the existing code organization patterns
  • Add appropriate error handling
  • Include JSDoc comments for public APIs
  • Ensure all tests pass