运行时依赖
安装命令
点击复制技能文档
钉钉待办技能
负责钉钉待办(Todo)的所有操作。本文件为策略指南,仅包含决策逻辑和工作流程。完整 API 请求格式见文末「references/API.md 查阅索引」。
工作流程(每次执行前) 读取配置 → 用一条 grep -E 命令一次性读取配置文件~/.dingtalk-技能s/config, 所有所需配置键值(配置文件跨会话保留,无需重复询问) 仅收集缺失配置 → 若配置文件不存在或缺少某项,一次性询问用户所有缺失的值,不要逐条问 持久化 → 将收集到的值写入 ~/.dingtalk-技能s/config 文件,后续无需再问 获取/复用 令牌 → 有效期内复用缓存(缓存 7000 秒,约 2 小时),避免重复请求;遇 401 重新获取 执行操作 → 凡是包含变量替换、管道或多行逻辑的命令,/tmp/.sh 再 bash /tmp/.sh 执行。不要把多行命令直接粘到终端里(终端工具会截断),也不要用 <<'EOF' 语法(heredoc 在工具中同样会被截断导致变量丢失)
凭证禁止在输出中完整打印,确认时仅显示前 4 位 + ***
所需配置 配置键 说明 如何获取 DINGTALK_应用_KEY 应用 应用Key 钉钉开放平台 → 应用管理 → 凭证信息 DINGTALK_应用_SECRET 应用 应用Secret 同上 DINGTALK_USER_ID 当前用户的企业员工 ID(userId) 管理后台 → 通讯录 → 成员管理 → 点击姓名查看(不是手机号、不是 unionId) DINGTALK_OPERATOR_ID 当前用户的 unionId 首次由脚本自动通过 userId 转换获取并写入 身份标识说明
钉钉有两种用户 ID,不同 API 使用不同的 ID:
标识 说明 如何获取 userId(= staffId) 企业内部员工 ID,最容易获取 管理后台 → 通讯录 → 成员管理 → 点击姓名查看;或调用手机号查询 API unionId 跨企业/跨应用唯一 通过 userId 调用 API 转换获取 待办 API 的路径参数 {unionId} 和查询参数 operatorId 均使用 unionId executorIds / participantIds(指派同事)也使用 unionId 因此配置中优先收集 userId(用户容易拿到),由脚本自动转换为 unionId userId → unionId 转换
需要旧版 访问_令牌(与新版不同):
# 1. 获取旧版 令牌 OLD_令牌=$(curl -s "https://oAPI.dingtalk.com/获取令牌?应用key=${应用_KEY}&应用secret=${应用_SECRET}" | grep -o '"访问_令牌":"[^"]"' | cut -d'"' -f4)
# 2. userId → unionId UNION_ID=$(curl -s -X POST "https://oAPI.dingtalk.com/topAPI/v2/user/获取?访问_令牌=${OLD_令牌}" \ -H 'Content-Type: 应用/json' \ -d "{\"userid\":\"${USER_ID}\"}" | grep -o '"unionid":"[^"]"' | cut -d'"' -f4)
# 3. 写入配置文件 echo "DINGTALK_OPERATOR_ID=$UNION_ID" >> ~/.dingtalk-技能s/config
⚠️ 注意:返回体中 结果.unionid(无下划线)有值,结果.union_id(有下划线)可能为空。
给同事创建待办时
如果用户要给同事创建待办(指定 executorIds),需要同事的 unionId。向用户询问同事的 userId(管理后台可查),然后用上述方法转换。
执行脚本模板 #!/bin/bash 设置 -e CONFIG=~/.dingtalk-技能s/config 应用_KEY=$(grep '^DINGTALK_应用_KEY=' "$CONFIG" | cut -d= -f2-) 应用_SECRET=$(grep '^DINGTALK_应用_SECRET=' "$CONFIG" | cut -d= -f2-) USER_ID=$(grep '^DINGTALK_USER_ID=' "$CONFIG" | cut -d= -f2-)
# 新版 令牌 缓存(用于待办 API) 缓存D_令牌=$(grep '^DINGTALK_访问_令牌=' "$CONFIG" 2>/dev/null | cut -d= -f2-) 令牌_EXPIRY=$(grep '^DINGTALK_令牌_EXPIRY=' "$CONFIG" 2>/dev/null | cut -d= -f2-) NOW=$(date +%s) if [ -n "$缓存D_令牌" ] && [ -n "$令牌_EXPIRY" ] && [ "$NOW" -lt "$令牌_EXPIRY" ]; then 令牌=$缓存D_令牌 else RESP=$(curl -s -X POST https://API.dingtalk.com/v1.0/oauth2/访问令牌 \ -H 'Content-Type: 应用/json' \ -d "{\"应用Key\":\"$应用_KEY\",\"应用Secret\":\"$应用_SECRET\"}") 令牌=$(echo "$RESP" | grep -o '"访问令牌":"[^"]"' | cut -d'"' -f4) sed -i '/^DINGTALK_访问_令牌=/d;/^DINGTALK_令牌_EXPIRY=/d' "$CONFIG" echo "DINGTALK_访问_令牌=$令牌" >> "$CONFIG" echo "DINGTALK_令牌_EXPIRY=$((NOW + 7000))" >> "$CONFIG" fi
# unionId:优先从配置读取,未存储时自动从 userId 转换并写入 UNION_ID=$(grep '^DINGTALK_OPERATOR_ID=' "$CONFIG" 2>/dev/null | cut -d= -f2-) if [ -z "$UNION_ID" ]; then OLD_令牌=$(curl -s "https://oAPI.dingtalk.com/获取令牌?应用key=${应用_KEY}&应用secret=${应用_SECRET}" | grep -o '"访问_令牌":"[^"]"' | cut -d'"' -f4) UNION_ID=$(curl -s -X POST "https://oAPI.dingtalk.com/topAPI/v2/user/获取?访问_令牌=${OLD_令牌}" \ -H 'Content-Type: 应用/json' \ -d "{\"userid\":\"${USER_ID}\"}" | grep -o '"unionid":"[^"]"' | cut -d'"' -f4) echo "DINGTALK_OPERATOR_ID=$UNION_ID" >> "$CONFIG" fi
# 在此追加具体 API 调用,例如创建待办: 结果=$(curl -s -X POST \ "https://API.dingtalk.com/v1.0/todo/users/${UNION_ID}/tasks?operatorId=${UNION_ID}" \ -H "x-acs-dingtalk-访问-令牌: $令牌" \ -H 'Content-Type: 应用/json' \ -d "{\"subject\":\"今天完成需求评审\"}") echo "$结果" TASK_ID=$(echo "$结果" | grep -o '"id":"[^"]*"' | head -1 | cut -d'"' -f4) echo "创建成功,taskId=$TASK_ID"
⚠️ 通过应用 API 创建的待办显示在钉钉「待办」的 Teambition 分类下,不是「个人」分类。 ⚠️ 通过 API 创建的任务无法在钉钉 UI 里手动删除,只能通过 API 删除。
references/API.md 查阅索引
确定好要做什么之后,用以下命令从 references/API.md 中提取对应章节的完整 API 细节(请求格式、参数说明、返回值示例):
# 身份标识与 userId ↔ unionId 转换(28 行) grep -A 28 "^## 身份标识" references/API.md
# 创建待办(含所有可选字段)(47 行) grep -A 47 "^## 1. 创建待办" references/API.md
# 获取待办详情(29 行) grep -A 29 "^## 2. 获取待办详情" references/API.md
# 查询待办列表(含分页)(42 行) grep -A 42 "^## 3. 查询待办列表" references/API.md
# 更新待办(25 行) grep -A 25 "^## 4. 更新待办" references/API.md
# 删除待办(16 行) grep -A 16 "^## 5. 删除待办" references/API.md
# 错误码表(9 行) grep -A 9 "^## 错误码" references/API.md
# 所需应用权限(7 行) grep -A 7 "^## 所需应用权限" references/API.md