Typescript SDK
@qwen-code/sdk
一个用于程序化访问 Qwen Code 的最小实验性 TypeScript SDK。
欢迎提交功能请求/问题/PR。
安装
npm install @qwen-code/sdk要求
- Node.js >= 20.0.0
- Qwen Code >= 0.4.0(稳定版)已安装并在 PATH 中可访问
nvm 用户注意:如果你使用 nvm 管理 Node.js 版本,SDK 可能无法自动检测到 Qwen Code 可执行文件。你应该显式设置
pathToQwenExecutable选项为qwen二进制文件的完整路径。
快速开始
import { query } from '@qwen-code/sdk';
// 单轮查询
const result = query({
prompt: '当前目录下有哪些文件?',
options: {
cwd: '/path/to/project',
},
});
// 遍历消息
for await (const message of result) {
if (message.type === 'assistant') {
console.log('助手:', message.message.content);
} else if (message.type === 'result') {
console.log('结果:', message.result);
}
}API 参考
query(config)
创建一个新的 Qwen Code 查询会话。
参数
prompt:string | AsyncIterable<SDKUserMessage>- 要发送的提示。使用字符串进行单轮查询,或使用异步可迭代对象进行多轮对话。options:QueryOptions- 查询会话的配置选项。
QueryOptions
| 选项 | 类型 | 默认值 | 描述 |
|---|---|---|---|
cwd | string | process.cwd() | 查询会话的工作目录。决定文件操作和命令执行的上下文环境。 |
model | string | - | 要使用的 AI 模型(例如 'qwen-max'、'qwen-plus'、'qwen-turbo')。优先级高于 OPENAI_MODEL 和 QWEN_MODEL 环境变量。 |
pathToQwenExecutable | string | 自动检测 | Qwen Code 可执行文件路径。支持多种格式:'qwen'(PATH 中的原生二进制文件)、'/path/to/qwen'(显式路径)、'/path/to/cli.js'(Node.js 包)、'node:/path/to/cli.js'(强制使用 Node.js 运行时)、'bun:/path/to/cli.js'(强制使用 Bun 运行时)。如果未提供,则从以下位置自动检测:QWEN_CODE_CLI_PATH 环境变量、~/.volta/bin/qwen、~/.npm-global/bin/qwen、/usr/local/bin/qwen、~/.local/bin/qwen、~/node_modules/.bin/qwen、~/.yarn/bin/qwen。 |
permissionMode | 'default' | 'plan' | 'auto-edit' | 'yolo' | 'default' | 控制工具执行审批的权限模式。详见 权限模式。 |
canUseTool | CanUseTool | - | 工具执行审批的自定义权限处理程序。当工具需要确认时调用。必须在 60 秒内响应,否则请求将被自动拒绝。详见 自定义权限处理程序。 |
env | Record<string, string> | - | 传递给 Qwen Code 进程的环境变量。与当前进程环境合并。 |
mcpServers | Record<string, McpServerConfig> | - | 要连接的 MCP(模型上下文协议)服务器。支持外部服务器(stdio/SSE/HTTP)和 SDK 内嵌服务器。外部服务器通过传输选项如 command、args、url、httpUrl 等进行配置。SDK 服务器使用 { type: 'sdk', name: string, instance: Server } 配置。 |
abortController | AbortController | - | 用于取消查询会话的控制器。调用 abortController.abort() 终止会话并清理资源。 |
debug | boolean | false | 启用调试模式以获取 CLI 进程的详细日志输出。 |
maxSessionTurns | number | -1(无限制) | 会话自动终止前的最大对话轮数。一轮包括用户消息和助手回复。 |
coreTools | string[] | - | 等效于 settings.json 中的 tool.core。如果指定,则只有这些工具可供 AI 使用。示例:['read_file', 'write_file', 'run_terminal_cmd']。 |
excludeTools | string[] | - | 等效于 settings.json 中的 tool.exclude。被排除的工具将立即返回权限错误。此设置具有最高优先级,覆盖所有其他权限设置。支持模式匹配:工具名称(如 'write_file')、工具类(如 'ShellTool')或 shell 命令前缀(如 'ShellTool(rm )')。 |
allowedTools | string[] | - | 等效于 settings.json 中的 tool.allowed。匹配的工具将跳过 canUseTool 回调并自动执行。仅适用于需要确认的工具。支持与 excludeTools 相同的模式匹配方式。 |
authType | 'openai' | 'qwen-oauth' | 'openai' | AI 服务的身份验证类型。在 SDK 中不建议使用 'qwen-oauth',因为凭证存储在 ~/.qwen 中且可能需要定期刷新。 |
agents | SubagentConfig[] | - | 会话期间可调用的子代理配置。子代理是针对特定任务或领域的专用 AI 代理。 |
includePartialMessages | boolean | false | 当设为 true 时,SDK 将在生成过程中发出不完整的消息,从而实现实时流式传输 AI 的响应内容。 |
超时
SDK 强制执行以下默认超时设置:
| 超时类型 | 默认值 | 描述 |
|---|---|---|
canUseTool | 1 分钟 | canUseTool 回调响应的最大时间。如果超过此时间,工具请求将被自动拒绝。 |
mcpRequest | 1 分钟 | SDK MCP 工具调用完成的最大时间。 |
controlRequest | 1 分钟 | 控制操作(如 initialize()、setModel()、setPermissionMode() 和 interrupt())完成的最大时间。 |
streamClose | 1 分钟 | 在多轮模式下与 SDK MCP 服务器通信时,等待初始化完成并关闭 CLI stdin 的最大时间。 |
你可以通过 timeout 选项自定义这些超时设置:
const query = qwen.query('Your prompt', {
timeout: {
canUseTool: 60000, // 权限回调 60 秒
mcpRequest: 600000, // MCP 工具调用 10 分钟
controlRequest: 60000, // 控制请求 60 秒
streamClose: 15000, // 流关闭等待 15 秒
},
});消息类型
SDK 提供了类型守卫来识别不同的消息类型:
import {
isSDKUserMessage,
isSDKAssistantMessage,
isSDKSystemMessage,
isSDKResultMessage,
isSDKPartialAssistantMessage,
} from '@qwen-code/sdk';
for await (const message of result) {
if (isSDKAssistantMessage(message)) {
// 处理助手消息
} else if (isSDKResultMessage(message)) {
// 处理结果消息
}
}查询实例方法
query() 返回的 Query 实例提供了以下几种方法:
const q = query({ prompt: 'Hello', options: {} });
// 获取会话 ID
const sessionId = q.getSessionId();
// 检查是否已关闭
const closed = q.isClosed();
// 中断当前操作
await q.interrupt();
// 会话中途更改权限模式
await q.setPermissionMode('yolo');
// 会话中途更改模型
await q.setModel('qwen-max');
// 关闭会话
await q.close();权限模式
SDK 支持不同的权限模式来控制工具执行:
default:除非通过canUseTool回调或在allowedTools中批准,否则写入工具会被拒绝。只读工具无需确认即可执行。plan:阻止所有写入工具,指示 AI 首先提出计划。auto-edit:自动批准编辑工具(edit、write_file),其他工具需要确认。yolo:所有工具自动执行,无需确认。
权限优先级链
excludeTools- 完全阻止工具permissionMode: 'plan'- 阻止非只读工具permissionMode: 'yolo'- 自动批准所有工具allowedTools- 自动批准匹配的工具canUseTool回调 - 自定义批准逻辑- 默认行为 - 在 SDK 模式下自动拒绝
示例
多轮对话
import { query, type SDKUserMessage } from '@qwen-code/sdk';
async function* generateMessages(): AsyncIterable<SDKUserMessage> {
yield {
type: 'user',
session_id: 'my-session',
message: { role: 'user', content: '创建一个 hello.txt 文件' },
parent_tool_use_id: null,
};
// 等待某些条件或用户输入
yield {
type: 'user',
session_id: 'my-session',
message: { role: 'user', content: '现在读取该文件内容' },
parent_tool_use_id: null,
};
}
const result = query({
prompt: generateMessages(),
options: {
permissionMode: 'auto-edit',
},
});
for await (const message of result) {
console.log(message);
}自定义权限处理器
import { query, type CanUseTool } from '@qwen-code/sdk';
const canUseTool: CanUseTool = async (toolName, input, { signal }) => {
// 允许所有读取操作
if (toolName.startsWith('read_')) {
return { behavior: 'allow', updatedInput: input };
}
// 提示用户进行写入操作(在实际应用中)
const userApproved = await promptUser(`允许 ${toolName} 吗?`);
if (userApproved) {
return { behavior: 'allow', updatedInput: input };
}
return { behavior: 'deny', message: '用户拒绝了该操作' };
};
const result = query({
prompt: '创建一个新文件',
options: {
canUseTool,
},
});使用外部 MCP 服务器
import { query } from '@qwen-code/sdk';
const result = query({
prompt: 'Use the custom tool from my MCP server',
options: {
mcpServers: {
'my-server': {
command: 'node',
args: ['path/to/mcp-server.js'],
env: { PORT: '3000' },
},
},
},
});使用 SDK 内置 MCP 服务器
SDK 提供了 tool 和 createSdkMcpServer 来创建与 SDK 应用程序在同一进程中运行的 MCP 服务器。当你希望向 AI 暴露自定义工具而不想运行独立的服务器进程时,这非常有用。
tool(name, description, inputSchema, handler)
创建一个带有 Zod schema 类型推断的工具定义。
| 参数 | 类型 | 描述 |
|---|---|---|
name | string | 工具名称(1-64个字符,以字母开头,可包含字母、数字和下划线) |
description | string | 对工具功能的人类可读描述 |
inputSchema | ZodRawShape | 定义工具输入参数的 Zod schema 对象 |
handler | (args, extra) => Promise<Result> | 异步函数,执行工具并返回 MCP 内容块 |
handler 必须返回一个具有以下结构的 CallToolResult 对象:
{
content: Array<
| { type: 'text'; text: string }
| { type: 'image'; data: string; mimeType: string }
| { type: 'resource'; uri: string; mimeType?: string; text?: string }
>;
isError?: boolean;
}createSdkMcpServer(options)
创建一个嵌入 SDK 的 MCP 服务器实例。
| 选项 | 类型 | 默认值 | 描述 |
|---|---|---|---|
name | string | 必填 | MCP 服务器的唯一名称 |
version | string | '1.0.0' | 服务器版本 |
tools | SdkMcpToolDefinition[] | - | 使用 tool() 创建的工具数组 |
返回一个 McpSdkServerConfigWithInstance 对象,可直接传递给 mcpServers 选项。
示例
import { z } from 'zod';
import { query, tool, createSdkMcpServer } from '@qwen-code/sdk';
// 使用 Zod schema 定义一个工具
const calculatorTool = tool(
'calculate_sum',
'将两个数字相加',
{ a: z.number(), b: z.number() },
async (args) => ({
content: [{ type: 'text', text: String(args.a + args.b) }],
}),
);
// 创建 MCP 服务器
const server = createSdkMcpServer({
name: 'calculator',
tools: [calculatorTool],
});
// 在查询中使用该服务器
const result = query({
prompt: '42 加 17 等于多少?',
options: {
permissionMode: 'yolo',
mcpServers: {
calculator: server,
},
},
});
for await (const message of result) {
console.log(message);
}中止查询
import { query, isAbortError } from '@qwen-code/sdk';
const abortController = new AbortController();
const result = query({
prompt: '长时间运行的任务...',
options: {
abortController,
},
});
// 5秒后中止
setTimeout(() => abortController.abort(), 5000);
try {
for await (const message of result) {
console.log(message);
}
} catch (error) {
if (isAbortError(error)) {
console.log('查询已被中止');
} else {
throw error;
}
}错误处理
SDK 提供了 AbortError 类来处理被中止的查询:
import { AbortError, isAbortError } from '@qwen-code/sdk';
try {
// ... 查询操作
} catch (error) {
if (isAbortError(error)) {
// 处理中止情况
} else {
// 处理其他错误
}
}Last updated on