【声明】本博客所有内容均为个人业余时间创作,所述技术案例均来自公开开源项目(如Github,Apache基金会),不涉及任何企业机密或未公开技术,如有侵权请联系删除
背景
上篇 blog
【Agent】【OpenCode】本地代理分析(四)
开始分析 end 事件,其中分析到了获取 Autorization 头部信息,并分析了其中的 JavaScript 细节:代理会尝试从请求头部信息中读取 authorization 字段,如果该字段不存在(或为假值),就用空字符串代替,然后本地代理会直接透传这个获取到的 authorization 字段,然后分析了 end 事件中的 option 对象,这个 options 对象是 Node.js 中使用的内置模块 https.request 发起 HTTPS 请求时必需的配置参数,定义了向哪个服务器,用什么方式,发送什么样的请求,并详细分析了其中的元素,比如 hostname,port,path,以及 method,下面继续分析
OpenCode
下面继续分析 options 对象剩余的元素

接下来是 headers: {...}:请求头,用于传递元数据,关键字段如下
'Authorization': authHeader:身份认证 ,这里的authHeader是从 OpenCode 原始请求里提取的 API Key'Content-Type': 'application/json':通知服务器 request body 是 JSON 格式,DashScope 要求 request body 需是 JSON,所以这里必须设置'Content-Length': Buffer.byteLength(body):声明 request body 的字节长度 ,注意,这里不能用body.length,因为body.length返回的是字符串字符数(UTF-16),而 HTTP 协议要求的是字节数(bytes),某些中文字符会占多个字节,Buffer.byteLength可以正确计算出真正的字节数(UTF-8),如果 Content-Length 不准确,服务器可能提前关闭连接,或返回 400 错误

综合以上字段,最终可以形成完整的 URL 请求示例如下
javascript
POST https://dashscope.aliyuncs.com/compatible-mode/v1/chat/completions
Headers:
Authorization: Bearer sk-xxxx
Content-Type: application/json
Content-Length: 123
Body:
{"model":"qwen-plus","messages":[...]}
OK,接着分析剩余部分

首先是这里的创建 HTTPS 请求对象
javascript
const proxyReq = https.request(options, (proxyRes) => {
// 当收到 DashScope 的响应头时触发
});
这是 Node.js 中使用 https 模块发起 HTTPS 请求的标准写法,其整体语法结构是函数调用 + 回调函数的典型模式
const requestObject = https.request(config, callback);
https.request 是 Node.js 内置模块 https 提供的方法,用于创建一个 HTTP/HTTPS 客户端请求对象,这里的第二个参数是个回调函数,当目标服务器返回响应头时,该回调会被自动调用
这里有四个关键点
https.request(...):来自 Node.js 标准库,负责构造一个 HTTPS 客户端请求对象,注意,构造对象并不会立即发送请求 ,必须要调用.write和.end才会真正发出请求,就好像写好了一封信(设置了地址,写了内容),但还没投进邮箱config(第一个参数):就是前面说过的options,定义消息的头部信息,配置请求要向哪个服务器发送,以及怎么发callback(第二个参数,回调函数):就是上面的(proxyRes) => {...},是个箭头函数 ,关于箭头函数之前 blog 【Agent】【OpenCode】本地代理分析(三) 已经介绍过
回调 (proxyRes) => {...} 里面的参数 proxyRes 详细分析下:
- 类型 :
http.IncomingMessage,是 Node.js 的内置类 - 含义:来自目标服务器的响应对象
该对象 proxyRes 包含如下关键属性
proxyRes.statusCode:响应码,比如 200,401,500 等proxyRes.headers:响应头部信息,比如{'content-type': 'application/json', ...}- 可读流,响应体数据通过
.on('data')或.pipe可以获取
OK,本篇先到这里,如有疑问,欢迎评论区留言讨论,祝各位功力大涨,技术更上一层楼!!!更多内容见下篇 blog
【Agent】【OpenCode】本地代理分析(六)