xshell 1.0.19 → 1.0.21

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/net.d.ts CHANGED
@@ -10,7 +10,7 @@ declare module 'tough-cookie' {
10
10
  }
11
11
  import './prototype.js';
12
12
  import type { Encoding } from './file.js';
13
- import { inspect } from './utils.js';
13
+ import { inspect, Lock } from './utils.js';
14
14
  export declare enum MyProxy {
15
15
  socks5 = "http://localhost:10080",
16
16
  whistle = "http://localhost:8899"
@@ -18,7 +18,7 @@ export declare enum MyProxy {
18
18
  export declare const cookies: {
19
19
  store: MemoryCookieStore;
20
20
  jar: CookieJar;
21
- get(domain_or_url: string, str?: boolean): Cookie[] | Promise<string> | Promise<Cookie[]>;
21
+ get(domain_or_url: string, str?: boolean): Promise<string> | Cookie[] | Promise<Cookie[]>;
22
22
  };
23
23
  export { Cookie };
24
24
  export interface BasicAuth {
@@ -155,8 +155,7 @@ export type MessageHandler = (message: Message, websocket?: WebSocket) => void |
155
155
  /** 二进制消息格式
156
156
  - json.length (小端序): 4 字节
157
157
  - json 数据
158
- - binary 数据
159
- */
158
+ - binary 数据 */
160
159
  export interface Message<TData extends any[] = any[]> {
161
160
  /** rpc id: 在 rpc 系统中认为是唯一的。用来在单个 websocket 连接上复用多个 rpc 请求。多个相同 id 的 message 组成一个请求流 */
162
161
  id?: number;
@@ -175,8 +174,7 @@ export interface Message<TData extends any[] = any[]> {
175
174
  里面是可序列化为 json 的 js 变量,或者是 Uint8Array
176
175
  Uint8Array 的参数处理后被替换为 Uint8Array.byteLength, 并将下标记录在 bins 中
177
176
 
178
- 注意: 数组中如果有 undefined 值,传输到对面会变成 null 值
179
- */
177
+ 注意: 数组中如果有 undefined 值,传输到对面会变成 null 值 */
180
178
  data?: TData;
181
179
  /** bins: data 中哪些下标对应的原始值是 Uint8Array 类型的,如: [0, 3] */
182
180
  bins?: number[];
@@ -184,35 +182,60 @@ export interface Message<TData extends any[] = any[]> {
184
182
  /** 通过创建 remote 对象对 websocket rpc 进行抽象
185
183
  调用方使用 remote.call 进行调用
186
184
  被调方在创建 remote 对象时传入 funcs 注册处理函数,并使用 remote.handle 方法处理 websocket message
187
- 未连接时自动连接,断开后自动重连 (保证执行 send 时已连接,否则报错) */
185
+ 未连接时自动连接,断开后自动重连 (保证执行 send 时已连接,否则报错)
186
+ 从设计上与 rpc 状态与底层连接的状态无关 (通过传入 url 创建的),底层连接断开后,只要检测到断线,会尝试重连,如果重连成功,则不影响调用的状态
187
+ @example
188
+ // Zero 继承自 Remote 并通过 call 实现了一些方法
189
+ let zero = new Zero({ local: true })
190
+
191
+ // 一元 rpc
192
+ await zero.repl_ts('1234')
193
+
194
+ // 订阅流
195
+ const id = genid()
196
+
197
+ zero.handlers.set(id, ({ data: [chunk] }: Message<[Uint8Array]>) => {
198
+ term.write(chunk)
199
+ })
200
+
201
+ zero.send({ id, func: 'subscribe_stdio' }) */
188
202
  export declare class Remote {
189
- url: string;
203
+ /** 是否为建立 rpc 连接的发起方 */
204
+ initiator: boolean;
205
+ /** websocket url */
206
+ url?: string;
190
207
  /** 主动发起 websocket 连接的客户端 (构造函数传 url) 有这个属性 */
191
- websocket?: WebSocket;
208
+ lwebsocket?: Lock<WebSocket>;
192
209
  /** 通过 rpc message.func 被调用的 rpc 函数 */
193
210
  funcs: Record<string, MessageHandler>;
194
211
  /** map<id, message handler>: 通过 rpc message.id 找到对应的 handler, unary rpc 接收方不需要设置 handlers, 发送方需要 */
195
212
  handlers: Map<number, MessageHandler>;
196
213
  print: boolean;
197
214
  pconnect: Promise<any>;
198
- get connected(): boolean;
199
- static parse<TData extends any[] = any[]>(array_buffer: ArrayBuffer): Message<TData>;
215
+ /** 使用 Uint8Array 作为参数更灵活 https://stackoverflow.com/a/74505197/7609214 */
216
+ static parse<TData extends any[] = any[]>(buffer: Uint8Array): Message<TData>;
200
217
  static pack({ id, func, data, done, error }: Message): Uint8Array;
201
- constructor({ url, funcs, websocket }?: {
218
+ /** 作为 rpc 发起方,可以通过传入 url 或者 websocket 定义远程 Remote
219
+ 作为 rpc 接收方,可以不传 url 和 websocket 定义本地 Remote */
220
+ constructor({ url, funcs, websocket, on_error }?: {
202
221
  url?: string;
203
222
  funcs?: Remote['funcs'];
204
223
  websocket?: WebSocket;
224
+ on_error?(error: WebSocketConnectionError, websocket: WebSocket): void;
205
225
  });
206
- /** 幂等,保证 websocket 已连接,否则抛出异常,不需要手动调用,在其它方法中已默认自动重连 */
207
- connect(): Promise<void>;
226
+ on_error?(error: WebSocketConnectionError, websocket: WebSocket): void;
227
+ /** 幂等,保证 websocket 已连接,否则抛出异常,不需要手动调用,在其它方法中已默认自动重连
228
+ 作为接收方需要传入使用的 websocket 连接,确保这个这个连接的状态 */
229
+ connect(websocket?: WebSocket): Promise<void>;
208
230
  disconnect(): void;
209
- /** 接收 websocket 连接的 remote 端必传 websocket 参数;发起端选传,如果传了必须等于 this.websocket
231
+ /** 接收 websocket 连接的本地 remote 必传 websocket 参数;发起端选传,如果传了必须等于 this.websocket_lock.resource
210
232
  发送或连接出错时自动清理 message.id 对应的 handler */
211
233
  send(message: Message, websocket?: WebSocket): Promise<void>;
212
234
  /** 处理接收到的 websocket message 并解析, 根据 id dispatch 到对应的 handler 进行处理
213
235
  如果 message.done == true 则清理 handler
214
- 如果 handler 返回了值,则包装为 message 发送 */
215
- handle(data: ArrayBuffer, websocket: WebSocket): Promise<void>;
236
+ 如果 handler 返回了值,则包装为 message 发送
237
+ 使用 Uint8Array 作为参数更灵活 https://stackoverflow.com/a/74505197/7609214 */
238
+ handle(data: Uint8Array, websocket: WebSocket): Promise<void>;
216
239
  /** 调用 remote 中的 func, 只适用于最简单的一元 rpc (请求, 响应) */
217
240
  call<TReturn extends any[] = any[]>(func: string, args?: any[]): Promise<TReturn>;
218
241
  }
package/net.js CHANGED
@@ -5,7 +5,7 @@ import { Cookie, CookieJar, MemoryCookieStore } from 'tough-cookie';
5
5
  import make_fetch_cookie from 'fetch-cookie';
6
6
  import { t } from './i18n/instance.js';
7
7
  import './prototype.js';
8
- import { inspect, concat, assert, genid, delay } from './utils.js';
8
+ import { inspect, concat, assert, genid, delay, Lock } from './utils.js';
9
9
  export var MyProxy;
10
10
  (function (MyProxy) {
11
11
  MyProxy["socks5"] = "http://localhost:10080";
@@ -67,7 +67,7 @@ export async function request(url, { method, queries, headers: _headers, body, t
67
67
  // --- headers, http/2 开始都用小写的 headers
68
68
  let headers = new Headers({
69
69
  'accept-language': 'zh-CN,zh;q=0.9,en-US;q=0.8,en;q=0.7,ja-JP;q=0.6,ja;q=0.5',
70
- 'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/110.0.0.0 Safari/537.36',
70
+ 'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/111.0.0.0 Safari/537.36',
71
71
  });
72
72
  if (body !== undefined)
73
73
  headers.set('content-type', type);
@@ -429,32 +429,48 @@ on_message, on_error, on_close }) {
429
429
  /** 通过创建 remote 对象对 websocket rpc 进行抽象
430
430
  调用方使用 remote.call 进行调用
431
431
  被调方在创建 remote 对象时传入 funcs 注册处理函数,并使用 remote.handle 方法处理 websocket message
432
- 未连接时自动连接,断开后自动重连 (保证执行 send 时已连接,否则报错) */
432
+ 未连接时自动连接,断开后自动重连 (保证执行 send 时已连接,否则报错)
433
+ 从设计上与 rpc 状态与底层连接的状态无关 (通过传入 url 创建的),底层连接断开后,只要检测到断线,会尝试重连,如果重连成功,则不影响调用的状态
434
+ @example
435
+ // Zero 继承自 Remote 并通过 call 实现了一些方法
436
+ let zero = new Zero({ local: true })
437
+
438
+ // 一元 rpc
439
+ await zero.repl_ts('1234')
440
+
441
+ // 订阅流
442
+ const id = genid()
443
+
444
+ zero.handlers.set(id, ({ data: [chunk] }: Message<[Uint8Array]>) => {
445
+ term.write(chunk)
446
+ })
447
+
448
+ zero.send({ id, func: 'subscribe_stdio' }) */
433
449
  export class Remote {
450
+ /** 是否为建立 rpc 连接的发起方 */
451
+ initiator;
452
+ /** websocket url */
434
453
  url;
435
454
  /** 主动发起 websocket 连接的客户端 (构造函数传 url) 有这个属性 */
436
- websocket;
455
+ lwebsocket;
437
456
  /** 通过 rpc message.func 被调用的 rpc 函数 */
438
457
  funcs;
439
458
  /** map<id, message handler>: 通过 rpc message.id 找到对应的 handler, unary rpc 接收方不需要设置 handlers, 发送方需要 */
440
459
  handlers = new Map();
441
460
  print = false;
442
461
  pconnect;
443
- get connected() {
444
- return this.websocket?.readyState === WebSocket.OPEN;
445
- }
446
- static parse(array_buffer) {
447
- const buf = new Uint8Array(array_buffer);
448
- const dv = new DataView(array_buffer);
462
+ /** 使用 Uint8Array 作为参数更灵活 https://stackoverflow.com/a/74505197/7609214 */
463
+ static parse(buffer) {
464
+ const dv = new DataView(buffer.buffer, buffer.byteOffset, buffer.byteLength);
449
465
  const len_json = dv.getUint32(0, true);
450
466
  let offset = 4 + len_json;
451
- let message = JSON.parse(decoder.decode(buf.subarray(4, offset)));
467
+ let message = JSON.parse(decoder.decode(buffer.subarray(4, offset)));
452
468
  if (message.bins) {
453
469
  const { bins } = message;
454
470
  let { data } = message;
455
471
  for (const ibin of bins) {
456
472
  const len_buf = data[ibin];
457
- data[ibin] = buf.subarray(offset, offset + len_buf);
473
+ data[ibin] = buffer.subarray(offset, offset + len_buf);
458
474
  offset += len_buf;
459
475
  }
460
476
  }
@@ -489,50 +505,69 @@ export class Remote {
489
505
  dv.setUint32(0, str_json.length, true);
490
506
  return concat([dv, str_json, ...bufs]);
491
507
  }
492
- constructor({ url, funcs = {}, websocket } = {}) {
508
+ /** 作为 rpc 发起方,可以通过传入 url 或者 websocket 定义远程 Remote
509
+ 作为 rpc 接收方,可以不传 url 和 websocket 定义本地 Remote */
510
+ constructor({ url, funcs = {}, websocket, on_error } = {}) {
511
+ assert(!(url && websocket), t('构建 Remote 时 url 和 websocket 最多只能传一个'));
512
+ this.initiator = Boolean(url || websocket);
493
513
  this.url = url;
494
514
  this.funcs = funcs;
495
- this.websocket = websocket;
515
+ if (on_error)
516
+ this.on_error = on_error;
517
+ this.lwebsocket = new Lock(websocket || null);
518
+ }
519
+ on_error(error, websocket) {
520
+ throw error;
496
521
  }
497
- /** 幂等,保证 websocket 已连接,否则抛出异常,不需要手动调用,在其它方法中已默认自动重连 */
498
- async connect() {
499
- if (this.connected)
522
+ /** 幂等,保证 websocket 已连接,否则抛出异常,不需要手动调用,在其它方法中已默认自动重连
523
+ 作为接收方需要传入使用的 websocket 连接,确保这个这个连接的状态 */
524
+ async connect(websocket) {
525
+ if (this.initiator) {
526
+ if (this.lwebsocket.resource?.readyState === WebSocket.OPEN)
527
+ return;
528
+ else if (!this.url)
529
+ throw new Error(t('创建 Remote 时传入的 websocket 连接已断开'));
530
+ // else 等待重连
531
+ }
532
+ else if (websocket.readyState === WebSocket.OPEN)
500
533
  return;
501
- const ptail = this.pconnect;
502
- let resolve;
503
- this.pconnect = new Promise((_resolve, _reject) => {
504
- resolve = _resolve;
534
+ else
535
+ throw new Error(t('传入的 websocket 连接已断开'));
536
+ // 假设有多个请求想要并发连接 websocket, 且此时 websocket 是断开的状态
537
+ // 应该排队依次连接,而不是后续的连接直接使用第一次连接的 promise,后续调用还是应该尝试重连(不止连接一次)
538
+ return this.lwebsocket.request(async (websocket) => {
539
+ // 保存的 rpc 状态在 this.handlers, 与 websocket 无关,因此即使断开重连也不影响 rpc 的运行,即
540
+ // 底层连接断开后自动重连对上层应该是无感知的,除非再次连接时失败
541
+ if (websocket?.readyState === WebSocket.OPEN)
542
+ return;
543
+ else if (!this.url)
544
+ throw new Error(t('创建 Remote 时传入的 websocket 连接已断开'));
545
+ else // 重连
546
+ this.lwebsocket.resource = await connect_websocket(this.url, {
547
+ on_message: (data, websocket) => {
548
+ this.handle(new Uint8Array(data), websocket);
549
+ },
550
+ on_error: this.on_error.bind(this)
551
+ });
505
552
  });
506
- await ptail;
507
- try {
508
- if (!this.connected)
509
- // 保存的 rpc 状态在 this.handlers, 与 websocket 无关,因此即使断开重连也不影响 rpc 的运行,即
510
- // 底层连接断开后自动重连对上层应该是无感知的,除非再次连接时失败
511
- this.websocket = await connect_websocket(this.url, { on_message: this.handle.bind(this) });
512
- }
513
- finally {
514
- resolve();
515
- }
516
553
  }
517
554
  disconnect() {
518
- this.websocket?.close(1000);
555
+ this.lwebsocket.resource?.close(1000);
519
556
  }
520
- /** 接收 websocket 连接的 remote 端必传 websocket 参数;发起端选传,如果传了必须等于 this.websocket
557
+ /** 接收 websocket 连接的本地 remote 必传 websocket 参数;发起端选传,如果传了必须等于 this.websocket_lock.resource
521
558
  发送或连接出错时自动清理 message.id 对应的 handler */
522
559
  async send(message, websocket) {
523
560
  try {
524
- if (this.url) {
525
- assert(!websocket || websocket === this.websocket);
561
+ if (this.initiator) {
562
+ assert(!websocket || websocket === this.lwebsocket.resource);
526
563
  await this.connect();
527
- websocket = this.websocket;
528
- }
529
- else {
530
- assert(websocket);
531
- if (websocket.readyState !== WebSocket.OPEN)
532
- throw new Error(t('remote.send(): websocket client 已断开'));
564
+ websocket = this.lwebsocket.resource;
533
565
  }
566
+ else
567
+ await this.connect(websocket);
534
568
  if (!message.id)
535
569
  message.id = genid();
570
+ // 不需要独占 websocket
536
571
  websocket.send(Remote.pack(message));
537
572
  }
538
573
  catch (error) {
@@ -543,7 +578,8 @@ export class Remote {
543
578
  }
544
579
  /** 处理接收到的 websocket message 并解析, 根据 id dispatch 到对应的 handler 进行处理
545
580
  如果 message.done == true 则清理 handler
546
- 如果 handler 返回了值,则包装为 message 发送 */
581
+ 如果 handler 返回了值,则包装为 message 发送
582
+ 使用 Uint8Array 作为参数更灵活 https://stackoverflow.com/a/74505197/7609214 */
547
583
  async handle(data, websocket) {
548
584
  const message = Remote.parse(data);
549
585
  const { id, func, done } = message;
package/net.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"net.js","sourceRoot":"","sources":["net.ts"],"names":[],"mappings":"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","sourcesContent":["import type { InspectOptions } from 'util'\n\nimport { fetch, ProxyAgent, FormData, Headers, type Response, type RequestInit } from 'undici'\n\nimport { WebSocket, type CloseEvent, type ErrorEvent } from 'ws'\n\nimport qs from 'qs'\n\nimport { Cookie, CookieJar, MemoryCookieStore } from 'tough-cookie'\n\ndeclare module 'tough-cookie' {\n interface MemoryCookieStore {\n idx: Record<string, any>\n }\n}\n\nimport make_fetch_cookie from 'fetch-cookie'\n\nimport { t } from './i18n/instance.js'\nimport './prototype.js'\nimport type { Encoding } from './file.js'\nimport { inspect, concat, assert, genid, delay } from './utils.js'\n\nexport enum MyProxy {\n socks5 = 'http://localhost:10080',\n whistle = 'http://localhost:8899',\n}\n\n\n// ------------------------------------ fetch, request\nlet cookie_store = new MemoryCookieStore()\n\nexport const cookies = {\n store: cookie_store,\n \n jar: new CookieJar(cookie_store),\n \n get (domain_or_url: string, str = false) {\n if (domain_or_url.startsWith('http'))\n if (str)\n return this.jar.getCookieString(domain_or_url)\n else\n return this.jar.getCookies(domain_or_url)\n \n let cookies: Cookie[]\n this.store.findCookies(domain_or_url, null, true, (error, _cookies) => {\n if (error)\n throw error\n cookies = _cookies\n })\n return cookies\n },\n}\n\nexport { Cookie }\n\n\nlet proxy_agents: Record<string, ProxyAgent> = { }\n\nconst fetch_cookie = make_fetch_cookie(fetch, cookies.jar, false)\n\n\nexport interface BasicAuth {\n type: 'basic',\n username: string\n password: string\n}\n\nexport interface BearerAuth {\n type: 'bearer',\n token: string\n}\n\n\nexport interface RequestOptions {\n method?: 'GET' | 'POST' | 'PUT' | 'HEAD' | 'DELETE' | 'PATCH'\n \n queries?: Record<string, any>\n \n headers?: Record<string, string> | Headers\n \n body?: string | Record<string, any> | NodeJS.ArrayBufferView | ArrayBuffer | \n AsyncIterable<Uint8Array> | Iterable<Uint8Array> | \n URLSearchParams | FormData\n \n type?: 'application/json' | 'application/x-www-form-urlencoded' | 'multipart/form-data'\n \n proxy?: boolean | MyProxy | string\n \n encoding?: Encoding | 'binary'\n \n retries?: true | number\n \n timeout?: number\n \n auth?: BasicAuth | BearerAuth\n \n cookies?: Record<string, string>\n}\n\nexport interface RequestRawOptions extends RequestOptions {\n raw: true\n}\n\n\nasync function fetch_retry (url: URL, options: RequestInit, timeout: number, retries = 0, count = 0): Promise<Response> {\n try {\n options.signal = AbortSignal.timeout(timeout)\n return await fetch_cookie(url, options)\n } catch (error) {\n if (\n count >= retries ||\n error.name !== 'TimeoutError' && !['ECONNRESET', 'ETIMEDOUT', 'ESOCKETTIMEDOUT'].includes(error.cause?.code)\n )\n throw error\n else {\n const duration = 2 ** count\n console.log(`${t('等待 {{duration}} 秒后重试 fetch ({{_count}}) …', { duration, _count: count }).yellow} ${url.toString().blue.underline}`)\n await delay(1000 * duration)\n return fetch_retry(url, options, timeout, retries, count + 1)\n }\n }\n}\n\n\nexport interface RequestError extends Error {\n url: URL\n \n options: RequestInit\n \n response?: {\n /** 状态码 */\n status: number\n \n url: string\n \n headers: Headers\n \n ok: boolean\n \n type: ResponseType\n \n text: string\n \n redirected: boolean\n }\n \n [inspect.custom]: Function\n}\n\n\n/** \n - url: 必须是完整 url\n - options?:\n - method?: `有 body 时为 POST, 否则为 GET` 'GET' | 'POST' | ···\n - queries?: 添加到 url 上的参数,是 Record<string, any>,true/false 会被转换为 0/1\n - headers?: http 请求头 (Record<string, string> 或者 Headers 类型),其中 key 必须是小写的\n - body?: http 请求体,可以是 string, Record<string, any> (会自动 JSON.stringify), ArrayBuffer(View), (Async)Iterable<Uint8Array> \n URLSearchParams (type 为 x-www-form-urlencoded), FormData (type 为 form-data)\n - type?: `'application/json'` 有 body 时设置 http 请求头中的 content-type 头\n - proxy?: `false` 通过代理发送请求\n - 为 true 时使用 MyProxy.socks5\n - 为非空 string 作为代理地址\n - 为 falsy 值时设为 false\n - encoding?: `根据网页 content-type: charset=gb18030 提取 || 'utf-8'` 传入 'binary' 时返回 ArrayBuffer\n - retries?: `false` 可以传入 true (默认 2 次) 或 重试次数\n - timeout?: `5 * 1000`\n - auth?: BasicAuth | BearerAuth\n - cookies?: 需要额外添加到请求的 cookies, 默认情况下携带了 MemoryCookieStore 中保存的之前 http 响应中的 cookies \n - raw?: `false` 传入后返回整个 response */\nexport async function request (url: string | URL): Promise<string>\nexport async function request (url: string | URL, options: RequestRawOptions): Promise<Response>\nexport async function request (url: string | URL, options: RequestOptions & { encoding: 'binary' }): Promise<ArrayBuffer>\nexport async function request (url: string | URL, options: RequestOptions): Promise<string>\nexport async function request (url: string | URL, {\n method,\n \n queries,\n \n headers: _headers,\n \n body,\n \n type = 'application/json',\n \n proxy,\n \n encoding,\n \n retries,\n \n timeout = 5 * 1000,\n \n auth,\n \n cookies: _cookies,\n \n raw = false,\n}: RequestOptions & { raw?: boolean } = { }) {\n url = new URL(url)\n \n if (queries)\n for (const key in queries) {\n let value = queries[key]\n if (typeof value === 'boolean')\n value = value ? '1' : '0'\n url.searchParams.append(key, value)\n }\n \n if (body !== undefined && !method)\n method = 'POST'\n \n if (retries === true)\n retries = 2\n \n \n // --- headers, http/2 开始都用小写的 headers\n let headers = new Headers({\n 'accept-language': 'zh-CN,zh;q=0.9,en-US;q=0.8,en;q=0.7,ja-JP;q=0.6,ja;q=0.5',\n 'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/110.0.0.0 Safari/537.36',\n })\n \n if (body !== undefined)\n headers.set('content-type', type)\n \n if (auth)\n headers.set(\n 'authorization',\n auth.type === 'basic' ? `Basic ${`${auth.username}:${auth.password}`.to_base64()}` : `Bearer ${auth.token}`\n )\n \n if (_cookies)\n Object.entries(_cookies)\n .map(([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`)\n .join('; ')\n \n if (_headers)\n if (_headers instanceof Headers)\n // @ts-ignore: ts 类型不支持,实际上已经有了\n for (const [key, value] of _headers)\n headers.set(key, value)\n else\n for (const key in _headers) {\n const value = _headers[key]\n if (!value.startsWith(':')) { // 可能在 http/2 的 response 中会有这样开头的保留 headers, 在透传时忽略比较好\n assert(key === key.toLowerCase(), t('传入 request 的 headers 参数中 key 应该都是小写的,实际为 {{key}}', { key }))\n headers[key] = value\n }\n }\n \n \n let options: RequestInit & { maxRedirect: number /* fetch-cookie 需要这个参数 */ } = {\n ... method ? { method } : { },\n \n dispatcher: (() => {\n if (proxy) {\n if (proxy === true)\n proxy = MyProxy.socks5\n \n return proxy_agents[proxy] ??= new ProxyAgent({ uri: proxy })\n }\n })(),\n \n keepalive: true,\n \n redirect: 'follow',\n \n maxRedirect: 6,\n \n credentials: 'include',\n \n headers,\n \n // --- body\n body: (() => {\n if (body === undefined)\n return\n \n switch (type) {\n case 'application/json':\n return typeof body === 'string' || \n ArrayBuffer.isView(body) || \n body instanceof ArrayBuffer || \n // 测试 Iterable<Uint8Array>, AsyncIterable<Uint8Array>\n (!Array.isArray(body) && (Symbol.iterator in Object(body) || Symbol.asyncIterator in Object(body))) ?\n body as (string | NodeJS.ArrayBufferView | ArrayBuffer | AsyncIterable<Uint8Array> | Iterable<Uint8Array>)\n :\n JSON.stringify(body)\n \n case 'application/x-www-form-urlencoded':\n return body instanceof URLSearchParams ? body : new URLSearchParams(body as Record<string, any>)\n \n case 'multipart/form-data':\n if (body instanceof FormData)\n return body\n else {\n let form = new FormData()\n for (const key in body as Record<string, any>) {\n let value = body[key]\n form.set(key, value)\n }\n return form\n }\n }\n })(),\n }\n \n \n let response: Response\n \n try {\n response = await fetch_retry(url, options, timeout, retries)\n \n if (!response.ok)\n throw Object.assign(\n new Error(t('状态码 {{status}} 非 2xx', { status: response.status })),\n { name: 'StatusCodeError' }\n )\n } catch (error) {\n error.url = url\n error.options = options\n \n if (response)\n error.response = {\n status: response.status,\n url: response.url,\n headers: response.headers,\n ok: response.ok,\n type: response.type,\n text: await response.text(),\n redirected: response.redirected\n }\n \n error[inspect.custom] = (depth: number, options: InspectOptions, inspect: Function) => {\n const { colors } = options\n \n let _method: string = method || 'GET'\n if (colors)\n _method = _method.red\n \n let _url = url.toString()\n if (colors)\n _url = _url.blue.underline\n \n let s = '\\n' +\n `${_method} ${_url}\\n`\n \n if (queries && Object.keys(queries).length) {\n let _query = t('请求参数:')\n if (colors)\n _query = _query.yellow\n \n s += _query + '\\n' +\n inspect(queries, options) + '\\n'\n }\n \n if (body !== undefined) {\n let _body = t('请求体:')\n if (colors)\n _body = _body.yellow\n \n s += _body + '\\n' +\n inspect(body, options) + '\\n'\n }\n \n \n if (error.name === 'StatusCodeError') {\n let _status = t('响应状态码:')\n if (colors)\n _status = _status.yellow\n \n let _code = String(response.status)\n if (colors)\n _code = _code.red\n \n s += _status + ' ' + _code + '\\n'\n } else if (error.name === 'TimeoutError') {\n let _timeout = t('超过等待时间:')\n if (colors)\n _timeout = _timeout.red\n \n s += `${_timeout} ${timeout} ms\\n`\n }\n \n \n if (response) {\n let _headers = t('响应头:')\n if (colors)\n _headers = _headers.yellow\n \n s += _headers + '\\n'\n \n for (const [key, value] of response.headers)\n s += `${key}: ${value}\\n`\n \n if ((error as RequestError).response.text) {\n let _body = t('响应体:')\n if (colors)\n _body = _body.yellow\n \n s += _body + '\\n' +\n (error as RequestError).response.text + '\\n'\n }\n }\n \n let _stack = t('调用栈:')\n if (colors)\n _stack = _stack.yellow\n \n s += _stack + '\\n'\n \n if (error.cause)\n s += inspect(error.cause, options) + '\\n'\n \n s += error.stack + '\\n' +\n '\\n'\n \n return s\n }\n \n throw error\n }\n \n if (raw)\n return response\n \n if (!response.body)\n return response.body\n \n // --- decode body\n if (encoding === 'binary')\n return response.arrayBuffer()\n \n encoding ||= /charset=(.*)/.exec(response.headers.get('content-type'))?.[1] as Encoding || 'utf-8'\n \n if (/utf-?8/i.test(encoding))\n return response.text()\n \n return new TextDecoder(encoding)\n .decode(await response.arrayBuffer())\n}\n\n\n/** 发起 http 请求并将响应体作为 json 解析 */\nexport async function request_json <T = any> (url: string | URL, options?: RequestOptions): Promise<T> {\n const resp = await request(url, options)\n if (!resp)\n return\n \n try {\n return JSON.parse(resp)\n } catch (error) {\n console.error(resp)\n throw error\n }\n}\n\n\n/** 使用 $.html(cheerio_element) 来获取 outer html */\nexport async function parse_html (html: string) {\n const { default: cheerio } = await import('cheerio')\n \n let $ = cheerio.load(html, { decodeEntities: false })\n \n Object.defineProperty($, inspect.custom, {\n configurable: true,\n enumerable: false,\n value () {\n return this.html()\n }\n })\n \n Object.defineProperty($.prototype, inspect.custom, {\n configurable: true,\n enumerable: false,\n value (this: cheerio.Cheerio) {\n if (this.length > 1)\n return this.map((index, element) => {\n if (typeof element === 'string') return element\n return $.html(element)\n }).get().join_lines()\n \n return this.toString()\n }\n })\n \n return $\n}\n\n\n/** 使用 $.html(cheerio_element) 来获取 outer html */\nexport async function request_page (url: string | URL, options?: RequestOptions) {\n return parse_html(\n await request(url, options)\n )\n}\n\n\nexport function to_curl (url: string | URL, { queries, headers, method, body, proxy, exe = true }: RequestOptions & { exe?: boolean } = { }) {\n if (proxy === true)\n proxy = process.env.http_proxy\n \n url = url.toString()\n \n if (!url.startsWith('http'))\n url = `http://${url}`\n \n return (exe ? 'curl.exe' : 'curl') + \n ' ' + ( url + (queries ? '?' : '') + qs.stringify(queries) ).quote() +\n // ( typeof proxy === 'undefined' ?\n // ''\n // :\n // ( proxy ? ' --proxy ' + proxy.quote() : ' --noproxy ' + '*'.quote())\n // ) +\n ( proxy ? ` --proxy ${proxy.quote()}` : '' ) +\n ( method && method !== 'GET' ? ` -X ${method}` : '' ) +\n ( headers ? Object.entries(headers).map( ([key, value]) => ' -H ' + `${key}: ${value}`.quote() ).join('') : '' ) +\n ( body ? ' -H ' + 'content-type: application/json'.quote() : '') +\n ( body ? ' --data ' + JSON.stringify(body).quote() : '')\n}\n\n\n\n\n// ------------------------------------ rpc client\n/** post json to http://localhost:8421/api/rpc\n - func: function name\n - args?: argument array\n - options?:\n - ignore?: `false` wait for execution but do not serialize result to response\n - async?: `false` do not wait for exec\n*/\nexport async function rpc (\n func: string, \n args?: any[], \n { url = 'http://localhost:8421/api/rpc', async: _async = false, ignore = false }: { url?: string, async?: boolean, ignore?: boolean } = { }\n) {\n if (!func)\n throw new Error('rpc argument error: no func')\n \n return request_json(url, {\n body: {\n func,\n args,\n async: _async,\n ignore,\n }\n })\n}\n\n\n\nlet decoder = new TextDecoder()\n\nlet encoder = new TextEncoder()\n\n\nexport { WebSocket }\n\n\nexport class WebSocketConnectionError extends Error {\n override name = 'WebSocketConnectionError'\n \n websocket: WebSocket\n \n event: CloseEvent | ErrorEvent\n \n type: 'close' | 'error'\n \n address?: string\n \n errno?: number\n \n port?: number\n \n syscall?: string\n \n /** close 事件时为 close code, error 事件为 error code */\n code?: string | number\n \n reason?: string\n \n \n constructor (websocket: WebSocket, event: CloseEvent | ErrorEvent, message: string = '') {\n super(`${websocket.url} ${t('连接出错了')}. ${message}`)\n \n this.websocket = websocket\n this.event = event\n this.type = event.type as 'close' | 'error'\n \n if (this.type === 'error') {\n const { error } = event as ErrorEvent\n this.address = error.address\n this.code = error.code\n this.errno = error.errno\n this.port = error.port\n this.syscall = error.syscall\n this.reason = error.reason\n \n this.stack = `${this.name}: ${this.message}\\n` +\n error.stack.slice(error.stack.indexOf('\\n') + 1) + '\\n'\n } else {\n this.code = (event as CloseEvent).code\n this.reason = (event as CloseEvent).reason\n }\n }\n}\n\n\n/** 连接 websocket url, 设置各种事件监听器。在 open 事件后 resolve, 返回 websocket \n 遇到 error 时会创建 WebSocketConnectionError: \n - reject 掉返回的 promise (若此时未 settle) \n - 作为参数调用 on_error (已 settle 且有 on_error 回调) \n 可以用 WebSocket.bufferedAmount 来显示大消息的发送进度 \n https://developer.mozilla.org/en-US/docs/Web/API/WebSocket/bufferedAmount\n - url\n - options:\n - protocols?\n - max_payload?: `8 GB`\n - on_message: 根据 websocket frame 的 opcode 不同 (text frame 或 binary frame),event 中的 data 对应为 ArrayBuffer 或者 string \n https://datatracker.ietf.org/doc/html/rfc6455#section-5.2\n - on_error?: 在 websocket 出错和非正常关闭 (close, error 事件) 时都调用,可以根据 error.type 来区分,error 的类型是 WebSocketConnectionError, \n type 为 'close' 时有 code 和 reason 属性 \n - on_close?: 和 websocket 的 'close' 事件不相同,只在正常关闭 (close code 为 1000) 时才调用,否则都会调用 on_error \n https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent#Status_codes */\nexport async function connect_websocket (\n url: string | URL,\n {\n protocols,\n max_payload = 2 ** 33, // 8 GB\n on_message,\n on_error,\n on_close\n }: {\n protocols?: string | string[]\n max_payload?: number\n on_message (data: ArrayBuffer | string, websocket: WebSocket): any\n on_error? (error: WebSocketConnectionError, websocket: WebSocket): any\n on_close? (event: CloseEvent, websocket: WebSocket): any\n }\n) {\n let websocket = new WebSocket(url, protocols, {\n maxPayload: max_payload,\n skipUTF8Validation: true\n })\n \n // https://stackoverflow.com/questions/11821096/what-is-the-difference-between-an-arraybuffer-and-a-blob/39951543\n websocket.binaryType = 'arraybuffer'\n \n return new Promise<WebSocket>((resolve, reject) => {\n let settled = false\n \n websocket.addEventListener('open', event => {\n console.log(\n websocket.url +\n (protocols ? \n ' ' + (\n typeof protocols === 'string' ? protocols : protocols.join(', ').bracket('square')\n ).bracket()\n :\n '') +\n t(' 已连接')\n )\n \n settled = true\n resolve(websocket)\n })\n \n websocket.addEventListener('close', event => {\n // https://blog.insiderattack.net/promises-next-ticks-and-immediates-nodejs-event-loop-part-3-9226cbe7a6aa\n // error 事件会先发生,然后 reject(error) 被执行,此时还未轮到外层的 await connect_websocket 执行(在微任务队列中),\n // 接着马上 close 事件也被调用,此时 settled,马上调用了 on_error 函数,弄乱了顺序\n // error 的错误信息比较多,而且通过 await 得到的栈也比较清晰,这里延后调用 on_close 和 on_error,放到微任务队列之后的 timers 队列中\n setTimeout(() => {\n if (event.code === 1000) // 正常关闭\n if (on_close)\n on_close(event, websocket)\n else\n console.log(`${websocket.url} ${t('已正常关闭')}`)\n else { // 异常关闭,认为发生了错误,进行错误处理\n assert(settled, t('websocket close 事件时应该已经 settled'))\n \n const error = new WebSocketConnectionError(websocket, event, `${t('连接被关闭')}, code: ${event.code}${ event.reason ? `, ${t('原因')}: ${event.reason}` : '' }`)\n \n if (on_error)\n on_error(error, websocket)\n else // 既然用户不传 on_error, 就当 unhandled error 抛出来\n throw error\n }\n })\n })\n \n websocket.addEventListener('error', event => {\n const error = new WebSocketConnectionError(websocket, event, event.error?.message)\n \n if (settled) {\n if (on_error)\n on_error(error, websocket)\n else // 既然用户不传 on_error, 就当 unhandled error 抛出来\n throw error\n } else {\n settled = true\n reject(error)\n }\n })\n \n websocket.addEventListener('message', event => {\n on_message(event.data as ArrayBuffer | string, websocket)\n })\n })\n}\n\n\n/** 接收到消息后的处理函数 \n 返回值可以是:\n - 数组: 会自动被封装为 { id: 相同, data: 返回值, done: true } 这样的消息并调用 websocket.send 将其发送\n - void: 什么都不做\n - 以上的 promise */\nexport type MessageHandler = (message: Message, websocket?: WebSocket) => void | any[] | Promise<void | any[]>\n\n\n/** 二进制消息格式 \n - json.length (小端序): 4 字节\n - json 数据\n - binary 数据\n*/\nexport interface Message <TData extends any[] = any[]> {\n /** rpc id: 在 rpc 系统中认为是唯一的。用来在单个 websocket 连接上复用多个 rpc 请求。多个相同 id 的 message 组成一个请求流 */\n id?: number\n \n /** 只在 rpc 发起时指定被调用的 function name,发起时 rpc 时必传 */\n func?: string\n \n /** 通过这个 flag 主动表明这是发往对方的最后一个 message, 对方可以销毁 handler 了 \n 并非强制,可以不说明,由双方的函数自己约定\n */\n done?: boolean\n \n /** 通知对方这里产生的错误,本质上类似 data 也是一种数据,并不代表 rpc 的结束,后续可能继续有 rpc message 交换 */\n error?: Error\n \n /** data 是一个数组, 作为:\n - rpc 发起方调用 func 的参数,或者请求流 message 携带的数据\n - 结果或者响应流的数据,传给请求发起方\n \n 里面是可序列化为 json 的 js 变量,或者是 Uint8Array \n Uint8Array 的参数处理后被替换为 Uint8Array.byteLength, 并将下标记录在 bins 中\n \n 注意: 数组中如果有 undefined 值,传输到对面会变成 null 值\n */\n data?: TData\n \n /** bins: data 中哪些下标对应的原始值是 Uint8Array 类型的,如: [0, 3] */\n bins?: number[]\n}\n\n/** 通过创建 remote 对象对 websocket rpc 进行抽象 \n 调用方使用 remote.call 进行调用 \n 被调方在创建 remote 对象时传入 funcs 注册处理函数,并使用 remote.handle 方法处理 websocket message \n 未连接时自动连接,断开后自动重连 (保证执行 send 时已连接,否则报错) */\nexport class Remote {\n url: string\n \n /** 主动发起 websocket 连接的客户端 (构造函数传 url) 有这个属性 */\n websocket?: WebSocket\n \n /** 通过 rpc message.func 被调用的 rpc 函数 */\n funcs: Record<string, MessageHandler>\n \n /** map<id, message handler>: 通过 rpc message.id 找到对应的 handler, unary rpc 接收方不需要设置 handlers, 发送方需要 */\n handlers = new Map<number, MessageHandler>()\n \n \n print = false\n \n pconnect: Promise<any>\n \n \n get connected () {\n return this.websocket?.readyState === WebSocket.OPEN\n }\n \n \n static parse <TData extends any[] = any[]> (array_buffer: ArrayBuffer) {\n const buf = new Uint8Array(array_buffer as ArrayBuffer)\n const dv = new DataView(array_buffer)\n \n const len_json = dv.getUint32(0, true)\n \n let offset = 4 + len_json\n \n let message: Message<TData> = JSON.parse(decoder.decode(buf.subarray(4, offset)))\n \n if (message.bins) {\n const { bins } = message\n let { data } = message\n \n for (const ibin of bins) {\n const len_buf = data[ibin]\n data[ibin] = buf.subarray(offset, offset + len_buf)\n offset += len_buf\n }\n }\n \n if (message.error)\n message.error = Object.assign(new Error(), message.error)\n \n return message\n }\n \n \n static pack ({ id, func, data = [ ], done, error }: Message) {\n let data_ = new Array(data.length)\n let bins: number[] = [ ]\n let bufs: Uint8Array[] = [ ]\n \n for (let i = 0; i < data.length; i++) {\n const item = data[i]\n if (item instanceof Uint8Array) {\n bins.push(i)\n bufs.push(item)\n data_[i] = item.length\n } else\n data_[i] = item\n }\n \n const data_json = {\n id,\n ... func ? { func } : { },\n ... done ? { done } : { },\n ... error ? { error } : { },\n ... data_.length ? { data: data_ } : { },\n ... bins.length ? { bins } : { },\n }\n \n const str_json = encoder.encode(JSON.stringify(data_json))\n \n let dv = new DataView(new ArrayBuffer(4))\n \n dv.setUint32(0, str_json.length, true)\n \n return concat([dv, str_json, ... bufs])\n }\n \n \n constructor ({\n url, funcs = { }, websocket\n }: {\n url?: string, funcs?: Remote['funcs'], websocket?: WebSocket\n } = { }) {\n this.url = url\n this.funcs = funcs\n this.websocket = websocket\n }\n \n \n /** 幂等,保证 websocket 已连接,否则抛出异常,不需要手动调用,在其它方法中已默认自动重连 */\n async connect () {\n if (this.connected)\n return\n \n const ptail = this.pconnect\n \n let resolve: () => void\n this.pconnect = new Promise<void>((_resolve, _reject) => {\n resolve = _resolve\n })\n \n await ptail\n \n try {\n if (!this.connected)\n // 保存的 rpc 状态在 this.handlers, 与 websocket 无关,因此即使断开重连也不影响 rpc 的运行,即\n // 底层连接断开后自动重连对上层应该是无感知的,除非再次连接时失败\n this.websocket = await connect_websocket(this.url, { on_message: this.handle.bind(this) })\n } finally {\n resolve()\n }\n }\n \n \n disconnect () {\n this.websocket?.close(1000)\n }\n \n \n /** 接收 websocket 连接的 remote 端必传 websocket 参数;发起端选传,如果传了必须等于 this.websocket \n 发送或连接出错时自动清理 message.id 对应的 handler */\n async send (message: Message, websocket?: WebSocket) {\n try {\n if (this.url) {\n assert(!websocket || websocket === this.websocket)\n await this.connect()\n websocket = this.websocket\n } else {\n assert(websocket)\n if (websocket.readyState !== WebSocket.OPEN)\n throw new Error(t('remote.send(): websocket client 已断开'))\n }\n \n if (!message.id)\n message.id = genid()\n \n websocket.send(Remote.pack(message))\n } catch (error) {\n if (message.id)\n this.handlers.delete(message.id)\n throw error\n }\n }\n \n \n /** 处理接收到的 websocket message 并解析, 根据 id dispatch 到对应的 handler 进行处理 \n 如果 message.done == true 则清理 handler \n 如果 handler 返回了值,则包装为 message 发送 */\n async handle (data: ArrayBuffer, websocket: WebSocket) {\n const message = Remote.parse(data)\n \n const { id, func, done } = message\n \n if (this.print)\n console.log(message)\n \n let handler: MessageHandler\n \n if (func)\n handler = this.funcs[func]\n else {\n handler = this.handlers.get(id)\n if (done)\n this.handlers.delete(id)\n }\n \n try {\n if (handler) {\n const data = await handler(message, websocket)\n if (data)\n await this.send({ id, data }, websocket)\n } else if (message.error)\n throw message.error\n else\n throw new Error(`${t('找不到 rpc handler')}: ${func ? `func: ${func.quote()}` : `id: ${id}`}`)\n } catch (error) {\n // handle 出错并不意味着 rpc 一定会结束,可能 error 是运行中的正常数据,所以不能清理 handler\n \n if (\n websocket.readyState === WebSocket.OPEN &&\n !message.error // 防止无限循环往对方发送 error, 只有在对方无错误时才可以发送\n )\n try { await this.send({ id, error, /* 不能设置 done 清理对面 handler, 理由同上 */ }, websocket) } catch { }\n \n // 再往上层抛出错误没有意义了,上层调用栈是 websocket.on('message') 之类的\n console.log(error)\n }\n }\n \n \n /** 调用 remote 中的 func, 只适用于最简单的一元 rpc (请求, 响应) */\n async call <TReturn extends any[] = any[]> (func: string, args?: any[]) {\n return new Promise<TReturn>(async (resolve, reject) => {\n const id = genid()\n \n this.handlers.set(id, (message: Message<TReturn>) => {\n const { error, data } = message\n if (error)\n reject(error)\n else\n resolve(data)\n this.handlers.delete(id)\n })\n \n try {\n await this.send({ id, func, data: args }) // 不需要 done: true, 因为对面的 remote.handlers 中不会有这个 id 的 handler\n } catch (error) {\n reject(error)\n }\n })\n }\n}\n\n"]}
1
+ {"version":3,"file":"net.js","sourceRoot":"","sources":["net.ts"],"names":[],"mappings":"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","sourcesContent":["import type { InspectOptions } from 'util'\n\nimport { fetch, ProxyAgent, FormData, Headers, type Response, type RequestInit } from 'undici'\n\nimport { WebSocket, type CloseEvent, type ErrorEvent } from 'ws'\n\nimport qs from 'qs'\n\nimport { Cookie, CookieJar, MemoryCookieStore } from 'tough-cookie'\n\ndeclare module 'tough-cookie' {\n interface MemoryCookieStore {\n idx: Record<string, any>\n }\n}\n\nimport make_fetch_cookie from 'fetch-cookie'\n\nimport { t } from './i18n/instance.js'\nimport './prototype.js'\nimport type { Encoding } from './file.js'\nimport { inspect, concat, assert, genid, delay, Lock } from './utils.js'\n\nexport enum MyProxy {\n socks5 = 'http://localhost:10080',\n whistle = 'http://localhost:8899',\n}\n\n\n// ------------------------------------ fetch, request\nlet cookie_store = new MemoryCookieStore()\n\nexport const cookies = {\n store: cookie_store,\n \n jar: new CookieJar(cookie_store),\n \n get (domain_or_url: string, str = false) {\n if (domain_or_url.startsWith('http'))\n if (str)\n return this.jar.getCookieString(domain_or_url)\n else\n return this.jar.getCookies(domain_or_url)\n \n let cookies: Cookie[]\n this.store.findCookies(domain_or_url, null, true, (error, _cookies) => {\n if (error)\n throw error\n cookies = _cookies\n })\n return cookies\n },\n}\n\nexport { Cookie }\n\n\nlet proxy_agents: Record<string, ProxyAgent> = { }\n\nconst fetch_cookie = make_fetch_cookie(fetch, cookies.jar, false)\n\n\nexport interface BasicAuth {\n type: 'basic',\n username: string\n password: string\n}\n\nexport interface BearerAuth {\n type: 'bearer',\n token: string\n}\n\n\nexport interface RequestOptions {\n method?: 'GET' | 'POST' | 'PUT' | 'HEAD' | 'DELETE' | 'PATCH'\n \n queries?: Record<string, any>\n \n headers?: Record<string, string> | Headers\n \n body?: string | Record<string, any> | NodeJS.ArrayBufferView | ArrayBuffer | \n AsyncIterable<Uint8Array> | Iterable<Uint8Array> | \n URLSearchParams | FormData\n \n type?: 'application/json' | 'application/x-www-form-urlencoded' | 'multipart/form-data'\n \n proxy?: boolean | MyProxy | string\n \n encoding?: Encoding | 'binary'\n \n retries?: true | number\n \n timeout?: number\n \n auth?: BasicAuth | BearerAuth\n \n cookies?: Record<string, string>\n}\n\nexport interface RequestRawOptions extends RequestOptions {\n raw: true\n}\n\n\nasync function fetch_retry (url: URL, options: RequestInit, timeout: number, retries = 0, count = 0): Promise<Response> {\n try {\n options.signal = AbortSignal.timeout(timeout)\n return await fetch_cookie(url, options)\n } catch (error) {\n if (\n count >= retries ||\n error.name !== 'TimeoutError' && !['ECONNRESET', 'ETIMEDOUT', 'ESOCKETTIMEDOUT'].includes(error.cause?.code)\n )\n throw error\n else {\n const duration = 2 ** count\n console.log(`${t('等待 {{duration}} 秒后重试 fetch ({{_count}}) …', { duration, _count: count }).yellow} ${url.toString().blue.underline}`)\n await delay(1000 * duration)\n return fetch_retry(url, options, timeout, retries, count + 1)\n }\n }\n}\n\n\nexport interface RequestError extends Error {\n url: URL\n \n options: RequestInit\n \n response?: {\n /** 状态码 */\n status: number\n \n url: string\n \n headers: Headers\n \n ok: boolean\n \n type: ResponseType\n \n text: string\n \n redirected: boolean\n }\n \n [inspect.custom]: Function\n}\n\n\n/** \n - url: 必须是完整 url\n - options?:\n - method?: `有 body 时为 POST, 否则为 GET` 'GET' | 'POST' | ···\n - queries?: 添加到 url 上的参数,是 Record<string, any>,true/false 会被转换为 0/1\n - headers?: http 请求头 (Record<string, string> 或者 Headers 类型),其中 key 必须是小写的\n - body?: http 请求体,可以是 string, Record<string, any> (会自动 JSON.stringify), ArrayBuffer(View), (Async)Iterable<Uint8Array> \n URLSearchParams (type 为 x-www-form-urlencoded), FormData (type 为 form-data)\n - type?: `'application/json'` 有 body 时设置 http 请求头中的 content-type 头\n - proxy?: `false` 通过代理发送请求\n - 为 true 时使用 MyProxy.socks5\n - 为非空 string 作为代理地址\n - 为 falsy 值时设为 false\n - encoding?: `根据网页 content-type: charset=gb18030 提取 || 'utf-8'` 传入 'binary' 时返回 ArrayBuffer\n - retries?: `false` 可以传入 true (默认 2 次) 或 重试次数\n - timeout?: `5 * 1000`\n - auth?: BasicAuth | BearerAuth\n - cookies?: 需要额外添加到请求的 cookies, 默认情况下携带了 MemoryCookieStore 中保存的之前 http 响应中的 cookies \n - raw?: `false` 传入后返回整个 response */\nexport async function request (url: string | URL): Promise<string>\nexport async function request (url: string | URL, options: RequestRawOptions): Promise<Response>\nexport async function request (url: string | URL, options: RequestOptions & { encoding: 'binary' }): Promise<ArrayBuffer>\nexport async function request (url: string | URL, options: RequestOptions): Promise<string>\nexport async function request (url: string | URL, {\n method,\n \n queries,\n \n headers: _headers,\n \n body,\n \n type = 'application/json',\n \n proxy,\n \n encoding,\n \n retries,\n \n timeout = 5 * 1000,\n \n auth,\n \n cookies: _cookies,\n \n raw = false,\n}: RequestOptions & { raw?: boolean } = { }) {\n url = new URL(url)\n \n if (queries)\n for (const key in queries) {\n let value = queries[key]\n if (typeof value === 'boolean')\n value = value ? '1' : '0'\n url.searchParams.append(key, value)\n }\n \n if (body !== undefined && !method)\n method = 'POST'\n \n if (retries === true)\n retries = 2\n \n \n // --- headers, http/2 开始都用小写的 headers\n let headers = new Headers({\n 'accept-language': 'zh-CN,zh;q=0.9,en-US;q=0.8,en;q=0.7,ja-JP;q=0.6,ja;q=0.5',\n 'user-agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/111.0.0.0 Safari/537.36',\n })\n \n if (body !== undefined)\n headers.set('content-type', type)\n \n if (auth)\n headers.set(\n 'authorization',\n auth.type === 'basic' ? `Basic ${`${auth.username}:${auth.password}`.to_base64()}` : `Bearer ${auth.token}`\n )\n \n if (_cookies)\n Object.entries(_cookies)\n .map(([key, value]) => `${encodeURIComponent(key)}=${encodeURIComponent(value)}`)\n .join('; ')\n \n if (_headers)\n if (_headers instanceof Headers)\n // @ts-ignore: ts 类型不支持,实际上已经有了\n for (const [key, value] of _headers)\n headers.set(key, value)\n else\n for (const key in _headers) {\n const value = _headers[key]\n if (!value.startsWith(':')) { // 可能在 http/2 的 response 中会有这样开头的保留 headers, 在透传时忽略比较好\n assert(key === key.toLowerCase(), t('传入 request 的 headers 参数中 key 应该都是小写的,实际为 {{key}}', { key }))\n headers[key] = value\n }\n }\n \n \n let options: RequestInit & { maxRedirect: number /* fetch-cookie 需要这个参数 */ } = {\n ... method ? { method } : { },\n \n dispatcher: (() => {\n if (proxy) {\n if (proxy === true)\n proxy = MyProxy.socks5\n \n return proxy_agents[proxy] ??= new ProxyAgent({ uri: proxy })\n }\n })(),\n \n keepalive: true,\n \n redirect: 'follow',\n \n maxRedirect: 6,\n \n credentials: 'include',\n \n headers,\n \n // --- body\n body: (() => {\n if (body === undefined)\n return\n \n switch (type) {\n case 'application/json':\n return typeof body === 'string' || \n ArrayBuffer.isView(body) || \n body instanceof ArrayBuffer || \n // 测试 Iterable<Uint8Array>, AsyncIterable<Uint8Array>\n (!Array.isArray(body) && (Symbol.iterator in Object(body) || Symbol.asyncIterator in Object(body))) ?\n body as (string | NodeJS.ArrayBufferView | ArrayBuffer | AsyncIterable<Uint8Array> | Iterable<Uint8Array>)\n :\n JSON.stringify(body)\n \n case 'application/x-www-form-urlencoded':\n return body instanceof URLSearchParams ? body : new URLSearchParams(body as Record<string, any>)\n \n case 'multipart/form-data':\n if (body instanceof FormData)\n return body\n else {\n let form = new FormData()\n for (const key in body as Record<string, any>) {\n let value = body[key]\n form.set(key, value)\n }\n return form\n }\n }\n })(),\n }\n \n \n let response: Response\n \n try {\n response = await fetch_retry(url, options, timeout, retries)\n \n if (!response.ok)\n throw Object.assign(\n new Error(t('状态码 {{status}} 非 2xx', { status: response.status })),\n { name: 'StatusCodeError' }\n )\n } catch (error) {\n error.url = url\n error.options = options\n \n if (response)\n error.response = {\n status: response.status,\n url: response.url,\n headers: response.headers,\n ok: response.ok,\n type: response.type,\n text: await response.text(),\n redirected: response.redirected\n }\n \n error[inspect.custom] = (depth: number, options: InspectOptions, inspect: Function) => {\n const { colors } = options\n \n let _method: string = method || 'GET'\n if (colors)\n _method = _method.red\n \n let _url = url.toString()\n if (colors)\n _url = _url.blue.underline\n \n let s = '\\n' +\n `${_method} ${_url}\\n`\n \n if (queries && Object.keys(queries).length) {\n let _query = t('请求参数:')\n if (colors)\n _query = _query.yellow\n \n s += _query + '\\n' +\n inspect(queries, options) + '\\n'\n }\n \n if (body !== undefined) {\n let _body = t('请求体:')\n if (colors)\n _body = _body.yellow\n \n s += _body + '\\n' +\n inspect(body, options) + '\\n'\n }\n \n \n if (error.name === 'StatusCodeError') {\n let _status = t('响应状态码:')\n if (colors)\n _status = _status.yellow\n \n let _code = String(response.status)\n if (colors)\n _code = _code.red\n \n s += _status + ' ' + _code + '\\n'\n } else if (error.name === 'TimeoutError') {\n let _timeout = t('超过等待时间:')\n if (colors)\n _timeout = _timeout.red\n \n s += `${_timeout} ${timeout} ms\\n`\n }\n \n \n if (response) {\n let _headers = t('响应头:')\n if (colors)\n _headers = _headers.yellow\n \n s += _headers + '\\n'\n \n for (const [key, value] of response.headers)\n s += `${key}: ${value}\\n`\n \n if ((error as RequestError).response.text) {\n let _body = t('响应体:')\n if (colors)\n _body = _body.yellow\n \n s += _body + '\\n' +\n (error as RequestError).response.text + '\\n'\n }\n }\n \n let _stack = t('调用栈:')\n if (colors)\n _stack = _stack.yellow\n \n s += _stack + '\\n'\n \n if (error.cause)\n s += inspect(error.cause, options) + '\\n'\n \n s += error.stack + '\\n' +\n '\\n'\n \n return s\n }\n \n throw error\n }\n \n if (raw)\n return response\n \n if (!response.body)\n return response.body\n \n // --- decode body\n if (encoding === 'binary')\n return response.arrayBuffer()\n \n encoding ||= /charset=(.*)/.exec(response.headers.get('content-type'))?.[1] as Encoding || 'utf-8'\n \n if (/utf-?8/i.test(encoding))\n return response.text()\n \n return new TextDecoder(encoding)\n .decode(await response.arrayBuffer())\n}\n\n\n/** 发起 http 请求并将响应体作为 json 解析 */\nexport async function request_json <T = any> (url: string | URL, options?: RequestOptions): Promise<T> {\n const resp = await request(url, options)\n if (!resp)\n return\n \n try {\n return JSON.parse(resp)\n } catch (error) {\n console.error(resp)\n throw error\n }\n}\n\n\n/** 使用 $.html(cheerio_element) 来获取 outer html */\nexport async function parse_html (html: string) {\n const { default: cheerio } = await import('cheerio')\n \n let $ = cheerio.load(html, { decodeEntities: false })\n \n Object.defineProperty($, inspect.custom, {\n configurable: true,\n enumerable: false,\n value () {\n return this.html()\n }\n })\n \n Object.defineProperty($.prototype, inspect.custom, {\n configurable: true,\n enumerable: false,\n value (this: cheerio.Cheerio) {\n if (this.length > 1)\n return this.map((index, element) => {\n if (typeof element === 'string') return element\n return $.html(element)\n }).get().join_lines()\n \n return this.toString()\n }\n })\n \n return $\n}\n\n\n/** 使用 $.html(cheerio_element) 来获取 outer html */\nexport async function request_page (url: string | URL, options?: RequestOptions) {\n return parse_html(\n await request(url, options)\n )\n}\n\n\nexport function to_curl (url: string | URL, { queries, headers, method, body, proxy, exe = true }: RequestOptions & { exe?: boolean } = { }) {\n if (proxy === true)\n proxy = process.env.http_proxy\n \n url = url.toString()\n \n if (!url.startsWith('http'))\n url = `http://${url}`\n \n return (exe ? 'curl.exe' : 'curl') + \n ' ' + ( url + (queries ? '?' : '') + qs.stringify(queries) ).quote() +\n // ( typeof proxy === 'undefined' ?\n // ''\n // :\n // ( proxy ? ' --proxy ' + proxy.quote() : ' --noproxy ' + '*'.quote())\n // ) +\n ( proxy ? ` --proxy ${proxy.quote()}` : '' ) +\n ( method && method !== 'GET' ? ` -X ${method}` : '' ) +\n ( headers ? Object.entries(headers).map( ([key, value]) => ' -H ' + `${key}: ${value}`.quote() ).join('') : '' ) +\n ( body ? ' -H ' + 'content-type: application/json'.quote() : '') +\n ( body ? ' --data ' + JSON.stringify(body).quote() : '')\n}\n\n\n\n\n// ------------------------------------ rpc client\n/** post json to http://localhost:8421/api/rpc\n - func: function name\n - args?: argument array\n - options?:\n - ignore?: `false` wait for execution but do not serialize result to response\n - async?: `false` do not wait for exec\n*/\nexport async function rpc (\n func: string, \n args?: any[], \n { url = 'http://localhost:8421/api/rpc', async: _async = false, ignore = false }: { url?: string, async?: boolean, ignore?: boolean } = { }\n) {\n if (!func)\n throw new Error('rpc argument error: no func')\n \n return request_json(url, {\n body: {\n func,\n args,\n async: _async,\n ignore,\n }\n })\n}\n\n\n\nlet decoder = new TextDecoder()\n\nlet encoder = new TextEncoder()\n\n\nexport { WebSocket }\n\n\nexport class WebSocketConnectionError extends Error {\n override name = 'WebSocketConnectionError'\n \n websocket: WebSocket\n \n event: CloseEvent | ErrorEvent\n \n type: 'close' | 'error'\n \n address?: string\n \n errno?: number\n \n port?: number\n \n syscall?: string\n \n /** close 事件时为 close code, error 事件为 error code */\n code?: string | number\n \n reason?: string\n \n \n constructor (websocket: WebSocket, event: CloseEvent | ErrorEvent, message: string = '') {\n super(`${websocket.url} ${t('连接出错了')}. ${message}`)\n \n this.websocket = websocket\n this.event = event\n this.type = event.type as 'close' | 'error'\n \n if (this.type === 'error') {\n const { error } = event as ErrorEvent\n this.address = error.address\n this.code = error.code\n this.errno = error.errno\n this.port = error.port\n this.syscall = error.syscall\n this.reason = error.reason\n \n this.stack = `${this.name}: ${this.message}\\n` +\n error.stack.slice(error.stack.indexOf('\\n') + 1) + '\\n'\n } else {\n this.code = (event as CloseEvent).code\n this.reason = (event as CloseEvent).reason\n }\n }\n}\n\n\n/** 连接 websocket url, 设置各种事件监听器。在 open 事件后 resolve, 返回 websocket \n 遇到 error 时会创建 WebSocketConnectionError: \n - reject 掉返回的 promise (若此时未 settle) \n - 作为参数调用 on_error (已 settle 且有 on_error 回调) \n 可以用 WebSocket.bufferedAmount 来显示大消息的发送进度 \n https://developer.mozilla.org/en-US/docs/Web/API/WebSocket/bufferedAmount\n - url\n - options:\n - protocols?\n - max_payload?: `8 GB`\n - on_message: 根据 websocket frame 的 opcode 不同 (text frame 或 binary frame),event 中的 data 对应为 ArrayBuffer 或者 string \n https://datatracker.ietf.org/doc/html/rfc6455#section-5.2\n - on_error?: 在 websocket 出错和非正常关闭 (close, error 事件) 时都调用,可以根据 error.type 来区分,error 的类型是 WebSocketConnectionError, \n type 为 'close' 时有 code 和 reason 属性 \n - on_close?: 和 websocket 的 'close' 事件不相同,只在正常关闭 (close code 为 1000) 时才调用,否则都会调用 on_error \n https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent#Status_codes */\nexport async function connect_websocket (\n url: string | URL,\n {\n protocols,\n max_payload = 2 ** 33, // 8 GB\n on_message,\n on_error,\n on_close\n }: {\n protocols?: string | string[]\n max_payload?: number\n on_message (data: ArrayBuffer | string, websocket: WebSocket): any\n on_error? (error: WebSocketConnectionError, websocket: WebSocket): any\n on_close? (event: CloseEvent, websocket: WebSocket): any\n }\n) {\n let websocket = new WebSocket(url, protocols, {\n maxPayload: max_payload,\n skipUTF8Validation: true\n })\n \n // https://stackoverflow.com/questions/11821096/what-is-the-difference-between-an-arraybuffer-and-a-blob/39951543\n websocket.binaryType = 'arraybuffer'\n \n return new Promise<WebSocket>((resolve, reject) => {\n let settled = false\n \n websocket.addEventListener('open', event => {\n console.log(\n websocket.url +\n (protocols ? \n ' ' + (\n typeof protocols === 'string' ? protocols : protocols.join(', ').bracket('square')\n ).bracket()\n :\n '') +\n t(' 已连接')\n )\n \n settled = true\n resolve(websocket)\n })\n \n websocket.addEventListener('close', event => {\n // https://blog.insiderattack.net/promises-next-ticks-and-immediates-nodejs-event-loop-part-3-9226cbe7a6aa\n // error 事件会先发生,然后 reject(error) 被执行,此时还未轮到外层的 await connect_websocket 执行(在微任务队列中),\n // 接着马上 close 事件也被调用,此时 settled,马上调用了 on_error 函数,弄乱了顺序\n // error 的错误信息比较多,而且通过 await 得到的栈也比较清晰,这里延后调用 on_close 和 on_error,放到微任务队列之后的 timers 队列中\n setTimeout(() => {\n if (event.code === 1000) // 正常关闭\n if (on_close)\n on_close(event, websocket)\n else\n console.log(`${websocket.url} ${t('已正常关闭')}`)\n else { // 异常关闭,认为发生了错误,进行错误处理\n assert(settled, t('websocket close 事件时应该已经 settled'))\n \n const error = new WebSocketConnectionError(websocket, event, `${t('连接被关闭')}, code: ${event.code}${ event.reason ? `, ${t('原因')}: ${event.reason}` : '' }`)\n \n if (on_error)\n on_error(error, websocket)\n else // 既然用户不传 on_error, 就当 unhandled error 抛出来\n throw error\n }\n })\n })\n \n websocket.addEventListener('error', event => {\n const error = new WebSocketConnectionError(websocket, event, event.error?.message)\n \n if (settled) {\n if (on_error)\n on_error(error, websocket)\n else // 既然用户不传 on_error, 就当 unhandled error 抛出来\n throw error\n } else {\n settled = true\n reject(error)\n }\n })\n \n websocket.addEventListener('message', event => {\n on_message(event.data as ArrayBuffer | string, websocket)\n })\n })\n}\n\n\n/** 接收到消息后的处理函数 \n 返回值可以是:\n - 数组: 会自动被封装为 { id: 相同, data: 返回值, done: true } 这样的消息并调用 websocket.send 将其发送\n - void: 什么都不做\n - 以上的 promise */\nexport type MessageHandler = (message: Message, websocket?: WebSocket) => void | any[] | Promise<void | any[]>\n\n\n/** 二进制消息格式 \n - json.length (小端序): 4 字节\n - json 数据\n - binary 数据 */\nexport interface Message <TData extends any[] = any[]> {\n /** rpc id: 在 rpc 系统中认为是唯一的。用来在单个 websocket 连接上复用多个 rpc 请求。多个相同 id 的 message 组成一个请求流 */\n id?: number\n \n /** 只在 rpc 发起时指定被调用的 function name,发起时 rpc 时必传 */\n func?: string\n \n /** 通过这个 flag 主动表明这是发往对方的最后一个 message, 对方可以销毁 handler 了 \n 并非强制,可以不说明,由双方的函数自己约定\n */\n done?: boolean\n \n /** 通知对方这里产生的错误,本质上类似 data 也是一种数据,并不代表 rpc 的结束,后续可能继续有 rpc message 交换 */\n error?: Error\n \n /** data 是一个数组, 作为:\n - rpc 发起方调用 func 的参数,或者请求流 message 携带的数据\n - 结果或者响应流的数据,传给请求发起方\n \n 里面是可序列化为 json 的 js 变量,或者是 Uint8Array \n Uint8Array 的参数处理后被替换为 Uint8Array.byteLength, 并将下标记录在 bins 中\n \n 注意: 数组中如果有 undefined 值,传输到对面会变成 null 值 */\n data?: TData\n \n /** bins: data 中哪些下标对应的原始值是 Uint8Array 类型的,如: [0, 3] */\n bins?: number[]\n}\n\n/** 通过创建 remote 对象对 websocket rpc 进行抽象 \n 调用方使用 remote.call 进行调用 \n 被调方在创建 remote 对象时传入 funcs 注册处理函数,并使用 remote.handle 方法处理 websocket message \n 未连接时自动连接,断开后自动重连 (保证执行 send 时已连接,否则报错) \n 从设计上与 rpc 状态与底层连接的状态无关 (通过传入 url 创建的),底层连接断开后,只要检测到断线,会尝试重连,如果重连成功,则不影响调用的状态 \n @example\n // Zero 继承自 Remote 并通过 call 实现了一些方法\n let zero = new Zero({ local: true })\n \n // 一元 rpc\n await zero.repl_ts('1234')\n \n // 订阅流\n const id = genid()\n \n zero.handlers.set(id, ({ data: [chunk] }: Message<[Uint8Array]>) => {\n term.write(chunk)\n })\n \n zero.send({ id, func: 'subscribe_stdio' }) */\nexport class Remote {\n /** 是否为建立 rpc 连接的发起方 */\n initiator: boolean\n \n /** websocket url */\n url?: string\n \n /** 主动发起 websocket 连接的客户端 (构造函数传 url) 有这个属性 */\n lwebsocket?: Lock<WebSocket>\n \n /** 通过 rpc message.func 被调用的 rpc 函数 */\n funcs: Record<string, MessageHandler>\n \n /** map<id, message handler>: 通过 rpc message.id 找到对应的 handler, unary rpc 接收方不需要设置 handlers, 发送方需要 */\n handlers = new Map<number, MessageHandler>()\n \n \n print = false\n \n pconnect: Promise<any>\n \n \n /** 使用 Uint8Array 作为参数更灵活 https://stackoverflow.com/a/74505197/7609214 */\n static parse <TData extends any[] = any[]> (buffer: Uint8Array) {\n const dv = new DataView(buffer.buffer, buffer.byteOffset, buffer.byteLength)\n \n const len_json = dv.getUint32(0, true)\n \n let offset = 4 + len_json\n \n let message: Message<TData> = JSON.parse(decoder.decode(buffer.subarray(4, offset)))\n \n if (message.bins) {\n const { bins } = message\n let { data } = message\n \n for (const ibin of bins) {\n const len_buf = data[ibin]\n data[ibin] = buffer.subarray(offset, offset + len_buf)\n offset += len_buf\n }\n }\n \n if (message.error)\n message.error = Object.assign(new Error(), message.error)\n \n return message\n }\n \n \n static pack ({ id, func, data = [ ], done, error }: Message) {\n let data_ = new Array(data.length)\n let bins: number[] = [ ]\n let bufs: Uint8Array[] = [ ]\n \n for (let i = 0; i < data.length; i++) {\n const item = data[i]\n if (item instanceof Uint8Array) {\n bins.push(i)\n bufs.push(item)\n data_[i] = item.length\n } else\n data_[i] = item\n }\n \n const data_json = {\n id,\n ... func ? { func } : { },\n ... done ? { done } : { },\n ... error ? { error } : { },\n ... data_.length ? { data: data_ } : { },\n ... bins.length ? { bins } : { },\n }\n \n const str_json = encoder.encode(JSON.stringify(data_json))\n \n let dv = new DataView(new ArrayBuffer(4))\n \n dv.setUint32(0, str_json.length, true)\n \n return concat([dv, str_json, ... bufs])\n }\n \n \n /** 作为 rpc 发起方,可以通过传入 url 或者 websocket 定义远程 Remote \n 作为 rpc 接收方,可以不传 url 和 websocket 定义本地 Remote */\n constructor ({\n url,\n funcs = { },\n websocket,\n on_error\n }: {\n url?: string\n funcs?: Remote['funcs']\n websocket?: WebSocket\n on_error? (error: WebSocketConnectionError, websocket: WebSocket): void\n } = { }) {\n assert(!(url && websocket), t('构建 Remote 时 url 和 websocket 最多只能传一个'))\n this.initiator = Boolean(url || websocket)\n this.url = url\n this.funcs = funcs\n if (on_error)\n this.on_error = on_error\n this.lwebsocket = new Lock(websocket || null)\n }\n \n \n on_error? (error: WebSocketConnectionError, websocket: WebSocket) {\n throw error\n }\n \n \n /** 幂等,保证 websocket 已连接,否则抛出异常,不需要手动调用,在其它方法中已默认自动重连 \n 作为接收方需要传入使用的 websocket 连接,确保这个这个连接的状态 */\n async connect (websocket?: WebSocket) {\n if (this.initiator) {\n if (this.lwebsocket.resource?.readyState === WebSocket.OPEN)\n return\n else if (!this.url)\n throw new Error(t('创建 Remote 时传入的 websocket 连接已断开'))\n // else 等待重连\n } else\n if (websocket.readyState === WebSocket.OPEN)\n return\n else\n throw new Error(t('传入的 websocket 连接已断开'))\n \n // 假设有多个请求想要并发连接 websocket, 且此时 websocket 是断开的状态\n // 应该排队依次连接,而不是后续的连接直接使用第一次连接的 promise,后续调用还是应该尝试重连(不止连接一次)\n \n return this.lwebsocket.request(async (websocket) => {\n // 保存的 rpc 状态在 this.handlers, 与 websocket 无关,因此即使断开重连也不影响 rpc 的运行,即\n // 底层连接断开后自动重连对上层应该是无感知的,除非再次连接时失败\n \n if (websocket?.readyState === WebSocket.OPEN)\n return\n else if (!this.url)\n throw new Error(t('创建 Remote 时传入的 websocket 连接已断开'))\n else // 重连\n this.lwebsocket.resource = await connect_websocket(this.url, {\n on_message: (data: ArrayBuffer, websocket) => {\n this.handle(new Uint8Array(data), websocket)\n },\n on_error: this.on_error.bind(this)\n })\n })\n }\n \n \n disconnect () {\n this.lwebsocket.resource?.close(1000)\n }\n \n \n /** 接收 websocket 连接的本地 remote 必传 websocket 参数;发起端选传,如果传了必须等于 this.websocket_lock.resource \n 发送或连接出错时自动清理 message.id 对应的 handler */\n async send (message: Message, websocket?: WebSocket) {\n try {\n if (this.initiator) {\n assert(!websocket || websocket === this.lwebsocket.resource)\n await this.connect()\n websocket = this.lwebsocket.resource\n } else\n await this.connect(websocket)\n \n if (!message.id)\n message.id = genid()\n \n // 不需要独占 websocket\n websocket.send(Remote.pack(message))\n } catch (error) {\n if (message.id)\n this.handlers.delete(message.id)\n throw error\n }\n }\n \n \n /** 处理接收到的 websocket message 并解析, 根据 id dispatch 到对应的 handler 进行处理 \n 如果 message.done == true 则清理 handler \n 如果 handler 返回了值,则包装为 message 发送 \n 使用 Uint8Array 作为参数更灵活 https://stackoverflow.com/a/74505197/7609214 */\n async handle (data: Uint8Array, websocket: WebSocket) {\n const message = Remote.parse(data)\n \n const { id, func, done } = message\n \n if (this.print)\n console.log(message)\n \n let handler: MessageHandler\n \n if (func)\n handler = this.funcs[func]\n else {\n handler = this.handlers.get(id)\n if (done)\n this.handlers.delete(id)\n }\n \n try {\n if (handler) {\n const data = await handler(message, websocket)\n if (data)\n await this.send({ id, data }, websocket)\n } else if (message.error)\n throw message.error\n else\n throw new Error(`${t('找不到 rpc handler')}: ${func ? `func: ${func.quote()}` : `id: ${id}`}`)\n } catch (error) {\n // handle 出错并不意味着 rpc 一定会结束,可能 error 是运行中的正常数据,所以不能清理 handler\n \n if (\n websocket.readyState === WebSocket.OPEN &&\n !message.error // 防止无限循环往对方发送 error, 只有在对方无错误时才可以发送\n )\n try { await this.send({ id, error, /* 不能设置 done 清理对面 handler, 理由同上 */ }, websocket) } catch { }\n \n // 再往上层抛出错误没有意义了,上层调用栈是 websocket.on('message') 之类的\n console.log(error)\n }\n }\n \n \n /** 调用 remote 中的 func, 只适用于最简单的一元 rpc (请求, 响应) */\n async call <TReturn extends any[] = any[]> (func: string, args?: any[]) {\n return new Promise<TReturn>(async (resolve, reject) => {\n const id = genid()\n \n this.handlers.set(id, (message: Message<TReturn>) => {\n const { error, data } = message\n if (error)\n reject(error)\n else\n resolve(data)\n this.handlers.delete(id)\n })\n \n try {\n await this.send({ id, func, data: args }) // 不需要 done: true, 因为对面的 remote.handlers 中不会有这个 id 的 handler\n } catch (error) {\n reject(error)\n }\n })\n }\n}\n\n"]}