@i.un/api-client 1.0.6 → 1.0.8
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 +128 -4
- package/dist/index.d.ts +128 -4
- package/dist/index.js +180 -9
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +162 -8
- package/dist/index.mjs.map +1 -1
- package/package.json +4 -3
package/dist/index.d.mts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { $Fetch, FetchOptions } from 'ofetch';
|
|
1
|
+
import { FetchContext, $Fetch, FetchOptions } from 'ofetch';
|
|
2
2
|
|
|
3
3
|
interface ApiResult<T> {
|
|
4
4
|
code: number;
|
|
@@ -24,9 +24,20 @@ interface CreateApiClientOptions {
|
|
|
24
24
|
isAuthError?: (code: number) => boolean;
|
|
25
25
|
unwrapResponse?<T>(result: unknown, returnFullResponse: boolean): T;
|
|
26
26
|
createErrorFromResult?(res: unknown): Error;
|
|
27
|
+
/**
|
|
28
|
+
* 自定义请求钩子
|
|
29
|
+
* 在请求发送前执行,可用于编码请求体等
|
|
30
|
+
*/
|
|
31
|
+
onRequest?: (context: FetchContext) => void | Promise<void>;
|
|
32
|
+
/**
|
|
33
|
+
* 自定义响应钩子
|
|
34
|
+
* 在响应返回后执行,可用于解码响应体等
|
|
35
|
+
*/
|
|
36
|
+
onResponse?: (context: FetchContext) => void | Promise<void>;
|
|
27
37
|
}
|
|
28
|
-
type RequestOptions = FetchOptions<"json"> & {
|
|
38
|
+
type RequestOptions = Omit<FetchOptions<"json">, "responseType"> & {
|
|
29
39
|
returnFullResponse?: boolean;
|
|
40
|
+
responseType?: "json" | "arrayBuffer" | "text" | "blob";
|
|
30
41
|
};
|
|
31
42
|
declare function createApiClient(options: CreateApiClientOptions): {
|
|
32
43
|
rawRequest: $Fetch;
|
|
@@ -122,6 +133,119 @@ interface ChainRequestRule {
|
|
|
122
133
|
* @param requests 链式执行规则列表
|
|
123
134
|
* @param handlers 网络适配器处理器列表 (按顺序匹配)
|
|
124
135
|
*/
|
|
125
|
-
declare function executeRequestChain<T>(requests: ChainRequestRule[], handlers?: NetworkHandler[], getChainRequests?: (context: Record<string, any>) => ChainRequestRule[] | null | undefined): Promise<T>;
|
|
136
|
+
declare function executeRequestChain<T>(requests: ChainRequestRule[], handlers?: NetworkHandler[], getChainRequests?: (context: Record<string, any>) => ChainRequestRule[] | null | undefined, initContext?: Record<string, any>): Promise<T>;
|
|
137
|
+
|
|
138
|
+
/**
|
|
139
|
+
* Protobuf 通用编解码模块
|
|
140
|
+
*
|
|
141
|
+
* 特性:
|
|
142
|
+
* - 支持加盐混淆(每次编码结果不同)
|
|
143
|
+
* - 直接输出二进制格式
|
|
144
|
+
* - 内置通用容器 schema(任意 JSON 数据)
|
|
145
|
+
* - 支持自定义编解码函数
|
|
146
|
+
*/
|
|
147
|
+
|
|
148
|
+
interface ProtobufCodecOptions {
|
|
149
|
+
/** 是否启用加盐 (默认 true) */
|
|
150
|
+
useSalt?: boolean;
|
|
151
|
+
/** 自定义盐值生成器 */
|
|
152
|
+
saltGenerator?: () => string;
|
|
153
|
+
}
|
|
154
|
+
interface ProtobufConfig {
|
|
155
|
+
/** 自定义编码函数 */
|
|
156
|
+
encode?: (data: unknown) => Uint8Array | Promise<Uint8Array>;
|
|
157
|
+
/** 自定义解码函数 */
|
|
158
|
+
decode?: <T>(buffer: Uint8Array) => T | Promise<T>;
|
|
159
|
+
/** 编码选项(仅内置编解码器使用) */
|
|
160
|
+
options?: ProtobufCodecOptions;
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* 使用内置 SaltedPayload schema 编码数据
|
|
164
|
+
*
|
|
165
|
+
* @param data - 任意 JSON 可序列化数据
|
|
166
|
+
* @param options - 编码选项
|
|
167
|
+
* @returns Uint8Array 二进制数据
|
|
168
|
+
*/
|
|
169
|
+
declare function encodeProtobuf<T>(data: T, options?: ProtobufCodecOptions): Uint8Array;
|
|
170
|
+
/**
|
|
171
|
+
* 使用内置 SaltedPayload schema 解码数据
|
|
172
|
+
*
|
|
173
|
+
* @param buffer - 二进制数据
|
|
174
|
+
* @returns 解码后的原始数据
|
|
175
|
+
*/
|
|
176
|
+
declare function decodeProtobuf<T>(buffer: Uint8Array | ArrayBuffer): T;
|
|
177
|
+
/**
|
|
178
|
+
* 创建 Protobuf 二进制响应
|
|
179
|
+
*
|
|
180
|
+
* @param data - 要编码的数据
|
|
181
|
+
* @param options - 编码选项和 CORS headers
|
|
182
|
+
* @returns Response 对象
|
|
183
|
+
*/
|
|
184
|
+
declare function protobufResponse<T>(data: T, options?: ProtobufCodecOptions & {
|
|
185
|
+
corsHeaders?: Record<string, string>;
|
|
186
|
+
}): Response;
|
|
187
|
+
/**
|
|
188
|
+
* 检查 Content-Type 是否是 Protobuf 格式
|
|
189
|
+
*/
|
|
190
|
+
declare function isProtobufContentType(contentType: string | null): boolean;
|
|
191
|
+
/**
|
|
192
|
+
* 规范化 protobuf 配置
|
|
193
|
+
* 将 boolean | ProtobufConfig 统一为 ProtobufConfig 或 null
|
|
194
|
+
*/
|
|
195
|
+
declare function normalizeProtobufConfig(config: boolean | ProtobufConfig | undefined): ProtobufConfig | null;
|
|
196
|
+
/**
|
|
197
|
+
* 解析 Protobuf 请求体 (Worker/Server 端使用)
|
|
198
|
+
*
|
|
199
|
+
* @param request - Request 对象
|
|
200
|
+
* @param customDecode - 可选的自定义解码函数
|
|
201
|
+
* @returns 解码后的数据
|
|
202
|
+
*/
|
|
203
|
+
declare function parseProtobufRequest<T>(request: Request, customDecode?: (buffer: Uint8Array) => T | Promise<T>): Promise<T>;
|
|
204
|
+
/**
|
|
205
|
+
* Protobuf 钩子选项
|
|
206
|
+
*/
|
|
207
|
+
interface CreateProtobufHooksOptions extends ProtobufCodecOptions {
|
|
208
|
+
/** 自定义编码函数 */
|
|
209
|
+
encode?: (data: unknown) => Uint8Array | Promise<Uint8Array>;
|
|
210
|
+
/** 自定义解码函数 */
|
|
211
|
+
decode?: <T>(buffer: Uint8Array) => T | Promise<T>;
|
|
212
|
+
}
|
|
213
|
+
/**
|
|
214
|
+
* 创建 Protobuf 编解码钩子
|
|
215
|
+
* 用于与 createApiClient 配合使用
|
|
216
|
+
*
|
|
217
|
+
* 钩子会自动:
|
|
218
|
+
* - 设置 responseType 为 arrayBuffer
|
|
219
|
+
* - 编码请求体为 Protobuf 格式
|
|
220
|
+
* - 根据响应 Content-Type 自动解码(protobuf 或 JSON)
|
|
221
|
+
*
|
|
222
|
+
* @example
|
|
223
|
+
* ```ts
|
|
224
|
+
* import { createApiClient } from '@i.un/api-client';
|
|
225
|
+
* import { createProtobufHooks } from '@i.un/api-client/protobuf';
|
|
226
|
+
*
|
|
227
|
+
* const { onRequest, onResponse } = createProtobufHooks();
|
|
228
|
+
*
|
|
229
|
+
* const client = createApiClient({
|
|
230
|
+
* baseURL: 'https://api.example.com',
|
|
231
|
+
* tokenStorage,
|
|
232
|
+
* onRequest,
|
|
233
|
+
* onResponse,
|
|
234
|
+
* });
|
|
235
|
+
*
|
|
236
|
+
* // 自动处理,无需手动指定 responseType
|
|
237
|
+
* const data = await client.post('/api/data', body);
|
|
238
|
+
* ```
|
|
239
|
+
*/
|
|
240
|
+
declare function createProtobufHooks(options?: CreateProtobufHooksOptions): {
|
|
241
|
+
/**
|
|
242
|
+
* 请求钩子:独立处理请求编码和响应类型设置
|
|
243
|
+
*/
|
|
244
|
+
onRequest(context: FetchContext): Promise<void>;
|
|
245
|
+
/**
|
|
246
|
+
* 响应钩子:解码 Protobuf 响应
|
|
247
|
+
*/
|
|
248
|
+
onResponse(context: FetchContext): Promise<void>;
|
|
249
|
+
};
|
|
126
250
|
|
|
127
|
-
export { type ApiError, type ApiResult, type ChainRequestRule, type CreateApiClientOptions, type HttpRequestSpec, type NetworkAdapter, type NetworkHandler, type TokenStorage, createApiClient, executeRequestChain, isApiError };
|
|
251
|
+
export { type ApiError, type ApiResult, type ChainRequestRule, type CreateApiClientOptions, type CreateProtobufHooksOptions, type HttpRequestSpec, type NetworkAdapter, type NetworkHandler, type ProtobufCodecOptions, type ProtobufConfig, type TokenStorage, createApiClient, createProtobufHooks, decodeProtobuf, encodeProtobuf, executeRequestChain, isApiError, isProtobufContentType, normalizeProtobufConfig, parseProtobufRequest, protobufResponse };
|
package/dist/index.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { $Fetch, FetchOptions } from 'ofetch';
|
|
1
|
+
import { FetchContext, $Fetch, FetchOptions } from 'ofetch';
|
|
2
2
|
|
|
3
3
|
interface ApiResult<T> {
|
|
4
4
|
code: number;
|
|
@@ -24,9 +24,20 @@ interface CreateApiClientOptions {
|
|
|
24
24
|
isAuthError?: (code: number) => boolean;
|
|
25
25
|
unwrapResponse?<T>(result: unknown, returnFullResponse: boolean): T;
|
|
26
26
|
createErrorFromResult?(res: unknown): Error;
|
|
27
|
+
/**
|
|
28
|
+
* 自定义请求钩子
|
|
29
|
+
* 在请求发送前执行,可用于编码请求体等
|
|
30
|
+
*/
|
|
31
|
+
onRequest?: (context: FetchContext) => void | Promise<void>;
|
|
32
|
+
/**
|
|
33
|
+
* 自定义响应钩子
|
|
34
|
+
* 在响应返回后执行,可用于解码响应体等
|
|
35
|
+
*/
|
|
36
|
+
onResponse?: (context: FetchContext) => void | Promise<void>;
|
|
27
37
|
}
|
|
28
|
-
type RequestOptions = FetchOptions<"json"> & {
|
|
38
|
+
type RequestOptions = Omit<FetchOptions<"json">, "responseType"> & {
|
|
29
39
|
returnFullResponse?: boolean;
|
|
40
|
+
responseType?: "json" | "arrayBuffer" | "text" | "blob";
|
|
30
41
|
};
|
|
31
42
|
declare function createApiClient(options: CreateApiClientOptions): {
|
|
32
43
|
rawRequest: $Fetch;
|
|
@@ -122,6 +133,119 @@ interface ChainRequestRule {
|
|
|
122
133
|
* @param requests 链式执行规则列表
|
|
123
134
|
* @param handlers 网络适配器处理器列表 (按顺序匹配)
|
|
124
135
|
*/
|
|
125
|
-
declare function executeRequestChain<T>(requests: ChainRequestRule[], handlers?: NetworkHandler[], getChainRequests?: (context: Record<string, any>) => ChainRequestRule[] | null | undefined): Promise<T>;
|
|
136
|
+
declare function executeRequestChain<T>(requests: ChainRequestRule[], handlers?: NetworkHandler[], getChainRequests?: (context: Record<string, any>) => ChainRequestRule[] | null | undefined, initContext?: Record<string, any>): Promise<T>;
|
|
137
|
+
|
|
138
|
+
/**
|
|
139
|
+
* Protobuf 通用编解码模块
|
|
140
|
+
*
|
|
141
|
+
* 特性:
|
|
142
|
+
* - 支持加盐混淆(每次编码结果不同)
|
|
143
|
+
* - 直接输出二进制格式
|
|
144
|
+
* - 内置通用容器 schema(任意 JSON 数据)
|
|
145
|
+
* - 支持自定义编解码函数
|
|
146
|
+
*/
|
|
147
|
+
|
|
148
|
+
interface ProtobufCodecOptions {
|
|
149
|
+
/** 是否启用加盐 (默认 true) */
|
|
150
|
+
useSalt?: boolean;
|
|
151
|
+
/** 自定义盐值生成器 */
|
|
152
|
+
saltGenerator?: () => string;
|
|
153
|
+
}
|
|
154
|
+
interface ProtobufConfig {
|
|
155
|
+
/** 自定义编码函数 */
|
|
156
|
+
encode?: (data: unknown) => Uint8Array | Promise<Uint8Array>;
|
|
157
|
+
/** 自定义解码函数 */
|
|
158
|
+
decode?: <T>(buffer: Uint8Array) => T | Promise<T>;
|
|
159
|
+
/** 编码选项(仅内置编解码器使用) */
|
|
160
|
+
options?: ProtobufCodecOptions;
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* 使用内置 SaltedPayload schema 编码数据
|
|
164
|
+
*
|
|
165
|
+
* @param data - 任意 JSON 可序列化数据
|
|
166
|
+
* @param options - 编码选项
|
|
167
|
+
* @returns Uint8Array 二进制数据
|
|
168
|
+
*/
|
|
169
|
+
declare function encodeProtobuf<T>(data: T, options?: ProtobufCodecOptions): Uint8Array;
|
|
170
|
+
/**
|
|
171
|
+
* 使用内置 SaltedPayload schema 解码数据
|
|
172
|
+
*
|
|
173
|
+
* @param buffer - 二进制数据
|
|
174
|
+
* @returns 解码后的原始数据
|
|
175
|
+
*/
|
|
176
|
+
declare function decodeProtobuf<T>(buffer: Uint8Array | ArrayBuffer): T;
|
|
177
|
+
/**
|
|
178
|
+
* 创建 Protobuf 二进制响应
|
|
179
|
+
*
|
|
180
|
+
* @param data - 要编码的数据
|
|
181
|
+
* @param options - 编码选项和 CORS headers
|
|
182
|
+
* @returns Response 对象
|
|
183
|
+
*/
|
|
184
|
+
declare function protobufResponse<T>(data: T, options?: ProtobufCodecOptions & {
|
|
185
|
+
corsHeaders?: Record<string, string>;
|
|
186
|
+
}): Response;
|
|
187
|
+
/**
|
|
188
|
+
* 检查 Content-Type 是否是 Protobuf 格式
|
|
189
|
+
*/
|
|
190
|
+
declare function isProtobufContentType(contentType: string | null): boolean;
|
|
191
|
+
/**
|
|
192
|
+
* 规范化 protobuf 配置
|
|
193
|
+
* 将 boolean | ProtobufConfig 统一为 ProtobufConfig 或 null
|
|
194
|
+
*/
|
|
195
|
+
declare function normalizeProtobufConfig(config: boolean | ProtobufConfig | undefined): ProtobufConfig | null;
|
|
196
|
+
/**
|
|
197
|
+
* 解析 Protobuf 请求体 (Worker/Server 端使用)
|
|
198
|
+
*
|
|
199
|
+
* @param request - Request 对象
|
|
200
|
+
* @param customDecode - 可选的自定义解码函数
|
|
201
|
+
* @returns 解码后的数据
|
|
202
|
+
*/
|
|
203
|
+
declare function parseProtobufRequest<T>(request: Request, customDecode?: (buffer: Uint8Array) => T | Promise<T>): Promise<T>;
|
|
204
|
+
/**
|
|
205
|
+
* Protobuf 钩子选项
|
|
206
|
+
*/
|
|
207
|
+
interface CreateProtobufHooksOptions extends ProtobufCodecOptions {
|
|
208
|
+
/** 自定义编码函数 */
|
|
209
|
+
encode?: (data: unknown) => Uint8Array | Promise<Uint8Array>;
|
|
210
|
+
/** 自定义解码函数 */
|
|
211
|
+
decode?: <T>(buffer: Uint8Array) => T | Promise<T>;
|
|
212
|
+
}
|
|
213
|
+
/**
|
|
214
|
+
* 创建 Protobuf 编解码钩子
|
|
215
|
+
* 用于与 createApiClient 配合使用
|
|
216
|
+
*
|
|
217
|
+
* 钩子会自动:
|
|
218
|
+
* - 设置 responseType 为 arrayBuffer
|
|
219
|
+
* - 编码请求体为 Protobuf 格式
|
|
220
|
+
* - 根据响应 Content-Type 自动解码(protobuf 或 JSON)
|
|
221
|
+
*
|
|
222
|
+
* @example
|
|
223
|
+
* ```ts
|
|
224
|
+
* import { createApiClient } from '@i.un/api-client';
|
|
225
|
+
* import { createProtobufHooks } from '@i.un/api-client/protobuf';
|
|
226
|
+
*
|
|
227
|
+
* const { onRequest, onResponse } = createProtobufHooks();
|
|
228
|
+
*
|
|
229
|
+
* const client = createApiClient({
|
|
230
|
+
* baseURL: 'https://api.example.com',
|
|
231
|
+
* tokenStorage,
|
|
232
|
+
* onRequest,
|
|
233
|
+
* onResponse,
|
|
234
|
+
* });
|
|
235
|
+
*
|
|
236
|
+
* // 自动处理,无需手动指定 responseType
|
|
237
|
+
* const data = await client.post('/api/data', body);
|
|
238
|
+
* ```
|
|
239
|
+
*/
|
|
240
|
+
declare function createProtobufHooks(options?: CreateProtobufHooksOptions): {
|
|
241
|
+
/**
|
|
242
|
+
* 请求钩子:独立处理请求编码和响应类型设置
|
|
243
|
+
*/
|
|
244
|
+
onRequest(context: FetchContext): Promise<void>;
|
|
245
|
+
/**
|
|
246
|
+
* 响应钩子:解码 Protobuf 响应
|
|
247
|
+
*/
|
|
248
|
+
onResponse(context: FetchContext): Promise<void>;
|
|
249
|
+
};
|
|
126
250
|
|
|
127
|
-
export { type ApiError, type ApiResult, type ChainRequestRule, type CreateApiClientOptions, type HttpRequestSpec, type NetworkAdapter, type NetworkHandler, type TokenStorage, createApiClient, executeRequestChain, isApiError };
|
|
251
|
+
export { type ApiError, type ApiResult, type ChainRequestRule, type CreateApiClientOptions, type CreateProtobufHooksOptions, type HttpRequestSpec, type NetworkAdapter, type NetworkHandler, type ProtobufCodecOptions, type ProtobufConfig, type TokenStorage, createApiClient, createProtobufHooks, decodeProtobuf, encodeProtobuf, executeRequestChain, isApiError, isProtobufContentType, normalizeProtobufConfig, parseProtobufRequest, protobufResponse };
|
package/dist/index.js
CHANGED
|
@@ -1,7 +1,9 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
+
var __create = Object.create;
|
|
2
3
|
var __defProp = Object.defineProperty;
|
|
3
4
|
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
5
|
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
5
7
|
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
8
|
var __export = (target, all) => {
|
|
7
9
|
for (var name in all)
|
|
@@ -15,14 +17,29 @@ var __copyProps = (to, from, except, desc) => {
|
|
|
15
17
|
}
|
|
16
18
|
return to;
|
|
17
19
|
};
|
|
20
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
21
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
|
22
|
+
// file that has been converted to a CommonJS file using a Babel-
|
|
23
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
24
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
25
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
26
|
+
mod
|
|
27
|
+
));
|
|
18
28
|
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
19
29
|
|
|
20
30
|
// src/index.ts
|
|
21
31
|
var index_exports = {};
|
|
22
32
|
__export(index_exports, {
|
|
23
33
|
createApiClient: () => createApiClient,
|
|
34
|
+
createProtobufHooks: () => createProtobufHooks,
|
|
35
|
+
decodeProtobuf: () => decodeProtobuf,
|
|
36
|
+
encodeProtobuf: () => encodeProtobuf,
|
|
24
37
|
executeRequestChain: () => executeRequestChain,
|
|
25
|
-
isApiError: () => isApiError
|
|
38
|
+
isApiError: () => isApiError,
|
|
39
|
+
isProtobufContentType: () => isProtobufContentType,
|
|
40
|
+
normalizeProtobufConfig: () => normalizeProtobufConfig,
|
|
41
|
+
parseProtobufRequest: () => parseProtobufRequest,
|
|
42
|
+
protobufResponse: () => protobufResponse
|
|
26
43
|
});
|
|
27
44
|
module.exports = __toCommonJS(index_exports);
|
|
28
45
|
|
|
@@ -92,17 +109,28 @@ function createApiClient(options) {
|
|
|
92
109
|
baseURL,
|
|
93
110
|
retry,
|
|
94
111
|
retryDelay,
|
|
95
|
-
async onRequest(
|
|
112
|
+
async onRequest(context) {
|
|
113
|
+
const { options: reqOptions } = context;
|
|
96
114
|
const token = await tokenStorage.getAccessToken();
|
|
97
|
-
const headers = new Headers(
|
|
115
|
+
const headers = new Headers(
|
|
116
|
+
reqOptions.headers
|
|
117
|
+
);
|
|
98
118
|
if (token) {
|
|
99
119
|
headers.set("Authorization", `Bearer ${token}`);
|
|
100
120
|
}
|
|
101
|
-
|
|
121
|
+
reqOptions.headers = headers;
|
|
122
|
+
if (options.onRequest) {
|
|
123
|
+
await options.onRequest(context);
|
|
124
|
+
}
|
|
102
125
|
},
|
|
103
|
-
onResponse(
|
|
126
|
+
async onResponse(context) {
|
|
127
|
+
const { response } = context;
|
|
104
128
|
if (response.status === 204) {
|
|
105
129
|
response._data = null;
|
|
130
|
+
return;
|
|
131
|
+
}
|
|
132
|
+
if (options.onResponse) {
|
|
133
|
+
await options.onResponse(context);
|
|
106
134
|
}
|
|
107
135
|
},
|
|
108
136
|
async onResponseError(context) {
|
|
@@ -274,8 +302,8 @@ function substituteVariables(target, context) {
|
|
|
274
302
|
}
|
|
275
303
|
return target;
|
|
276
304
|
}
|
|
277
|
-
async function executeRequestChain(requests, handlers = [], getChainRequests) {
|
|
278
|
-
const context = {};
|
|
305
|
+
async function executeRequestChain(requests, handlers = [], getChainRequests, initContext) {
|
|
306
|
+
const context = initContext || {};
|
|
279
307
|
let lastResult = null;
|
|
280
308
|
for (const rule of requests) {
|
|
281
309
|
try {
|
|
@@ -304,7 +332,8 @@ async function executeRequestChain(requests, handlers = [], getChainRequests) {
|
|
|
304
332
|
rawData = await executeRequestChain(
|
|
305
333
|
chainRequests,
|
|
306
334
|
handlers,
|
|
307
|
-
getChainRequests
|
|
335
|
+
getChainRequests,
|
|
336
|
+
JSON.parse(JSON.stringify(context))
|
|
308
337
|
);
|
|
309
338
|
}
|
|
310
339
|
}
|
|
@@ -328,10 +357,152 @@ async function executeRequestChain(requests, handlers = [], getChainRequests) {
|
|
|
328
357
|
}
|
|
329
358
|
return lastResult;
|
|
330
359
|
}
|
|
360
|
+
|
|
361
|
+
// src/protobuf.ts
|
|
362
|
+
var import_protobufjs = __toESM(require("protobufjs"));
|
|
363
|
+
var SALTED_SCHEMA = `
|
|
364
|
+
syntax = "proto3";
|
|
365
|
+
|
|
366
|
+
message SaltedPayload {
|
|
367
|
+
string salt = 1;
|
|
368
|
+
int64 timestamp = 2;
|
|
369
|
+
bytes data = 3;
|
|
370
|
+
}
|
|
371
|
+
`;
|
|
372
|
+
var encoder = new TextEncoder();
|
|
373
|
+
var decoder = new TextDecoder();
|
|
374
|
+
var saltedPayloadType = null;
|
|
375
|
+
function getSaltedPayloadType() {
|
|
376
|
+
if (!saltedPayloadType) {
|
|
377
|
+
const root = import_protobufjs.default.parse(SALTED_SCHEMA).root;
|
|
378
|
+
saltedPayloadType = root.lookupType("SaltedPayload");
|
|
379
|
+
}
|
|
380
|
+
return saltedPayloadType;
|
|
381
|
+
}
|
|
382
|
+
function generateSalt() {
|
|
383
|
+
if (typeof crypto !== "undefined" && crypto.randomUUID) {
|
|
384
|
+
return crypto.randomUUID();
|
|
385
|
+
}
|
|
386
|
+
return Math.random().toString(36).substring(2) + Date.now().toString(36);
|
|
387
|
+
}
|
|
388
|
+
function encodeProtobuf(data, options = {}) {
|
|
389
|
+
const { useSalt = true, saltGenerator = generateSalt } = options;
|
|
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();
|
|
400
|
+
}
|
|
401
|
+
function decodeProtobuf(buffer) {
|
|
402
|
+
const type = getSaltedPayloadType();
|
|
403
|
+
const uint8 = buffer instanceof ArrayBuffer ? new Uint8Array(buffer) : buffer;
|
|
404
|
+
const decoded = type.decode(uint8);
|
|
405
|
+
const jsonString = decoder.decode(decoded.data);
|
|
406
|
+
return JSON.parse(jsonString);
|
|
407
|
+
}
|
|
408
|
+
function protobufResponse(data, options) {
|
|
409
|
+
const buffer = encodeProtobuf(data, options);
|
|
410
|
+
const arrayBuffer = buffer.buffer.slice(
|
|
411
|
+
buffer.byteOffset,
|
|
412
|
+
buffer.byteOffset + buffer.byteLength
|
|
413
|
+
);
|
|
414
|
+
return new Response(arrayBuffer, {
|
|
415
|
+
headers: {
|
|
416
|
+
"Content-Type": "application/x-protobuf",
|
|
417
|
+
...options?.corsHeaders
|
|
418
|
+
}
|
|
419
|
+
});
|
|
420
|
+
}
|
|
421
|
+
function isProtobufContentType(contentType) {
|
|
422
|
+
return !!contentType?.includes("application/x-protobuf");
|
|
423
|
+
}
|
|
424
|
+
function normalizeProtobufConfig(config) {
|
|
425
|
+
if (!config) {
|
|
426
|
+
return null;
|
|
427
|
+
}
|
|
428
|
+
if (config === true) {
|
|
429
|
+
return {
|
|
430
|
+
encode: (data) => encodeProtobuf(data),
|
|
431
|
+
decode: (buffer) => decodeProtobuf(buffer)
|
|
432
|
+
};
|
|
433
|
+
}
|
|
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);
|
|
446
|
+
}
|
|
447
|
+
function createProtobufHooks(options = {}) {
|
|
448
|
+
const encode = options.encode || ((data) => encodeProtobuf(data, options));
|
|
449
|
+
const decode = options.decode || ((buffer) => decodeProtobuf(buffer));
|
|
450
|
+
return {
|
|
451
|
+
/**
|
|
452
|
+
* 请求钩子:独立处理请求编码和响应类型设置
|
|
453
|
+
*/
|
|
454
|
+
async onRequest(context) {
|
|
455
|
+
const { options: reqOptions } = context;
|
|
456
|
+
const headers = reqOptions.headers instanceof Headers ? reqOptions.headers : new Headers(reqOptions.headers);
|
|
457
|
+
const contentType = headers.get("Content-Type");
|
|
458
|
+
const isProtobufRequest = isProtobufContentType(contentType);
|
|
459
|
+
if (isProtobufRequest && reqOptions.body && !(reqOptions.body instanceof Uint8Array)) {
|
|
460
|
+
const encoded = await encode(reqOptions.body);
|
|
461
|
+
reqOptions.body = encoded;
|
|
462
|
+
}
|
|
463
|
+
const accept = headers.get("Accept");
|
|
464
|
+
const prefersProtobufResponse = isProtobufContentType(accept);
|
|
465
|
+
if (prefersProtobufResponse && !reqOptions.responseType) {
|
|
466
|
+
reqOptions.responseType = "arrayBuffer";
|
|
467
|
+
}
|
|
468
|
+
reqOptions.headers = headers;
|
|
469
|
+
},
|
|
470
|
+
/**
|
|
471
|
+
* 响应钩子:解码 Protobuf 响应
|
|
472
|
+
*/
|
|
473
|
+
async onResponse(context) {
|
|
474
|
+
const { response } = context;
|
|
475
|
+
if (!response?._data) {
|
|
476
|
+
return;
|
|
477
|
+
}
|
|
478
|
+
const contentType = response.headers.get("Content-Type");
|
|
479
|
+
if (isProtobufContentType(contentType)) {
|
|
480
|
+
const buffer = response._data;
|
|
481
|
+
if (buffer && buffer.byteLength > 0) {
|
|
482
|
+
response._data = await decode(new Uint8Array(buffer));
|
|
483
|
+
}
|
|
484
|
+
} else if (response._data instanceof ArrayBuffer) {
|
|
485
|
+
const text = decoder.decode(response._data);
|
|
486
|
+
try {
|
|
487
|
+
response._data = JSON.parse(text);
|
|
488
|
+
} catch {
|
|
489
|
+
response._data = text;
|
|
490
|
+
}
|
|
491
|
+
}
|
|
492
|
+
}
|
|
493
|
+
};
|
|
494
|
+
}
|
|
331
495
|
// Annotate the CommonJS export names for ESM import in node:
|
|
332
496
|
0 && (module.exports = {
|
|
333
497
|
createApiClient,
|
|
498
|
+
createProtobufHooks,
|
|
499
|
+
decodeProtobuf,
|
|
500
|
+
encodeProtobuf,
|
|
334
501
|
executeRequestChain,
|
|
335
|
-
isApiError
|
|
502
|
+
isApiError,
|
|
503
|
+
isProtobufContentType,
|
|
504
|
+
normalizeProtobufConfig,
|
|
505
|
+
parseProtobufRequest,
|
|
506
|
+
protobufResponse
|
|
336
507
|
});
|
|
337
508
|
//# 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"],"sourcesContent":["export * from \"./client\";\nexport * from \"./chain\";\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\ntype RequestOptions = FetchOptions<\"json\"> & { returnFullResponse?: boolean };\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({ options }: FetchContext) {\n const token = await tokenStorage.getAccessToken();\n\n const headers = new Headers(options.headers as HeadersInit | undefined);\n\n if (token) {\n headers.set(\"Authorization\", `Bearer ${token}`);\n }\n\n options.headers = headers;\n },\n onResponse({ response }) {\n // 不在这里处理 code,统一交给 fetchApi 层处理\n if (response.status === 204) {\n response._data = null;\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 RequestOptions);\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): Promise<T> {\n const context: Record<string, any> = {};\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 );\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"],"mappings":"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","names":["options"]}
|
|
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"]}
|
package/dist/index.mjs
CHANGED
|
@@ -66,17 +66,28 @@ function createApiClient(options) {
|
|
|
66
66
|
baseURL,
|
|
67
67
|
retry,
|
|
68
68
|
retryDelay,
|
|
69
|
-
async onRequest(
|
|
69
|
+
async onRequest(context) {
|
|
70
|
+
const { options: reqOptions } = context;
|
|
70
71
|
const token = await tokenStorage.getAccessToken();
|
|
71
|
-
const headers = new Headers(
|
|
72
|
+
const headers = new Headers(
|
|
73
|
+
reqOptions.headers
|
|
74
|
+
);
|
|
72
75
|
if (token) {
|
|
73
76
|
headers.set("Authorization", `Bearer ${token}`);
|
|
74
77
|
}
|
|
75
|
-
|
|
78
|
+
reqOptions.headers = headers;
|
|
79
|
+
if (options.onRequest) {
|
|
80
|
+
await options.onRequest(context);
|
|
81
|
+
}
|
|
76
82
|
},
|
|
77
|
-
onResponse(
|
|
83
|
+
async onResponse(context) {
|
|
84
|
+
const { response } = context;
|
|
78
85
|
if (response.status === 204) {
|
|
79
86
|
response._data = null;
|
|
87
|
+
return;
|
|
88
|
+
}
|
|
89
|
+
if (options.onResponse) {
|
|
90
|
+
await options.onResponse(context);
|
|
80
91
|
}
|
|
81
92
|
},
|
|
82
93
|
async onResponseError(context) {
|
|
@@ -248,8 +259,8 @@ function substituteVariables(target, context) {
|
|
|
248
259
|
}
|
|
249
260
|
return target;
|
|
250
261
|
}
|
|
251
|
-
async function executeRequestChain(requests, handlers = [], getChainRequests) {
|
|
252
|
-
const context = {};
|
|
262
|
+
async function executeRequestChain(requests, handlers = [], getChainRequests, initContext) {
|
|
263
|
+
const context = initContext || {};
|
|
253
264
|
let lastResult = null;
|
|
254
265
|
for (const rule of requests) {
|
|
255
266
|
try {
|
|
@@ -278,7 +289,8 @@ async function executeRequestChain(requests, handlers = [], getChainRequests) {
|
|
|
278
289
|
rawData = await executeRequestChain(
|
|
279
290
|
chainRequests,
|
|
280
291
|
handlers,
|
|
281
|
-
getChainRequests
|
|
292
|
+
getChainRequests,
|
|
293
|
+
JSON.parse(JSON.stringify(context))
|
|
282
294
|
);
|
|
283
295
|
}
|
|
284
296
|
}
|
|
@@ -302,9 +314,151 @@ async function executeRequestChain(requests, handlers = [], getChainRequests) {
|
|
|
302
314
|
}
|
|
303
315
|
return lastResult;
|
|
304
316
|
}
|
|
317
|
+
|
|
318
|
+
// src/protobuf.ts
|
|
319
|
+
import protobuf from "protobufjs";
|
|
320
|
+
var SALTED_SCHEMA = `
|
|
321
|
+
syntax = "proto3";
|
|
322
|
+
|
|
323
|
+
message SaltedPayload {
|
|
324
|
+
string salt = 1;
|
|
325
|
+
int64 timestamp = 2;
|
|
326
|
+
bytes data = 3;
|
|
327
|
+
}
|
|
328
|
+
`;
|
|
329
|
+
var encoder = new TextEncoder();
|
|
330
|
+
var decoder = new TextDecoder();
|
|
331
|
+
var saltedPayloadType = null;
|
|
332
|
+
function getSaltedPayloadType() {
|
|
333
|
+
if (!saltedPayloadType) {
|
|
334
|
+
const root = protobuf.parse(SALTED_SCHEMA).root;
|
|
335
|
+
saltedPayloadType = root.lookupType("SaltedPayload");
|
|
336
|
+
}
|
|
337
|
+
return saltedPayloadType;
|
|
338
|
+
}
|
|
339
|
+
function generateSalt() {
|
|
340
|
+
if (typeof crypto !== "undefined" && crypto.randomUUID) {
|
|
341
|
+
return crypto.randomUUID();
|
|
342
|
+
}
|
|
343
|
+
return Math.random().toString(36).substring(2) + Date.now().toString(36);
|
|
344
|
+
}
|
|
345
|
+
function encodeProtobuf(data, options = {}) {
|
|
346
|
+
const { useSalt = true, saltGenerator = generateSalt } = options;
|
|
347
|
+
const type = getSaltedPayloadType();
|
|
348
|
+
const jsonString = JSON.stringify(data);
|
|
349
|
+
const dataBytes = encoder.encode(jsonString);
|
|
350
|
+
const payload = {
|
|
351
|
+
salt: useSalt ? saltGenerator() : "",
|
|
352
|
+
timestamp: Date.now(),
|
|
353
|
+
data: dataBytes
|
|
354
|
+
};
|
|
355
|
+
const message = type.create(payload);
|
|
356
|
+
return type.encode(message).finish();
|
|
357
|
+
}
|
|
358
|
+
function decodeProtobuf(buffer) {
|
|
359
|
+
const type = getSaltedPayloadType();
|
|
360
|
+
const uint8 = buffer instanceof ArrayBuffer ? new Uint8Array(buffer) : buffer;
|
|
361
|
+
const decoded = type.decode(uint8);
|
|
362
|
+
const jsonString = decoder.decode(decoded.data);
|
|
363
|
+
return JSON.parse(jsonString);
|
|
364
|
+
}
|
|
365
|
+
function protobufResponse(data, options) {
|
|
366
|
+
const buffer = encodeProtobuf(data, options);
|
|
367
|
+
const arrayBuffer = buffer.buffer.slice(
|
|
368
|
+
buffer.byteOffset,
|
|
369
|
+
buffer.byteOffset + buffer.byteLength
|
|
370
|
+
);
|
|
371
|
+
return new Response(arrayBuffer, {
|
|
372
|
+
headers: {
|
|
373
|
+
"Content-Type": "application/x-protobuf",
|
|
374
|
+
...options?.corsHeaders
|
|
375
|
+
}
|
|
376
|
+
});
|
|
377
|
+
}
|
|
378
|
+
function isProtobufContentType(contentType) {
|
|
379
|
+
return !!contentType?.includes("application/x-protobuf");
|
|
380
|
+
}
|
|
381
|
+
function normalizeProtobufConfig(config) {
|
|
382
|
+
if (!config) {
|
|
383
|
+
return null;
|
|
384
|
+
}
|
|
385
|
+
if (config === true) {
|
|
386
|
+
return {
|
|
387
|
+
encode: (data) => encodeProtobuf(data),
|
|
388
|
+
decode: (buffer) => decodeProtobuf(buffer)
|
|
389
|
+
};
|
|
390
|
+
}
|
|
391
|
+
return {
|
|
392
|
+
encode: config.encode || ((data) => encodeProtobuf(data, config.options)),
|
|
393
|
+
decode: config.decode || ((buffer) => decodeProtobuf(buffer))
|
|
394
|
+
};
|
|
395
|
+
}
|
|
396
|
+
async function parseProtobufRequest(request, customDecode) {
|
|
397
|
+
const buffer = await request.arrayBuffer();
|
|
398
|
+
const uint8 = new Uint8Array(buffer);
|
|
399
|
+
if (customDecode) {
|
|
400
|
+
return customDecode(uint8);
|
|
401
|
+
}
|
|
402
|
+
return decodeProtobuf(uint8);
|
|
403
|
+
}
|
|
404
|
+
function createProtobufHooks(options = {}) {
|
|
405
|
+
const encode = options.encode || ((data) => encodeProtobuf(data, options));
|
|
406
|
+
const decode = options.decode || ((buffer) => decodeProtobuf(buffer));
|
|
407
|
+
return {
|
|
408
|
+
/**
|
|
409
|
+
* 请求钩子:独立处理请求编码和响应类型设置
|
|
410
|
+
*/
|
|
411
|
+
async onRequest(context) {
|
|
412
|
+
const { options: reqOptions } = context;
|
|
413
|
+
const headers = reqOptions.headers instanceof Headers ? reqOptions.headers : new Headers(reqOptions.headers);
|
|
414
|
+
const contentType = headers.get("Content-Type");
|
|
415
|
+
const isProtobufRequest = isProtobufContentType(contentType);
|
|
416
|
+
if (isProtobufRequest && reqOptions.body && !(reqOptions.body instanceof Uint8Array)) {
|
|
417
|
+
const encoded = await encode(reqOptions.body);
|
|
418
|
+
reqOptions.body = encoded;
|
|
419
|
+
}
|
|
420
|
+
const accept = headers.get("Accept");
|
|
421
|
+
const prefersProtobufResponse = isProtobufContentType(accept);
|
|
422
|
+
if (prefersProtobufResponse && !reqOptions.responseType) {
|
|
423
|
+
reqOptions.responseType = "arrayBuffer";
|
|
424
|
+
}
|
|
425
|
+
reqOptions.headers = headers;
|
|
426
|
+
},
|
|
427
|
+
/**
|
|
428
|
+
* 响应钩子:解码 Protobuf 响应
|
|
429
|
+
*/
|
|
430
|
+
async onResponse(context) {
|
|
431
|
+
const { response } = context;
|
|
432
|
+
if (!response?._data) {
|
|
433
|
+
return;
|
|
434
|
+
}
|
|
435
|
+
const contentType = response.headers.get("Content-Type");
|
|
436
|
+
if (isProtobufContentType(contentType)) {
|
|
437
|
+
const buffer = response._data;
|
|
438
|
+
if (buffer && buffer.byteLength > 0) {
|
|
439
|
+
response._data = await decode(new Uint8Array(buffer));
|
|
440
|
+
}
|
|
441
|
+
} else if (response._data instanceof ArrayBuffer) {
|
|
442
|
+
const text = decoder.decode(response._data);
|
|
443
|
+
try {
|
|
444
|
+
response._data = JSON.parse(text);
|
|
445
|
+
} catch {
|
|
446
|
+
response._data = text;
|
|
447
|
+
}
|
|
448
|
+
}
|
|
449
|
+
}
|
|
450
|
+
};
|
|
451
|
+
}
|
|
305
452
|
export {
|
|
306
453
|
createApiClient,
|
|
454
|
+
createProtobufHooks,
|
|
455
|
+
decodeProtobuf,
|
|
456
|
+
encodeProtobuf,
|
|
307
457
|
executeRequestChain,
|
|
308
|
-
isApiError
|
|
458
|
+
isApiError,
|
|
459
|
+
isProtobufContentType,
|
|
460
|
+
normalizeProtobufConfig,
|
|
461
|
+
parseProtobufRequest,
|
|
462
|
+
protobufResponse
|
|
309
463
|
};
|
|
310
464
|
//# sourceMappingURL=index.mjs.map
|
package/dist/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/client.ts","../src/chain.ts"],"sourcesContent":["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\ntype RequestOptions = FetchOptions<\"json\"> & { returnFullResponse?: boolean };\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({ options }: FetchContext) {\n const token = await tokenStorage.getAccessToken();\n\n const headers = new Headers(options.headers as HeadersInit | undefined);\n\n if (token) {\n headers.set(\"Authorization\", `Bearer ${token}`);\n }\n\n options.headers = headers;\n },\n onResponse({ response }) {\n // 不在这里处理 code,统一交给 fetchApi 层处理\n if (response.status === 204) {\n response._data = null;\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 RequestOptions);\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): Promise<T> {\n const context: Record<string, any> = {};\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 );\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"],"mappings":"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","names":["options"]}
|
|
1
|
+
{"version":3,"sources":["../src/client.ts","../src/chain.ts","../src/protobuf.ts"],"sourcesContent":["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"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@i.un/api-client",
|
|
3
|
-
"version": "1.0.
|
|
3
|
+
"version": "1.0.8",
|
|
4
4
|
"description": "Universal API client for i.un services",
|
|
5
5
|
"license": "MIT",
|
|
6
6
|
"author": "nova <www.nova@gmail.com>",
|
|
@@ -37,7 +37,8 @@
|
|
|
37
37
|
"clean": "rm -rf dist"
|
|
38
38
|
},
|
|
39
39
|
"dependencies": {
|
|
40
|
-
"ofetch": ">=1.4.0"
|
|
40
|
+
"ofetch": ">=1.4.0",
|
|
41
|
+
"protobufjs": "^8.0.0"
|
|
41
42
|
},
|
|
42
43
|
"devDependencies": {
|
|
43
44
|
"tsup": "^8.0.0",
|
|
@@ -52,4 +53,4 @@
|
|
|
52
53
|
"publishConfig": {
|
|
53
54
|
"access": "public"
|
|
54
55
|
}
|
|
55
|
-
}
|
|
56
|
+
}
|