| name | tapd-idle |
| description | TAPD 项目管理平台集成。管理需求、缺陷、任务、工时、迭代、测试用例、Wiki等。**必须使用此技能当**: (1) 用户提到 TAPD、看板、敏捷、需求、缺陷、任务、迭代、工时、测试用例 (2) 需要查询/创建/更新项目实体或查看进度统计 (3) 用户说"看看我的任务"、"这周工时"、"有多少bug"、"需求进度"、"前端组任务"、"缺陷统计"等未明确提到TAPD但需要项目数据的场景。即使用户只说"看看任务"、"最近有什么bug"也应主动触发此技能。 |
TAPD Skill
TAPD 敏捷研发管理平台集成,通过 Python 脚本调用 TAPD API。
Agent 使用指南
何时使用此 Skill
当用户需要与 TAPD 平台交互时,主动调用此 skill:
触发关键词:
- TAPD、看板、敏捷、需求管理
- 查询/创建/更新 需求、缺陷、任务、迭代
- 工时填写、评论管理
- 测试用例、Wiki 管理
Agent 工作流程
- 理解用户意图 → 确定 TAPD 操作类型
- 构建命令 → 使用
python scripts/tapd.py <command> [参数]
- 执行脚本 → 使用 Bash 工具运行
- 处理结果 → 解析 JSON 输出,向用户报告
示例对话
用户: "查看需求 1167459320001114969 的详情"
Agent:
1. 识别操作:查询需求详情
2. 执行: python scripts/tapd.py get_stories_or_tasks --workspace_id 67459320 --entity_type stories --id 1167459320001114969
3. 解析返回的需求信息并向用户报告
环境配置
使用前需要配置以下环境变量:
export TAPD_ACCESS_TOKEN="你的个人访问令牌"
export TAPD_API_USER="API账号"
export TAPD_API_PASSWORD="API密钥"
export TAPD_API_BASE_URL="https://api.tapd.cn"
export TAPD_BASE_URL="https://www.tapd.cn"
export CURRENT_USER_NICK="你的昵称"
常用项目 ID
| 项目名称 | workspace_id |
|---|
| 聚宝赞 | 50372234 |
常用用户组 role_id
注意: get_group_members 命令需要管理员权限,普通用户可使用 get_workspace_users 结合 role_id 筛选特定组成员。
| 用户组名称 | role_id |
|---|
| 管理员 | 1000000000000000002 |
| 普通成员 | 1000000000000000089 |
| QA部 | 1150372234001000002 |
| 产品 | 1150372234001000004 |
| 客成 | 1150372234001000008 |
| 商品组 | 1150372234001000009 |
| 资金组 | 1150372234001000010 |
| 交易组 | 1150372234001000011 |
| 架构组 | 1150372234001000015 |
| 主管 | 1150372234001000016 |
| 场景组 | 1150372234001000017 |
| 用户组 | 1150372234001000018 |
| 前端组 | 1150372234001000020 |
| 运维 | 1150372234001000021 |
命令速查表
项目与用户
| 命令 | 说明 |
|---|
get_user_participant_projects | 获取用户参与的项目列表 |
get_workspace_info | 获取项目信息 |
get_workitem_types | 获取需求类别 |
get_roles | 获取用户组ID对照关系 |
get_workspace_users | 获取项目成员列表 |
get_group_members | 获取指定用户组的成员 |
需求/任务
| 命令 | 说明 |
|---|
get_stories_or_tasks | 查询需求/任务 |
create_story_or_task | 创建需求/任务 |
update_story_or_task | 更新需求/任务 |
get_story_or_task_count | 获取数量 |
get_stories_fields_lable | 字段中英文对照 |
get_stories_fields_info | 字段及候选值 |
缺陷
| 命令 | 说明 |
|---|
get_bug | 查询缺陷 |
create_bug | 创建缺陷 |
update_bug | 更新缺陷 |
get_bug_count | 获取数量 |
迭代
| 命令 | 说明 |
|---|
get_iterations | 查询迭代 |
create_iteration | 创建迭代 |
update_iteration | 更新迭代 |
评论
| 命令 | 说明 |
|---|
get_comments | 查询评论 |
create_comments | 创建评论 |
update_comments | 更新评论 |
工时
| 命令 | 说明 |
|---|
get_timesheets | 查询工时 |
add_timesheets | 填写工时 |
update_timesheets | 更新工时 |
其他命令
| 分类 | 命令 |
|---|
| 附件 | get_entity_attachments, get_image |
| 自定义字段 | get_entity_custom_fields |
| 工作流 | get_workflows_status_map, get_workflows_all_transitions, get_workflows_last_steps |
| 测试用例 | get_tcases, create_or_update_tcases, create_tcases_batch |
| Wiki | get_wiki, create_wiki, update_wiki |
| 关联 | get_related_bugs, entity_relations |
| 源码 | get_commit_msg |
| 消息 | send_qiwei_message |
使用方式
python scripts/tapd.py <command> [参数]
所有命令默认输出 JSON 格式结果。
使用示例
查询需求
python scripts/tapd.py get_stories_or_tasks --workspace_id 123 --entity_type stories --id 1167459320001114969
python scripts/tapd.py get_stories_or_tasks --workspace_id 123 --entity_type stories --name "%登录%" --limit 20
python scripts/tapd.py get_stories_or_tasks --workspace_id 123 --entity_type stories --v_status "已验收"
查询任务
重要: 使用 --owner 参数指定中文姓名查询任务可能返回空结果,建议先查询所有任务再筛选。
任务时间相关字段:
| 字段 | 说明 |
|---|
begin | 预计开始时间 |
due | 预计结束时间 |
effort | 预估工时 |
effort_completed | 完成工时 |
remain | 剩余工时 |
created | 创建时间 |
modified | 最后修改时间 |
completed | 完成时间 |
任务状态值:open(未开始)、progressing(进行中)、done(已完成)
python scripts/tapd.py get_stories_or_tasks --workspace_id 50372234 --entity_type tasks \
--fields "id,name,owner,status,begin,due,effort,effort_completed,remain,created,modified"
前端组任务统计(推荐方式)
重要: 直接使用 --owner 参数指定中文姓名可能匹配失败,建议查询所有任务后在结果中筛选。
前端组成员通过 role_id 动态查询,role_id 1150372234001000020 对应前端组。
分状态查询并筛选前端组成员:
python scripts/tapd.py get_stories_or_tasks --workspace_id 50372234 --entity_type tasks \
--fields "id,name,owner,status,begin,due,effort,effort_completed,remain" \
--status done --limit 200
python scripts/tapd.py get_stories_or_tasks --workspace_id 50372234 --entity_type tasks \
--fields "id,name,owner,status,begin,due,effort,effort_completed,remain" \
--status progressing --limit 200
python scripts/tapd.py get_stories_or_tasks --workspace_id 50372234 --entity_type tasks \
--fields "id,name,owner,status,begin,due,effort,effort_completed,remain" \
--status open --limit 200
完整统计脚本(本周任务进度,按成员、时间范围、工时、状态):
{
python scripts/tapd.py get_workspace_users --workspace_id 50372234
sleep 1
python scripts/tapd.py get_stories_or_tasks --workspace_id 50372234 --entity_type tasks \
--fields "id,name,owner,status,begin,due,effort,effort_completed" \
--status done --limit 200
sleep 2
python scripts/tapd.py get_stories_or_tasks --workspace_id 50372234 --entity_type tasks \
--fields "id,name,owner,status,begin,due,effort,effort_completed" \
--status progressing --limit 200
} | python3 -c "
import json
import sys
from datetime import datetime
# 前端组 role_id
FRONTEND_ROLE_ID = '1150372234001000020'
# 本周时间范围
week_start = datetime.strptime('2026-03-03', '%Y-%m-%d')
week_end = datetime.strptime('2026-03-10', '%Y-%m-%d')
# 解析所有输入
all_json = []
for line in sys.stdin:
line = line.strip()
if line and line.startswith('{'):
try:
all_json.append(json.loads(line))
except:
pass
# 提取前端组成员
frontend_members = set()
for data in all_json:
if data.get('status') == 1:
for item in data.get('data', []):
user = item.get('UserWorkspace', {})
if FRONTEND_ROLE_ID in user.get('role_id', []):
name = user.get('name') or user.get('user')
if name:
frontend_members.add(name)
# 合并任务数据
all_tasks = []
for data in all_json:
all_tasks.extend(data.get('data', []))
# 筛选前端组成员任务
member_stats = {}
for item in all_tasks:
task = item.get('Task', {})
if not task:
continue
owner = task.get('owner', '').rstrip(';')
# 筛选前端组成员
if owner not in frontend_members:
continue
effort = float(task.get('effort', 0) or 0)
effort_completed = float(task.get('effort_completed', 0) or 0)
status = task.get('status', '')
if owner not in member_stats:
member_stats[owner] = {'effort': 0, 'completed': 0, 'count': 0, 'status': {}}
member_stats[owner]['effort'] += effort
member_stats[owner]['completed'] += effort_completed
member_stats[owner]['count'] += 1
member_stats[owner]['status'][status] = member_stats[owner]['status'].get(status, 0) + 1
# 输出统计结果
print('前端组本周任务进度统计')
print('=' * 70)
print(f\"{'成员':<10} {'任务数':<6} {'预估工时':<8} {'完成工时':<8} {'状态分布'}\")
print('-' * 70)
for member, stats in sorted(member_stats.items(), key=lambda x: -x[1]['effort']):
status_str = ', '.join([f\"{k}:{v}\" for k, v in stats['status'].items()])
print(f\"{member:<10} {stats['count']:<6} {stats['effort']:<8.0f} {stats['completed']:<8.0f} {status_str}\")
print('-' * 70)
print(f\"{'合计':<10} {sum(s['count'] for s in member_stats.values()):<6} {sum(s['effort'] for s in member_stats.values()):<8.0f} {sum(s['completed'] for s in member_stats.values()):<8.0f}\")
"
筛选本周时间范围内的任务
{
python scripts/tapd.py get_workspace_users --workspace_id 50372234
sleep 1
python scripts/tapd.py get_stories_or_tasks --workspace_id 50372234 --entity_type tasks \
--fields "id,name,owner,status,begin,due,effort,effort_completed,remain" \
--limit 500
} | python3 -c "
import json
import sys
from datetime import datetime
FRONTEND_ROLE_ID = '1150372234001000020'
all_json = []
for line in sys.stdin:
line = line.strip()
if line and line.startswith('{'):
try:
all_json.append(json.loads(line))
except:
pass
# 提取前端组成员
frontend_members = set()
for data in all_json:
if data.get('status') == 1:
for item in data.get('data', []):
user = item.get('UserWorkspace', {})
if FRONTEND_ROLE_ID in user.get('role_id', []):
name = user.get('name') or user.get('user')
if name:
frontend_members.add(name)
week_start = datetime.strptime('2026-03-03', '%Y-%m-%d')
week_end = datetime.strptime('2026-03-10', '%Y-%m-%d')
print('本周任务列表(预计时间在范围内):')
for data in all_json:
for item in data.get('data', []):
task = item.get('Task', {})
if not task:
continue
owner = task.get('owner', '').rstrip(';')
if owner not in frontend_members:
continue
begin = task.get('begin', '')
due = task.get('due', '')
effort = task.get('effort', '0')
if begin:
begin_dt = datetime.strptime(begin[:10], '%Y-%m-%d')
if week_start <= begin_dt <= week_end:
print(f\" {owner} | {task.get('name')[:30]} | {begin}~{due} | 预估: {effort}h\")
"
创建需求
python scripts/tapd.py create_story_or_task --workspace_id 123 \
--name "用户登录功能" \
--description "## 需求描述\n用户可以通过账号密码登录系统" \
--priority_label "高" \
--owner "zhangsan" \
--iteration_name "Sprint 1"
更新需求状态
python scripts/tapd.py update_story_or_task --workspace_id 123 \
--id 1167459320001114969 \
--v_status "实现中"
查询缺陷
python scripts/tapd.py get_bug --workspace_id 123 --title "%登录失败%" --priority_label "高"
缺陷查询参数说明
重要: TAPD API 参数与脚本参数对应关系:
--owner → API 参数 current_owner(处理人)
--developer → API 参数 de(开发人员)
| 脚本参数 | API 参数 | 说明 |
|---|
--owner | current_owner | 处理人 |
--developer | de | 开发人员 |
--created | created | 创建时间范围 |
--version_report | version_report | 发现版本 |
--closed | status | 结束状态别名(closed|rejected|suspended) |
--open | status | 未结束状态别名(unconfirmed|in_progress|resolved|verified) |
--exclude_status | - | 排除指定状态(结果过滤) |
缺陷状态说明
| 英文状态 | 中文状态 | 类型 |
|---|
unconfirmed | 缺陷池 | 未结束 |
in_progress | 修复中 | 未结束 |
resolved | 待测试 | 未结束 |
verified | 测试中 | 未结束 |
closed | 完成 | 结束 |
rejected | 拒绝 | 结束 |
suspended | 关闭 | 结束 |
缺陷查询示例
按开发人员过滤:使用 --developer 参数
python scripts/tapd.py get_bug --workspace_id 50372234 --developer "张三"
按处理人过滤:使用 --owner 参数
python scripts/tapd.py get_bug --workspace_id 50372234 --owner "张三"
按状态别名查询:使用 --closed 或 --open 参数
python scripts/tapd.py get_bug --workspace_id 50372234 --closed --developer "陈方杰"
python scripts/tapd.py get_bug --workspace_id 50372234 --open --developer "陈方杰"
排除指定状态:使用 --exclude_status 参数
python scripts/tapd.py get_bug --workspace_id 50372234 --exclude_status "closed|suspended" --developer "陈方杰"
按时间范围查询:
python scripts/tapd.py get_bug --workspace_id 50372234 \
--created "2026-02-01~2026-02-28"
按发现版本查询:
python scripts/tapd.py get_bug --workspace_id 50372234 \
--version_report "线上版本"
python scripts/tapd.py get_bug --workspace_id 50372234 \
--version_report "测试版本"
人员缺陷统计
查询某人某月的所有缺陷:
python scripts/tapd.py get_bug --workspace_id 50372234 \
--developer "姓名" \
--created "2026-02-01~2026-02-28" \
--fields "id,title,severity,status,version_report,created"
统计某人缺陷数量:
python scripts/tapd.py get_bug_count --workspace_id 50372234 \
--developer "姓名" \
--created "2026-02-01~2026-02-28"
区分线上/测试缺陷:
python scripts/tapd.py get_bug_count --workspace_id 50372234 \
--developer "姓名" \
--version_report "线上版本" \
--created "2026-02-01~2026-02-28"
python scripts/tapd.py get_bug_count --workspace_id 50372234 \
--developer "姓名" \
--version_report "测试版本" \
--created "2026-02-01~2026-02-28"
创建缺陷
python scripts/tapd.py create_bug --workspace_id 123 \
--title "登录页面显示异常" \
--description "输入正确密码后提示错误" \
--priority_label "高" \
--severity "严重"
迭代管理
python scripts/tapd.py get_iterations --workspace_id 123
python scripts/tapd.py create_iteration --workspace_id 123 \
--name "Sprint 1" \
--startdate "2024-01-01" \
--enddate "2024-01-14" \
--creator "zhangsan"
工时管理
python scripts/tapd.py get_timesheets --workspace_id 123 --entity_type story --entity_id 1167459320001114969
python scripts/tapd.py get_timesheets --workspace_id 123 --spentdate "2026-03-10"
python scripts/tapd.py get_timesheets --workspace_id 123 --spentdate "2026-03-03~2026-03-10"
python scripts/tapd.py get_timesheets --workspace_id 123 --owner "张三" --spentdate "2026-03-10"
python scripts/tapd.py add_timesheets --workspace_id 123 \
--entity_type story \
--entity_id 1167459320001114969 \
--timespent "4" \
--spentdate "2024-01-08" \
--memo "开发登录功能"
工时统计示例
按成员筛选前端组工时:
{
python scripts/tapd.py get_workspace_users --workspace_id 50372234
sleep 1
python scripts/tapd.py get_timesheets --workspace_id 50372234 --spentdate "2026-03-03~2026-03-10"
} | python3 -c "
import json
import sys
FRONTEND_ROLE_ID = '1150372234001000020'
all_json = []
for line in sys.stdin:
line = line.strip()
if line and line.startswith('{'):
try:
all_json.append(json.loads(line))
except:
pass
# 提取前端组成员
frontend_members = set()
for data in all_json:
if data.get('status') == 1:
for item in data.get('data', []):
user = item.get('UserWorkspace', {})
if FRONTEND_ROLE_ID in user.get('role_id', []):
name = user.get('name') or user.get('user')
if name:
frontend_members.add(name)
member_stats = {}
for data in all_json:
for item in data.get('data', []):
ts = item.get('Timesheet', {})
if not ts:
continue
owner = ts.get('owner', '')
timespent = float(ts.get('timespent', 0))
spentdate = ts.get('spentdate', '')
if owner in frontend_members:
if owner not in member_stats:
member_stats[owner] = {'total_hours': 0, 'days': {}}
member_stats[owner]['total_hours'] += timespent
if spentdate not in member_stats[owner]['days']:
member_stats[owner]['days'][spentdate] = 0
member_stats[owner]['days'][spentdate] += timespent
print('前端组本周工时统计')
for member, stats in sorted(member_stats.items(), key=lambda x: -x[1]['total_hours']):
print(f\"{member}: {stats['total_hours']:.0f}h\")
print(f\"总计: {sum(s['total_hours'] for s in member_stats.values()):.0f}h\")
"
评论管理
python scripts/tapd.py get_comments --workspace_id 123 \
--entry_type stories \
--entry_id 1167459320001114969
python scripts/tapd.py create_comments --workspace_id 123 \
--entry_type stories \
--entry_id 1167459320001114969 \
--description "看起来不错,可以继续完善"
用户组与成员
python scripts/tapd.py get_roles --workspace_id 123
python scripts/tapd.py get_workspace_users --workspace_id 123
python scripts/tapd.py get_group_members --workspace_id 123 --group_name "前端开发"
python scripts/tapd.py get_workspace_users --workspace_id 50372234 | python3 -c "
import json
import sys
data = json.load(sys.stdin)
if data.get('status') == 1:
members = []
for item in data.get('data', []):
user = item.get('UserWorkspace', {})
if '1150372234001000020' in user.get('role_id', []):
members.append({
'user': user.get('user'),
'name': user.get('name') or user.get('user') or 'Unknown'
})
print(f'前端组共有 {len(members)} 人:')
for m in sorted(members, key=lambda x: x['name'] or ''):
print(f\" - {m['name']} ({m['user']})\")
"
常用命令速查
python scripts/tapd.py get_stories_or_tasks --workspace_id $WS_ID --entity_type stories
python scripts/tapd.py create_story_or_task --workspace_id $WS_ID --name "标题"
python scripts/tapd.py update_story_or_task --workspace_id $WS_ID --id $ID --v_status "状态"
python scripts/tapd.py get_bug --workspace_id $WS_ID
python scripts/tapd.py create_bug --workspace_id $WS_ID --title "标题"
python scripts/tapd.py get_iterations --workspace_id $WS_ID
python scripts/tapd.py create_iteration --workspace_id $WS_ID --name "Sprint X" --startdate "2024-01-01" --enddate "2024-01-14"
python scripts/tapd.py add_timesheets --workspace_id $WS_ID --entity_type story --entity_id $ID --timespent 4 --spentdate "2024-01-08"
python scripts/tapd.py create_comments --workspace_id $WS_ID --entry_type stories --entry_id $ID --description "评论内容"
状态值说明
| 类型 | 字段 | 可用值 |
|---|
| 需求优先级 | priority_label | High / Middle / Low / Nice To Have |
| 缺陷优先级 | priority_label | urgent / high / medium / low / insignificant |
| 缺陷严重程度 | severity | fatal / serious / normal / prompt / advice |
| 任务状态 | status | open / progressing / done |
| 迭代状态 | status | open / done |
图片处理
当获取需求详情时,get_stories_or_tasks 命令会自动解析 description 中的图片并获取下载链接。
返回结果包含 images 字段:
{
"data": [
{
"Story": { "id": "1167459320001114969", "name": "需求标题", ... },
"images": [
{
"path": "/tfl/captures/2026-01/tapd_67459320_base64_1767668922_121.png",
"download_url": "https://file.tapd.cn/attachments/tmp_download/...?salt=...&time=...",
"filename": "tapd_67459320_base64_1767668922_121.png"
}
]
}
]
}
处理步骤:
- 从返回结果中读取
images 数组
- 使用
download_url 访问或下载图片
- 图片链接有效期约 300 秒
手动获取图片(备用方式):
python scripts/tapd.py get_image --workspace_id 67459320 --image_path "/tfl/captures/2026-01/tapd_xxx.png"
文件结构
scripts/
├── tapd.py # 统一入口脚本(46个子命令)
├── tapd_client.py # TAPD API 客户端
└── requirements.txt
参考文档
详细的 API 使用说明请参阅 references/ 目录:
stories-tasks.md - 需求/任务操作指南
bugs.md - 缺陷操作指南
iterations.md - 迭代操作指南
fields.md - 字段配置指南
testcases.md - 测试用例操作指南