Skip to content

Commit 91ee142

Browse files
authored
Fix bug OpenAI tooluse,Improve error messaging for deferred-loading tools under OpenAI‑compatible models. (#199)
* fix: reorder tool and user messages for OpenAI API compatibility (#168) Fixes #168 OpenAI requires that an assistant message with tool_calls be immediately followed by tool messages. Previously, convertInternalUserMessage output user content before tool results, causing 400 errors. Now tool messages are pushed first. * fix: 修复OpenAI兼容层中deferred tools处理问题 提交描述: 修复了在使用OpenAI兼容API时TaskCreate工具调用失败的问题。 问题: - 当使用OpenAI兼容API模型时,调用TaskCreate工具出现"InputValidationError: The required parameter `subject` is missing"错误 - OpenAI兼容层没有正确处理deferred tools的过滤逻辑,导致工具schema没有被正确发送给模型 修复: 1. 在OpenAI兼容层中添加了与Anthropic API路径一致的deferred tools处理逻辑 2. 导入必要的工具搜索相关函数: isToolSearchEnabled, extractDiscoveredToolNames, isDeferredTool等 3. 实现工具过滤逻辑: - 检查工具搜索是否启用 - 构建deferred tools集合 - 过滤工具列表: 只包含非deferred工具或已发现的deferred工具 - 为deferred tools设置deferLoading标志 4. 修正了extractDiscoveredToolNames函数的导入路径错误 影响: - 解决了TaskCreate工具调用时的参数验证错误 - 确保OpenAI兼容层与Anthropic API路径在处理deferred tools时行为一致 - 支持工具搜索功能在OpenAI兼容模式下正常工作 修改的文件: - src/services/api/openai/index.ts - 主要修复文件 测试建议: 1. 使用OpenAI兼容API模型时,TaskCreate工具应该可以正常调用 2. 如果工具搜索功能启用,可能需要先使用ToolSearchTool来发现TaskCreate工具 3. 验证工具调用时不再出现"InputValidationError"错误 这个修复确保了当使用OpenAI兼容API(如Ollama、DeepSeek、vLLM等)时,deferred tools(如TaskCreate)能够被正确处理,解决了工具调用失败的问题。 * fix: 更新工具模式未发送提示,增加OpenAI兼容模型使用指南
1 parent d52300f commit 91ee142

2 files changed

Lines changed: 206 additions & 3 deletions

File tree

docs/openai-task-tools.md

Lines changed: 190 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,190 @@
1+
# OpenAI兼容模型中task工具使用指南
2+
3+
## 问题描述
4+
5+
当使用OpenAI兼容模型(如DeepSeek、Ollama、vLLM等)时,调用task工具(TaskGet、TaskCreate、TaskUpdate、TaskList)可能会出现以下错误:
6+
7+
```
8+
Error: InputValidationError: TaskGet failed due to the following issues:
9+
The required parameter `taskId` is missing
10+
An unexpected parameter `task_id` was provided
11+
12+
This tool's schema was not sent to the API — it was not in the discovered-tool set derived from message history. Without the schema in your prompt, typed parameters (arrays, numbers, booleans) get emitted as strings and the client-side parser rejects them. Load the tool first: call ToolSearch with query "select:TaskGet", then retry this call.
13+
```
14+
15+
## 问题原因
16+
17+
### 1. 延迟加载工具(Deferred Tools)
18+
task工具都是延迟加载的(`shouldDefer: true`),这意味着:
19+
- 工具的模式(schema)不会在初始API调用中发送
20+
- 需要先通过`ToolSearch`工具发现
21+
- 只有在被发现后,工具模式才会被发送给API
22+
23+
### 2. 参数名转换问题
24+
- task工具使用驼峰命名:`taskId`
25+
- OpenAI兼容模型可能输出蛇形命名:`task_id`
26+
- 当工具模式没有被发送时,模型会猜测参数名,可能导致不匹配
27+
28+
## 解决方案
29+
30+
### 方案1:先使用ToolSearch(推荐)
31+
在使用task工具之前,先调用`ToolSearch`工具:
32+
33+
```javascript
34+
// 第一步:发现task工具
35+
ToolSearch("select:TaskGet,TaskCreate,TaskUpdate,TaskList")
36+
37+
// 第二步:正常使用task工具
38+
TaskCreate({ subject: "任务标题", description: "任务描述" })
39+
TaskGet({ taskId: "1" })
40+
TaskUpdate({ taskId: "1", status: "completed" })
41+
TaskList()
42+
```
43+
44+
### 方案2:批量发现所有task工具
45+
```javascript
46+
// 一次性发现所有task工具
47+
ToolSearch("select:TaskGet,TaskCreate,TaskUpdate,TaskList")
48+
49+
// 然后可以任意使用task工具
50+
const task = await TaskCreate({ subject: "新任务", description: "任务描述" })
51+
console.log(`创建的任务ID: ${task.id}`)
52+
53+
const taskList = await TaskList()
54+
console.log(`当前有 ${taskList.tasks.length} 个任务`)
55+
```
56+
57+
### 方案3:单独发现特定工具
58+
```javascript
59+
// 只发现需要的工具
60+
ToolSearch("select:TaskGet")
61+
62+
// 然后使用该工具
63+
TaskGet({ taskId: "1" })
64+
```
65+
66+
## 参数名注意事项
67+
68+
在使用OpenAI兼容模型时,请注意参数名格式:
69+
70+
### ✅ 正确(驼峰命名)
71+
```javascript
72+
TaskGet({ taskId: "1" })
73+
TaskCreate({ subject: "标题", description: "描述" })
74+
TaskUpdate({ taskId: "1", status: "completed" })
75+
```
76+
77+
### ❌ 错误(蛇形命名)
78+
```javascript
79+
TaskGet({ task_id: "1" }) // 错误:应该使用taskId
80+
TaskCreate({ subject: "标题", description: "描述" }) // 正确
81+
TaskUpdate({ task_id: "1", status: "completed" }) // 错误:应该使用taskId
82+
```
83+
84+
## 常见问题解答
85+
86+
### Q1: 为什么需要先使用ToolSearch?
87+
A: task工具是延迟加载的,它们的模式只有在被`ToolSearch`工具发现后才会发送给API。没有工具模式,模型无法知道正确的参数名和类型。
88+
89+
### Q2: 每次会话都需要使用ToolSearch吗?
90+
A: 是的,每次新的会话都需要先使用ToolSearch发现工具。工具发现状态不会在会话之间保留。
91+
92+
### Q3: 使用Anthropic官方模型也需要这样吗?
93+
A: 通常不需要。Anthropic官方模型对延迟加载工具的处理更智能,但为了兼容性,建议在使用task工具前都先使用ToolSearch。
94+
95+
### Q4: 可以一次性发现所有工具吗?
96+
A: 可以,使用`ToolSearch("select:TaskGet,TaskCreate,TaskUpdate,TaskList")`可以一次性发现所有task工具。
97+
98+
### Q5: 如果忘记使用ToolSearch会怎样?
99+
A: 会收到参数验证错误,提示需要先使用ToolSearch。按照错误信息的指导操作即可。
100+
101+
## 最佳实践
102+
103+
1. **会话开始时发现工具**:在开始使用task工具前,先调用ToolSearch
104+
2. **批量发现**:一次性发现所有需要的task工具
105+
3. **检查参数名**:确保使用正确的驼峰命名参数
106+
4. **查看错误信息**:如果遇到错误,仔细阅读错误信息中的指导
107+
108+
## 示例工作流
109+
110+
```javascript
111+
// 1. 开始新会话
112+
// 2. 发现task工具
113+
ToolSearch("select:TaskGet,TaskCreate,TaskUpdate,TaskList")
114+
115+
// 3. 创建任务
116+
const newTask = await TaskCreate({
117+
subject: "修复OpenAI兼容性问题",
118+
description: "解决task工具在OpenAI兼容模型下的参数名问题"
119+
})
120+
121+
// 4. 获取任务详情
122+
const taskDetails = await TaskGet({ taskId: newTask.id })
123+
124+
// 5. 更新任务状态
125+
await TaskUpdate({
126+
taskId: newTask.id,
127+
status: "in_progress",
128+
activeForm: "修复OpenAI兼容性问题"
129+
})
130+
131+
// 6. 查看所有任务
132+
const allTasks = await TaskList()
133+
console.log(`当前有 ${allTasks.tasks.length} 个任务`)
134+
135+
// 7. 完成任务
136+
await TaskUpdate({
137+
taskId: newTask.id,
138+
status: "completed"
139+
})
140+
```
141+
142+
## 故障排除
143+
144+
### 错误:参数名不匹配
145+
**症状**`taskId`参数缺失,发现`task_id`参数
146+
**解决**:确保使用驼峰命名的`taskId`,而不是蛇形命名的`task_id`
147+
148+
### 错误:工具模式未发送
149+
**症状**`This tool's schema was not sent to the API`
150+
**解决**:先使用`ToolSearch("select:工具名")`发现工具
151+
152+
### 错误:工具不可用
153+
**症状**:工具调用失败,没有具体错误信息
154+
**解决**:检查工具是否启用(通过`isTodoV2Enabled()`),确保环境变量设置正确
155+
156+
## 相关配置
157+
158+
### 环境变量
159+
```bash
160+
# 启用OpenAI兼容模式
161+
export CLAUDE_CODE_USE_OPENAI=1
162+
export OPENAI_API_KEY=your-api-key
163+
export OPENAI_BASE_URL=https://api.deepseek.com
164+
165+
# 配置模型映射
166+
export OPENAI_DEFAULT_SONNET_MODEL=deepseek-chat
167+
export OPENAI_DEFAULT_OPUS_MODEL=deepseek-chat
168+
export OPENAI_DEFAULT_HAIKU_MODEL=deepseek-chat
169+
```
170+
171+
### 设置文件
172+
通过`/login`命令配置OpenAI兼容模式后,设置会保存在`~/.claude/settings.json`
173+
```json
174+
{
175+
"modelType": "openai",
176+
"openai": {
177+
"baseURL": "https://api.deepseek.com",
178+
"apiKey": "your-api-key",
179+
"models": {
180+
"haiku": "deepseek-chat",
181+
"sonnet": "deepseek-chat",
182+
"opus": "deepseek-chat"
183+
}
184+
}
185+
}
186+
```
187+
188+
## 总结
189+
190+
在使用OpenAI兼容模型时,task工具需要先通过`ToolSearch`发现才能正常使用。遵循"先发现,后使用"的原则,并注意参数名的正确格式(驼峰命名),可以确保task工具在OpenAI兼容模型下正常工作。

src/services/tools/toolExecution.ts

Lines changed: 16 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -589,10 +589,23 @@ export function buildSchemaNotSentHint(
589589
if (!isDeferredTool(tool)) return null
590590
const discovered = extractDiscoveredToolNames(messages)
591591
if (discovered.has(tool.name)) return null
592+
593+
const toolDisplayName = tool.userFacingName
594+
? tool.userFacingName(undefined)
595+
: tool.name
596+
592597
return (
593-
`\n\nThis tool's schema was not sent to the API — it was not in the discovered-tool set derived from message history. ` +
594-
`Without the schema in your prompt, typed parameters (arrays, numbers, booleans) get emitted as strings and the client-side parser rejects them. ` +
595-
`Load the tool first: call ${TOOL_SEARCH_TOOL_NAME} with query "select:${tool.name}", then retry this call.`
598+
`\n\nTool "${toolDisplayName}" is deferred-loading and needs to be discovered before use.\n` +
599+
`When using OpenAI-compatible models (DeepSeek, Ollama, etc.), follow these steps:\n` +
600+
`1. First discover the tool with ToolSearch: ${TOOL_SEARCH_TOOL_NAME}("select:${tool.name}")\n` +
601+
`2. Then call ${toolDisplayName} tool\n` +
602+
`\nExample:\n` +
603+
`${TOOL_SEARCH_TOOL_NAME}("select:${tool.name}") → ${toolDisplayName}({ ... })\n` +
604+
`\nImportant notes:\n` +
605+
`• Use camelCase parameter names (e.g., taskId), not snake_case (task_id)\n` +
606+
`• All task tools (TaskGet, TaskCreate, TaskUpdate, TaskList) need to be discovered first\n` +
607+
`• You can discover them all at once: ${TOOL_SEARCH_TOOL_NAME}("select:TaskGet,TaskCreate,TaskUpdate,TaskList")\n` +
608+
`\nSee docs/openai-task-tools.md for detailed guide.`
596609
)
597610
}
598611

0 commit comments

Comments
 (0)