Skip to Content
开发者指南Typescript SDK

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

选项类型默认值描述
cwdstringprocess.cwd()查询会话的工作目录。决定文件操作和命令执行的上下文环境。
modelstring-要使用的 AI 模型(例如 'qwen-max''qwen-plus''qwen-turbo')。优先级高于 OPENAI_MODELQWEN_MODEL 环境变量。
pathToQwenExecutablestring自动检测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'控制工具执行审批的权限模式。详见 权限模式
canUseToolCanUseTool-工具执行审批的自定义权限处理程序。当工具需要确认时调用。必须在 60 秒内响应,否则请求将被自动拒绝。详见 自定义权限处理程序
envRecord<string, string>-传递给 Qwen Code 进程的环境变量。与当前进程环境合并。
mcpServersRecord<string, McpServerConfig>-要连接的 MCP(模型上下文协议)服务器。支持外部服务器(stdio/SSE/HTTP)和 SDK 内嵌服务器。外部服务器通过传输选项如 commandargsurlhttpUrl 等进行配置。SDK 服务器使用 { type: 'sdk', name: string, instance: Server } 配置。
abortControllerAbortController-用于取消查询会话的控制器。调用 abortController.abort() 终止会话并清理资源。
debugbooleanfalse启用调试模式以获取 CLI 进程的详细日志输出。
maxSessionTurnsnumber-1(无限制)会话自动终止前的最大对话轮数。一轮包括用户消息和助手回复。
coreToolsstring[]-等效于 settings.json 中的 tool.core。如果指定,则只有这些工具可供 AI 使用。示例:['read_file', 'write_file', 'run_terminal_cmd']
excludeToolsstring[]-等效于 settings.json 中的 tool.exclude。被排除的工具将立即返回权限错误。此设置具有最高优先级,覆盖所有其他权限设置。支持模式匹配:工具名称(如 'write_file')、工具类(如 'ShellTool')或 shell 命令前缀(如 'ShellTool(rm )')。
allowedToolsstring[]-等效于 settings.json 中的 tool.allowed。匹配的工具将跳过 canUseTool 回调并自动执行。仅适用于需要确认的工具。支持与 excludeTools 相同的模式匹配方式。
authType'openai' | 'qwen-oauth''openai'AI 服务的身份验证类型。在 SDK 中不建议使用 'qwen-oauth',因为凭证存储在 ~/.qwen 中且可能需要定期刷新。
agentsSubagentConfig[]-会话期间可调用的子代理配置。子代理是针对特定任务或领域的专用 AI 代理。
includePartialMessagesbooleanfalse当设为 true 时,SDK 将在生成过程中发出不完整的消息,从而实现实时流式传输 AI 的响应内容。

超时

SDK 强制执行以下默认超时设置:

超时类型默认值描述
canUseTool1 分钟canUseTool 回调响应的最大时间。如果超过此时间,工具请求将被自动拒绝。
mcpRequest1 分钟SDK MCP 工具调用完成的最大时间。
controlRequest1 分钟控制操作(如 initialize()setModel()setPermissionMode()interrupt())完成的最大时间。
streamClose1 分钟在多轮模式下与 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:所有工具自动执行,无需确认。

权限优先级链

  1. excludeTools - 完全阻止工具
  2. permissionMode: 'plan' - 阻止非只读工具
  3. permissionMode: 'yolo' - 自动批准所有工具
  4. allowedTools - 自动批准匹配的工具
  5. canUseTool 回调 - 自定义批准逻辑
  6. 默认行为 - 在 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 提供了 toolcreateSdkMcpServer 来创建与 SDK 应用程序在同一进程中运行的 MCP 服务器。当你希望向 AI 暴露自定义工具而不想运行独立的服务器进程时,这非常有用。

tool(name, description, inputSchema, handler)

创建一个带有 Zod schema 类型推断的工具定义。

参数类型描述
namestring工具名称(1-64个字符,以字母开头,可包含字母、数字和下划线)
descriptionstring对工具功能的人类可读描述
inputSchemaZodRawShape定义工具输入参数的 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 服务器实例。

选项类型默认值描述
namestring必填MCP 服务器的唯一名称
versionstring'1.0.0'服务器版本
toolsSdkMcpToolDefinition[]-使用 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