@i.un/api-client 1.0.8 → 1.1.0
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/dist/index.d.mts +61 -92
- package/dist/index.d.ts +61 -92
- package/dist/index.js +249 -86
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +243 -82
- package/dist/index.mjs.map +1 -1
- package/package.json +4 -2
package/dist/index.js
CHANGED
|
@@ -32,14 +32,16 @@ var index_exports = {};
|
|
|
32
32
|
__export(index_exports, {
|
|
33
33
|
createApiClient: () => createApiClient,
|
|
34
34
|
createProtobufHooks: () => createProtobufHooks,
|
|
35
|
-
|
|
36
|
-
|
|
35
|
+
decodeSecure: () => decodeSecure,
|
|
36
|
+
encodeSecure: () => encodeSecure,
|
|
37
37
|
executeRequestChain: () => executeRequestChain,
|
|
38
|
+
getObfuscationKey: () => getObfuscationKey,
|
|
38
39
|
isApiError: () => isApiError,
|
|
39
40
|
isProtobufContentType: () => isProtobufContentType,
|
|
40
41
|
normalizeProtobufConfig: () => normalizeProtobufConfig,
|
|
41
|
-
|
|
42
|
-
|
|
42
|
+
parseSecureRequest: () => parseSecureRequest,
|
|
43
|
+
secureResponse: () => secureResponse,
|
|
44
|
+
xorTransform: () => xorTransform
|
|
43
45
|
});
|
|
44
46
|
module.exports = __toCommonJS(index_exports);
|
|
45
47
|
|
|
@@ -358,55 +360,237 @@ async function executeRequestChain(requests, handlers = [], getChainRequests, in
|
|
|
358
360
|
return lastResult;
|
|
359
361
|
}
|
|
360
362
|
|
|
361
|
-
// src/
|
|
362
|
-
var
|
|
363
|
-
var
|
|
364
|
-
|
|
363
|
+
// src/secure.js
|
|
364
|
+
var $protobuf = __toESM(require("protobufjs/minimal"));
|
|
365
|
+
var $Reader = $protobuf.Reader;
|
|
366
|
+
var $Writer = $protobuf.Writer;
|
|
367
|
+
var $util = $protobuf.util;
|
|
368
|
+
var $root = $protobuf.roots["default"] || ($protobuf.roots["default"] = {});
|
|
369
|
+
var secure = $root.secure = (() => {
|
|
370
|
+
const secure2 = {};
|
|
371
|
+
secure2.SecurePayload = (function() {
|
|
372
|
+
function SecurePayload(properties) {
|
|
373
|
+
if (properties) {
|
|
374
|
+
for (let keys = Object.keys(properties), i = 0; i < keys.length; ++i)
|
|
375
|
+
if (properties[keys[i]] != null)
|
|
376
|
+
this[keys[i]] = properties[keys[i]];
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
SecurePayload.prototype.ts = $util.Long ? $util.Long.fromBits(0, 0, false) : 0;
|
|
380
|
+
SecurePayload.prototype.data = $util.newBuffer([]);
|
|
381
|
+
SecurePayload.create = function create(properties) {
|
|
382
|
+
return new SecurePayload(properties);
|
|
383
|
+
};
|
|
384
|
+
SecurePayload.encode = function encode(message, writer) {
|
|
385
|
+
if (!writer)
|
|
386
|
+
writer = $Writer.create();
|
|
387
|
+
if (message.ts != null && Object.hasOwnProperty.call(message, "ts"))
|
|
388
|
+
writer.uint32(
|
|
389
|
+
/* id 1, wireType 0 =*/
|
|
390
|
+
8
|
|
391
|
+
).int64(message.ts);
|
|
392
|
+
if (message.data != null && Object.hasOwnProperty.call(message, "data"))
|
|
393
|
+
writer.uint32(
|
|
394
|
+
/* id 2, wireType 2 =*/
|
|
395
|
+
18
|
|
396
|
+
).bytes(message.data);
|
|
397
|
+
return writer;
|
|
398
|
+
};
|
|
399
|
+
SecurePayload.encodeDelimited = function encodeDelimited(message, writer) {
|
|
400
|
+
return this.encode(message, writer).ldelim();
|
|
401
|
+
};
|
|
402
|
+
SecurePayload.decode = function decode(reader, length, error) {
|
|
403
|
+
if (!(reader instanceof $Reader))
|
|
404
|
+
reader = $Reader.create(reader);
|
|
405
|
+
let end = length === void 0 ? reader.len : reader.pos + length, message = new $root.secure.SecurePayload();
|
|
406
|
+
while (reader.pos < end) {
|
|
407
|
+
let tag = reader.uint32();
|
|
408
|
+
if (tag === error)
|
|
409
|
+
break;
|
|
410
|
+
switch (tag >>> 3) {
|
|
411
|
+
case 1: {
|
|
412
|
+
message.ts = reader.int64();
|
|
413
|
+
break;
|
|
414
|
+
}
|
|
415
|
+
case 2: {
|
|
416
|
+
message.data = reader.bytes();
|
|
417
|
+
break;
|
|
418
|
+
}
|
|
419
|
+
default:
|
|
420
|
+
reader.skipType(tag & 7);
|
|
421
|
+
break;
|
|
422
|
+
}
|
|
423
|
+
}
|
|
424
|
+
return message;
|
|
425
|
+
};
|
|
426
|
+
SecurePayload.decodeDelimited = function decodeDelimited(reader) {
|
|
427
|
+
if (!(reader instanceof $Reader))
|
|
428
|
+
reader = new $Reader(reader);
|
|
429
|
+
return this.decode(reader, reader.uint32());
|
|
430
|
+
};
|
|
431
|
+
SecurePayload.verify = function verify(message) {
|
|
432
|
+
if (typeof message !== "object" || message === null)
|
|
433
|
+
return "object expected";
|
|
434
|
+
if (message.ts != null && message.hasOwnProperty("ts")) {
|
|
435
|
+
if (!$util.isInteger(message.ts) && !(message.ts && $util.isInteger(message.ts.low) && $util.isInteger(message.ts.high)))
|
|
436
|
+
return "ts: integer|Long expected";
|
|
437
|
+
}
|
|
438
|
+
if (message.data != null && message.hasOwnProperty("data")) {
|
|
439
|
+
if (!(message.data && typeof message.data.length === "number" || $util.isString(message.data)))
|
|
440
|
+
return "data: buffer expected";
|
|
441
|
+
}
|
|
442
|
+
return null;
|
|
443
|
+
};
|
|
444
|
+
SecurePayload.fromObject = function fromObject(object) {
|
|
445
|
+
if (object instanceof $root.secure.SecurePayload)
|
|
446
|
+
return object;
|
|
447
|
+
let message = new $root.secure.SecurePayload();
|
|
448
|
+
if (object.ts != null) {
|
|
449
|
+
if ($util.Long)
|
|
450
|
+
(message.ts = $util.Long.fromValue(object.ts)).unsigned = false;
|
|
451
|
+
else if (typeof object.ts === "string")
|
|
452
|
+
message.ts = parseInt(object.ts, 10);
|
|
453
|
+
else if (typeof object.ts === "number")
|
|
454
|
+
message.ts = object.ts;
|
|
455
|
+
else if (typeof object.ts === "object")
|
|
456
|
+
message.ts = new $util.LongBits(object.ts.low >>> 0, object.ts.high >>> 0).toNumber();
|
|
457
|
+
}
|
|
458
|
+
if (object.data != null) {
|
|
459
|
+
if (typeof object.data === "string")
|
|
460
|
+
$util.base64.decode(object.data, message.data = $util.newBuffer($util.base64.length(object.data)), 0);
|
|
461
|
+
else if (object.data.length >= 0)
|
|
462
|
+
message.data = object.data;
|
|
463
|
+
}
|
|
464
|
+
return message;
|
|
465
|
+
};
|
|
466
|
+
SecurePayload.toObject = function toObject(message, options) {
|
|
467
|
+
if (!options)
|
|
468
|
+
options = {};
|
|
469
|
+
let object = {};
|
|
470
|
+
if (options.defaults) {
|
|
471
|
+
if ($util.Long) {
|
|
472
|
+
let long = new $util.Long(0, 0, false);
|
|
473
|
+
object.ts = options.longs === String ? long.toString() : options.longs === Number ? long.toNumber() : long;
|
|
474
|
+
} else
|
|
475
|
+
object.ts = options.longs === String ? "0" : 0;
|
|
476
|
+
if (options.bytes === String)
|
|
477
|
+
object.data = "";
|
|
478
|
+
else {
|
|
479
|
+
object.data = [];
|
|
480
|
+
if (options.bytes !== Array)
|
|
481
|
+
object.data = $util.newBuffer(object.data);
|
|
482
|
+
}
|
|
483
|
+
}
|
|
484
|
+
if (message.ts != null && message.hasOwnProperty("ts"))
|
|
485
|
+
if (typeof message.ts === "number")
|
|
486
|
+
object.ts = options.longs === String ? String(message.ts) : message.ts;
|
|
487
|
+
else
|
|
488
|
+
object.ts = options.longs === String ? $util.Long.prototype.toString.call(message.ts) : options.longs === Number ? new $util.LongBits(message.ts.low >>> 0, message.ts.high >>> 0).toNumber() : message.ts;
|
|
489
|
+
if (message.data != null && message.hasOwnProperty("data"))
|
|
490
|
+
object.data = options.bytes === String ? $util.base64.encode(message.data, 0, message.data.length) : options.bytes === Array ? Array.prototype.slice.call(message.data) : message.data;
|
|
491
|
+
return object;
|
|
492
|
+
};
|
|
493
|
+
SecurePayload.prototype.toJSON = function toJSON() {
|
|
494
|
+
return this.constructor.toObject(this, $protobuf.util.toJSONOptions);
|
|
495
|
+
};
|
|
496
|
+
SecurePayload.getTypeUrl = function getTypeUrl(typeUrlPrefix) {
|
|
497
|
+
if (typeUrlPrefix === void 0) {
|
|
498
|
+
typeUrlPrefix = "type.googleapis.com";
|
|
499
|
+
}
|
|
500
|
+
return typeUrlPrefix + "/secure.SecurePayload";
|
|
501
|
+
};
|
|
502
|
+
return SecurePayload;
|
|
503
|
+
})();
|
|
504
|
+
return secure2;
|
|
505
|
+
})();
|
|
365
506
|
|
|
366
|
-
|
|
367
|
-
string salt = 1;
|
|
368
|
-
int64 timestamp = 2;
|
|
369
|
-
bytes data = 3;
|
|
370
|
-
}
|
|
371
|
-
`;
|
|
507
|
+
// src/protobuf.ts
|
|
372
508
|
var encoder = new TextEncoder();
|
|
373
509
|
var decoder = new TextDecoder();
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
if (
|
|
377
|
-
|
|
378
|
-
|
|
510
|
+
function xorTransform(data, key) {
|
|
511
|
+
const keyBytes = typeof key === "string" ? encoder.encode(key) : key;
|
|
512
|
+
if (keyBytes.length === 0) return data;
|
|
513
|
+
for (let i = 0; i < data.length; i++) {
|
|
514
|
+
data[i] ^= keyBytes[i % keyBytes.length];
|
|
379
515
|
}
|
|
380
|
-
return
|
|
516
|
+
return data;
|
|
381
517
|
}
|
|
382
|
-
function
|
|
383
|
-
|
|
384
|
-
|
|
518
|
+
async function getObfuscationKey(keyOption) {
|
|
519
|
+
const defaultKey = "api-client-default-key";
|
|
520
|
+
if (!keyOption) return defaultKey;
|
|
521
|
+
if (typeof keyOption === "function") {
|
|
522
|
+
return await keyOption();
|
|
385
523
|
}
|
|
386
|
-
return
|
|
524
|
+
return keyOption;
|
|
387
525
|
}
|
|
388
|
-
function
|
|
389
|
-
|
|
390
|
-
const type = getSaltedPayloadType();
|
|
391
|
-
const jsonString = JSON.stringify(data);
|
|
392
|
-
const dataBytes = encoder.encode(jsonString);
|
|
393
|
-
const payload = {
|
|
394
|
-
salt: useSalt ? saltGenerator() : "",
|
|
395
|
-
timestamp: Date.now(),
|
|
396
|
-
data: dataBytes
|
|
397
|
-
};
|
|
398
|
-
const message = type.create(payload);
|
|
399
|
-
return type.encode(message).finish();
|
|
526
|
+
function getInternalSecureType() {
|
|
527
|
+
return secure.SecurePayload;
|
|
400
528
|
}
|
|
401
|
-
function
|
|
402
|
-
const
|
|
403
|
-
|
|
529
|
+
async function encodeSecure(data, options = {}) {
|
|
530
|
+
const {
|
|
531
|
+
obfuscate = true,
|
|
532
|
+
obfuscationKey,
|
|
533
|
+
protoType,
|
|
534
|
+
transform,
|
|
535
|
+
encode: customEncode
|
|
536
|
+
} = options;
|
|
537
|
+
let buffer;
|
|
538
|
+
const processedData = transform?.beforeEncode ? transform.beforeEncode(data) : data;
|
|
539
|
+
if (customEncode) {
|
|
540
|
+
buffer = await customEncode(data);
|
|
541
|
+
} else {
|
|
542
|
+
const type = protoType || getInternalSecureType();
|
|
543
|
+
const payload = protoType ? processedData : {
|
|
544
|
+
// 默认容器模式:将预处理后的数据封装进信封
|
|
545
|
+
ts: Date.now(),
|
|
546
|
+
data: processedData instanceof Uint8Array ? processedData : encoder.encode(JSON.stringify(processedData))
|
|
547
|
+
};
|
|
548
|
+
const message = type.create(payload);
|
|
549
|
+
buffer = type.encode(message).finish();
|
|
550
|
+
}
|
|
551
|
+
if (!obfuscate) return buffer;
|
|
552
|
+
const finalKey = await getObfuscationKey(obfuscationKey);
|
|
553
|
+
return xorTransform(buffer, finalKey);
|
|
554
|
+
}
|
|
555
|
+
async function decodeSecure(buffer, options = {}) {
|
|
556
|
+
const {
|
|
557
|
+
obfuscate = true,
|
|
558
|
+
obfuscationKey,
|
|
559
|
+
protoType,
|
|
560
|
+
transform,
|
|
561
|
+
decode: customDecode
|
|
562
|
+
} = options;
|
|
563
|
+
let uint8 = buffer instanceof ArrayBuffer ? new Uint8Array(buffer) : new Uint8Array(buffer.buffer, buffer.byteOffset, buffer.byteLength);
|
|
564
|
+
if (uint8.length > 0 && obfuscate) {
|
|
565
|
+
const finalKey = await getObfuscationKey(obfuscationKey);
|
|
566
|
+
uint8 = xorTransform(uint8, finalKey);
|
|
567
|
+
}
|
|
568
|
+
if (customDecode) {
|
|
569
|
+
return await customDecode(uint8);
|
|
570
|
+
}
|
|
571
|
+
const type = protoType || getInternalSecureType();
|
|
404
572
|
const decoded = type.decode(uint8);
|
|
405
|
-
const
|
|
406
|
-
|
|
573
|
+
const plainObj = type.toObject(decoded, {
|
|
574
|
+
longs: String,
|
|
575
|
+
enums: String,
|
|
576
|
+
bytes: Uint8Array,
|
|
577
|
+
defaults: true
|
|
578
|
+
});
|
|
579
|
+
if (transform?.afterDecode) {
|
|
580
|
+
return transform.afterDecode(plainObj);
|
|
581
|
+
}
|
|
582
|
+
if (!protoType && plainObj.data) {
|
|
583
|
+
const jsonString = decoder.decode(plainObj.data);
|
|
584
|
+
try {
|
|
585
|
+
return JSON.parse(jsonString);
|
|
586
|
+
} catch {
|
|
587
|
+
return jsonString;
|
|
588
|
+
}
|
|
589
|
+
}
|
|
590
|
+
return plainObj;
|
|
407
591
|
}
|
|
408
|
-
function
|
|
409
|
-
const buffer =
|
|
592
|
+
async function secureResponse(data, options) {
|
|
593
|
+
const buffer = await encodeSecure(data, options);
|
|
410
594
|
const arrayBuffer = buffer.buffer.slice(
|
|
411
595
|
buffer.byteOffset,
|
|
412
596
|
buffer.byteOffset + buffer.byteLength
|
|
@@ -418,65 +602,42 @@ function protobufResponse(data, options) {
|
|
|
418
602
|
}
|
|
419
603
|
});
|
|
420
604
|
}
|
|
605
|
+
async function parseSecureRequest(request, customDecode) {
|
|
606
|
+
const buffer = await request.arrayBuffer();
|
|
607
|
+
return decodeSecure(buffer, { decode: customDecode });
|
|
608
|
+
}
|
|
421
609
|
function isProtobufContentType(contentType) {
|
|
422
610
|
return !!contentType?.includes("application/x-protobuf");
|
|
423
611
|
}
|
|
424
|
-
function normalizeProtobufConfig(config) {
|
|
425
|
-
if (!config)
|
|
426
|
-
return null;
|
|
427
|
-
}
|
|
612
|
+
function normalizeProtobufConfig(config, options) {
|
|
613
|
+
if (!config) return null;
|
|
428
614
|
if (config === true) {
|
|
429
615
|
return {
|
|
430
|
-
encode: (data) =>
|
|
431
|
-
decode: (buffer) =>
|
|
616
|
+
encode: (data) => encodeSecure(data, options),
|
|
617
|
+
decode: (buffer) => decodeSecure(buffer, options)
|
|
432
618
|
};
|
|
433
619
|
}
|
|
434
|
-
return
|
|
435
|
-
encode: config.encode || ((data) => encodeProtobuf(data, config.options)),
|
|
436
|
-
decode: config.decode || ((buffer) => decodeProtobuf(buffer))
|
|
437
|
-
};
|
|
438
|
-
}
|
|
439
|
-
async function parseProtobufRequest(request, customDecode) {
|
|
440
|
-
const buffer = await request.arrayBuffer();
|
|
441
|
-
const uint8 = new Uint8Array(buffer);
|
|
442
|
-
if (customDecode) {
|
|
443
|
-
return customDecode(uint8);
|
|
444
|
-
}
|
|
445
|
-
return decodeProtobuf(uint8);
|
|
620
|
+
return config;
|
|
446
621
|
}
|
|
447
622
|
function createProtobufHooks(options = {}) {
|
|
448
|
-
const encode =
|
|
449
|
-
const decode =
|
|
623
|
+
const encode = (data) => encodeSecure(data, options);
|
|
624
|
+
const decode = (buffer) => decodeSecure(buffer, options);
|
|
450
625
|
return {
|
|
451
|
-
/**
|
|
452
|
-
* 请求钩子:独立处理请求编码和响应类型设置
|
|
453
|
-
*/
|
|
454
626
|
async onRequest(context) {
|
|
455
627
|
const { options: reqOptions } = context;
|
|
456
628
|
const headers = reqOptions.headers instanceof Headers ? reqOptions.headers : new Headers(reqOptions.headers);
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
if (isProtobufRequest && reqOptions.body && !(reqOptions.body instanceof Uint8Array)) {
|
|
460
|
-
const encoded = await encode(reqOptions.body);
|
|
461
|
-
reqOptions.body = encoded;
|
|
629
|
+
if (isProtobufContentType(headers.get("Content-Type")) && reqOptions.body && !(reqOptions.body instanceof Uint8Array)) {
|
|
630
|
+
reqOptions.body = await encode(reqOptions.body);
|
|
462
631
|
}
|
|
463
|
-
|
|
464
|
-
const prefersProtobufResponse = isProtobufContentType(accept);
|
|
465
|
-
if (prefersProtobufResponse && !reqOptions.responseType) {
|
|
632
|
+
if (isProtobufContentType(headers.get("Accept")) && !reqOptions.responseType) {
|
|
466
633
|
reqOptions.responseType = "arrayBuffer";
|
|
467
634
|
}
|
|
468
635
|
reqOptions.headers = headers;
|
|
469
636
|
},
|
|
470
|
-
/**
|
|
471
|
-
* 响应钩子:解码 Protobuf 响应
|
|
472
|
-
*/
|
|
473
637
|
async onResponse(context) {
|
|
474
638
|
const { response } = context;
|
|
475
|
-
if (!response?._data)
|
|
476
|
-
|
|
477
|
-
}
|
|
478
|
-
const contentType = response.headers.get("Content-Type");
|
|
479
|
-
if (isProtobufContentType(contentType)) {
|
|
639
|
+
if (!response?._data) return;
|
|
640
|
+
if (isProtobufContentType(response.headers.get("Content-Type"))) {
|
|
480
641
|
const buffer = response._data;
|
|
481
642
|
if (buffer && buffer.byteLength > 0) {
|
|
482
643
|
response._data = await decode(new Uint8Array(buffer));
|
|
@@ -496,13 +657,15 @@ function createProtobufHooks(options = {}) {
|
|
|
496
657
|
0 && (module.exports = {
|
|
497
658
|
createApiClient,
|
|
498
659
|
createProtobufHooks,
|
|
499
|
-
|
|
500
|
-
|
|
660
|
+
decodeSecure,
|
|
661
|
+
encodeSecure,
|
|
501
662
|
executeRequestChain,
|
|
663
|
+
getObfuscationKey,
|
|
502
664
|
isApiError,
|
|
503
665
|
isProtobufContentType,
|
|
504
666
|
normalizeProtobufConfig,
|
|
505
|
-
|
|
506
|
-
|
|
667
|
+
parseSecureRequest,
|
|
668
|
+
secureResponse,
|
|
669
|
+
xorTransform
|
|
507
670
|
});
|
|
508
671
|
//# sourceMappingURL=index.js.map
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts","../src/client.ts","../src/chain.ts","../src/protobuf.ts"],"sourcesContent":["export * from \"./client\";\nexport * from \"./chain\";\nexport * from \"./protobuf\";\n","import {\n ofetch,\n type FetchOptions,\n type FetchContext,\n type $Fetch,\n} from \"ofetch\";\n\nexport interface ApiResult<T> {\n code: number;\n data: T;\n message: string;\n}\n\nexport interface ApiError extends Error {\n code: number; // 业务错误码\n data?: unknown; // 后端返回的 data 字段(如果有)\n status?: number; // HTTP 状态码(网络错误时)\n}\n\n// 类型守卫:判断是否是 API 业务错误\nexport const isApiError = (error: unknown): error is ApiError => {\n return error instanceof Error && \"code\" in error;\n};\n\nexport interface TokenStorage {\n getAccessToken: () => Promise<string> | string;\n setAccessToken: (token: string) => Promise<void> | void;\n}\n\nexport interface CreateApiClientOptions {\n baseURL: string;\n tokenStorage: TokenStorage;\n refreshToken?: (() => Promise<string>) | string | false;\n retry?: number; // 重试次数,默认 1\n retryDelay?: number; // 重试间隔,默认 1s\n isAuthError?: (code: number) => boolean;\n unwrapResponse?<T>(result: unknown, returnFullResponse: boolean): T;\n createErrorFromResult?(res: unknown): Error;\n /**\n * 自定义请求钩子\n * 在请求发送前执行,可用于编码请求体等\n */\n onRequest?: (context: FetchContext) => void | Promise<void>;\n /**\n * 自定义响应钩子\n * 在响应返回后执行,可用于解码响应体等\n */\n onResponse?: (context: FetchContext) => void | Promise<void>;\n}\n\ntype RequestOptions = Omit<FetchOptions<\"json\">, \"responseType\"> & {\n returnFullResponse?: boolean;\n responseType?: \"json\" | \"arrayBuffer\" | \"text\" | \"blob\";\n};\n\n// 解包后端统一响应格式(默认实现)\nconst defaultUnwrapResponse = <T>(\n result: unknown,\n returnFullResponse = false,\n): T => {\n if (result && typeof result === \"object\" && \"code\" in result) {\n const body = result as Record<string, unknown>;\n if (body.code === 0) {\n return returnFullResponse ? (body as T) : (body.data as T);\n }\n }\n return result as T;\n};\n\nconst defaultCreateErrorFromResult = (res: ApiResult<unknown>): ApiError => {\n const error = new Error(res.message || \"Request failed\") as ApiError;\n error.code = res.code;\n error.data = res.data;\n return error;\n};\n\nconst extractAccessToken = (data: unknown): string => {\n if (typeof data === \"string\" && data) {\n return data;\n }\n\n if (!data || typeof data !== \"object\") {\n throw new Error(\n \"Invalid refresh token response: data is not an object or string\",\n );\n }\n\n const anyData = data as Record<string, unknown>;\n\n const accessToken =\n anyData.access_token ?? anyData.accessToken ?? anyData.token;\n\n if (typeof accessToken === \"string\" && accessToken) {\n return accessToken;\n }\n\n throw new Error(\n \"Invalid refresh token response: no access_token or token field found\",\n );\n};\n\n// 全局共享的刷新状态,以 tokenStorage 为 Key\n// 这样即使有多个 ApiClient 实例,只要它们共用同一个 tokenStorage,刷新逻辑就是单例的\nconst refreshingPromises = new WeakMap<TokenStorage, Promise<string>>();\n\nexport function createApiClient(options: CreateApiClientOptions) {\n const {\n baseURL,\n tokenStorage,\n refreshToken = false,\n retry = 1,\n retryDelay = 1000,\n isAuthError = (code: number) => code === 401,\n unwrapResponse = defaultUnwrapResponse,\n createErrorFromResult = defaultCreateErrorFromResult,\n } = options;\n\n const refreshTokenFn: (() => Promise<string>) | null = !refreshToken\n ? null\n : typeof refreshToken === \"string\"\n ? async () => {\n const res = await ofetch<ApiResult<unknown>>(refreshToken, {\n baseURL,\n method: \"POST\",\n retry,\n retryDelay,\n });\n\n if (res.code !== 0) {\n throw createErrorFromResult(res as ApiResult<unknown>);\n }\n\n return extractAccessToken(res.data);\n }\n : refreshToken;\n\n const rawRequest = ofetch.create({\n baseURL,\n retry,\n retryDelay,\n\n async onRequest(context: FetchContext) {\n const { options: reqOptions } = context;\n const token = await tokenStorage.getAccessToken();\n\n const headers = new Headers(\n reqOptions.headers as HeadersInit | undefined,\n );\n\n if (token) {\n headers.set(\"Authorization\", `Bearer ${token}`);\n }\n\n reqOptions.headers = headers;\n\n // 执行用户自定义钩子\n if (options.onRequest) {\n await options.onRequest(context);\n }\n },\n async onResponse(context) {\n const { response } = context;\n // 不在这里处理 code,统一交给 fetchApi 层处理\n if (response.status === 204) {\n response._data = null;\n return;\n }\n\n // 执行用户自定义钩子\n if (options.onResponse) {\n await options.onResponse(context);\n }\n },\n\n async onResponseError(context: FetchContext) {\n // 后端统一返回 HTTP 200,此处只处理网络层错误(如超时、断网)\n const { response } = context;\n const message =\n (response?._data as Record<string, unknown>)?.message ||\n `HTTP ${response?.status || \"Network Error\"}`;\n\n const error = new Error(message as string) as ApiError;\n error.status = response?.status;\n error.data = response?._data;\n throw error;\n },\n }) as $Fetch;\n\n async function request<T = unknown>(\n url: string,\n options: RequestOptions & { _retry?: boolean } = {},\n ): Promise<T> {\n // 提取自定义选项,避免传递给 $fetch\n const { returnFullResponse, _retry, ...fetchOptions } = options;\n\n // const res = await rawRequest<ApiResult<T>>(url, fetchOptions);\n const res = (await rawRequest(url, fetchOptions as FetchOptions)) as ApiResult<T>;\n\n if (res.code === 0) {\n return unwrapResponse<T>(res, !!returnFullResponse);\n // if (returnFullResponse) {\n // return res as unknown as T;\n // }\n\n // return res.data;\n }\n\n if (isAuthError(res.code) && !_retry && refreshTokenFn) {\n try {\n let refreshingPromise = refreshingPromises.get(tokenStorage);\n\n if (!refreshingPromise) {\n refreshingPromise = refreshTokenFn().finally(() => {\n refreshingPromises.delete(tokenStorage);\n });\n refreshingPromises.set(tokenStorage, refreshingPromise);\n }\n\n const newToken = await refreshingPromise;\n\n if (newToken) {\n await tokenStorage.setAccessToken(newToken);\n }\n\n return await request<T>(url, {\n ...(options || {}),\n _retry: true,\n } as any);\n } catch (e) {\n await tokenStorage.setAccessToken(\"\");\n throw createErrorFromResult(res as ApiResult<unknown>);\n }\n }\n\n throw createErrorFromResult(res as ApiResult<unknown>);\n }\n\n async function get<T = unknown>(\n url: string,\n params: FetchOptions[\"query\"] = {},\n options?: RequestOptions,\n ) {\n return request<T>(url, {\n ...options,\n method: \"GET\",\n query: params,\n } as any);\n }\n\n async function post<T = unknown>(\n url: string,\n body: FetchOptions[\"body\"] = {},\n options?: RequestOptions,\n ) {\n return request<T>(url, {\n ...options,\n method: \"POST\",\n body,\n } as any);\n }\n\n async function put<T = unknown>(\n url: string,\n body: FetchOptions[\"body\"] = {},\n options?: RequestOptions,\n ) {\n return request<T>(url, {\n ...options,\n method: \"PUT\",\n body,\n } as any);\n }\n\n async function patch<T = unknown>(\n url: string,\n body: FetchOptions[\"body\"] = {},\n options?: RequestOptions,\n ) {\n return request<T>(url, {\n ...options,\n method: \"PATCH\",\n body,\n } as any);\n }\n\n async function del<T = unknown>(\n url: string,\n params: FetchOptions[\"query\"] = {},\n options?: RequestOptions,\n ) {\n return request<T>(url, {\n ...options,\n method: \"DELETE\",\n query: params,\n } as any);\n }\n\n return {\n rawRequest,\n request,\n get,\n post,\n put,\n patch,\n del,\n };\n}\n","/**\n * Request Chain Runner\n * 负责解析和执行通用的 HTTP 请求链 (Chain Execution)\n * 适用于任何需要链式 HTTP 请求编排的场景\n */\n\n// ============================================================================\n// Type Definitions\n// ============================================================================\n\n/** Supported HTTP Methods */\ntype HttpMethod = \"GET\" | \"POST\" | \"PUT\" | \"DELETE\" | \"PATCH\";\n\n/** HTTP 请求参数 */\nexport interface HttpRequestSpec {\n method: HttpMethod;\n url: string;\n headers?: Record<string, string>;\n body?: any;\n}\n\n/**\n * 网络适配器接口\n * 定义实际发送请求的能力\n */\n/**\n * 网络适配器接口\n * 定义实际发送请求的能力\n */\nexport interface NetworkAdapter {\n get<T>(\n url: string,\n params?: any,\n headers?: Record<string, string>\n ): Promise<T>;\n post<T>(\n url: string,\n body?: any,\n headers?: Record<string, string>\n ): Promise<T>;\n put?<T>(\n url: string,\n body?: any,\n headers?: Record<string, string>\n ): Promise<T>;\n delete?<T>(\n url: string,\n params?: any,\n headers?: Record<string, string>\n ): Promise<T>;\n patch?<T>(\n url: string,\n body?: any,\n headers?: Record<string, string>\n ): Promise<T>;\n}\n\n/**\n * 网络处理器接口 (Strategy Pattern)\n * 组合了\"匹配规则\"和\"执行能力\"\n */\nexport interface NetworkHandler {\n /** 处理器名称 (用于调试) */\n name?: string;\n\n /** 判断该 URL 是否由此适配器处理 */\n shouldHandle(url: string, method: HttpMethod): boolean;\n\n /** 具体的网络适配器 */\n adapter: NetworkAdapter;\n}\n\n/**\n * 请求链规则 (Chain Step)\n */\nexport interface ChainRequestRule {\n /**\n * 结果存储的键名 (Context Key)\n * Fetch 结果存储: context[key] = response\n */\n key?: string;\n\n /**\n * HTTP 请求详情\n */\n request: HttpRequestSpec;\n\n /**\n * 数据提取选择器\n * 用于从响应中提取特定数据, e.g., \"elements.0\"\n */\n selector?: string;\n\n /** 是否允许请求失败 (默认 false) */\n optional?: boolean;\n\n /**\n * 是否包含上下文\n * 如果为 true,请求 Body 将自动合并当前 Context 和 Payload\n */\n includeContext?: boolean;\n\n /**\n * 上下文筛选列表 (优先级高于 includeContext)\n * 指定需要合并到 Body 中的 Context Key 列表\n * e.g., [\"rawProfile\", \"rawCompany\"]\n */\n pickContext?: string[];\n\n /** 额外 Payload (仅当 includeContext 为 true 或 pickContext 存在时合并) */\n payload?: Record<string, any>;\n}\n\n// ============================================================================\n// Internal Utilities\n// ============================================================================\n\n/**\n * 执行 HTTP 请求\n * 遍历 handlers 数组,找到第一个能处理该 URL 的适配器执行\n *\n * @param spec 请求详情\n * @param handlers 网络处理器链\n */\nasync function executeChainRequest<T>(\n spec: HttpRequestSpec,\n handlers: NetworkHandler[] = []\n): Promise<T | null> {\n // Normalize method to upper case for robustness\n const method = spec.method.toUpperCase() as HttpMethod;\n\n // 1. 遍历处理器数组 (责任链)\n for (const handler of handlers) {\n if (handler.shouldHandle(spec.url, method)) {\n try {\n const { adapter } = handler;\n const headers = spec.headers;\n\n switch (method) {\n case \"GET\":\n return await adapter.get<T>(spec.url, spec.body, headers);\n case \"POST\":\n return await adapter.post<T>(spec.url, spec.body, headers);\n case \"PUT\":\n if (!adapter.put)\n throw new Error(`Adapter ${handler.name} missing PUT method`);\n return await adapter.put<T>(spec.url, spec.body, headers);\n case \"DELETE\":\n if (!adapter.delete)\n throw new Error(`Adapter ${handler.name} missing DELETE method`);\n return await adapter.delete<T>(spec.url, spec.body, headers);\n case \"PATCH\":\n if (!adapter.patch)\n throw new Error(`Adapter ${handler.name} missing PATCH method`);\n return await adapter.patch<T>(spec.url, spec.body, headers);\n default:\n throw new Error(`Unsupported method: ${method}`);\n }\n } catch (err) {\n console.warn(\n `Handler [${handler.name || \"Anonymous\"}] failed for ${spec.url}`,\n err\n );\n // 如果需要继续尝试下一个 handler,可以在这里 continue,但通常由第一个匹配者全权负责\n return null;\n }\n }\n }\n\n // 2. 默认行为 (Fallback): 绝对路径自动走标准 fetch\n if (spec.url.startsWith(\"http://\") || spec.url.startsWith(\"https://\")) {\n return await executeFallbackFetch<T>(spec);\n }\n\n console.warn(`No handler found for url: ${spec.url}`);\n return null;\n}\n\n/**\n * 标准 Fetch 回退实现\n */\nasync function executeFallbackFetch<T>(\n spec: HttpRequestSpec\n): Promise<T | null> {\n try {\n const response = await fetch(spec.url, {\n method: spec.method,\n headers: spec.headers,\n body: spec.body ? JSON.stringify(spec.body) : undefined,\n credentials: \"include\",\n });\n\n if (!response.ok) {\n console.log(`External Request failed: ${response.status} ${spec.url}`);\n return null;\n }\n\n return await response.json();\n } catch (err) {\n console.log(\"External Request error:\", err);\n return null;\n }\n}\n\n/**\n * 根据路径获取对象值\n */\nfunction getByPath(obj: any, path: string): any {\n if (!path) return obj;\n return path.split(\".\").reduce((acc, part) => acc && acc[part], obj);\n}\n\n/**\n * 变量替换 (Internal)\n * 支持字符串 {{key}} 和 {{key.prop}}\n */\nfunction substituteVariables(target: any, context: any): any {\n if (typeof target === \"string\") {\n return target.replace(/\\{\\{([\\w\\.]+)\\}\\}/g, (_, path) => {\n const val = getByPath(context, path);\n return val !== undefined ? String(val) : \"\";\n });\n }\n\n if (Array.isArray(target)) {\n return target.map((item) => substituteVariables(item, context));\n }\n\n if (target && typeof target === \"object\") {\n const result: any = {};\n for (const key in target) {\n result[key] = substituteVariables(target[key], context);\n }\n return result;\n }\n\n return target;\n}\n\n// ============================================================================\n// Main Runner\n// ============================================================================\n\n/**\n * 请求链执行器 (Request Chain Engine)\n * 支持链式请求、上下文累积、智能路由、变量替换\n *\n * @param requests 链式执行规则列表\n * @param handlers 网络适配器处理器列表 (按顺序匹配)\n */\nexport async function executeRequestChain<T>(\n requests: ChainRequestRule[],\n handlers: NetworkHandler[] = [],\n getChainRequests?: (\n context: Record<string, any>\n ) => ChainRequestRule[] | null | undefined,\n initContext?: Record<string, any>\n): Promise<T> {\n const context: Record<string, any> = initContext || {};\n let lastResult: any = null;\n\n for (const rule of requests) {\n try {\n // 1. 准备 Context Data (用于 Submit)\n let contextData: Record<string, any> = {};\n if (rule.pickContext) {\n rule.pickContext.forEach((key) => {\n if (key in context) contextData[key] = context[key];\n });\n } else if (rule.includeContext) {\n contextData = context;\n }\n const hasContextData = Object.keys(contextData).length > 0;\n\n // 2. 准备 Request Spec (合并 Payload 和 Context)\n let requestSpec = { ...rule.request };\n if (hasContextData || rule.payload) {\n requestSpec.body = {\n ...(rule.payload || {}),\n ...contextData,\n ...(requestSpec.body || {}),\n };\n }\n\n // 3. 变量替换\n requestSpec = substituteVariables(requestSpec, context);\n\n // 4. 执行请求 (传入 handlers)\n let rawData = await executeChainRequest<any>(requestSpec, handlers);\n\n if (getChainRequests) {\n const chainRequests = getChainRequests(rawData);\n if (chainRequests && chainRequests.length > 0) {\n rawData = await executeRequestChain(\n chainRequests,\n handlers,\n getChainRequests,\n JSON.parse(JSON.stringify(context))\n );\n }\n }\n\n lastResult = rawData;\n\n if (rawData) {\n // 5. 存储结果\n if (rule.key) {\n const data = rule.selector\n ? getByPath(rawData, rule.selector)\n : rawData;\n context[rule.key] = data;\n }\n } else if (!rule.optional) {\n throw new Error(\n `Failed to fetch required data for key: ${rule.key || \"unknown\"}`\n );\n }\n } catch (err) {\n if (!rule.optional) {\n throw err;\n }\n console.warn(`Optional request failed for rule:`, rule, err);\n }\n }\n\n return lastResult as T;\n}\n","/**\n * Protobuf 通用编解码模块\n *\n * 特性:\n * - 支持加盐混淆(每次编码结果不同)\n * - 直接输出二进制格式\n * - 内置通用容器 schema(任意 JSON 数据)\n * - 支持自定义编解码函数\n */\n\nimport protobuf from \"protobufjs\";\nimport type { FetchContext } from \"ofetch\";\n\n// ============================================================================\n// Schema 定义\n// ============================================================================\n\n/**\n * 内置的加盐消息 Schema\n * 支持任意 JSON 数据的安全传输\n */\nconst SALTED_SCHEMA = `\nsyntax = \"proto3\";\n\nmessage SaltedPayload {\n string salt = 1;\n int64 timestamp = 2;\n bytes data = 3;\n}\n`;\n\nconst encoder = new TextEncoder();\nconst decoder = new TextDecoder();\n\n// ============================================================================\n// 类型定义\n// ============================================================================\n\nexport interface ProtobufCodecOptions {\n /** 是否启用加盐 (默认 true) */\n useSalt?: boolean;\n /** 自定义盐值生成器 */\n saltGenerator?: () => string;\n}\n\nexport interface ProtobufConfig {\n /** 自定义编码函数 */\n encode?: (data: unknown) => Uint8Array | Promise<Uint8Array>;\n /** 自定义解码函数 */\n decode?: <T>(buffer: Uint8Array) => T | Promise<T>;\n /** 编码选项(仅内置编解码器使用) */\n options?: ProtobufCodecOptions;\n}\n\ninterface SaltedPayload {\n salt: string;\n timestamp: number;\n data: Uint8Array;\n}\n\n// ============================================================================\n// Schema 缓存\n// ============================================================================\n\nlet saltedPayloadType: protobuf.Type | null = null;\n\nfunction getSaltedPayloadType(): protobuf.Type {\n if (!saltedPayloadType) {\n const root = protobuf.parse(SALTED_SCHEMA).root;\n saltedPayloadType = root.lookupType(\"SaltedPayload\");\n }\n return saltedPayloadType;\n}\n\n// ============================================================================\n// 工具函数\n// ============================================================================\n\n/**\n * 生成随机盐值\n */\nfunction generateSalt(): string {\n if (typeof crypto !== \"undefined\" && crypto.randomUUID) {\n return crypto.randomUUID();\n }\n // Fallback for older environments\n return Math.random().toString(36).substring(2) + Date.now().toString(36);\n}\n\n// ============================================================================\n// 内置编解码器\n// ============================================================================\n\n/**\n * 使用内置 SaltedPayload schema 编码数据\n *\n * @param data - 任意 JSON 可序列化数据\n * @param options - 编码选项\n * @returns Uint8Array 二进制数据\n */\nexport function encodeProtobuf<T>(\n data: T,\n options: ProtobufCodecOptions = {},\n): Uint8Array {\n const { useSalt = true, saltGenerator = generateSalt } = options;\n\n const type = getSaltedPayloadType();\n const jsonString = JSON.stringify(data);\n const dataBytes = encoder.encode(jsonString);\n\n const payload: SaltedPayload = {\n salt: useSalt ? saltGenerator() : \"\",\n timestamp: Date.now(),\n data: dataBytes,\n };\n\n const message = type.create(payload);\n return type.encode(message).finish();\n}\n\n/**\n * 使用内置 SaltedPayload schema 解码数据\n *\n * @param buffer - 二进制数据\n * @returns 解码后的原始数据\n */\nexport function decodeProtobuf<T>(buffer: Uint8Array | ArrayBuffer): T {\n const type = getSaltedPayloadType();\n const uint8 = buffer instanceof ArrayBuffer ? new Uint8Array(buffer) : buffer;\n const decoded = type.decode(uint8) as unknown as SaltedPayload;\n const jsonString = decoder.decode(decoded.data);\n return JSON.parse(jsonString) as T;\n}\n\n// ============================================================================\n// HTTP Response 工具 (Worker/Server 端使用)\n// ============================================================================\n\n/**\n * 创建 Protobuf 二进制响应\n *\n * @param data - 要编码的数据\n * @param options - 编码选项和 CORS headers\n * @returns Response 对象\n */\nexport function protobufResponse<T>(\n data: T,\n options?: ProtobufCodecOptions & { corsHeaders?: Record<string, string> },\n): Response {\n const buffer = encodeProtobuf(data, options);\n // 使用类型断言确保兼容 Response 构造函数\n const arrayBuffer = buffer.buffer.slice(\n buffer.byteOffset,\n buffer.byteOffset + buffer.byteLength,\n ) as ArrayBuffer;\n return new Response(arrayBuffer, {\n headers: {\n \"Content-Type\": \"application/x-protobuf\",\n ...options?.corsHeaders,\n },\n });\n}\n\n/**\n * 检查 Content-Type 是否是 Protobuf 格式\n */\nexport function isProtobufContentType(contentType: string | null): boolean {\n return !!contentType?.includes(\"application/x-protobuf\");\n}\n\n// ============================================================================\n// 规范化配置\n// ============================================================================\n\n/**\n * 规范化 protobuf 配置\n * 将 boolean | ProtobufConfig 统一为 ProtobufConfig 或 null\n */\nexport function normalizeProtobufConfig(\n config: boolean | ProtobufConfig | undefined,\n): ProtobufConfig | null {\n if (!config) {\n return null;\n }\n\n if (config === true) {\n return {\n encode: (data) => encodeProtobuf(data),\n decode: <T>(buffer: Uint8Array) => decodeProtobuf<T>(buffer),\n };\n }\n\n return {\n encode: config.encode || ((data) => encodeProtobuf(data, config.options)),\n decode:\n config.decode || (<T>(buffer: Uint8Array) => decodeProtobuf<T>(buffer)),\n };\n}\n\n// ============================================================================\n// 请求体处理\n// ============================================================================\n\n/**\n * 解析 Protobuf 请求体 (Worker/Server 端使用)\n *\n * @param request - Request 对象\n * @param customDecode - 可选的自定义解码函数\n * @returns 解码后的数据\n */\nexport async function parseProtobufRequest<T>(\n request: Request,\n customDecode?: (buffer: Uint8Array) => T | Promise<T>,\n): Promise<T> {\n const buffer = await request.arrayBuffer();\n const uint8 = new Uint8Array(buffer);\n\n if (customDecode) {\n return customDecode(uint8);\n }\n\n return decodeProtobuf<T>(uint8);\n}\n\n// ============================================================================\n// API Client 钩子\n// ============================================================================\n\n/**\n * Protobuf 钩子选项\n */\nexport interface CreateProtobufHooksOptions extends ProtobufCodecOptions {\n /** 自定义编码函数 */\n encode?: (data: unknown) => Uint8Array | Promise<Uint8Array>;\n /** 自定义解码函数 */\n decode?: <T>(buffer: Uint8Array) => T | Promise<T>;\n}\n\n/**\n * 创建 Protobuf 编解码钩子\n * 用于与 createApiClient 配合使用\n *\n * 钩子会自动:\n * - 设置 responseType 为 arrayBuffer\n * - 编码请求体为 Protobuf 格式\n * - 根据响应 Content-Type 自动解码(protobuf 或 JSON)\n *\n * @example\n * ```ts\n * import { createApiClient } from '@i.un/api-client';\n * import { createProtobufHooks } from '@i.un/api-client/protobuf';\n *\n * const { onRequest, onResponse } = createProtobufHooks();\n *\n * const client = createApiClient({\n * baseURL: 'https://api.example.com',\n * tokenStorage,\n * onRequest,\n * onResponse,\n * });\n *\n * // 自动处理,无需手动指定 responseType\n * const data = await client.post('/api/data', body);\n * ```\n */\nexport function createProtobufHooks(options: CreateProtobufHooksOptions = {}) {\n const encode =\n options.encode || ((data: unknown) => encodeProtobuf(data, options));\n const decode =\n options.decode || (<T>(buffer: Uint8Array) => decodeProtobuf<T>(buffer));\n\n return {\n /**\n * 请求钩子:独立处理请求编码和响应类型设置\n */\n async onRequest(context: FetchContext) {\n const { options: reqOptions } = context;\n\n const headers =\n reqOptions.headers instanceof Headers\n ? reqOptions.headers\n : new Headers(reqOptions.headers as HeadersInit | undefined);\n\n // 1. 处理请求体编码 (Content-Type)\n const contentType = headers.get(\"Content-Type\");\n const isProtobufRequest = isProtobufContentType(contentType);\n\n if (\n isProtobufRequest &&\n reqOptions.body &&\n !(reqOptions.body instanceof Uint8Array)\n ) {\n const encoded = await encode(reqOptions.body);\n reqOptions.body = encoded;\n }\n\n // 2. 处理响应期望类型 (Accept)\n // 如果 Accept 包含 protobuf,或者用户显式要求 arrayBuffer,则设置 responseType\n const accept = headers.get(\"Accept\");\n const prefersProtobufResponse = isProtobufContentType(accept);\n\n if (prefersProtobufResponse && !reqOptions.responseType) {\n reqOptions.responseType = \"arrayBuffer\";\n }\n\n reqOptions.headers = headers;\n },\n\n /**\n * 响应钩子:解码 Protobuf 响应\n */\n async onResponse(context: FetchContext) {\n const { response } = context;\n // 跳过空响应 (如 204 No Content)\n if (!response?._data) {\n return;\n }\n\n const contentType = response.headers.get(\"Content-Type\");\n\n if (isProtobufContentType(contentType)) {\n const buffer = response._data as ArrayBuffer;\n if (buffer && buffer.byteLength > 0) {\n response._data = await decode(new Uint8Array(buffer));\n }\n } else if (response._data instanceof ArrayBuffer) {\n // 如果响应不是 protobuf(比如普通 JSON),需要手动解析\n const text = decoder.decode(response._data);\n try {\n response._data = JSON.parse(text);\n } catch {\n response._data = text;\n }\n }\n },\n };\n}\n"],"mappings":"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","names":["options","protobuf"]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts","../src/client.ts","../src/chain.ts","../src/secure.js","../src/protobuf.ts"],"sourcesContent":["export * from \"./client\";\nexport * from \"./chain\";\nexport * from \"./protobuf\";\n","import {\n ofetch,\n type FetchOptions,\n type FetchContext,\n type $Fetch,\n} from \"ofetch\";\n\nexport interface ApiResult<T> {\n code: number;\n data: T;\n message: string;\n}\n\nexport interface ApiError extends Error {\n code: number; // 业务错误码\n data?: unknown; // 后端返回的 data 字段(如果有)\n status?: number; // HTTP 状态码(网络错误时)\n}\n\n// 类型守卫:判断是否是 API 业务错误\nexport const isApiError = (error: unknown): error is ApiError => {\n return error instanceof Error && \"code\" in error;\n};\n\nexport interface TokenStorage {\n getAccessToken: () => Promise<string> | string;\n setAccessToken: (token: string) => Promise<void> | void;\n}\n\nexport interface CreateApiClientOptions {\n baseURL: string;\n tokenStorage: TokenStorage;\n refreshToken?: (() => Promise<string>) | string | false;\n retry?: number; // 重试次数,默认 1\n retryDelay?: number; // 重试间隔,默认 1s\n isAuthError?: (code: number) => boolean;\n unwrapResponse?<T>(result: unknown, returnFullResponse: boolean): T;\n createErrorFromResult?(res: unknown): Error;\n /**\n * 自定义请求钩子\n * 在请求发送前执行,可用于编码请求体等\n */\n onRequest?: (context: FetchContext) => void | Promise<void>;\n /**\n * 自定义响应钩子\n * 在响应返回后执行,可用于解码响应体等\n */\n onResponse?: (context: FetchContext) => void | Promise<void>;\n}\n\ntype RequestOptions = Omit<FetchOptions<\"json\">, \"responseType\"> & {\n returnFullResponse?: boolean;\n responseType?: \"json\" | \"arrayBuffer\" | \"text\" | \"blob\";\n};\n\n// 解包后端统一响应格式(默认实现)\nconst defaultUnwrapResponse = <T>(\n result: unknown,\n returnFullResponse = false,\n): T => {\n if (result && typeof result === \"object\" && \"code\" in result) {\n const body = result as Record<string, unknown>;\n if (body.code === 0) {\n return returnFullResponse ? (body as T) : (body.data as T);\n }\n }\n return result as T;\n};\n\nconst defaultCreateErrorFromResult = (res: ApiResult<unknown>): ApiError => {\n const error = new Error(res.message || \"Request failed\") as ApiError;\n error.code = res.code;\n error.data = res.data;\n return error;\n};\n\nconst extractAccessToken = (data: unknown): string => {\n if (typeof data === \"string\" && data) {\n return data;\n }\n\n if (!data || typeof data !== \"object\") {\n throw new Error(\n \"Invalid refresh token response: data is not an object or string\",\n );\n }\n\n const anyData = data as Record<string, unknown>;\n\n const accessToken =\n anyData.access_token ?? anyData.accessToken ?? anyData.token;\n\n if (typeof accessToken === \"string\" && accessToken) {\n return accessToken;\n }\n\n throw new Error(\n \"Invalid refresh token response: no access_token or token field found\",\n );\n};\n\n// 全局共享的刷新状态,以 tokenStorage 为 Key\n// 这样即使有多个 ApiClient 实例,只要它们共用同一个 tokenStorage,刷新逻辑就是单例的\nconst refreshingPromises = new WeakMap<TokenStorage, Promise<string>>();\n\nexport function createApiClient(options: CreateApiClientOptions) {\n const {\n baseURL,\n tokenStorage,\n refreshToken = false,\n retry = 1,\n retryDelay = 1000,\n isAuthError = (code: number) => code === 401,\n unwrapResponse = defaultUnwrapResponse,\n createErrorFromResult = defaultCreateErrorFromResult,\n } = options;\n\n const refreshTokenFn: (() => Promise<string>) | null = !refreshToken\n ? null\n : typeof refreshToken === \"string\"\n ? async () => {\n const res = await ofetch<ApiResult<unknown>>(refreshToken, {\n baseURL,\n method: \"POST\",\n retry,\n retryDelay,\n });\n\n if (res.code !== 0) {\n throw createErrorFromResult(res as ApiResult<unknown>);\n }\n\n return extractAccessToken(res.data);\n }\n : refreshToken;\n\n const rawRequest = ofetch.create({\n baseURL,\n retry,\n retryDelay,\n\n async onRequest(context: FetchContext) {\n const { options: reqOptions } = context;\n const token = await tokenStorage.getAccessToken();\n\n const headers = new Headers(\n reqOptions.headers as HeadersInit | undefined,\n );\n\n if (token) {\n headers.set(\"Authorization\", `Bearer ${token}`);\n }\n\n reqOptions.headers = headers;\n\n // 执行用户自定义钩子\n if (options.onRequest) {\n await options.onRequest(context);\n }\n },\n async onResponse(context) {\n const { response } = context;\n // 不在这里处理 code,统一交给 fetchApi 层处理\n if (response.status === 204) {\n response._data = null;\n return;\n }\n\n // 执行用户自定义钩子\n if (options.onResponse) {\n await options.onResponse(context);\n }\n },\n\n async onResponseError(context: FetchContext) {\n // 后端统一返回 HTTP 200,此处只处理网络层错误(如超时、断网)\n const { response } = context;\n const message =\n (response?._data as Record<string, unknown>)?.message ||\n `HTTP ${response?.status || \"Network Error\"}`;\n\n const error = new Error(message as string) as ApiError;\n error.status = response?.status;\n error.data = response?._data;\n throw error;\n },\n }) as $Fetch;\n\n async function request<T = unknown>(\n url: string,\n options: RequestOptions & { _retry?: boolean } = {},\n ): Promise<T> {\n // 提取自定义选项,避免传递给 $fetch\n const { returnFullResponse, _retry, ...fetchOptions } = options;\n\n // const res = await rawRequest<ApiResult<T>>(url, fetchOptions);\n const res = (await rawRequest(url, fetchOptions as FetchOptions)) as ApiResult<T>;\n\n if (res.code === 0) {\n return unwrapResponse<T>(res, !!returnFullResponse);\n // if (returnFullResponse) {\n // return res as unknown as T;\n // }\n\n // return res.data;\n }\n\n if (isAuthError(res.code) && !_retry && refreshTokenFn) {\n try {\n let refreshingPromise = refreshingPromises.get(tokenStorage);\n\n if (!refreshingPromise) {\n refreshingPromise = refreshTokenFn().finally(() => {\n refreshingPromises.delete(tokenStorage);\n });\n refreshingPromises.set(tokenStorage, refreshingPromise);\n }\n\n const newToken = await refreshingPromise;\n\n if (newToken) {\n await tokenStorage.setAccessToken(newToken);\n }\n\n return await request<T>(url, {\n ...(options || {}),\n _retry: true,\n } as any);\n } catch (e) {\n await tokenStorage.setAccessToken(\"\");\n throw createErrorFromResult(res as ApiResult<unknown>);\n }\n }\n\n throw createErrorFromResult(res as ApiResult<unknown>);\n }\n\n async function get<T = unknown>(\n url: string,\n params: FetchOptions[\"query\"] = {},\n options?: RequestOptions,\n ) {\n return request<T>(url, {\n ...options,\n method: \"GET\",\n query: params,\n } as any);\n }\n\n async function post<T = unknown>(\n url: string,\n body: FetchOptions[\"body\"] = {},\n options?: RequestOptions,\n ) {\n return request<T>(url, {\n ...options,\n method: \"POST\",\n body,\n } as any);\n }\n\n async function put<T = unknown>(\n url: string,\n body: FetchOptions[\"body\"] = {},\n options?: RequestOptions,\n ) {\n return request<T>(url, {\n ...options,\n method: \"PUT\",\n body,\n } as any);\n }\n\n async function patch<T = unknown>(\n url: string,\n body: FetchOptions[\"body\"] = {},\n options?: RequestOptions,\n ) {\n return request<T>(url, {\n ...options,\n method: \"PATCH\",\n body,\n } as any);\n }\n\n async function del<T = unknown>(\n url: string,\n params: FetchOptions[\"query\"] = {},\n options?: RequestOptions,\n ) {\n return request<T>(url, {\n ...options,\n method: \"DELETE\",\n query: params,\n } as any);\n }\n\n return {\n rawRequest,\n request,\n get,\n post,\n put,\n patch,\n del,\n };\n}\n","/**\n * Request Chain Runner\n * 负责解析和执行通用的 HTTP 请求链 (Chain Execution)\n * 适用于任何需要链式 HTTP 请求编排的场景\n */\n\n// ============================================================================\n// Type Definitions\n// ============================================================================\n\n/** Supported HTTP Methods */\ntype HttpMethod = \"GET\" | \"POST\" | \"PUT\" | \"DELETE\" | \"PATCH\";\n\n/** HTTP 请求参数 */\nexport interface HttpRequestSpec {\n method: HttpMethod;\n url: string;\n headers?: Record<string, string>;\n body?: any;\n}\n\n/**\n * 网络适配器接口\n * 定义实际发送请求的能力\n */\n/**\n * 网络适配器接口\n * 定义实际发送请求的能力\n */\nexport interface NetworkAdapter {\n get<T>(\n url: string,\n params?: any,\n headers?: Record<string, string>\n ): Promise<T>;\n post<T>(\n url: string,\n body?: any,\n headers?: Record<string, string>\n ): Promise<T>;\n put?<T>(\n url: string,\n body?: any,\n headers?: Record<string, string>\n ): Promise<T>;\n delete?<T>(\n url: string,\n params?: any,\n headers?: Record<string, string>\n ): Promise<T>;\n patch?<T>(\n url: string,\n body?: any,\n headers?: Record<string, string>\n ): Promise<T>;\n}\n\n/**\n * 网络处理器接口 (Strategy Pattern)\n * 组合了\"匹配规则\"和\"执行能力\"\n */\nexport interface NetworkHandler {\n /** 处理器名称 (用于调试) */\n name?: string;\n\n /** 判断该 URL 是否由此适配器处理 */\n shouldHandle(url: string, method: HttpMethod): boolean;\n\n /** 具体的网络适配器 */\n adapter: NetworkAdapter;\n}\n\n/**\n * 请求链规则 (Chain Step)\n */\nexport interface ChainRequestRule {\n /**\n * 结果存储的键名 (Context Key)\n * Fetch 结果存储: context[key] = response\n */\n key?: string;\n\n /**\n * HTTP 请求详情\n */\n request: HttpRequestSpec;\n\n /**\n * 数据提取选择器\n * 用于从响应中提取特定数据, e.g., \"elements.0\"\n */\n selector?: string;\n\n /** 是否允许请求失败 (默认 false) */\n optional?: boolean;\n\n /**\n * 是否包含上下文\n * 如果为 true,请求 Body 将自动合并当前 Context 和 Payload\n */\n includeContext?: boolean;\n\n /**\n * 上下文筛选列表 (优先级高于 includeContext)\n * 指定需要合并到 Body 中的 Context Key 列表\n * e.g., [\"rawProfile\", \"rawCompany\"]\n */\n pickContext?: string[];\n\n /** 额外 Payload (仅当 includeContext 为 true 或 pickContext 存在时合并) */\n payload?: Record<string, any>;\n}\n\n// ============================================================================\n// Internal Utilities\n// ============================================================================\n\n/**\n * 执行 HTTP 请求\n * 遍历 handlers 数组,找到第一个能处理该 URL 的适配器执行\n *\n * @param spec 请求详情\n * @param handlers 网络处理器链\n */\nasync function executeChainRequest<T>(\n spec: HttpRequestSpec,\n handlers: NetworkHandler[] = []\n): Promise<T | null> {\n // Normalize method to upper case for robustness\n const method = spec.method.toUpperCase() as HttpMethod;\n\n // 1. 遍历处理器数组 (责任链)\n for (const handler of handlers) {\n if (handler.shouldHandle(spec.url, method)) {\n try {\n const { adapter } = handler;\n const headers = spec.headers;\n\n switch (method) {\n case \"GET\":\n return await adapter.get<T>(spec.url, spec.body, headers);\n case \"POST\":\n return await adapter.post<T>(spec.url, spec.body, headers);\n case \"PUT\":\n if (!adapter.put)\n throw new Error(`Adapter ${handler.name} missing PUT method`);\n return await adapter.put<T>(spec.url, spec.body, headers);\n case \"DELETE\":\n if (!adapter.delete)\n throw new Error(`Adapter ${handler.name} missing DELETE method`);\n return await adapter.delete<T>(spec.url, spec.body, headers);\n case \"PATCH\":\n if (!adapter.patch)\n throw new Error(`Adapter ${handler.name} missing PATCH method`);\n return await adapter.patch<T>(spec.url, spec.body, headers);\n default:\n throw new Error(`Unsupported method: ${method}`);\n }\n } catch (err) {\n console.warn(\n `Handler [${handler.name || \"Anonymous\"}] failed for ${spec.url}`,\n err\n );\n // 如果需要继续尝试下一个 handler,可以在这里 continue,但通常由第一个匹配者全权负责\n return null;\n }\n }\n }\n\n // 2. 默认行为 (Fallback): 绝对路径自动走标准 fetch\n if (spec.url.startsWith(\"http://\") || spec.url.startsWith(\"https://\")) {\n return await executeFallbackFetch<T>(spec);\n }\n\n console.warn(`No handler found for url: ${spec.url}`);\n return null;\n}\n\n/**\n * 标准 Fetch 回退实现\n */\nasync function executeFallbackFetch<T>(\n spec: HttpRequestSpec\n): Promise<T | null> {\n try {\n const response = await fetch(spec.url, {\n method: spec.method,\n headers: spec.headers,\n body: spec.body ? JSON.stringify(spec.body) : undefined,\n credentials: \"include\",\n });\n\n if (!response.ok) {\n console.log(`External Request failed: ${response.status} ${spec.url}`);\n return null;\n }\n\n return await response.json();\n } catch (err) {\n console.log(\"External Request error:\", err);\n return null;\n }\n}\n\n/**\n * 根据路径获取对象值\n */\nfunction getByPath(obj: any, path: string): any {\n if (!path) return obj;\n return path.split(\".\").reduce((acc, part) => acc && acc[part], obj);\n}\n\n/**\n * 变量替换 (Internal)\n * 支持字符串 {{key}} 和 {{key.prop}}\n */\nfunction substituteVariables(target: any, context: any): any {\n if (typeof target === \"string\") {\n return target.replace(/\\{\\{([\\w\\.]+)\\}\\}/g, (_, path) => {\n const val = getByPath(context, path);\n return val !== undefined ? String(val) : \"\";\n });\n }\n\n if (Array.isArray(target)) {\n return target.map((item) => substituteVariables(item, context));\n }\n\n if (target && typeof target === \"object\") {\n const result: any = {};\n for (const key in target) {\n result[key] = substituteVariables(target[key], context);\n }\n return result;\n }\n\n return target;\n}\n\n// ============================================================================\n// Main Runner\n// ============================================================================\n\n/**\n * 请求链执行器 (Request Chain Engine)\n * 支持链式请求、上下文累积、智能路由、变量替换\n *\n * @param requests 链式执行规则列表\n * @param handlers 网络适配器处理器列表 (按顺序匹配)\n */\nexport async function executeRequestChain<T>(\n requests: ChainRequestRule[],\n handlers: NetworkHandler[] = [],\n getChainRequests?: (\n context: Record<string, any>\n ) => ChainRequestRule[] | null | undefined,\n initContext?: Record<string, any>\n): Promise<T> {\n const context: Record<string, any> = initContext || {};\n let lastResult: any = null;\n\n for (const rule of requests) {\n try {\n // 1. 准备 Context Data (用于 Submit)\n let contextData: Record<string, any> = {};\n if (rule.pickContext) {\n rule.pickContext.forEach((key) => {\n if (key in context) contextData[key] = context[key];\n });\n } else if (rule.includeContext) {\n contextData = context;\n }\n const hasContextData = Object.keys(contextData).length > 0;\n\n // 2. 准备 Request Spec (合并 Payload 和 Context)\n let requestSpec = { ...rule.request };\n if (hasContextData || rule.payload) {\n requestSpec.body = {\n ...(rule.payload || {}),\n ...contextData,\n ...(requestSpec.body || {}),\n };\n }\n\n // 3. 变量替换\n requestSpec = substituteVariables(requestSpec, context);\n\n // 4. 执行请求 (传入 handlers)\n let rawData = await executeChainRequest<any>(requestSpec, handlers);\n\n if (getChainRequests) {\n const chainRequests = getChainRequests(rawData);\n if (chainRequests && chainRequests.length > 0) {\n rawData = await executeRequestChain(\n chainRequests,\n handlers,\n getChainRequests,\n JSON.parse(JSON.stringify(context))\n );\n }\n }\n\n lastResult = rawData;\n\n if (rawData) {\n // 5. 存储结果\n if (rule.key) {\n const data = rule.selector\n ? getByPath(rawData, rule.selector)\n : rawData;\n context[rule.key] = data;\n }\n } else if (!rule.optional) {\n throw new Error(\n `Failed to fetch required data for key: ${rule.key || \"unknown\"}`\n );\n }\n } catch (err) {\n if (!rule.optional) {\n throw err;\n }\n console.warn(`Optional request failed for rule:`, rule, err);\n }\n }\n\n return lastResult as T;\n}\n","/*eslint-disable block-scoped-var, id-length, no-control-regex, no-magic-numbers, no-prototype-builtins, no-redeclare, no-shadow, no-var, sort-vars*/\nimport * as $protobuf from \"protobufjs/minimal\";\n\n// Common aliases\nconst $Reader = $protobuf.Reader, $Writer = $protobuf.Writer, $util = $protobuf.util;\n\n// Exported root namespace\nconst $root = $protobuf.roots[\"default\"] || ($protobuf.roots[\"default\"] = {});\n\nexport const secure = $root.secure = (() => {\n\n /**\n * Namespace secure.\n * @exports secure\n * @namespace\n */\n const secure = {};\n\n secure.SecurePayload = (function() {\n\n /**\n * Properties of a SecurePayload.\n * @memberof secure\n * @interface ISecurePayload\n * @property {number|Long|null} [ts] SecurePayload ts\n * @property {Uint8Array|null} [data] SecurePayload data\n */\n\n /**\n * Constructs a new SecurePayload.\n * @memberof secure\n * @classdesc Represents a SecurePayload.\n * @implements ISecurePayload\n * @constructor\n * @param {secure.ISecurePayload=} [properties] Properties to set\n */\n function SecurePayload(properties) {\n if (properties)\n for (let keys = Object.keys(properties), i = 0; i < keys.length; ++i)\n if (properties[keys[i]] != null)\n this[keys[i]] = properties[keys[i]];\n }\n\n /**\n * SecurePayload ts.\n * @member {number|Long} ts\n * @memberof secure.SecurePayload\n * @instance\n */\n SecurePayload.prototype.ts = $util.Long ? $util.Long.fromBits(0,0,false) : 0;\n\n /**\n * SecurePayload data.\n * @member {Uint8Array} data\n * @memberof secure.SecurePayload\n * @instance\n */\n SecurePayload.prototype.data = $util.newBuffer([]);\n\n /**\n * Creates a new SecurePayload instance using the specified properties.\n * @function create\n * @memberof secure.SecurePayload\n * @static\n * @param {secure.ISecurePayload=} [properties] Properties to set\n * @returns {secure.SecurePayload} SecurePayload instance\n */\n SecurePayload.create = function create(properties) {\n return new SecurePayload(properties);\n };\n\n /**\n * Encodes the specified SecurePayload message. Does not implicitly {@link secure.SecurePayload.verify|verify} messages.\n * @function encode\n * @memberof secure.SecurePayload\n * @static\n * @param {secure.ISecurePayload} message SecurePayload message or plain object to encode\n * @param {$protobuf.Writer} [writer] Writer to encode to\n * @returns {$protobuf.Writer} Writer\n */\n SecurePayload.encode = function encode(message, writer) {\n if (!writer)\n writer = $Writer.create();\n if (message.ts != null && Object.hasOwnProperty.call(message, \"ts\"))\n writer.uint32(/* id 1, wireType 0 =*/8).int64(message.ts);\n if (message.data != null && Object.hasOwnProperty.call(message, \"data\"))\n writer.uint32(/* id 2, wireType 2 =*/18).bytes(message.data);\n return writer;\n };\n\n /**\n * Encodes the specified SecurePayload message, length delimited. Does not implicitly {@link secure.SecurePayload.verify|verify} messages.\n * @function encodeDelimited\n * @memberof secure.SecurePayload\n * @static\n * @param {secure.ISecurePayload} message SecurePayload message or plain object to encode\n * @param {$protobuf.Writer} [writer] Writer to encode to\n * @returns {$protobuf.Writer} Writer\n */\n SecurePayload.encodeDelimited = function encodeDelimited(message, writer) {\n return this.encode(message, writer).ldelim();\n };\n\n /**\n * Decodes a SecurePayload message from the specified reader or buffer.\n * @function decode\n * @memberof secure.SecurePayload\n * @static\n * @param {$protobuf.Reader|Uint8Array} reader Reader or buffer to decode from\n * @param {number} [length] Message length if known beforehand\n * @returns {secure.SecurePayload} SecurePayload\n * @throws {Error} If the payload is not a reader or valid buffer\n * @throws {$protobuf.util.ProtocolError} If required fields are missing\n */\n SecurePayload.decode = function decode(reader, length, error) {\n if (!(reader instanceof $Reader))\n reader = $Reader.create(reader);\n let end = length === undefined ? reader.len : reader.pos + length, message = new $root.secure.SecurePayload();\n while (reader.pos < end) {\n let tag = reader.uint32();\n if (tag === error)\n break;\n switch (tag >>> 3) {\n case 1: {\n message.ts = reader.int64();\n break;\n }\n case 2: {\n message.data = reader.bytes();\n break;\n }\n default:\n reader.skipType(tag & 7);\n break;\n }\n }\n return message;\n };\n\n /**\n * Decodes a SecurePayload message from the specified reader or buffer, length delimited.\n * @function decodeDelimited\n * @memberof secure.SecurePayload\n * @static\n * @param {$protobuf.Reader|Uint8Array} reader Reader or buffer to decode from\n * @returns {secure.SecurePayload} SecurePayload\n * @throws {Error} If the payload is not a reader or valid buffer\n * @throws {$protobuf.util.ProtocolError} If required fields are missing\n */\n SecurePayload.decodeDelimited = function decodeDelimited(reader) {\n if (!(reader instanceof $Reader))\n reader = new $Reader(reader);\n return this.decode(reader, reader.uint32());\n };\n\n /**\n * Verifies a SecurePayload message.\n * @function verify\n * @memberof secure.SecurePayload\n * @static\n * @param {Object.<string,*>} message Plain object to verify\n * @returns {string|null} `null` if valid, otherwise the reason why it is not\n */\n SecurePayload.verify = function verify(message) {\n if (typeof message !== \"object\" || message === null)\n return \"object expected\";\n if (message.ts != null && message.hasOwnProperty(\"ts\"))\n if (!$util.isInteger(message.ts) && !(message.ts && $util.isInteger(message.ts.low) && $util.isInteger(message.ts.high)))\n return \"ts: integer|Long expected\";\n if (message.data != null && message.hasOwnProperty(\"data\"))\n if (!(message.data && typeof message.data.length === \"number\" || $util.isString(message.data)))\n return \"data: buffer expected\";\n return null;\n };\n\n /**\n * Creates a SecurePayload message from a plain object. Also converts values to their respective internal types.\n * @function fromObject\n * @memberof secure.SecurePayload\n * @static\n * @param {Object.<string,*>} object Plain object\n * @returns {secure.SecurePayload} SecurePayload\n */\n SecurePayload.fromObject = function fromObject(object) {\n if (object instanceof $root.secure.SecurePayload)\n return object;\n let message = new $root.secure.SecurePayload();\n if (object.ts != null)\n if ($util.Long)\n (message.ts = $util.Long.fromValue(object.ts)).unsigned = false;\n else if (typeof object.ts === \"string\")\n message.ts = parseInt(object.ts, 10);\n else if (typeof object.ts === \"number\")\n message.ts = object.ts;\n else if (typeof object.ts === \"object\")\n message.ts = new $util.LongBits(object.ts.low >>> 0, object.ts.high >>> 0).toNumber();\n if (object.data != null)\n if (typeof object.data === \"string\")\n $util.base64.decode(object.data, message.data = $util.newBuffer($util.base64.length(object.data)), 0);\n else if (object.data.length >= 0)\n message.data = object.data;\n return message;\n };\n\n /**\n * Creates a plain object from a SecurePayload message. Also converts values to other types if specified.\n * @function toObject\n * @memberof secure.SecurePayload\n * @static\n * @param {secure.SecurePayload} message SecurePayload\n * @param {$protobuf.IConversionOptions} [options] Conversion options\n * @returns {Object.<string,*>} Plain object\n */\n SecurePayload.toObject = function toObject(message, options) {\n if (!options)\n options = {};\n let object = {};\n if (options.defaults) {\n if ($util.Long) {\n let long = new $util.Long(0, 0, false);\n object.ts = options.longs === String ? long.toString() : options.longs === Number ? long.toNumber() : long;\n } else\n object.ts = options.longs === String ? \"0\" : 0;\n if (options.bytes === String)\n object.data = \"\";\n else {\n object.data = [];\n if (options.bytes !== Array)\n object.data = $util.newBuffer(object.data);\n }\n }\n if (message.ts != null && message.hasOwnProperty(\"ts\"))\n if (typeof message.ts === \"number\")\n object.ts = options.longs === String ? String(message.ts) : message.ts;\n else\n object.ts = options.longs === String ? $util.Long.prototype.toString.call(message.ts) : options.longs === Number ? new $util.LongBits(message.ts.low >>> 0, message.ts.high >>> 0).toNumber() : message.ts;\n if (message.data != null && message.hasOwnProperty(\"data\"))\n object.data = options.bytes === String ? $util.base64.encode(message.data, 0, message.data.length) : options.bytes === Array ? Array.prototype.slice.call(message.data) : message.data;\n return object;\n };\n\n /**\n * Converts this SecurePayload to JSON.\n * @function toJSON\n * @memberof secure.SecurePayload\n * @instance\n * @returns {Object.<string,*>} JSON object\n */\n SecurePayload.prototype.toJSON = function toJSON() {\n return this.constructor.toObject(this, $protobuf.util.toJSONOptions);\n };\n\n /**\n * Gets the default type url for SecurePayload\n * @function getTypeUrl\n * @memberof secure.SecurePayload\n * @static\n * @param {string} [typeUrlPrefix] your custom typeUrlPrefix(default \"type.googleapis.com\")\n * @returns {string} The default type url\n */\n SecurePayload.getTypeUrl = function getTypeUrl(typeUrlPrefix) {\n if (typeUrlPrefix === undefined) {\n typeUrlPrefix = \"type.googleapis.com\";\n }\n return typeUrlPrefix + \"/secure.SecurePayload\";\n };\n\n return SecurePayload;\n })();\n\n return secure;\n})();\n\nexport { $root as default };\n","/**\n * Protobuf 安全通信模块\n *\n * 提供基于 Protobuf 的二进制序列化、XOR 混淆加密以及与 API Client 的无缝集成。\n */\n\nimport protobuf from \"protobufjs/light\";\nimport { secure } from \"./secure.js\";\nimport type { FetchContext } from \"ofetch\";\n\n// ============================================================================\n// 1. 类型定义 (Types)\n// ============================================================================\n\n/**\n * 基础编解码配置\n * 控制如何将数据转换为二进制,以及是否混淆\n */\nexport interface ProtobufCodecOptions {\n /** 是否启用二进制 XOR 混淆 (默认 true) */\n obfuscate?: boolean;\n /**\n * 混淆密钥\n * 可以是静态字符串/字节数组,也可以是动态获取密钥的同步或异步函数\n */\n obfuscationKey?:\n | string\n | Uint8Array\n | (() => string | Uint8Array | Promise<string | Uint8Array>);\n /** 预编译的 Protobuf 类型对象 (推荐,性能最高) */\n protoType?: any;\n /** 数据转换钩子:处理业务模型与 Proto 结构不一致的情况 */\n transform?: {\n beforeEncode?: (data: any) => any;\n afterDecode?: (payload: any) => any;\n };\n}\n\n/**\n * 完全自定义编解码接口\n * 用于外部项目完全接管序列化过程,同时复用基础库的混淆外壳\n */\nexport interface ProtobufCustomCodec {\n /** 外部定义的编码逻辑 (返回原始二进制) */\n encode?: (data: any) => Uint8Array | Promise<Uint8Array>;\n /** 外部定义的解码逻辑 (返回原始对象) */\n decode?: <T>(buffer: Uint8Array) => T | Promise<T>;\n}\n\n/**\n * 集成配置项\n * 用于 createProtobufHooks 或 API Client 全局配置\n */\nexport interface ProtobufHooksOptions\n extends ProtobufCodecOptions, ProtobufCustomCodec {}\n\n/**\n * API Client 内部使用的标准化配置\n */\nexport interface ProtobufConfig {\n encode: (data: any) => Uint8Array | Promise<Uint8Array>;\n decode: <T>(buffer: Uint8Array) => T | Promise<T>;\n options?: ProtobufCodecOptions;\n}\n\n// ============================================================================\n// 2. 内部工具函数 (Internal Utilities)\n// ============================================================================\n\nconst encoder = new TextEncoder();\nconst decoder = new TextDecoder();\n\n/** 简单的二进制异或混淆转换 */\nexport function xorTransform(\n data: Uint8Array,\n key: string | Uint8Array,\n): Uint8Array {\n const keyBytes = typeof key === \"string\" ? encoder.encode(key) : key;\n if (keyBytes.length === 0) return data;\n\n for (let i = 0; i < data.length; i++) {\n data[i] ^= keyBytes[i % keyBytes.length];\n }\n return data;\n}\n\n/** 获取混淆密钥(支持异步/函数) */\nexport async function getObfuscationKey(\n keyOption?: ProtobufCodecOptions[\"obfuscationKey\"],\n): Promise<string | Uint8Array> {\n const defaultKey = \"api-client-default-key\";\n if (!keyOption) return defaultKey;\n if (typeof keyOption === \"function\") {\n return await keyOption();\n }\n return keyOption;\n}\n\n/** 获取内置的 SecurePayload 类型(使用预编译代码) */\nfunction getInternalSecureType(): protobuf.Type {\n return secure.SecurePayload as unknown as protobuf.Type;\n}\n\n// ============================================================================\n// 3. 核心编解码逻辑 (Core Codecs)\n// ============================================================================\n\n/**\n * 编码安全载荷\n *\n * 流程:业务数据 -> (自定义编码 / Proto 序列化) -> 二进制混淆\n */\nexport async function encodeSecure<T>(\n data: T,\n options: ProtobufHooksOptions = {},\n): Promise<Uint8Array> {\n const {\n obfuscate = true,\n obfuscationKey,\n protoType,\n transform,\n encode: customEncode,\n } = options;\n\n let buffer: Uint8Array;\n\n // 1. 预处理阶段:无论后续走哪条路径,只要定义了 beforeEncode 就先执行\n const processedData = transform?.beforeEncode\n ? transform.beforeEncode(data)\n : data;\n\n // 1. 序列化阶段\n if (customEncode) {\n buffer = await customEncode(data);\n } else {\n const type = protoType || getInternalSecureType();\n\n // 构造最终要交给 Protobuf 序列化的对象\n const payload = protoType\n ? processedData // 自定义模式:直接使用预处理后的数据\n : {\n // 默认容器模式:将预处理后的数据封装进信封\n ts: Date.now(),\n data:\n processedData instanceof Uint8Array\n ? processedData\n : encoder.encode(JSON.stringify(processedData)),\n };\n\n const message = type.create(payload);\n buffer = type.encode(message).finish();\n }\n\n // 2. 混淆阶段\n if (!obfuscate) return buffer;\n const finalKey = await getObfuscationKey(obfuscationKey);\n return xorTransform(buffer, finalKey);\n}\n\n/**\n * 解码安全载荷\n *\n * 流程:二进制流 -> 二进制反混淆 -> (自定义解码 / Proto 反序列化) -> 业务数据\n */\nexport async function decodeSecure<T>(\n buffer: Uint8Array | ArrayBuffer,\n options: ProtobufHooksOptions = {},\n): Promise<T> {\n const {\n obfuscate = true,\n obfuscationKey,\n protoType,\n transform,\n decode: customDecode,\n } = options;\n\n // let uint8 = buffer instanceof ArrayBuffer ? new Uint8Array(buffer) : buffer;\n let uint8 =\n buffer instanceof ArrayBuffer\n ? new Uint8Array(buffer)\n : new Uint8Array(buffer.buffer, buffer.byteOffset, buffer.byteLength);\n\n // 1. 反混淆阶段\n if (uint8.length > 0 && obfuscate) {\n const finalKey = await getObfuscationKey(obfuscationKey);\n uint8 = xorTransform(uint8, finalKey);\n }\n\n // 2. 反序列化阶段\n if (customDecode) {\n return await customDecode<T>(uint8);\n }\n\n const type = protoType || getInternalSecureType();\n const decoded = type.decode(uint8);\n const plainObj = type.toObject(decoded, {\n longs: String,\n enums: String,\n bytes: Uint8Array as any,\n defaults: true,\n });\n\n // 3. 转换阶段\n if (transform?.afterDecode) {\n return transform.afterDecode(plainObj);\n }\n\n // 内置容器模式的额外还原逻辑\n if (!protoType && plainObj.data) {\n const jsonString = decoder.decode(plainObj.data as Uint8Array);\n try {\n return JSON.parse(jsonString) as T;\n } catch {\n return jsonString as unknown as T;\n }\n }\n\n return plainObj as T;\n}\n\n// ============================================================================\n// 4. HTTP/环境适配工具 (HTTP Tools)\n// ============================================================================\n\n/** 创建安全响应 (Worker/Server 端使用) */\nexport async function secureResponse<T>(\n data: T,\n options?: ProtobufCodecOptions & { corsHeaders?: Record<string, string> },\n): Promise<Response> {\n const buffer = await encodeSecure(data, options);\n const arrayBuffer = buffer.buffer.slice(\n buffer.byteOffset,\n buffer.byteOffset + buffer.byteLength,\n ) as ArrayBuffer;\n\n return new Response(arrayBuffer, {\n headers: {\n \"Content-Type\": \"application/x-protobuf\",\n ...options?.corsHeaders,\n },\n });\n}\n\n/** 解析安全请求体 (Worker/Server 端使用) */\nexport async function parseSecureRequest<T>(\n request: Request,\n customDecode?: ProtobufCustomCodec[\"decode\"],\n): Promise<T> {\n const buffer = await request.arrayBuffer();\n return decodeSecure<T>(buffer, { decode: customDecode });\n}\n\n/** 检查是否为 Protobuf 响应头 */\nexport function isProtobufContentType(contentType: string | null): boolean {\n return !!contentType?.includes(\"application/x-protobuf\");\n}\n\n/** 标准化配置对象 */\nexport function normalizeProtobufConfig(\n config: boolean | ProtobufConfig | undefined,\n options?: ProtobufHooksOptions, // 接收全局配置\n): ProtobufConfig | null {\n if (!config) return null;\n if (config === true) {\n return {\n encode: (data) => encodeSecure(data, options),\n decode: <T>(buffer: Uint8Array) => decodeSecure<T>(buffer, options),\n };\n }\n return config;\n}\n\n// ============================================================================\n// 5. API Client 钩子逻辑 (Integration Hooks)\n// ============================================================================\n\n/** 创建 Protobuf 编解码钩子 (用于与 createApiClient 配合使用) */\nexport function createProtobufHooks(options: ProtobufHooksOptions = {}) {\n const encode = (data: any) => encodeSecure(data, options);\n const decode = <T>(buffer: Uint8Array) => decodeSecure<T>(buffer, options);\n\n return {\n async onRequest(context: FetchContext) {\n const { options: reqOptions } = context;\n const headers =\n reqOptions.headers instanceof Headers\n ? reqOptions.headers\n : new Headers(reqOptions.headers as HeadersInit | undefined);\n\n // 自动编码请求体\n if (\n isProtobufContentType(headers.get(\"Content-Type\")) &&\n reqOptions.body &&\n !(reqOptions.body instanceof Uint8Array)\n ) {\n reqOptions.body = await encode(reqOptions.body);\n }\n\n // 自动设置期望响应类型\n if (\n isProtobufContentType(headers.get(\"Accept\")) &&\n !reqOptions.responseType\n ) {\n reqOptions.responseType = \"arrayBuffer\";\n }\n\n reqOptions.headers = headers;\n },\n\n async onResponse(context: FetchContext) {\n const { response } = context;\n if (!response?._data) return;\n\n if (isProtobufContentType(response.headers.get(\"Content-Type\"))) {\n const buffer = response._data as ArrayBuffer;\n if (buffer && buffer.byteLength > 0) {\n response._data = await decode(new Uint8Array(buffer));\n }\n } else if (response._data instanceof ArrayBuffer) {\n const text = decoder.decode(response._data);\n try {\n response._data = JSON.parse(text);\n } catch {\n response._data = text;\n }\n }\n },\n };\n}\n"],"mappings":"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","names":["options","secure"]}
|