sa2kit 1.0.0 → 1.0.2
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/UniversalFileService-CEZRJ87g.d.mts +727 -0
- package/dist/UniversalFileService-CEZRJ87g.d.ts +727 -0
- package/dist/api/index.d.mts +248 -0
- package/dist/api/index.d.ts +248 -0
- package/dist/api/index.js +294 -0
- package/dist/api/index.js.map +1 -0
- package/dist/api/index.mjs +290 -0
- package/dist/api/index.mjs.map +1 -0
- package/dist/auth/client/index.d.mts +52 -3
- package/dist/auth/client/index.d.ts +52 -3
- package/dist/auth/components/index.d.mts +149 -4
- package/dist/auth/components/index.d.ts +149 -4
- package/dist/auth/components/index.js +243 -9
- package/dist/auth/components/index.js.map +1 -1
- package/dist/auth/components/index.mjs +237 -4
- package/dist/auth/components/index.mjs.map +1 -1
- package/dist/auth/hooks/index.d.mts +31 -2
- package/dist/auth/hooks/index.d.ts +31 -2
- package/dist/auth/index.d.mts +5 -5
- package/dist/auth/index.d.ts +5 -5
- package/dist/auth/index.js +49 -17
- package/dist/auth/index.mjs +1 -1
- package/dist/auth/routes/index.d.mts +103 -5
- package/dist/auth/routes/index.d.ts +103 -5
- package/dist/auth/routes/index.js +37 -5
- package/dist/auth/routes/index.mjs +1 -1
- package/dist/chunk-42IJ7HEI.js +573 -0
- package/dist/chunk-42IJ7HEI.js.map +1 -0
- package/dist/chunk-7XLFSPDG.mjs +31 -0
- package/dist/chunk-7XLFSPDG.mjs.map +1 -0
- package/dist/chunk-GCVOKQZP.js +36 -0
- package/dist/chunk-GCVOKQZP.js.map +1 -0
- package/dist/chunk-IBLB7ARJ.mjs +560 -0
- package/dist/chunk-IBLB7ARJ.mjs.map +1 -0
- package/dist/{chunk-6FNUWAIV.js → chunk-LX4XX6W7.js} +54 -8
- package/dist/chunk-LX4XX6W7.js.map +1 -0
- package/dist/{chunk-HXFFYNIF.mjs → chunk-T5OZHYVM.mjs} +54 -8
- package/dist/chunk-T5OZHYVM.mjs.map +1 -0
- package/dist/config/server/index.d.mts +1533 -0
- package/dist/config/server/index.d.ts +1533 -0
- package/dist/config/server/index.js +1177 -0
- package/dist/config/server/index.js.map +1 -0
- package/dist/config/server/index.mjs +1138 -0
- package/dist/config/server/index.mjs.map +1 -0
- package/dist/i18n/index.d.mts +2 -1
- package/dist/i18n/index.d.ts +2 -1
- package/dist/i18n/index.js +125 -61
- package/dist/i18n/index.js.map +1 -1
- package/dist/i18n/index.mjs +126 -62
- package/dist/i18n/index.mjs.map +1 -1
- package/dist/index.js +6 -6
- package/dist/index.mjs +1 -1
- package/dist/mmd/index.d.mts +346 -0
- package/dist/mmd/index.d.ts +346 -0
- package/dist/mmd/index.js +1535 -0
- package/dist/mmd/index.js.map +1 -0
- package/dist/mmd/index.mjs +1503 -0
- package/dist/mmd/index.mjs.map +1 -0
- package/dist/storage/index.d.mts +1 -0
- package/dist/storage/index.d.ts +1 -0
- package/dist/storage/index.js +9 -9
- package/dist/storage/index.mjs +1 -1
- package/dist/{index-8VoHap_4.d.mts → types-CroexXnI.d.ts} +38 -44
- package/dist/{index-8VoHap_4.d.ts → types-DmsXCWvm.d.mts} +38 -44
- package/dist/{types-DAxQ1MeY.d.ts → types-Dt0oqeFM.d.mts} +1 -1
- package/dist/{types-DT8LVCvE.d.mts → types-zK6kDzDQ.d.ts} +1 -1
- package/dist/universalExport/index.js +17 -32
- package/dist/universalExport/index.js.map +1 -1
- package/dist/universalExport/index.mjs +2 -29
- package/dist/universalExport/index.mjs.map +1 -1
- package/dist/universalExport/server/index.d.mts +849 -8
- package/dist/universalExport/server/index.d.ts +849 -8
- package/dist/universalExport/server/index.js +1382 -2
- package/dist/universalExport/server/index.js.map +1 -1
- package/dist/universalExport/server/index.mjs +1355 -3
- package/dist/universalExport/server/index.mjs.map +1 -1
- package/dist/universalFile/index.d.mts +54 -3
- package/dist/universalFile/index.d.ts +54 -3
- package/dist/universalFile/index.js +272 -0
- package/dist/universalFile/index.js.map +1 -1
- package/dist/universalFile/index.mjs +267 -1
- package/dist/universalFile/index.mjs.map +1 -1
- package/dist/universalFile/server/index.d.mts +2541 -469
- package/dist/universalFile/server/index.d.ts +2541 -469
- package/dist/universalFile/server/index.js +830 -64
- package/dist/universalFile/server/index.js.map +1 -1
- package/dist/universalFile/server/index.mjs +803 -66
- package/dist/universalFile/server/index.mjs.map +1 -1
- package/package.json +47 -23
- package/dist/chunk-6FNUWAIV.js.map +0 -1
- package/dist/chunk-APY57REU.js +0 -300
- package/dist/chunk-APY57REU.js.map +0 -1
- package/dist/chunk-C64RY2OW.mjs +0 -295
- package/dist/chunk-C64RY2OW.mjs.map +0 -1
- package/dist/chunk-HXFFYNIF.mjs.map +0 -1
- package/dist/types-CoGG1rNV.d.mts +0 -258
- package/dist/types-CoGG1rNV.d.ts +0 -258
- package/dist/types-DW9qar-w.d.mts +0 -52
- package/dist/types-DW9qar-w.d.ts +0 -52
|
@@ -0,0 +1,294 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
require('../chunk-DGUM43GV.js');
|
|
4
|
+
|
|
5
|
+
// src/api/types.ts
|
|
6
|
+
var DEFAULT_STORAGE_KEYS = {
|
|
7
|
+
AUTH_TOKEN: "auth_token",
|
|
8
|
+
USER_DATA: "user_data",
|
|
9
|
+
REFRESH_TOKEN: "refresh_token"
|
|
10
|
+
};
|
|
11
|
+
var DEFAULT_API_ROUTES = {
|
|
12
|
+
auth: {
|
|
13
|
+
login: "/auth/login",
|
|
14
|
+
logout: "/auth/logout",
|
|
15
|
+
register: "/auth/register",
|
|
16
|
+
me: "/auth/me"
|
|
17
|
+
},
|
|
18
|
+
users: {
|
|
19
|
+
list: "/users",
|
|
20
|
+
detail: (id) => `/users/${id}`,
|
|
21
|
+
update: (id) => `/users/${id}`,
|
|
22
|
+
delete: (id) => `/users/${id}`
|
|
23
|
+
}
|
|
24
|
+
};
|
|
25
|
+
|
|
26
|
+
// src/api/BaseApiClient.ts
|
|
27
|
+
var BaseApiClient = class {
|
|
28
|
+
constructor(config) {
|
|
29
|
+
this.token = null;
|
|
30
|
+
this.user = null;
|
|
31
|
+
this.baseUrl = config.baseUrl;
|
|
32
|
+
this.storage = config.storage;
|
|
33
|
+
this.request = config.request;
|
|
34
|
+
this.storageKeys = {
|
|
35
|
+
...DEFAULT_STORAGE_KEYS,
|
|
36
|
+
...config.storageKeys || {}
|
|
37
|
+
};
|
|
38
|
+
this.routes = {
|
|
39
|
+
...DEFAULT_API_ROUTES,
|
|
40
|
+
...config.routes || {},
|
|
41
|
+
auth: {
|
|
42
|
+
...DEFAULT_API_ROUTES.auth,
|
|
43
|
+
...config.routes?.auth || {}
|
|
44
|
+
},
|
|
45
|
+
users: {
|
|
46
|
+
...DEFAULT_API_ROUTES.users,
|
|
47
|
+
...config.routes?.users || {}
|
|
48
|
+
}
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* 初始化 - 从存储中加载 token 和用户信息
|
|
53
|
+
*/
|
|
54
|
+
async init() {
|
|
55
|
+
try {
|
|
56
|
+
this.token = await this.storage.getItem(this.storageKeys.AUTH_TOKEN);
|
|
57
|
+
const userData = await this.storage.getItem(this.storageKeys.USER_DATA);
|
|
58
|
+
if (userData) {
|
|
59
|
+
this.user = JSON.parse(userData);
|
|
60
|
+
}
|
|
61
|
+
} catch (error) {
|
|
62
|
+
console.error("Failed to load auth data:", error);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* 设置认证 token
|
|
67
|
+
*/
|
|
68
|
+
async setToken(token) {
|
|
69
|
+
this.token = token;
|
|
70
|
+
if (token) {
|
|
71
|
+
await this.storage.setItem(this.storageKeys.AUTH_TOKEN, token);
|
|
72
|
+
} else {
|
|
73
|
+
await this.storage.removeItem(this.storageKeys.AUTH_TOKEN);
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* 设置用户信息
|
|
78
|
+
*/
|
|
79
|
+
async setUser(user) {
|
|
80
|
+
this.user = user;
|
|
81
|
+
if (user) {
|
|
82
|
+
await this.storage.setItem(this.storageKeys.USER_DATA, JSON.stringify(user));
|
|
83
|
+
} else {
|
|
84
|
+
await this.storage.removeItem(this.storageKeys.USER_DATA);
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* 获取当前 token
|
|
89
|
+
*/
|
|
90
|
+
getToken() {
|
|
91
|
+
return this.token;
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* 获取当前用户
|
|
95
|
+
*/
|
|
96
|
+
getUser() {
|
|
97
|
+
return this.user;
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* 检查是否已登录
|
|
101
|
+
*/
|
|
102
|
+
async isAuthenticated() {
|
|
103
|
+
return !!this.token;
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* 清除用户数据
|
|
107
|
+
*/
|
|
108
|
+
async clearUserData() {
|
|
109
|
+
await this.storage.removeItem(this.storageKeys.AUTH_TOKEN);
|
|
110
|
+
await this.storage.removeItem(this.storageKeys.USER_DATA);
|
|
111
|
+
if (this.storageKeys.REFRESH_TOKEN) {
|
|
112
|
+
await this.storage.removeItem(this.storageKeys.REFRESH_TOKEN);
|
|
113
|
+
}
|
|
114
|
+
this.token = null;
|
|
115
|
+
this.user = null;
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* 发送请求的通用方法
|
|
119
|
+
*/
|
|
120
|
+
async sendRequest(config) {
|
|
121
|
+
try {
|
|
122
|
+
const headers = {
|
|
123
|
+
"Content-Type": "application/json",
|
|
124
|
+
...config.headers || {}
|
|
125
|
+
};
|
|
126
|
+
if (this.token) {
|
|
127
|
+
headers["Authorization"] = `Bearer ${this.token}`;
|
|
128
|
+
}
|
|
129
|
+
const response = await this.request.request({
|
|
130
|
+
...config,
|
|
131
|
+
url: `${this.baseUrl}${config.url}`,
|
|
132
|
+
headers
|
|
133
|
+
});
|
|
134
|
+
return response;
|
|
135
|
+
} catch (error) {
|
|
136
|
+
console.error("API request error:", error);
|
|
137
|
+
return {
|
|
138
|
+
success: false,
|
|
139
|
+
error: error instanceof Error ? error.message : "\u7F51\u7EDC\u9519\u8BEF\uFF0C\u8BF7\u91CD\u8BD5"
|
|
140
|
+
};
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* 发送 GET 请求
|
|
145
|
+
*/
|
|
146
|
+
async get(url, params) {
|
|
147
|
+
return this.sendRequest({ url, method: "GET", params });
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* 发送 POST 请求
|
|
151
|
+
*/
|
|
152
|
+
async post(url, body) {
|
|
153
|
+
return this.sendRequest({ url, method: "POST", body });
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* 发送 PUT 请求
|
|
157
|
+
*/
|
|
158
|
+
async put(url, body) {
|
|
159
|
+
return this.sendRequest({ url, method: "PUT", body });
|
|
160
|
+
}
|
|
161
|
+
/**
|
|
162
|
+
* 发送 PATCH 请求
|
|
163
|
+
*/
|
|
164
|
+
async patch(url, body) {
|
|
165
|
+
return this.sendRequest({ url, method: "PATCH", body });
|
|
166
|
+
}
|
|
167
|
+
/**
|
|
168
|
+
* 发送 DELETE 请求
|
|
169
|
+
*/
|
|
170
|
+
async delete(url) {
|
|
171
|
+
return this.sendRequest({ url, method: "DELETE" });
|
|
172
|
+
}
|
|
173
|
+
// ==================== 默认认证方法(子类可重写) ====================
|
|
174
|
+
/**
|
|
175
|
+
* 用户注册
|
|
176
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
177
|
+
*/
|
|
178
|
+
async register(email, password, username) {
|
|
179
|
+
if (!this.routes.auth?.register) {
|
|
180
|
+
throw new Error("Register route is not configured");
|
|
181
|
+
}
|
|
182
|
+
const response = await this.post(this.routes.auth.register, {
|
|
183
|
+
email,
|
|
184
|
+
password,
|
|
185
|
+
username
|
|
186
|
+
});
|
|
187
|
+
if (response.success && response.data) {
|
|
188
|
+
await this.setToken(response.data.token);
|
|
189
|
+
await this.setUser(response.data.user);
|
|
190
|
+
await this.onAuthSuccess?.(response.data.user, response.data.token);
|
|
191
|
+
} else if (!response.success) {
|
|
192
|
+
await this.onAuthError?.(response.error || "Registration failed");
|
|
193
|
+
}
|
|
194
|
+
return response;
|
|
195
|
+
}
|
|
196
|
+
/**
|
|
197
|
+
* 用户登录
|
|
198
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
199
|
+
*/
|
|
200
|
+
async login(email, password) {
|
|
201
|
+
if (!this.routes.auth?.login) {
|
|
202
|
+
throw new Error("Login route is not configured");
|
|
203
|
+
}
|
|
204
|
+
const response = await this.post(this.routes.auth.login, {
|
|
205
|
+
email,
|
|
206
|
+
password
|
|
207
|
+
});
|
|
208
|
+
if (response.success && response.data) {
|
|
209
|
+
await this.setToken(response.data.token);
|
|
210
|
+
await this.setUser(response.data.user);
|
|
211
|
+
await this.onAuthSuccess?.(response.data.user, response.data.token);
|
|
212
|
+
} else if (!response.success) {
|
|
213
|
+
await this.onAuthError?.(response.error || "Login failed");
|
|
214
|
+
}
|
|
215
|
+
return response;
|
|
216
|
+
}
|
|
217
|
+
/**
|
|
218
|
+
* 用户退出登录
|
|
219
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
220
|
+
*/
|
|
221
|
+
async logout() {
|
|
222
|
+
if (this.routes.auth?.logout) {
|
|
223
|
+
await this.post(this.routes.auth.logout);
|
|
224
|
+
}
|
|
225
|
+
await this.clearUserData();
|
|
226
|
+
await this.onLogout?.();
|
|
227
|
+
}
|
|
228
|
+
/**
|
|
229
|
+
* 获取当前用户信息
|
|
230
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
231
|
+
*/
|
|
232
|
+
async getCurrentUser() {
|
|
233
|
+
if (!this.routes.auth?.me) {
|
|
234
|
+
throw new Error("Current user route is not configured");
|
|
235
|
+
}
|
|
236
|
+
const response = await this.get(this.routes.auth.me);
|
|
237
|
+
if (response.success && response.data) {
|
|
238
|
+
const userData = response.data.user || response.data;
|
|
239
|
+
await this.setUser(userData);
|
|
240
|
+
return {
|
|
241
|
+
...response,
|
|
242
|
+
data: userData
|
|
243
|
+
};
|
|
244
|
+
}
|
|
245
|
+
return response;
|
|
246
|
+
}
|
|
247
|
+
// ==================== 默认用户管理方法(子类可重写) ====================
|
|
248
|
+
/**
|
|
249
|
+
* 获取用户列表
|
|
250
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
251
|
+
*/
|
|
252
|
+
async getUsers(params) {
|
|
253
|
+
if (!this.routes.users?.list) {
|
|
254
|
+
throw new Error("Users list route is not configured");
|
|
255
|
+
}
|
|
256
|
+
return this.get(this.routes.users.list, params);
|
|
257
|
+
}
|
|
258
|
+
/**
|
|
259
|
+
* 获取用户详情
|
|
260
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
261
|
+
*/
|
|
262
|
+
async getUserById(userId) {
|
|
263
|
+
if (!this.routes.users?.detail) {
|
|
264
|
+
throw new Error("User detail route is not configured");
|
|
265
|
+
}
|
|
266
|
+
return this.get(this.routes.users.detail(userId));
|
|
267
|
+
}
|
|
268
|
+
/**
|
|
269
|
+
* 更新用户信息
|
|
270
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
271
|
+
*/
|
|
272
|
+
async updateUser(userId, data) {
|
|
273
|
+
if (!this.routes.users?.update) {
|
|
274
|
+
throw new Error("User update route is not configured");
|
|
275
|
+
}
|
|
276
|
+
return this.put(this.routes.users.update(userId), data);
|
|
277
|
+
}
|
|
278
|
+
/**
|
|
279
|
+
* 删除用户
|
|
280
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
281
|
+
*/
|
|
282
|
+
async deleteUser(userId) {
|
|
283
|
+
if (!this.routes.users?.delete) {
|
|
284
|
+
throw new Error("User delete route is not configured");
|
|
285
|
+
}
|
|
286
|
+
return this.delete(this.routes.users.delete(userId));
|
|
287
|
+
}
|
|
288
|
+
};
|
|
289
|
+
|
|
290
|
+
exports.BaseApiClient = BaseApiClient;
|
|
291
|
+
exports.DEFAULT_API_ROUTES = DEFAULT_API_ROUTES;
|
|
292
|
+
exports.DEFAULT_STORAGE_KEYS = DEFAULT_STORAGE_KEYS;
|
|
293
|
+
//# sourceMappingURL=index.js.map
|
|
294
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/api/types.ts","../../src/api/BaseApiClient.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["/**\n * API 通用类型定义\n * Common API Types\n */\n\n/**\n * API 响应类型\n * Generic API Response Format\n */\nexport interface ApiResponse<T = any> {\n success: boolean;\n data?: T;\n error?: string;\n message?: string;\n meta?: {\n pagination?: {\n page: number;\n limit: number;\n total: number;\n pages: number;\n };\n [key: string]: any;\n };\n}\n\n/**\n * 分页参数类型\n * Pagination Parameters\n */\nexport interface PaginationParams {\n page?: number;\n limit?: number;\n search?: string;\n sortBy?: string;\n sortOrder?: 'asc' | 'desc';\n}\n\n/**\n * 认证响应类型(泛型)\n * Generic Auth Response\n */\nexport interface AuthResponse<TUser = any> {\n user: TUser;\n token: string;\n}\n\n/**\n * 默认存储键常量\n * Default Storage Keys\n */\nexport const DEFAULT_STORAGE_KEYS = {\n AUTH_TOKEN: 'auth_token',\n USER_DATA: 'user_data',\n REFRESH_TOKEN: 'refresh_token',\n} as const;\n\n/**\n * 存储键类型(可扩展)\n * Storage Keys Type\n */\nexport type StorageKeys = typeof DEFAULT_STORAGE_KEYS;\n\n/**\n * API 路由配置\n * API Routes Configuration\n */\nexport interface ApiRoutes {\n /** 认证相关路由 */\n auth?: {\n login?: string;\n logout?: string;\n register?: string;\n me?: string;\n };\n /** 用户相关路由 */\n users?: {\n list?: string;\n detail?: (id: string) => string;\n update?: (id: string) => string;\n delete?: (id: string) => string;\n };\n /** 自定义路由 */\n [key: string]: any;\n}\n\n/**\n * 默认 API 路由\n */\nexport const DEFAULT_API_ROUTES: ApiRoutes = {\n auth: {\n login: '/auth/login',\n logout: '/auth/logout',\n register: '/auth/register',\n me: '/auth/me',\n },\n users: {\n list: '/users',\n detail: (id: string) => `/users/${id}`,\n update: (id: string) => `/users/${id}`,\n delete: (id: string) => `/users/${id}`,\n },\n};\n","/**\n * 基础 API 客户端(泛型)\n * Base API Client (Generic)\n *\n * 提供统一的 API 调用逻辑,通过适配器模式支持多平台\n * 使用泛型 TUser 支持不同项目的用户类型\n */\n\nimport type { StorageAdapter } from '../storage';\nimport type { RequestAdapter, RequestConfig } from '../request';\nimport type { ApiResponse, AuthResponse, StorageKeys, ApiRoutes } from './types';\nimport { DEFAULT_STORAGE_KEYS, DEFAULT_API_ROUTES } from './types';\n\n/**\n * 基础 API 客户端配置\n */\nexport interface BaseApiClientConfig {\n /** 基础 URL */\n baseUrl: string;\n /** 存储适配器 */\n storage: StorageAdapter;\n /** 请求适配器 */\n request: RequestAdapter;\n /** 自定义存储键(可选) */\n storageKeys?: Partial<StorageKeys>;\n /** API 路由配置(可选) */\n routes?: ApiRoutes;\n}\n\n/**\n * 基础 API 客户端\n * @template TUser 用户类型(泛型)\n */\nexport class BaseApiClient<TUser = any> {\n protected token: string | null = null;\n protected user: TUser | null = null;\n protected baseUrl: string;\n protected storage: StorageAdapter;\n protected request: RequestAdapter;\n protected storageKeys: StorageKeys;\n protected routes: ApiRoutes;\n\n constructor(config: BaseApiClientConfig) {\n this.baseUrl = config.baseUrl;\n this.storage = config.storage;\n this.request = config.request;\n this.storageKeys = {\n ...DEFAULT_STORAGE_KEYS,\n ...(config.storageKeys || {}),\n };\n this.routes = {\n ...DEFAULT_API_ROUTES,\n ...(config.routes || {}),\n auth: {\n ...DEFAULT_API_ROUTES.auth,\n ...(config.routes?.auth || {}),\n },\n users: {\n ...DEFAULT_API_ROUTES.users,\n ...(config.routes?.users || {}),\n },\n };\n }\n\n /**\n * 初始化 - 从存储中加载 token 和用户信息\n */\n async init(): Promise<void> {\n try {\n this.token = await this.storage.getItem(this.storageKeys.AUTH_TOKEN);\n const userData = await this.storage.getItem(this.storageKeys.USER_DATA);\n if (userData) {\n this.user = JSON.parse(userData) as TUser;\n }\n } catch (error) {\n console.error('Failed to load auth data:', error);\n }\n }\n\n /**\n * 设置认证 token\n */\n async setToken(token: string | null): Promise<void> {\n this.token = token;\n if (token) {\n await this.storage.setItem(this.storageKeys.AUTH_TOKEN, token);\n } else {\n await this.storage.removeItem(this.storageKeys.AUTH_TOKEN);\n }\n }\n\n /**\n * 设置用户信息\n */\n async setUser(user: TUser | null): Promise<void> {\n this.user = user;\n if (user) {\n await this.storage.setItem(this.storageKeys.USER_DATA, JSON.stringify(user));\n } else {\n await this.storage.removeItem(this.storageKeys.USER_DATA);\n }\n }\n\n /**\n * 获取当前 token\n */\n getToken(): string | null {\n return this.token;\n }\n\n /**\n * 获取当前用户\n */\n getUser(): TUser | null {\n return this.user;\n }\n\n /**\n * 检查是否已登录\n */\n async isAuthenticated(): Promise<boolean> {\n return !!this.token;\n }\n\n /**\n * 清除用户数据\n */\n async clearUserData(): Promise<void> {\n await this.storage.removeItem(this.storageKeys.AUTH_TOKEN);\n await this.storage.removeItem(this.storageKeys.USER_DATA);\n if (this.storageKeys.REFRESH_TOKEN) {\n await this.storage.removeItem(this.storageKeys.REFRESH_TOKEN);\n }\n this.token = null;\n this.user = null;\n }\n\n /**\n * 发送请求的通用方法\n */\n protected async sendRequest<T = any>(config: RequestConfig): Promise<ApiResponse<T>> {\n try {\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n ...(config.headers || {}),\n };\n\n // 添加认证 token\n if (this.token) {\n headers['Authorization'] = `Bearer ${this.token}`;\n }\n\n const response = await this.request.request<ApiResponse<T>>({\n ...config,\n url: `${this.baseUrl}${config.url}`,\n headers,\n });\n\n return response;\n } catch (error) {\n console.error('API request error:', error);\n return {\n success: false,\n error: error instanceof Error ? error.message : '网络错误,请重试',\n };\n }\n }\n\n /**\n * 发送 GET 请求\n */\n async get<T = any>(url: string, params?: Record<string, any>): Promise<ApiResponse<T>> {\n return this.sendRequest({ url, method: 'GET', params });\n }\n\n /**\n * 发送 POST 请求\n */\n async post<T = any>(url: string, body?: any): Promise<ApiResponse<T>> {\n return this.sendRequest({ url, method: 'POST', body });\n }\n\n /**\n * 发送 PUT 请求\n */\n async put<T = any>(url: string, body?: any): Promise<ApiResponse<T>> {\n return this.sendRequest({ url, method: 'PUT', body });\n }\n\n /**\n * 发送 PATCH 请求\n */\n async patch<T = any>(url: string, body?: any): Promise<ApiResponse<T>> {\n return this.sendRequest({ url, method: 'PATCH', body });\n }\n\n /**\n * 发送 DELETE 请求\n */\n async delete<T = any>(url: string): Promise<ApiResponse<T>> {\n return this.sendRequest({ url, method: 'DELETE' });\n }\n\n // ==================== 默认认证方法(子类可重写) ====================\n\n /**\n * 用户注册\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async register(\n email: string,\n password: string,\n username: string\n ): Promise<ApiResponse<AuthResponse<TUser>>> {\n if (!this.routes.auth?.register) {\n throw new Error('Register route is not configured');\n }\n\n const response = await this.post<AuthResponse<TUser>>(this.routes.auth.register, {\n email,\n password,\n username,\n });\n\n if (response.success && response.data) {\n await this.setToken(response.data.token);\n await this.setUser(response.data.user);\n await this.onAuthSuccess?.(response.data.user, response.data.token);\n } else if (!response.success) {\n await this.onAuthError?.(response.error || 'Registration failed');\n }\n\n return response;\n }\n\n /**\n * 用户登录\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async login(email: string, password: string): Promise<ApiResponse<AuthResponse<TUser>>> {\n if (!this.routes.auth?.login) {\n throw new Error('Login route is not configured');\n }\n\n const response = await this.post<AuthResponse<TUser>>(this.routes.auth.login, {\n email,\n password,\n });\n\n if (response.success && response.data) {\n await this.setToken(response.data.token);\n await this.setUser(response.data.user);\n await this.onAuthSuccess?.(response.data.user, response.data.token);\n } else if (!response.success) {\n await this.onAuthError?.(response.error || 'Login failed');\n }\n\n return response;\n }\n\n /**\n * 用户退出登录\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async logout(): Promise<void> {\n if (this.routes.auth?.logout) {\n await this.post<void>(this.routes.auth.logout);\n }\n\n // 无论请求成功与否,都清除本地数据\n await this.clearUserData();\n await this.onLogout?.();\n }\n\n /**\n * 获取当前用户信息\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async getCurrentUser(): Promise<ApiResponse<TUser>> {\n if (!this.routes.auth?.me) {\n throw new Error('Current user route is not configured');\n }\n\n const response = await this.get<any>(this.routes.auth.me);\n\n // 统一处理响应格式:自动展开 response.data.user 为 response.data\n if (response.success && response.data) {\n // 如果 data 是嵌套的 {user: {...}, session: {...}} 格式,提取 user\n const userData = response.data.user || response.data;\n\n await this.setUser(userData);\n\n return {\n ...response,\n data: userData,\n };\n }\n\n return response;\n }\n\n // ==================== 默认用户管理方法(子类可重写) ====================\n\n /**\n * 获取用户列表\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async getUsers(params?: {\n page?: number;\n limit?: number;\n search?: string;\n }): Promise<ApiResponse<{ users: TUser[]; total: number }>> {\n if (!this.routes.users?.list) {\n throw new Error('Users list route is not configured');\n }\n\n return this.get(this.routes.users.list, params);\n }\n\n /**\n * 获取用户详情\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async getUserById(userId: string): Promise<ApiResponse<TUser>> {\n if (!this.routes.users?.detail) {\n throw new Error('User detail route is not configured');\n }\n\n return this.get(this.routes.users.detail(userId));\n }\n\n /**\n * 更新用户信息\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async updateUser(userId: string, data: Partial<TUser>): Promise<ApiResponse<TUser>> {\n if (!this.routes.users?.update) {\n throw new Error('User update route is not configured');\n }\n\n return this.put(this.routes.users.update(userId), data);\n }\n\n /**\n * 删除用户\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async deleteUser(userId: string): Promise<ApiResponse<void>> {\n if (!this.routes.users?.delete) {\n throw new Error('User delete route is not configured');\n }\n\n return this.delete(this.routes.users.delete(userId));\n }\n\n // ==================== 子类可重写的钩子方法 ====================\n\n /**\n * 请求前钩子(子类可重写)\n */\n protected async onBeforeRequest?(config: RequestConfig): Promise<RequestConfig>;\n\n /**\n * 请求后钩子(子类可重写)\n */\n protected async onAfterRequest?<T>(response: ApiResponse<T>): Promise<ApiResponse<T>>;\n\n /**\n * 请求错误钩子(子类可重写)\n */\n protected async onRequestError?(error: Error): Promise<void>;\n\n /**\n * 认证成功钩子(子类可重写)\n */\n protected async onAuthSuccess?(user: TUser, token: string): Promise<void>;\n\n /**\n * 认证失败钩子(子类可重写)\n */\n protected async onAuthError?(error: string): Promise<void>;\n\n /**\n * 登出钩子(子类可重写)\n */\n protected async onLogout?(): Promise<void>;\n}\n\n"]}
|
|
@@ -0,0 +1,290 @@
|
|
|
1
|
+
import '../chunk-BJTO5JO5.mjs';
|
|
2
|
+
|
|
3
|
+
// src/api/types.ts
|
|
4
|
+
var DEFAULT_STORAGE_KEYS = {
|
|
5
|
+
AUTH_TOKEN: "auth_token",
|
|
6
|
+
USER_DATA: "user_data",
|
|
7
|
+
REFRESH_TOKEN: "refresh_token"
|
|
8
|
+
};
|
|
9
|
+
var DEFAULT_API_ROUTES = {
|
|
10
|
+
auth: {
|
|
11
|
+
login: "/auth/login",
|
|
12
|
+
logout: "/auth/logout",
|
|
13
|
+
register: "/auth/register",
|
|
14
|
+
me: "/auth/me"
|
|
15
|
+
},
|
|
16
|
+
users: {
|
|
17
|
+
list: "/users",
|
|
18
|
+
detail: (id) => `/users/${id}`,
|
|
19
|
+
update: (id) => `/users/${id}`,
|
|
20
|
+
delete: (id) => `/users/${id}`
|
|
21
|
+
}
|
|
22
|
+
};
|
|
23
|
+
|
|
24
|
+
// src/api/BaseApiClient.ts
|
|
25
|
+
var BaseApiClient = class {
|
|
26
|
+
constructor(config) {
|
|
27
|
+
this.token = null;
|
|
28
|
+
this.user = null;
|
|
29
|
+
this.baseUrl = config.baseUrl;
|
|
30
|
+
this.storage = config.storage;
|
|
31
|
+
this.request = config.request;
|
|
32
|
+
this.storageKeys = {
|
|
33
|
+
...DEFAULT_STORAGE_KEYS,
|
|
34
|
+
...config.storageKeys || {}
|
|
35
|
+
};
|
|
36
|
+
this.routes = {
|
|
37
|
+
...DEFAULT_API_ROUTES,
|
|
38
|
+
...config.routes || {},
|
|
39
|
+
auth: {
|
|
40
|
+
...DEFAULT_API_ROUTES.auth,
|
|
41
|
+
...config.routes?.auth || {}
|
|
42
|
+
},
|
|
43
|
+
users: {
|
|
44
|
+
...DEFAULT_API_ROUTES.users,
|
|
45
|
+
...config.routes?.users || {}
|
|
46
|
+
}
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* 初始化 - 从存储中加载 token 和用户信息
|
|
51
|
+
*/
|
|
52
|
+
async init() {
|
|
53
|
+
try {
|
|
54
|
+
this.token = await this.storage.getItem(this.storageKeys.AUTH_TOKEN);
|
|
55
|
+
const userData = await this.storage.getItem(this.storageKeys.USER_DATA);
|
|
56
|
+
if (userData) {
|
|
57
|
+
this.user = JSON.parse(userData);
|
|
58
|
+
}
|
|
59
|
+
} catch (error) {
|
|
60
|
+
console.error("Failed to load auth data:", error);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
/**
|
|
64
|
+
* 设置认证 token
|
|
65
|
+
*/
|
|
66
|
+
async setToken(token) {
|
|
67
|
+
this.token = token;
|
|
68
|
+
if (token) {
|
|
69
|
+
await this.storage.setItem(this.storageKeys.AUTH_TOKEN, token);
|
|
70
|
+
} else {
|
|
71
|
+
await this.storage.removeItem(this.storageKeys.AUTH_TOKEN);
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
/**
|
|
75
|
+
* 设置用户信息
|
|
76
|
+
*/
|
|
77
|
+
async setUser(user) {
|
|
78
|
+
this.user = user;
|
|
79
|
+
if (user) {
|
|
80
|
+
await this.storage.setItem(this.storageKeys.USER_DATA, JSON.stringify(user));
|
|
81
|
+
} else {
|
|
82
|
+
await this.storage.removeItem(this.storageKeys.USER_DATA);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* 获取当前 token
|
|
87
|
+
*/
|
|
88
|
+
getToken() {
|
|
89
|
+
return this.token;
|
|
90
|
+
}
|
|
91
|
+
/**
|
|
92
|
+
* 获取当前用户
|
|
93
|
+
*/
|
|
94
|
+
getUser() {
|
|
95
|
+
return this.user;
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* 检查是否已登录
|
|
99
|
+
*/
|
|
100
|
+
async isAuthenticated() {
|
|
101
|
+
return !!this.token;
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* 清除用户数据
|
|
105
|
+
*/
|
|
106
|
+
async clearUserData() {
|
|
107
|
+
await this.storage.removeItem(this.storageKeys.AUTH_TOKEN);
|
|
108
|
+
await this.storage.removeItem(this.storageKeys.USER_DATA);
|
|
109
|
+
if (this.storageKeys.REFRESH_TOKEN) {
|
|
110
|
+
await this.storage.removeItem(this.storageKeys.REFRESH_TOKEN);
|
|
111
|
+
}
|
|
112
|
+
this.token = null;
|
|
113
|
+
this.user = null;
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* 发送请求的通用方法
|
|
117
|
+
*/
|
|
118
|
+
async sendRequest(config) {
|
|
119
|
+
try {
|
|
120
|
+
const headers = {
|
|
121
|
+
"Content-Type": "application/json",
|
|
122
|
+
...config.headers || {}
|
|
123
|
+
};
|
|
124
|
+
if (this.token) {
|
|
125
|
+
headers["Authorization"] = `Bearer ${this.token}`;
|
|
126
|
+
}
|
|
127
|
+
const response = await this.request.request({
|
|
128
|
+
...config,
|
|
129
|
+
url: `${this.baseUrl}${config.url}`,
|
|
130
|
+
headers
|
|
131
|
+
});
|
|
132
|
+
return response;
|
|
133
|
+
} catch (error) {
|
|
134
|
+
console.error("API request error:", error);
|
|
135
|
+
return {
|
|
136
|
+
success: false,
|
|
137
|
+
error: error instanceof Error ? error.message : "\u7F51\u7EDC\u9519\u8BEF\uFF0C\u8BF7\u91CD\u8BD5"
|
|
138
|
+
};
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
/**
|
|
142
|
+
* 发送 GET 请求
|
|
143
|
+
*/
|
|
144
|
+
async get(url, params) {
|
|
145
|
+
return this.sendRequest({ url, method: "GET", params });
|
|
146
|
+
}
|
|
147
|
+
/**
|
|
148
|
+
* 发送 POST 请求
|
|
149
|
+
*/
|
|
150
|
+
async post(url, body) {
|
|
151
|
+
return this.sendRequest({ url, method: "POST", body });
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* 发送 PUT 请求
|
|
155
|
+
*/
|
|
156
|
+
async put(url, body) {
|
|
157
|
+
return this.sendRequest({ url, method: "PUT", body });
|
|
158
|
+
}
|
|
159
|
+
/**
|
|
160
|
+
* 发送 PATCH 请求
|
|
161
|
+
*/
|
|
162
|
+
async patch(url, body) {
|
|
163
|
+
return this.sendRequest({ url, method: "PATCH", body });
|
|
164
|
+
}
|
|
165
|
+
/**
|
|
166
|
+
* 发送 DELETE 请求
|
|
167
|
+
*/
|
|
168
|
+
async delete(url) {
|
|
169
|
+
return this.sendRequest({ url, method: "DELETE" });
|
|
170
|
+
}
|
|
171
|
+
// ==================== 默认认证方法(子类可重写) ====================
|
|
172
|
+
/**
|
|
173
|
+
* 用户注册
|
|
174
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
175
|
+
*/
|
|
176
|
+
async register(email, password, username) {
|
|
177
|
+
if (!this.routes.auth?.register) {
|
|
178
|
+
throw new Error("Register route is not configured");
|
|
179
|
+
}
|
|
180
|
+
const response = await this.post(this.routes.auth.register, {
|
|
181
|
+
email,
|
|
182
|
+
password,
|
|
183
|
+
username
|
|
184
|
+
});
|
|
185
|
+
if (response.success && response.data) {
|
|
186
|
+
await this.setToken(response.data.token);
|
|
187
|
+
await this.setUser(response.data.user);
|
|
188
|
+
await this.onAuthSuccess?.(response.data.user, response.data.token);
|
|
189
|
+
} else if (!response.success) {
|
|
190
|
+
await this.onAuthError?.(response.error || "Registration failed");
|
|
191
|
+
}
|
|
192
|
+
return response;
|
|
193
|
+
}
|
|
194
|
+
/**
|
|
195
|
+
* 用户登录
|
|
196
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
197
|
+
*/
|
|
198
|
+
async login(email, password) {
|
|
199
|
+
if (!this.routes.auth?.login) {
|
|
200
|
+
throw new Error("Login route is not configured");
|
|
201
|
+
}
|
|
202
|
+
const response = await this.post(this.routes.auth.login, {
|
|
203
|
+
email,
|
|
204
|
+
password
|
|
205
|
+
});
|
|
206
|
+
if (response.success && response.data) {
|
|
207
|
+
await this.setToken(response.data.token);
|
|
208
|
+
await this.setUser(response.data.user);
|
|
209
|
+
await this.onAuthSuccess?.(response.data.user, response.data.token);
|
|
210
|
+
} else if (!response.success) {
|
|
211
|
+
await this.onAuthError?.(response.error || "Login failed");
|
|
212
|
+
}
|
|
213
|
+
return response;
|
|
214
|
+
}
|
|
215
|
+
/**
|
|
216
|
+
* 用户退出登录
|
|
217
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
218
|
+
*/
|
|
219
|
+
async logout() {
|
|
220
|
+
if (this.routes.auth?.logout) {
|
|
221
|
+
await this.post(this.routes.auth.logout);
|
|
222
|
+
}
|
|
223
|
+
await this.clearUserData();
|
|
224
|
+
await this.onLogout?.();
|
|
225
|
+
}
|
|
226
|
+
/**
|
|
227
|
+
* 获取当前用户信息
|
|
228
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
229
|
+
*/
|
|
230
|
+
async getCurrentUser() {
|
|
231
|
+
if (!this.routes.auth?.me) {
|
|
232
|
+
throw new Error("Current user route is not configured");
|
|
233
|
+
}
|
|
234
|
+
const response = await this.get(this.routes.auth.me);
|
|
235
|
+
if (response.success && response.data) {
|
|
236
|
+
const userData = response.data.user || response.data;
|
|
237
|
+
await this.setUser(userData);
|
|
238
|
+
return {
|
|
239
|
+
...response,
|
|
240
|
+
data: userData
|
|
241
|
+
};
|
|
242
|
+
}
|
|
243
|
+
return response;
|
|
244
|
+
}
|
|
245
|
+
// ==================== 默认用户管理方法(子类可重写) ====================
|
|
246
|
+
/**
|
|
247
|
+
* 获取用户列表
|
|
248
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
249
|
+
*/
|
|
250
|
+
async getUsers(params) {
|
|
251
|
+
if (!this.routes.users?.list) {
|
|
252
|
+
throw new Error("Users list route is not configured");
|
|
253
|
+
}
|
|
254
|
+
return this.get(this.routes.users.list, params);
|
|
255
|
+
}
|
|
256
|
+
/**
|
|
257
|
+
* 获取用户详情
|
|
258
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
259
|
+
*/
|
|
260
|
+
async getUserById(userId) {
|
|
261
|
+
if (!this.routes.users?.detail) {
|
|
262
|
+
throw new Error("User detail route is not configured");
|
|
263
|
+
}
|
|
264
|
+
return this.get(this.routes.users.detail(userId));
|
|
265
|
+
}
|
|
266
|
+
/**
|
|
267
|
+
* 更新用户信息
|
|
268
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
269
|
+
*/
|
|
270
|
+
async updateUser(userId, data) {
|
|
271
|
+
if (!this.routes.users?.update) {
|
|
272
|
+
throw new Error("User update route is not configured");
|
|
273
|
+
}
|
|
274
|
+
return this.put(this.routes.users.update(userId), data);
|
|
275
|
+
}
|
|
276
|
+
/**
|
|
277
|
+
* 删除用户
|
|
278
|
+
* 默认实现,子类可重写以添加自定义逻辑
|
|
279
|
+
*/
|
|
280
|
+
async deleteUser(userId) {
|
|
281
|
+
if (!this.routes.users?.delete) {
|
|
282
|
+
throw new Error("User delete route is not configured");
|
|
283
|
+
}
|
|
284
|
+
return this.delete(this.routes.users.delete(userId));
|
|
285
|
+
}
|
|
286
|
+
};
|
|
287
|
+
|
|
288
|
+
export { BaseApiClient, DEFAULT_API_ROUTES, DEFAULT_STORAGE_KEYS };
|
|
289
|
+
//# sourceMappingURL=index.mjs.map
|
|
290
|
+
//# sourceMappingURL=index.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/api/types.ts","../../src/api/BaseApiClient.ts"],"names":[],"mappings":"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","file":"index.mjs","sourcesContent":["/**\n * API 通用类型定义\n * Common API Types\n */\n\n/**\n * API 响应类型\n * Generic API Response Format\n */\nexport interface ApiResponse<T = any> {\n success: boolean;\n data?: T;\n error?: string;\n message?: string;\n meta?: {\n pagination?: {\n page: number;\n limit: number;\n total: number;\n pages: number;\n };\n [key: string]: any;\n };\n}\n\n/**\n * 分页参数类型\n * Pagination Parameters\n */\nexport interface PaginationParams {\n page?: number;\n limit?: number;\n search?: string;\n sortBy?: string;\n sortOrder?: 'asc' | 'desc';\n}\n\n/**\n * 认证响应类型(泛型)\n * Generic Auth Response\n */\nexport interface AuthResponse<TUser = any> {\n user: TUser;\n token: string;\n}\n\n/**\n * 默认存储键常量\n * Default Storage Keys\n */\nexport const DEFAULT_STORAGE_KEYS = {\n AUTH_TOKEN: 'auth_token',\n USER_DATA: 'user_data',\n REFRESH_TOKEN: 'refresh_token',\n} as const;\n\n/**\n * 存储键类型(可扩展)\n * Storage Keys Type\n */\nexport type StorageKeys = typeof DEFAULT_STORAGE_KEYS;\n\n/**\n * API 路由配置\n * API Routes Configuration\n */\nexport interface ApiRoutes {\n /** 认证相关路由 */\n auth?: {\n login?: string;\n logout?: string;\n register?: string;\n me?: string;\n };\n /** 用户相关路由 */\n users?: {\n list?: string;\n detail?: (id: string) => string;\n update?: (id: string) => string;\n delete?: (id: string) => string;\n };\n /** 自定义路由 */\n [key: string]: any;\n}\n\n/**\n * 默认 API 路由\n */\nexport const DEFAULT_API_ROUTES: ApiRoutes = {\n auth: {\n login: '/auth/login',\n logout: '/auth/logout',\n register: '/auth/register',\n me: '/auth/me',\n },\n users: {\n list: '/users',\n detail: (id: string) => `/users/${id}`,\n update: (id: string) => `/users/${id}`,\n delete: (id: string) => `/users/${id}`,\n },\n};\n","/**\n * 基础 API 客户端(泛型)\n * Base API Client (Generic)\n *\n * 提供统一的 API 调用逻辑,通过适配器模式支持多平台\n * 使用泛型 TUser 支持不同项目的用户类型\n */\n\nimport type { StorageAdapter } from '../storage';\nimport type { RequestAdapter, RequestConfig } from '../request';\nimport type { ApiResponse, AuthResponse, StorageKeys, ApiRoutes } from './types';\nimport { DEFAULT_STORAGE_KEYS, DEFAULT_API_ROUTES } from './types';\n\n/**\n * 基础 API 客户端配置\n */\nexport interface BaseApiClientConfig {\n /** 基础 URL */\n baseUrl: string;\n /** 存储适配器 */\n storage: StorageAdapter;\n /** 请求适配器 */\n request: RequestAdapter;\n /** 自定义存储键(可选) */\n storageKeys?: Partial<StorageKeys>;\n /** API 路由配置(可选) */\n routes?: ApiRoutes;\n}\n\n/**\n * 基础 API 客户端\n * @template TUser 用户类型(泛型)\n */\nexport class BaseApiClient<TUser = any> {\n protected token: string | null = null;\n protected user: TUser | null = null;\n protected baseUrl: string;\n protected storage: StorageAdapter;\n protected request: RequestAdapter;\n protected storageKeys: StorageKeys;\n protected routes: ApiRoutes;\n\n constructor(config: BaseApiClientConfig) {\n this.baseUrl = config.baseUrl;\n this.storage = config.storage;\n this.request = config.request;\n this.storageKeys = {\n ...DEFAULT_STORAGE_KEYS,\n ...(config.storageKeys || {}),\n };\n this.routes = {\n ...DEFAULT_API_ROUTES,\n ...(config.routes || {}),\n auth: {\n ...DEFAULT_API_ROUTES.auth,\n ...(config.routes?.auth || {}),\n },\n users: {\n ...DEFAULT_API_ROUTES.users,\n ...(config.routes?.users || {}),\n },\n };\n }\n\n /**\n * 初始化 - 从存储中加载 token 和用户信息\n */\n async init(): Promise<void> {\n try {\n this.token = await this.storage.getItem(this.storageKeys.AUTH_TOKEN);\n const userData = await this.storage.getItem(this.storageKeys.USER_DATA);\n if (userData) {\n this.user = JSON.parse(userData) as TUser;\n }\n } catch (error) {\n console.error('Failed to load auth data:', error);\n }\n }\n\n /**\n * 设置认证 token\n */\n async setToken(token: string | null): Promise<void> {\n this.token = token;\n if (token) {\n await this.storage.setItem(this.storageKeys.AUTH_TOKEN, token);\n } else {\n await this.storage.removeItem(this.storageKeys.AUTH_TOKEN);\n }\n }\n\n /**\n * 设置用户信息\n */\n async setUser(user: TUser | null): Promise<void> {\n this.user = user;\n if (user) {\n await this.storage.setItem(this.storageKeys.USER_DATA, JSON.stringify(user));\n } else {\n await this.storage.removeItem(this.storageKeys.USER_DATA);\n }\n }\n\n /**\n * 获取当前 token\n */\n getToken(): string | null {\n return this.token;\n }\n\n /**\n * 获取当前用户\n */\n getUser(): TUser | null {\n return this.user;\n }\n\n /**\n * 检查是否已登录\n */\n async isAuthenticated(): Promise<boolean> {\n return !!this.token;\n }\n\n /**\n * 清除用户数据\n */\n async clearUserData(): Promise<void> {\n await this.storage.removeItem(this.storageKeys.AUTH_TOKEN);\n await this.storage.removeItem(this.storageKeys.USER_DATA);\n if (this.storageKeys.REFRESH_TOKEN) {\n await this.storage.removeItem(this.storageKeys.REFRESH_TOKEN);\n }\n this.token = null;\n this.user = null;\n }\n\n /**\n * 发送请求的通用方法\n */\n protected async sendRequest<T = any>(config: RequestConfig): Promise<ApiResponse<T>> {\n try {\n const headers: Record<string, string> = {\n 'Content-Type': 'application/json',\n ...(config.headers || {}),\n };\n\n // 添加认证 token\n if (this.token) {\n headers['Authorization'] = `Bearer ${this.token}`;\n }\n\n const response = await this.request.request<ApiResponse<T>>({\n ...config,\n url: `${this.baseUrl}${config.url}`,\n headers,\n });\n\n return response;\n } catch (error) {\n console.error('API request error:', error);\n return {\n success: false,\n error: error instanceof Error ? error.message : '网络错误,请重试',\n };\n }\n }\n\n /**\n * 发送 GET 请求\n */\n async get<T = any>(url: string, params?: Record<string, any>): Promise<ApiResponse<T>> {\n return this.sendRequest({ url, method: 'GET', params });\n }\n\n /**\n * 发送 POST 请求\n */\n async post<T = any>(url: string, body?: any): Promise<ApiResponse<T>> {\n return this.sendRequest({ url, method: 'POST', body });\n }\n\n /**\n * 发送 PUT 请求\n */\n async put<T = any>(url: string, body?: any): Promise<ApiResponse<T>> {\n return this.sendRequest({ url, method: 'PUT', body });\n }\n\n /**\n * 发送 PATCH 请求\n */\n async patch<T = any>(url: string, body?: any): Promise<ApiResponse<T>> {\n return this.sendRequest({ url, method: 'PATCH', body });\n }\n\n /**\n * 发送 DELETE 请求\n */\n async delete<T = any>(url: string): Promise<ApiResponse<T>> {\n return this.sendRequest({ url, method: 'DELETE' });\n }\n\n // ==================== 默认认证方法(子类可重写) ====================\n\n /**\n * 用户注册\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async register(\n email: string,\n password: string,\n username: string\n ): Promise<ApiResponse<AuthResponse<TUser>>> {\n if (!this.routes.auth?.register) {\n throw new Error('Register route is not configured');\n }\n\n const response = await this.post<AuthResponse<TUser>>(this.routes.auth.register, {\n email,\n password,\n username,\n });\n\n if (response.success && response.data) {\n await this.setToken(response.data.token);\n await this.setUser(response.data.user);\n await this.onAuthSuccess?.(response.data.user, response.data.token);\n } else if (!response.success) {\n await this.onAuthError?.(response.error || 'Registration failed');\n }\n\n return response;\n }\n\n /**\n * 用户登录\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async login(email: string, password: string): Promise<ApiResponse<AuthResponse<TUser>>> {\n if (!this.routes.auth?.login) {\n throw new Error('Login route is not configured');\n }\n\n const response = await this.post<AuthResponse<TUser>>(this.routes.auth.login, {\n email,\n password,\n });\n\n if (response.success && response.data) {\n await this.setToken(response.data.token);\n await this.setUser(response.data.user);\n await this.onAuthSuccess?.(response.data.user, response.data.token);\n } else if (!response.success) {\n await this.onAuthError?.(response.error || 'Login failed');\n }\n\n return response;\n }\n\n /**\n * 用户退出登录\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async logout(): Promise<void> {\n if (this.routes.auth?.logout) {\n await this.post<void>(this.routes.auth.logout);\n }\n\n // 无论请求成功与否,都清除本地数据\n await this.clearUserData();\n await this.onLogout?.();\n }\n\n /**\n * 获取当前用户信息\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async getCurrentUser(): Promise<ApiResponse<TUser>> {\n if (!this.routes.auth?.me) {\n throw new Error('Current user route is not configured');\n }\n\n const response = await this.get<any>(this.routes.auth.me);\n\n // 统一处理响应格式:自动展开 response.data.user 为 response.data\n if (response.success && response.data) {\n // 如果 data 是嵌套的 {user: {...}, session: {...}} 格式,提取 user\n const userData = response.data.user || response.data;\n\n await this.setUser(userData);\n\n return {\n ...response,\n data: userData,\n };\n }\n\n return response;\n }\n\n // ==================== 默认用户管理方法(子类可重写) ====================\n\n /**\n * 获取用户列表\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async getUsers(params?: {\n page?: number;\n limit?: number;\n search?: string;\n }): Promise<ApiResponse<{ users: TUser[]; total: number }>> {\n if (!this.routes.users?.list) {\n throw new Error('Users list route is not configured');\n }\n\n return this.get(this.routes.users.list, params);\n }\n\n /**\n * 获取用户详情\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async getUserById(userId: string): Promise<ApiResponse<TUser>> {\n if (!this.routes.users?.detail) {\n throw new Error('User detail route is not configured');\n }\n\n return this.get(this.routes.users.detail(userId));\n }\n\n /**\n * 更新用户信息\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async updateUser(userId: string, data: Partial<TUser>): Promise<ApiResponse<TUser>> {\n if (!this.routes.users?.update) {\n throw new Error('User update route is not configured');\n }\n\n return this.put(this.routes.users.update(userId), data);\n }\n\n /**\n * 删除用户\n * 默认实现,子类可重写以添加自定义逻辑\n */\n async deleteUser(userId: string): Promise<ApiResponse<void>> {\n if (!this.routes.users?.delete) {\n throw new Error('User delete route is not configured');\n }\n\n return this.delete(this.routes.users.delete(userId));\n }\n\n // ==================== 子类可重写的钩子方法 ====================\n\n /**\n * 请求前钩子(子类可重写)\n */\n protected async onBeforeRequest?(config: RequestConfig): Promise<RequestConfig>;\n\n /**\n * 请求后钩子(子类可重写)\n */\n protected async onAfterRequest?<T>(response: ApiResponse<T>): Promise<ApiResponse<T>>;\n\n /**\n * 请求错误钩子(子类可重写)\n */\n protected async onRequestError?(error: Error): Promise<void>;\n\n /**\n * 认证成功钩子(子类可重写)\n */\n protected async onAuthSuccess?(user: TUser, token: string): Promise<void>;\n\n /**\n * 认证失败钩子(子类可重写)\n */\n protected async onAuthError?(error: string): Promise<void>;\n\n /**\n * 登出钩子(子类可重写)\n */\n protected async onLogout?(): Promise<void>;\n}\n\n"]}
|