{/* 本页面由 website/scripts/generate-skill-docs.py 从技能 SKILL.md 自动生成。请编辑源 SKILL.md 而非本页面。 */}
Linear
Linear:通过 GraphQL + curl 管理问题、项目、团队。
技能元数据
| 来源 | 内置(默认安装) |
| 路径 | skills/productivity/linear |
| 版本 | 1.0.0 |
| 作者 | Hermes Agent |
| 许可证 | MIT |
| 平台 | linux, macos, windows |
| 标签 | Linear, Project Management, Issues, GraphQL, API, Productivity |
参考:完整 SKILL.md
:::info 以下是此技能被触发时 Hermes 加载的完整技能定义。这是技能激活时代理所看到的指令。 :::
Linear — 问题与项目管理
通过 GraphQL API 使用 curl 直接管理 Linear 的问题、项目和团队。无需 MCP 服务器、OAuth 流程或额外依赖。
设置
- 从 Linear Settings > Account > Security & access > Personal API keys 获取个人 API 密钥(URL:https://linear.app/settings/account/security)。注意:组织级别的 Settings > API 页面仅显示 OAuth 应用和工作空间成员密钥,不显示个人密钥。
- 在环境中设置
LINEAR_API_KEY(通过hermes setup或你的环境配置)
API 基础
- 端点:
https://api.linear.app/graphql(POST) - 认证头:
Authorization: $LINEAR_API_KEY(API 密钥无需 “Bearer” 前缀) - 所有请求均为 POST,使用
Content-Type: application/json - UUID 和短标识符(例如
ENG-123)均可用于issue(id:)
基础 curl 模式:
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ viewer { id name } }"}' | python3 -m json.toolPython 辅助脚本(更人性化的替代方案)
对于无需手写 GraphQL 的快速单行命令,此技能附带了一个使用 Python 标准库的 CLI 脚本 scripts/linear_api.py。零依赖。相同认证方式(读取 LINEAR_API_KEY)。
SCRIPT=$(dirname "$(find ~/.hermes -path '*skills/productivity/linear/scripts/linear_api.py' 2>/dev/null | head -1)")/linear_api.py
python3 "$SCRIPT" whoami
python3 "$SCRIPT" list-teams
python3 "$SCRIPT" get-issue ENG-42
python3 "$SCRIPT" get-document 38359beef67c # 通过 URL 中的 slugId 获取文档
python3 "$SCRIPT" raw 'query { viewer { name } }'所有子命令:whoami、list-teams、list-projects、list-states、list-issues、get-issue、search-issues、create-issue、update-issue、update-status、add-comment、list-documents、get-document、search-documents、raw。使用 --help 查看参数。
当你想要快速答案而不想编写 GraphQL 时使用此脚本。当你需要脚本未封装查询或想内联组合筛选条件时使用 curl。
工作流状态
Linear 使用带有 type 字段的 WorkflowState 对象。6 种状态类型:
| 类型 | 描述 |
|---|---|
triage | 待审核的传入问题 |
backlog | 已确认但尚未规划 |
unstarted | 已规划/就绪但尚未开始 |
started | 正在积极处理 |
completed | 已完成 |
canceled | 不做 |
每个团队都有自己的命名状态(例如,“In Progress” 的类型为 started)。要更改问题的状态,你需要目标状态的 stateId(UUID)——首先查询工作流状态。
优先级值: 0 = 无,1 = 紧急,2 = 高,3 = 中,4 = 低
常用查询
获取当前用户
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ viewer { id name email } }"}' | python3 -m json.tool列出团队
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ teams { nodes { id name key } } }"}' | python3 -m json.tool列出团队的工作流状态
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ workflowStates(filter: { team: { key: { eq: \\\"ENG\\\" } } }) { nodes { id name type } } }"}' | python3 -m json.tool列出问题(前 20 个)
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ issues(first: 20) { nodes { identifier title priority state { name type } assignee { name } team { key } url } pageInfo { hasNextPage endCursor } } }"}' | python3 -m json.tool列出分配给我的问题
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ viewer { assignedIssues(first: 25) { nodes { identifier title state { name type } priority url } } } }"}' | python3 -m json.tool获取单个问题(按标识符如 ENG-123)
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ issue(id: \\\"ENG-123\\\") { id identifier title description priority state { id name type } assignee { id name } team { key } project { name } labels { nodes { name } } comments { nodes { body user { name } createdAt } } url } }"}' | python3 -m json.tool按文本搜索问题
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ issueSearch(query: \\\"bug login\\\", first: 10) { nodes { identifier title state { name } assignee { name } url } } }"}' | python3 -m json.tool按状态类型筛选问题
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ issues(filter: { state: { type: { in: [\\\"started\\\"] } } }, first: 20) { nodes { identifier title state { name } assignee { name } } } }"}' | python3 -m json.tool按团队和负责人筛选
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ issues(filter: { team: { key: { eq: \\\"ENG\\\" } }, assignee: { email: { eq: \\\"user@example.com\\\" } } }, first: 20) { nodes { identifier title state { name } priority } } }"}' | python3 -m json.tool列出项目
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ projects(first: 20) { nodes { id name description progress lead { name } teams { nodes { key } } url } } }"}' | python3 -m json.tool列出团队成员
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ users { nodes { id name email active } } }"}' | python3 -m json.tool列出标签
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ issueLabels { nodes { id name color } } }"}' | python3 -m json.tool常用变更操作
创建问题
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{
"query": "mutation($input: IssueCreateInput!) { issueCreate(input: $input) { success issue { id identifier title url } } }",
"variables": {
"input": {
"teamId": "TEAM_UUID",
"title": "Fix login bug",
"description": "Users cannot login with SSO",
"priority": 2
}
}
}' | python3 -m json.tool更新问题状态
首先从上述工作流状态查询中获取目标状态 UUID,然后:
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "mutation { issueUpdate(id: \\\"ENG-123\\\", input: { stateId: \\\"STATE_UUID\\\" }) { success issue { identifier state { name type } } } }"}' | python3 -m json.tool分配问题
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "mutation { issueUpdate(id: \\\"ENG-123\\\", input: { assigneeId: \\\"USER_UUID\\\" }) { success issue { identifier assignee { name } } } }"}' | python3 -m json.tool设置优先级
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "mutation { issueUpdate(id: \\\"ENG-123\\\", input: { priority: 1 }) { success issue { identifier priority } } }"}' | python3 -m json.tool添加评论
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "mutation { commentCreate(input: { issueId: \\\"ISSUE_UUID\\\", body: \\\"Investigated. Root cause is X.\\\" }) { success comment { id body } } }"}' | python3 -m json.tool设置截止日期
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "mutation { issueUpdate(id: \\\"ENG-123\\\", input: { dueDate: \\\"2026-04-01\\\" }) { success issue { identifier dueDate } } }"}' | python3 -m json.tool为问题添加标签
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "mutation { issueUpdate(id: \\\"ENG-123\\\", input: { labelIds: [\\\"LABEL_UUID_1\\\", \\\"LABEL_UUID_2\\\"] }) { success issue { identifier labels { nodes { name } } } } }"}' | python3 -m json.tool将问题添加到项目
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "mutation { issueUpdate(id: \\\"ENG-123\\\", input: { projectId: \\\"PROJECT_UUID\\\" }) { success issue { identifier project { name } } } }"}' | python3 -m json.tool创建项目
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{
"query": "mutation($input: ProjectCreateInput!) { projectCreate(input: $input) { success project { id name url } } }",
"variables": {
"input": {
"name": "Q2 Auth Overhaul",
"description": "Replace legacy auth with OAuth2 and PKCE",
"teamIds": ["TEAM_UUID"]
}
}
}' | python3 -m json.tool文档
Linear **文档(Documents)**是与问题并列存储的散文式文档(RFC、规格说明、笔记)。它们有自己的 documents 根查询和 document(id:) 单条获取。
文档 URL 和 slugId
文档 URL 形如:
https://linear.app/<workspace>/document/<slug>-<hexSlugId>
尾部的十六进制段是 slugId。例如:https://linear.app/nousresearch/document/rfc-hermes-permission-gateway-discord-38359beef67c → slugId 为 38359beef67c。
重要的架构细节: Markdown 正文在 content 字段中。ProseMirror JSON 在 contentState 中(不是 contentData——该字段不存在,API 会返回 400)。
按 slugId 获取文档
document(id:) 仅接受 UUID。要通过 URL 的十六进制 slug 获取,请筛选集合:
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "query($s: String!) { documents(filter: { slugId: { eq: $s } }, first: 1) { nodes { id title content contentState slugId url creator { name } project { name } updatedAt } } }", "variables": {"s": "38359beef67c"}}' \
| python3 -m json.tool或通过 Python 辅助脚本:
python3 scripts/linear_api.py get-document 38359beef67c按 UUID 获取文档
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ document(id: \\\"11700cff-b514-4db3-afcc-3ed1afacba1c\\\") { title content url } }"}' \
| python3 -m json.tool列出最近文档
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ documents(first: 25, orderBy: updatedAt) { nodes { id title slugId url updatedAt project { name } } } }"}' \
| python3 -m json.tool按标题搜索文档
Linear 的架构没有 searchDocuments 根查询。改用标题子字符串筛选:
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ documents(filter: { title: { containsIgnoreCase: \\\"RFC\\\" } }, first: 25) { nodes { title slugId url } } }"}' \
| python3 -m json.tool分页
Linear 使用 Relay 风格的游标分页:
# 第一页
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ issues(first: 20) { nodes { identifier title } pageInfo { hasNextPage endCursor } } }"}' | python3 -m json.tool
# 下一页——使用上一响应中的 endCursor
curl -s -X POST https://api.linear.app/graphql \
-H "Authorization: $LINEAR_API_KEY" \
-H "Content-Type: application/json" \
-d '{"query": "{ issues(first: 20, after: \\\"CURSOR_FROM_PREVIOUS\\\") { nodes { identifier title } pageInfo { hasNextPage endCursor } } }"}' | python3 -m json.tool默认页面大小:50。最大:250。始终使用 first: N 限制结果。
筛选参考
比较器:eq、neq、in、nin、lt、lte、gt、gte、contains、startsWith、containsIgnoreCase
使用 or: [...] 组合筛选条件实现 OR 逻辑(筛选对象内默认为 AND)。
典型工作流程
- 查询团队以获取团队 ID 和密钥
- 查询工作流状态以获取目标团队的状态 UUID
- 列出或搜索问题以查找需要处理的内容
- 创建问题时提供团队 ID、标题、描述、优先级
- 更新状态通过将
stateId设置为目标工作流状态 - 添加评论以跟踪进度
- 标记完成通过将
stateId设置为团队的”completed”类型状态
速率限制
- 每个 API 密钥每小时 5,000 请求
- 每小时 3,000,000 复杂度点数
- 使用
first: N限制结果并降低复杂度成本 - 监控
X-RateLimit-Requests-Remaining响应头
重要提示
- 始终使用带有
curl的terminal工具进行 API 调用——不要使用web_extract或browser - 始终检查 GraphQL 响应中的
errors数组——HTTP 200 仍可能包含错误 - 如果在创建问题时省略了
stateId,Linear 默认为第一个 backlog 状态 description字段支持 Markdown- 使用
python3 -m json.tool或jq格式化 JSON 响应以提高可读性