sa2kit 1.6.57 → 1.6.58

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.
Files changed (145) hide show
  1. package/dist/AliyunOSSProvider-KSYW2IOG.js +15 -0
  2. package/dist/{AliyunOSSProvider-2FARPAQD.js.map → AliyunOSSProvider-KSYW2IOG.js.map} +1 -1
  3. package/dist/AliyunOSSProvider-TBK3G7YK.mjs +6 -0
  4. package/dist/{AliyunOSSProvider-UMVGVBDJ.mjs.map → AliyunOSSProvider-TBK3G7YK.mjs.map} +1 -1
  5. package/dist/LocalStorageProvider-2DGYRQAB.mjs +6 -0
  6. package/dist/{LocalStorageProvider-PYOHETJV.mjs.map → LocalStorageProvider-2DGYRQAB.mjs.map} +1 -1
  7. package/dist/LocalStorageProvider-SSRW3ZJW.js +15 -0
  8. package/dist/{LocalStorageProvider-JQF5WK5H.js.map → LocalStorageProvider-SSRW3ZJW.js.map} +1 -1
  9. package/dist/UniversalFileService-336GFY6N.mjs +6 -0
  10. package/dist/{UniversalFileService-TNYKO6JN.mjs.map → UniversalFileService-336GFY6N.mjs.map} +1 -1
  11. package/dist/UniversalFileService-J6ET6KZK.js +15 -0
  12. package/dist/{UniversalFileService-RBV6EN5J.js.map → UniversalFileService-J6ET6KZK.js.map} +1 -1
  13. package/dist/calendar/index.js +11 -11
  14. package/dist/calendar/index.mjs +4 -4
  15. package/dist/chunk-25OFOKNF.js +171 -0
  16. package/dist/chunk-25OFOKNF.js.map +1 -0
  17. package/dist/chunk-3DXPQ4YV.mjs +165 -0
  18. package/dist/chunk-3DXPQ4YV.mjs.map +1 -0
  19. package/dist/{chunk-4VJQZSPU.mjs → chunk-3NHAT7D4.mjs} +3 -4
  20. package/dist/chunk-3NHAT7D4.mjs.map +1 -0
  21. package/dist/{chunk-3JW4X3AC.mjs → chunk-622Y6LTH.mjs} +3 -3
  22. package/dist/{chunk-3JW4X3AC.mjs.map → chunk-622Y6LTH.mjs.map} +1 -1
  23. package/dist/chunk-CIVO4R6N.mjs +37 -0
  24. package/dist/chunk-CIVO4R6N.mjs.map +1 -0
  25. package/dist/{chunk-6BJ76BYC.mjs → chunk-EGJPS7OL.mjs} +3 -3
  26. package/dist/{chunk-6BJ76BYC.mjs.map → chunk-EGJPS7OL.mjs.map} +1 -1
  27. package/dist/{chunk-U7AQC2Z7.js → chunk-EKIOJEJK.js} +1179 -2002
  28. package/dist/chunk-EKIOJEJK.js.map +1 -0
  29. package/dist/{chunk-53WLQ22S.js → chunk-EOWTOG7Y.js} +6 -6
  30. package/dist/{chunk-53WLQ22S.js.map → chunk-EOWTOG7Y.js.map} +1 -1
  31. package/dist/chunk-HDMIOOZY.mjs +546 -0
  32. package/dist/chunk-HDMIOOZY.mjs.map +1 -0
  33. package/dist/{chunk-MZKATHB7.js → chunk-HHVDOIPV.js} +4 -4
  34. package/dist/{chunk-MZKATHB7.js.map → chunk-HHVDOIPV.js.map} +1 -1
  35. package/dist/chunk-HJ6MH7J7.js +552 -0
  36. package/dist/chunk-HJ6MH7J7.js.map +1 -0
  37. package/dist/chunk-KH6RQ4J5.js +28 -0
  38. package/dist/chunk-KH6RQ4J5.js.map +1 -0
  39. package/dist/{chunk-35CXIK5Y.js → chunk-NCOXT7SK.js} +11 -11
  40. package/dist/{chunk-35CXIK5Y.js.map → chunk-NCOXT7SK.js.map} +1 -1
  41. package/dist/chunk-NZZZUMMX.mjs +784 -0
  42. package/dist/chunk-NZZZUMMX.mjs.map +1 -0
  43. package/dist/{chunk-OBIPI4GU.mjs → chunk-OFYBMMWT.mjs} +4 -4
  44. package/dist/{chunk-OBIPI4GU.mjs.map → chunk-OFYBMMWT.mjs.map} +1 -1
  45. package/dist/{chunk-W2NCOORK.js → chunk-PVLLRDUT.js} +951 -145
  46. package/dist/chunk-PVLLRDUT.js.map +1 -0
  47. package/dist/chunk-Q5EDCKQA.js +336 -0
  48. package/dist/chunk-Q5EDCKQA.js.map +1 -0
  49. package/dist/{chunk-6NMIKAE7.mjs → chunk-SNBILYSH.mjs} +5 -5
  50. package/dist/{chunk-6NMIKAE7.mjs.map → chunk-SNBILYSH.mjs.map} +1 -1
  51. package/dist/{chunk-IPY26RQH.js → chunk-UVHPCLP6.js} +5 -5
  52. package/dist/{chunk-IPY26RQH.js.map → chunk-UVHPCLP6.js.map} +1 -1
  53. package/dist/{chunk-V6BXO6ZS.mjs → chunk-WGD2NBVR.mjs} +835 -38
  54. package/dist/chunk-WGD2NBVR.mjs.map +1 -0
  55. package/dist/{chunk-HYZ5C6FY.mjs → chunk-WS3QZYBI.mjs} +1175 -1996
  56. package/dist/chunk-WS3QZYBI.mjs.map +1 -0
  57. package/dist/{chunk-UR5TU4MW.mjs → chunk-XLR6QUDR.mjs} +3 -3
  58. package/dist/{chunk-UR5TU4MW.mjs.map → chunk-XLR6QUDR.mjs.map} +1 -1
  59. package/dist/chunk-YMS6BPXS.js +807 -0
  60. package/dist/chunk-YMS6BPXS.js.map +1 -0
  61. package/dist/chunk-YOTQG4NP.mjs +314 -0
  62. package/dist/chunk-YOTQG4NP.mjs.map +1 -0
  63. package/dist/chunk-ZGVB35L2.mjs +25 -0
  64. package/dist/chunk-ZGVB35L2.mjs.map +1 -0
  65. package/dist/chunk-ZRAW3HXA.js +43 -0
  66. package/dist/chunk-ZRAW3HXA.js.map +1 -0
  67. package/dist/{chunk-4XXIBWCO.js → chunk-ZRWED7Q6.js} +66 -66
  68. package/dist/{chunk-4XXIBWCO.js.map → chunk-ZRWED7Q6.js.map} +1 -1
  69. package/dist/{chunk-DVENFCQY.js → chunk-ZWQJSZEY.js} +4 -5
  70. package/dist/chunk-ZWQJSZEY.js.map +1 -0
  71. package/dist/components/index.js +104 -104
  72. package/dist/components/index.mjs +4 -4
  73. package/dist/index.js +146 -148
  74. package/dist/index.js.map +1 -1
  75. package/dist/index.mjs +8 -11
  76. package/dist/index.mjs.map +1 -1
  77. package/dist/logger/index.js +7 -7
  78. package/dist/logger/index.mjs +1 -4
  79. package/dist/mikuFusionGame/index.js +4 -4
  80. package/dist/mikuFusionGame/index.mjs +3 -3
  81. package/dist/portfolio/index.js +10 -10
  82. package/dist/portfolio/index.mjs +5 -5
  83. package/dist/showmasterpiece/db/index.js +42 -42
  84. package/dist/showmasterpiece/db/index.mjs +1 -1
  85. package/dist/showmasterpiece/index.js +143 -144
  86. package/dist/showmasterpiece/index.js.map +1 -1
  87. package/dist/showmasterpiece/index.mjs +7 -9
  88. package/dist/showmasterpiece/index.mjs.map +1 -1
  89. package/dist/showmasterpiece/logic/index.js +19 -19
  90. package/dist/showmasterpiece/logic/index.mjs +2 -2
  91. package/dist/showmasterpiece/server/index.js +42 -42
  92. package/dist/showmasterpiece/server/index.mjs +1 -1
  93. package/dist/showmasterpiece/ui/web/index.js +34 -34
  94. package/dist/showmasterpiece/ui/web/index.mjs +6 -6
  95. package/dist/universalExport/server/index.js +2 -4
  96. package/dist/universalExport/server/index.js.map +1 -1
  97. package/dist/universalExport/server/index.mjs +1 -3
  98. package/dist/universalExport/server/index.mjs.map +1 -1
  99. package/dist/universalFile/index.js +6 -9
  100. package/dist/universalFile/index.js.map +1 -1
  101. package/dist/universalFile/index.mjs +1 -5
  102. package/dist/universalFile/index.mjs.map +1 -1
  103. package/dist/universalFile/server/index.js +31 -64
  104. package/dist/universalFile/server/index.js.map +1 -1
  105. package/dist/universalFile/server/index.mjs +7 -42
  106. package/dist/universalFile/server/index.mjs.map +1 -1
  107. package/dist/utils/index.js +11 -11
  108. package/dist/utils/index.mjs +2 -2
  109. package/package.json +1 -1
  110. package/dist/AliyunOSSProvider-2FARPAQD.js +0 -15
  111. package/dist/AliyunOSSProvider-UMVGVBDJ.mjs +0 -9
  112. package/dist/LocalStorageProvider-JQF5WK5H.js +0 -15
  113. package/dist/LocalStorageProvider-PYOHETJV.mjs +0 -9
  114. package/dist/UniversalFileService-RBV6EN5J.js +0 -15
  115. package/dist/UniversalFileService-TNYKO6JN.mjs +0 -9
  116. package/dist/chunk-4NFOSCM6.js +0 -34
  117. package/dist/chunk-4NFOSCM6.js.map +0 -1
  118. package/dist/chunk-4VJQZSPU.mjs.map +0 -1
  119. package/dist/chunk-6AHYPPUP.js +0 -344
  120. package/dist/chunk-6AHYPPUP.js.map +0 -1
  121. package/dist/chunk-76V7EKBX.mjs +0 -796
  122. package/dist/chunk-76V7EKBX.mjs.map +0 -1
  123. package/dist/chunk-ACLOJXXE.js +0 -195
  124. package/dist/chunk-ACLOJXXE.js.map +0 -1
  125. package/dist/chunk-AEXPAH7Z.mjs +0 -32
  126. package/dist/chunk-AEXPAH7Z.mjs.map +0 -1
  127. package/dist/chunk-CFGX3EKK.js +0 -560
  128. package/dist/chunk-CFGX3EKK.js.map +0 -1
  129. package/dist/chunk-D2HXMGXS.js +0 -46
  130. package/dist/chunk-D2HXMGXS.js.map +0 -1
  131. package/dist/chunk-DVENFCQY.js.map +0 -1
  132. package/dist/chunk-HYZ5C6FY.mjs.map +0 -1
  133. package/dist/chunk-K7WNCB4V.mjs +0 -554
  134. package/dist/chunk-K7WNCB4V.mjs.map +0 -1
  135. package/dist/chunk-L4ZYBFB2.mjs +0 -44
  136. package/dist/chunk-L4ZYBFB2.mjs.map +0 -1
  137. package/dist/chunk-M4HGHTIC.js +0 -820
  138. package/dist/chunk-M4HGHTIC.js.map +0 -1
  139. package/dist/chunk-PXWDQFWV.mjs +0 -192
  140. package/dist/chunk-PXWDQFWV.mjs.map +0 -1
  141. package/dist/chunk-U7AQC2Z7.js.map +0 -1
  142. package/dist/chunk-V6BXO6ZS.mjs.map +0 -1
  143. package/dist/chunk-VTGPHE4Z.mjs +0 -322
  144. package/dist/chunk-VTGPHE4Z.mjs.map +0 -1
  145. package/dist/chunk-W2NCOORK.js.map +0 -1
@@ -1,554 +0,0 @@
1
- import { init_types, StorageProviderError } from './chunk-AEXPAH7Z.mjs';
2
- import { init_logger, createLogger } from './chunk-PXWDQFWV.mjs';
3
- import { __esm } from './chunk-WMJKH4XE.mjs';
4
- import OSS from 'ali-oss';
5
-
6
- var logger, AliyunOSSProvider;
7
- var init_AliyunOSSProvider = __esm({
8
- "src/universalFile/server/providers/AliyunOSSProvider.ts"() {
9
- init_logger();
10
- init_types();
11
- logger = createLogger("AliyunOSSProvider");
12
- AliyunOSSProvider = class {
13
- constructor() {
14
- this.type = "aliyun-oss";
15
- this.config = null;
16
- this.client = null;
17
- this.isInitialized = false;
18
- }
19
- /**
20
- * 初始化存储提供者
21
- */
22
- async initialize(config) {
23
- return this.reinitialize(config);
24
- }
25
- /**
26
- * 重新初始化存储提供者(支持配置热更新)
27
- */
28
- async reinitialize(config) {
29
- logger.info("\u{1F504} [AliyunOSSProvider] \u91CD\u65B0\u521D\u59CB\u5316OSS\u5BA2\u6237\u7AEF", config);
30
- if (config.type !== "aliyun-oss") {
31
- throw new StorageProviderError("\u914D\u7F6E\u7C7B\u578B\u4E0D\u5339\u914D\uFF1A\u671F\u671B aliyun-oss");
32
- }
33
- const newConfig = config;
34
- const configChanged = !this.config || this.config.region !== newConfig.region || this.config.bucket !== newConfig.bucket || this.config.accessKeyId !== newConfig.accessKeyId || this.config.accessKeySecret !== newConfig.accessKeySecret || this.config.customDomain !== newConfig.customDomain || this.config.secure !== newConfig.secure || this.config.internal !== newConfig.internal;
35
- if (configChanged) {
36
- logger.info("\u{1F504} [AliyunOSSProvider] \u68C0\u6D4B\u5230\u914D\u7F6E\u53D8\u5316\uFF0C\u91CD\u65B0\u521D\u59CB\u5316OSS\u5BA2\u6237\u7AEF");
37
- logger.info(
38
- "\u2601\uFE0F [AliyunOSSProvider] \u65B0\u914D\u7F6E: bucket=" + newConfig.bucket + ", region=" + newConfig.region
39
- );
40
- } else if (this.isInitialized) {
41
- logger.info("\u2139\uFE0F [AliyunOSSProvider] \u914D\u7F6E\u672A\u53D8\u5316\uFF0C\u8DF3\u8FC7\u91CD\u65B0\u521D\u59CB\u5316");
42
- return;
43
- }
44
- this.config = newConfig;
45
- logger.info("\u2601\uFE0F [AliyunOSSProvider] " + (this.isInitialized ? "\u91CD\u65B0" : "") + "\u521D\u59CB\u5316\u963F\u91CC\u4E91OSS");
46
- logger.info("\u2601\uFE0F [AliyunOSSProvider] " + (this.config ? JSON.stringify(this.config) : ""));
47
- try {
48
- this.validateConfig();
49
- if (!this.config.region || typeof this.config.region !== "string") {
50
- throw new Error("OSS region \u5FC5\u987B\u662F\u6709\u6548\u7684\u5B57\u7B26\u4E32");
51
- }
52
- if (!this.config.bucket || typeof this.config.bucket !== "string") {
53
- throw new Error("OSS bucket \u5FC5\u987B\u662F\u6709\u6548\u7684\u5B57\u7B26\u4E32");
54
- }
55
- if (!this.config.accessKeyId || typeof this.config.accessKeyId !== "string") {
56
- throw new Error("OSS accessKeyId \u5FC5\u987B\u662F\u6709\u6548\u7684\u5B57\u7B26\u4E32");
57
- }
58
- if (!this.config.accessKeySecret || typeof this.config.accessKeySecret !== "string") {
59
- throw new Error("OSS accessKeySecret \u5FC5\u987B\u662F\u6709\u6548\u7684\u5B57\u7B26\u4E32");
60
- }
61
- logger.info(`\u2601\uFE0F [AliyunOSSProvider] \u521B\u5EFAOSS\u5BA2\u6237\u7AEF\u914D\u7F6E:`, {
62
- region: this.config.region,
63
- bucket: this.config.bucket,
64
- hasAccessKeyId: !!this.config.accessKeyId,
65
- hasAccessKeySecret: !!this.config.accessKeySecret,
66
- secure: this.config.secure !== false,
67
- internal: this.config.internal || false,
68
- cname: !!this.config.customDomain,
69
- endpoint: this.config.customDomain || "\u9ED8\u8BA4\u7AEF\u70B9"
70
- });
71
- this.client = new OSS({
72
- region: this.config.region,
73
- bucket: this.config.bucket,
74
- accessKeyId: this.config.accessKeyId,
75
- accessKeySecret: this.config.accessKeySecret,
76
- secure: this.config.secure !== false,
77
- // 默认使用HTTPS
78
- internal: this.config.internal || false,
79
- // 默认使用公网
80
- timeout: 3e5,
81
- // 5分钟超时
82
- cname: !!this.config.customDomain,
83
- // 是否使用自定义域名
84
- endpoint: this.config.customDomain || void 0
85
- });
86
- if (!this.client) {
87
- throw new Error("OSS\u5BA2\u6237\u7AEF\u521B\u5EFA\u5931\u8D25");
88
- }
89
- logger.info(`\u2601\uFE0F [AliyunOSSProvider] OSS\u5BA2\u6237\u7AEF\u521B\u5EFA\u6210\u529F`);
90
- this.isInitialized = true;
91
- logger.info("\u2705 [AliyunOSSProvider] \u963F\u91CC\u4E91OSS" + (configChanged ? "\u91CD\u65B0" : "") + "\u521D\u59CB\u5316\u5B8C\u6210");
92
- } catch (error) {
93
- logger.error("\u274C [AliyunOSSProvider] \u963F\u91CC\u4E91OSS\u521D\u59CB\u5316\u5931\u8D25:", error);
94
- throw new StorageProviderError(`\u963F\u91CC\u4E91OSS\u521D\u59CB\u5316\u5931\u8D25`);
95
- }
96
- }
97
- /**
98
- * 上传文件
99
- */
100
- async upload(fileInfo, filePath) {
101
- this.ensureInitialized();
102
- const startTime = Date.now();
103
- logger.info("\u{1F4E4} [AliyunOSSProvider] \u5F00\u59CB\u4E0A\u4F20\u6587\u4EF6\u5230OSS: " + filePath);
104
- try {
105
- const buffer = Buffer.from(await fileInfo.file.arrayBuffer());
106
- const options = {
107
- headers: {
108
- "Content-Type": fileInfo.file.type || "application/octet-stream",
109
- "Content-Length": fileInfo.file.size.toString()
110
- },
111
- meta: {
112
- uid: 0,
113
- // 必需字段
114
- pid: 0,
115
- // 必需字段
116
- originalName: encodeURIComponent(fileInfo.file.name),
117
- moduleId: fileInfo.moduleId,
118
- businessId: fileInfo.businessId || "",
119
- uploadTime: (/* @__PURE__ */ new Date()).toISOString(),
120
- // 对元数据进行编码处理,避免中文字符问题
121
- ...this.encodeMetadata(fileInfo.metadata || {})
122
- }
123
- };
124
- let result;
125
- if (fileInfo.file.size > 100 * 1024 * 1024) {
126
- logger.info(
127
- "\u{1F4E6} [AliyunOSSProvider] \u4F7F\u7528\u5206\u7247\u4E0A\u4F20\u5927\u6587\u4EF6: " + filePath + ", \u5927\u5C0F: " + fileInfo.file.size
128
- );
129
- result = await this.multipartUpload(filePath, buffer, options);
130
- } else {
131
- logger.info(
132
- "\u{1F4E4} [AliyunOSSProvider] \u4F7F\u7528\u666E\u901A\u4E0A\u4F20: " + filePath + ", \u5927\u5C0F: " + fileInfo.file.size
133
- );
134
- result = await this.client?.put(filePath, buffer, options);
135
- }
136
- const accessUrl = this.generateAccessUrl(filePath);
137
- const uploadTime = Date.now() - startTime;
138
- logger.info("\u2705 [AliyunOSSProvider] \u6587\u4EF6\u4E0A\u4F20\u5B8C\u6210: " + filePath + ", \u8017\u65F6: " + uploadTime + "ms");
139
- return {
140
- success: true,
141
- path: filePath,
142
- url: accessUrl,
143
- size: fileInfo.file.size,
144
- data: {
145
- etag: result.data ? JSON.stringify(result.data) : "",
146
- requestId: result.res?.rt || 0,
147
- uploadTime,
148
- ossUrl: result.url || result.name
149
- }
150
- };
151
- } catch (error) {
152
- logger.error("\u274C [AliyunOSSProvider] \u6587\u4EF6\u4E0A\u4F20\u5931\u8D25: " + filePath + ":", error);
153
- return {
154
- success: false,
155
- error: this.formatOSSError(error)
156
- };
157
- }
158
- }
159
- /**
160
- * 下载文件
161
- */
162
- async download(filePath) {
163
- this.ensureInitialized();
164
- logger.info("\u{1F4E5} [AliyunOSSProvider] \u5F00\u59CB\u4ECEOSS\u4E0B\u8F7D\u6587\u4EF6: " + filePath);
165
- try {
166
- const result = await this.client?.get(filePath);
167
- if (!result.content || !Buffer.isBuffer(result.content)) {
168
- throw new StorageProviderError("\u4E0B\u8F7D\u7684\u6587\u4EF6\u5185\u5BB9\u683C\u5F0F\u9519\u8BEF");
169
- }
170
- logger.info(
171
- "\u2705 [AliyunOSSProvider] \u6587\u4EF6\u4E0B\u8F7D\u5B8C\u6210: " + filePath + ", \u5927\u5C0F: " + result.content.length
172
- );
173
- return result.content;
174
- } catch (error) {
175
- logger.error("\u274C [AliyunOSSProvider] \u6587\u4EF6\u4E0B\u8F7D\u5931\u8D25: " + filePath + ":", error);
176
- if (this.isOSSError(error) && error.code === "NoSuchKey") {
177
- throw new StorageProviderError(`\u6587\u4EF6\u4E0D\u5B58\u5728`);
178
- }
179
- throw new StorageProviderError(`\u6587\u4EF6\u4E0B\u8F7D\u5931\u8D25`);
180
- }
181
- }
182
- /**
183
- * 删除文件
184
- */
185
- async delete(filePath) {
186
- this.ensureInitialized();
187
- logger.info("\u{1F5D1}\uFE0F [AliyunOSSProvider] \u5F00\u59CB\u4ECEOSS\u5220\u9664\u6587\u4EF6: " + filePath);
188
- try {
189
- const result = await this.client?.delete(filePath);
190
- logger.info("\u2705 [AliyunOSSProvider] \u6587\u4EF6\u5220\u9664\u5B8C\u6210: " + filePath);
191
- return {
192
- success: true,
193
- data: {
194
- requestId: result?.res?.rt || 0,
195
- deletedPath: filePath
196
- }
197
- };
198
- } catch (error) {
199
- logger.error("\u274C [AliyunOSSProvider] \u6587\u4EF6\u5220\u9664\u5931\u8D25: " + filePath + ":", error);
200
- if (this.isOSSError(error) && error.code === "NoSuchKey") {
201
- logger.warn("\u26A0\uFE0F [AliyunOSSProvider] \u6587\u4EF6\u4E0D\u5B58\u5728: " + filePath);
202
- return {
203
- success: true,
204
- data: { reason: "file_not_exists" }
205
- };
206
- }
207
- return {
208
- success: false,
209
- error: this.formatOSSError(error)
210
- };
211
- }
212
- }
213
- /**
214
- * 获取文件信息
215
- */
216
- async getFileInfo(filePath) {
217
- this.ensureInitialized();
218
- try {
219
- const result = await this.client?.head(filePath);
220
- return {
221
- success: true,
222
- size: parseInt(String(result.meta["content-length"] || "0")),
223
- data: {
224
- etag: result.meta.etag || "",
225
- lastModified: result.meta["last-modified"] || "",
226
- contentType: result.meta["content-type"],
227
- meta: result.meta,
228
- size: parseInt(String(result.meta["content-length"] || "0"))
229
- }
230
- };
231
- } catch (error) {
232
- if (this.isOSSError(error) && error.code === "NoSuchKey") {
233
- return {
234
- success: false,
235
- error: "\u6587\u4EF6\u4E0D\u5B58\u5728"
236
- };
237
- }
238
- return {
239
- success: false,
240
- error: this.formatOSSError(error)
241
- };
242
- }
243
- }
244
- /**
245
- * 生成访问URL
246
- */
247
- async getAccessUrl(filePath, expiresIn) {
248
- this.ensureInitialized();
249
- try {
250
- const isImage = /\.(jpg|jpeg|png|gif|webp|svg|bmp|ico)$/i.test(filePath);
251
- if (isImage) {
252
- return this.generateAccessUrl(filePath);
253
- } else {
254
- const expires = expiresIn || 3600;
255
- const signedUrl = this.client?.signatureUrl(filePath, {
256
- expires,
257
- method: "GET"
258
- });
259
- return signedUrl || "";
260
- }
261
- } catch (error) {
262
- logger.error("\u274C [AliyunOSSProvider] \u751F\u6210\u8BBF\u95EEURL\u5931\u8D25: " + filePath + ":", error);
263
- throw new StorageProviderError(`\u751F\u6210\u8BBF\u95EEURL\u5931\u8D25`);
264
- }
265
- }
266
- /**
267
- * 生成预签名上传URL
268
- */
269
- async getUploadUrl(filePath, expiresIn) {
270
- this.ensureInitialized();
271
- try {
272
- const expires = expiresIn || 3600;
273
- const signedUrl = this.client?.signatureUrl(filePath, {
274
- expires,
275
- method: "PUT"
276
- });
277
- return signedUrl || "";
278
- } catch (error) {
279
- logger.error("\u274C [AliyunOSSProvider] \u751F\u6210\u4E0A\u4F20URL\u5931\u8D25: " + filePath + ":", error);
280
- throw new StorageProviderError(`\u751F\u6210\u4E0A\u4F20URL\u5931\u8D25`);
281
- }
282
- }
283
- /**
284
- * 检查文件是否存在
285
- */
286
- async exists(filePath) {
287
- this.ensureInitialized();
288
- try {
289
- await this.client?.head(filePath);
290
- return true;
291
- } catch (error) {
292
- if (this.isOSSError(error) && error.code === "NoSuchKey") {
293
- return false;
294
- }
295
- logger.warn("\u26A0\uFE0F [AliyunOSSProvider] \u68C0\u67E5\u6587\u4EF6\u5B58\u5728\u6027\u65F6\u51FA\u9519: " + filePath + ":", error);
296
- return false;
297
- }
298
- }
299
- /**
300
- * 列出文件(详细信息)
301
- */
302
- async listFiles(prefix, delimiter = "/", maxKeys = 1e3) {
303
- this.ensureInitialized();
304
- try {
305
- const options = {
306
- prefix,
307
- delimiter,
308
- "max-keys": String(maxKeys)
309
- };
310
- const result = await this.client?.list(options, {});
311
- const files = (result.objects || []).map((obj) => ({
312
- name: obj.name,
313
- url: this.generateAccessUrl(obj.name),
314
- size: obj.size,
315
- lastModified: obj.lastModified,
316
- etag: obj.etag,
317
- type: "file"
318
- }));
319
- const folders = result.prefixes || [];
320
- return {
321
- files,
322
- folders,
323
- nextMarker: result.nextMarker
324
- };
325
- } catch (error) {
326
- logger.error("\u274C [AliyunOSSProvider] \u5217\u51FA\u6587\u4EF6\u5931\u8D25: " + prefix + ":", error);
327
- throw new StorageProviderError(`\u5217\u51FA\u6587\u4EF6\u5931\u8D25`);
328
- }
329
- }
330
- /**
331
- * 列出文件
332
- */
333
- async list(prefix, maxKeys) {
334
- this.ensureInitialized();
335
- try {
336
- const options = {
337
- prefix,
338
- "max-keys": String(maxKeys || 1e3)
339
- };
340
- const result = await this.client?.list(options, {});
341
- return result.objects?.map((obj) => obj.name) || [];
342
- } catch (error) {
343
- logger.error("\u274C [AliyunOSSProvider] \u5217\u51FA\u6587\u4EF6\u5931\u8D25: " + prefix + ":", error);
344
- return [];
345
- }
346
- }
347
- // ============= 私有方法 =============
348
- /**
349
- * 确保已初始化
350
- */
351
- ensureInitialized() {
352
- if (!this.isInitialized || !this.client || !this.config) {
353
- throw new StorageProviderError("OSS\u5B58\u50A8\u63D0\u4F9B\u8005\u672A\u521D\u59CB\u5316");
354
- }
355
- }
356
- /**
357
- * 验证配置
358
- */
359
- validateConfig() {
360
- if (!this.config) {
361
- throw new StorageProviderError("OSS\u914D\u7F6E\u4E3A\u7A7A");
362
- }
363
- const required = ["region", "bucket", "accessKeyId", "accessKeySecret"];
364
- const missing = required.filter((key) => !this.config[key]);
365
- if (missing.length > 0) {
366
- throw new StorageProviderError(`OSS\u914D\u7F6E\u7F3A\u5C11\u5FC5\u9700\u9879`);
367
- }
368
- }
369
- /**
370
- * 测试连接
371
- */
372
- async testConnection() {
373
- }
374
- /**
375
- * 分片上传大文件
376
- */
377
- async multipartUpload(filePath, buffer, options) {
378
- logger.info(`\u{1F4E6} [AliyunOSSProvider] \u4F7F\u7528\u591A\u5206\u7247\u4E0A\u4F20`);
379
- const result = await this.client?.multipartUpload(filePath, buffer, {
380
- partSize: 10 * 1024 * 1024,
381
- // 10MB per chunk
382
- parallel: 4,
383
- // 并发数
384
- progress: (p) => {
385
- if (p % 0.1 < 0.01) {
386
- logger.info("\u{1F4E6} [AliyunOSSProvider] \u4E0A\u4F20\u8FDB\u5EA6: " + (p * 100).toFixed(1) + "%");
387
- }
388
- },
389
- meta: options.meta,
390
- headers: options.headers
391
- });
392
- return {
393
- name: result?.name || "",
394
- url: result?.name || "",
395
- // OSS返回的是object名称
396
- data: result?.data || {},
397
- res: result?.res || {}
398
- };
399
- }
400
- /**
401
- * 生成公开访问URL
402
- */
403
- generateAccessUrl(filePath) {
404
- if (!this.config) {
405
- throw new StorageProviderError("OSS\u914D\u7F6E\u4E3A\u7A7A");
406
- }
407
- const normalizedPath = filePath.startsWith("/") ? filePath.substring(1) : filePath;
408
- if (this.config.customDomain) {
409
- const protocol = this.config.secure !== false ? "https" : "http";
410
- const url = protocol + "://" + this.config.customDomain + "/" + normalizedPath;
411
- logger.info("\u{1F517} [AliyunOSSProvider] \u4F7F\u7528\u81EA\u5B9A\u4E49\u57DF\u540D: " + url);
412
- return url;
413
- } else {
414
- const protocol = this.config.secure !== false ? "https" : "http";
415
- const url = protocol + "://" + this.config.bucket + "." + this.config.region + ".aliyuncs.com/" + normalizedPath;
416
- logger.info("\u{1F517} [AliyunOSSProvider] \u4F7F\u7528\u9ED8\u8BA4OSS\u57DF\u540D: " + url);
417
- return url;
418
- }
419
- }
420
- /**
421
- * 判断是否为OSS错误
422
- */
423
- isOSSError(error) {
424
- if (!error || typeof error !== "object") {
425
- return false;
426
- }
427
- return typeof error.code === "string" && typeof (error.name || "") === "string" && typeof (error.message || "") === "string";
428
- }
429
- /**
430
- * 格式化OSS错误信息
431
- */
432
- formatOSSError(error) {
433
- if (error instanceof Error) {
434
- return error.message;
435
- }
436
- if (typeof error === "string") {
437
- return error;
438
- }
439
- return "\u672A\u77E5\u9519\u8BEF";
440
- }
441
- /**
442
- * 流式上传(可选实现)
443
- */
444
- async uploadStream(readableStream, filePath, contentType, contentLength) {
445
- this.ensureInitialized();
446
- const startTime = Date.now();
447
- logger.info("\u{1F4E4} [AliyunOSSProvider] \u5F00\u59CB\u6D41\u5F0F\u4E0A\u4F20\u6587\u4EF6\u5230OSS: " + filePath);
448
- try {
449
- const options = {
450
- timeout: 3e5,
451
- mime: contentType || "application/octet-stream",
452
- meta: { uid: 0, pid: 0 },
453
- callback: { url: "", body: "" },
454
- headers: {}
455
- };
456
- if (contentLength) {
457
- options.headers["Content-Length"] = contentLength.toString();
458
- }
459
- const result = await this.client?.putStream(filePath, readableStream, options);
460
- const accessUrl = this.generateAccessUrl(filePath);
461
- const uploadTime = Date.now() - startTime;
462
- logger.info("\u2705 [AliyunOSSProvider] \u6D41\u5F0F\u4E0A\u4F20\u5B8C\u6210: " + filePath + ", \u8017\u65F6: " + uploadTime + "ms");
463
- return {
464
- success: true,
465
- path: filePath,
466
- url: accessUrl,
467
- size: contentLength,
468
- data: {
469
- name: result.name,
470
- requestId: result.res?.rt || 0,
471
- uploadTime,
472
- ossUrl: result.url || result.name
473
- }
474
- };
475
- } catch (error) {
476
- logger.error("\u274C [AliyunOSSProvider] \u6D41\u5F0F\u4E0A\u4F20\u5931\u8D25: " + filePath + ":", error);
477
- return {
478
- success: false,
479
- error: this.formatOSSError(error)
480
- };
481
- }
482
- }
483
- /**
484
- * 批量删除文件
485
- */
486
- async batchDelete(filePaths) {
487
- this.ensureInitialized();
488
- logger.info("\u{1F5D1}\uFE0F [AliyunOSSProvider] \u5F00\u59CB\u6279\u91CF\u5220\u9664\u6587\u4EF6\uFF0C\u6570\u91CF: " + filePaths.length);
489
- try {
490
- const result = await this.client?.deleteMulti(filePaths, {
491
- quiet: false
492
- // 返回删除结果
493
- });
494
- logger.info("\u2705 [AliyunOSSProvider] \u6279\u91CF\u5220\u9664\u5B8C\u6210\uFF0C\u6210\u529F: " + (result.deleted?.length || 0));
495
- return {
496
- success: true,
497
- data: {
498
- deleted: result.deleted,
499
- requestId: result.res?.rt || 0
500
- }
501
- };
502
- } catch (error) {
503
- logger.error(`\u274C [AliyunOSSProvider] \u6279\u91CF\u5220\u9664\u5931\u8D25:`, error);
504
- return {
505
- success: false,
506
- error: this.formatOSSError(error)
507
- };
508
- }
509
- }
510
- /**
511
- * 复制文件
512
- */
513
- async copy(sourcePath, targetPath) {
514
- this.ensureInitialized();
515
- logger.info("\u{1F4CB} [AliyunOSSProvider] \u5F00\u59CB\u590D\u5236\u6587\u4EF6: " + sourcePath + " -> " + targetPath);
516
- try {
517
- const result = await this.client?.copy(targetPath, sourcePath);
518
- logger.info("\u2705 [AliyunOSSProvider] \u6587\u4EF6\u590D\u5236\u5B8C\u6210: " + sourcePath + " -> " + targetPath);
519
- return {
520
- success: true,
521
- data: {
522
- etag: result.data?.etag,
523
- lastModified: result.data?.lastModified,
524
- requestId: result.res?.rt || 0
525
- }
526
- };
527
- } catch (error) {
528
- logger.error("\u274C [AliyunOSSProvider] \u6587\u4EF6\u590D\u5236\u5931\u8D25: " + sourcePath + " -> " + targetPath + ":", error);
529
- return {
530
- success: false,
531
- error: this.formatOSSError(error)
532
- };
533
- }
534
- }
535
- /**
536
- * 编码元数据,避免中文字符在HTTP头部中的问题
537
- */
538
- encodeMetadata(metadata) {
539
- const encoded = {};
540
- for (const [key, value] of Object.entries(metadata)) {
541
- if (value !== null && value !== void 0) {
542
- const stringValue = String(value);
543
- encoded[key] = encodeURIComponent(stringValue);
544
- }
545
- }
546
- return encoded;
547
- }
548
- };
549
- }
550
- });
551
-
552
- export { AliyunOSSProvider, init_AliyunOSSProvider };
553
- //# sourceMappingURL=chunk-K7WNCB4V.mjs.map
554
- //# sourceMappingURL=chunk-K7WNCB4V.mjs.map
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/universalFile/server/providers/AliyunOSSProvider.ts"],"names":[],"mappings":"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","file":"chunk-K7WNCB4V.mjs","sourcesContent":["/**\n * 阿里云OSS存储提供者实现\n */\n\nimport OSS from 'ali-oss';\nimport { createLogger } from '../../../logger';\n\nimport type {\n IStorageProvider,\n StorageConfig,\n AliyunOSSConfig,\n StorageResult,\n UploadFileInfo,\n StorageType,\n} from '../types';\n\nimport { StorageProviderError } from '../types';\n\nconst logger = createLogger('AliyunOSSProvider');\n\n/**\n * 阿里云OSS存储提供者\n */\nexport class AliyunOSSProvider implements IStorageProvider {\n readonly type: StorageType = 'aliyun-oss';\n\n private config: AliyunOSSConfig | null = null;\n private client: OSS | null = null;\n private isInitialized = false;\n\n /**\n * 初始化存储提供者\n */\n async initialize(config: StorageConfig): Promise<void> {\n return this.reinitialize(config);\n }\n\n /**\n * 重新初始化存储提供者(支持配置热更新)\n */\n async reinitialize(config: StorageConfig): Promise<void> {\n logger.info('🔄 [AliyunOSSProvider] 重新初始化OSS客户端', config);\n if (config.type !== 'aliyun-oss') {\n throw new StorageProviderError('配置类型不匹配:期望 aliyun-oss');\n }\n\n const newConfig = config as AliyunOSSConfig;\n\n // 检查配置是否发生变化\n const configChanged =\n !this.config ||\n this.config.region !== newConfig.region ||\n this.config.bucket !== newConfig.bucket ||\n this.config.accessKeyId !== newConfig.accessKeyId ||\n this.config.accessKeySecret !== newConfig.accessKeySecret ||\n this.config.customDomain !== newConfig.customDomain ||\n this.config.secure !== newConfig.secure ||\n this.config.internal !== newConfig.internal;\n\n if (configChanged) {\n logger.info('🔄 [AliyunOSSProvider] 检测到配置变化,重新初始化OSS客户端');\n logger.info(\n '☁️ [AliyunOSSProvider] 新配置: bucket=' + (newConfig.bucket) + ', region=' + (newConfig.region)\n );\n } else if (this.isInitialized) {\n logger.info('ℹ️ [AliyunOSSProvider] 配置未变化,跳过重新初始化');\n return;\n }\n\n this.config = newConfig;\n\n logger.info('☁️ [AliyunOSSProvider] ' + (this.isInitialized ? '重新' : '') + '初始化阿里云OSS');\n logger.info('☁️ [AliyunOSSProvider] ' + (this.config ? JSON.stringify(this.config) : ''));\n\n try {\n // 验证必需的配置项\n this.validateConfig();\n\n // 确保配置值是有效的字符串\n if (!this.config.region || typeof this.config.region !== 'string') {\n throw new Error('OSS region 必须是有效的字符串');\n }\n if (!this.config.bucket || typeof this.config.bucket !== 'string') {\n throw new Error('OSS bucket 必须是有效的字符串');\n }\n if (!this.config.accessKeyId || typeof this.config.accessKeyId !== 'string') {\n throw new Error('OSS accessKeyId 必须是有效的字符串');\n }\n if (!this.config.accessKeySecret || typeof this.config.accessKeySecret !== 'string') {\n throw new Error('OSS accessKeySecret 必须是有效的字符串');\n }\n\n logger.info(`☁️ [AliyunOSSProvider] 创建OSS客户端配置:`, {\n region: this.config.region,\n bucket: this.config.bucket,\n hasAccessKeyId: !!this.config.accessKeyId,\n hasAccessKeySecret: !!this.config.accessKeySecret,\n secure: this.config.secure !== false,\n internal: this.config.internal || false,\n cname: !!this.config.customDomain,\n endpoint: this.config.customDomain || '默认端点',\n });\n\n // 创建OSS客户端\n this.client = new OSS({\n region: this.config.region,\n bucket: this.config.bucket,\n accessKeyId: this.config.accessKeyId,\n accessKeySecret: this.config.accessKeySecret,\n secure: this.config.secure !== false, // 默认使用HTTPS\n internal: this.config.internal || false, // 默认使用公网\n timeout: 300000, // 5分钟超时\n cname: !!this.config.customDomain, // 是否使用自定义域名\n endpoint: this.config.customDomain || undefined,\n });\n\n if (!this.client) {\n throw new Error('OSS客户端创建失败');\n }\n logger.info(`☁️ [AliyunOSSProvider] OSS客户端创建成功`);\n this.isInitialized = true;\n logger.info('✅ [AliyunOSSProvider] 阿里云OSS' + (configChanged ? '重新' : '') + '初始化完成');\n } catch (error) {\n logger.error('❌ [AliyunOSSProvider] 阿里云OSS初始化失败:', error);\n throw new StorageProviderError(`阿里云OSS初始化失败`);\n }\n }\n\n /**\n * 上传文件\n */\n async upload(fileInfo: UploadFileInfo, filePath: string): Promise<StorageResult> {\n this.ensureInitialized();\n\n const startTime = Date.now();\n logger.info('📤 [AliyunOSSProvider] 开始上传文件到OSS: ' + (filePath));\n\n try {\n // 将File对象转换为Buffer\n const buffer = Buffer.from(await fileInfo.file.arrayBuffer());\n\n // 构建上传选项\n const options: any = {\n headers: {\n 'Content-Type': fileInfo.file.type || 'application/octet-stream',\n 'Content-Length': fileInfo.file.size.toString(),\n },\n meta: {\n uid: 0, // 必需字段\n pid: 0, // 必需字段\n originalName: encodeURIComponent(fileInfo.file.name),\n moduleId: fileInfo.moduleId,\n businessId: fileInfo.businessId || '',\n uploadTime: new Date().toISOString(),\n // 对元数据进行编码处理,避免中文字符问题\n ...this.encodeMetadata(fileInfo.metadata || {}),\n },\n };\n\n // 根据文件大小选择上传方式\n let result: any;\n\n if (fileInfo.file.size > 100 * 1024 * 1024) {\n // 大于100MB使用分片上传\n logger.info(\n '📦 [AliyunOSSProvider] 使用分片上传大文件: ' + (filePath) + ', 大小: ' + (fileInfo.file.size)\n );\n result = await this.multipartUpload(filePath, buffer, options);\n } else {\n logger.info(\n '📤 [AliyunOSSProvider] 使用普通上传: ' + (filePath) + ', 大小: ' + (fileInfo.file.size)\n );\n result = await this.client?.put(filePath, buffer, options);\n }\n\n // 生成访问URL\n const accessUrl = this.generateAccessUrl(filePath);\n\n const uploadTime = Date.now() - startTime;\n logger.info('✅ [AliyunOSSProvider] 文件上传完成: ' + (filePath) + ', 耗时: ' + (uploadTime) + 'ms');\n\n return {\n success: true,\n path: filePath,\n url: accessUrl,\n size: fileInfo.file.size,\n data: {\n etag: result.data ? JSON.stringify(result.data) : '',\n requestId: result.res?.rt || 0,\n uploadTime,\n ossUrl: result.url || result.name,\n },\n };\n } catch (error) {\n logger.error('❌ [AliyunOSSProvider] 文件上传失败: ' + (filePath) + ':', error);\n\n return {\n success: false,\n error: this.formatOSSError(error),\n };\n }\n }\n\n /**\n * 下载文件\n */\n async download(filePath: string): Promise<Buffer> {\n this.ensureInitialized();\n\n logger.info('📥 [AliyunOSSProvider] 开始从OSS下载文件: ' + (filePath));\n\n try {\n const result = await this.client?.get(filePath) as any;\n\n if (!result.content || !Buffer.isBuffer(result.content)) {\n throw new StorageProviderError('下载的文件内容格式错误');\n }\n\n logger.info(\n '✅ [AliyunOSSProvider] 文件下载完成: ' + (filePath) + ', 大小: ' + (result.content.length)\n );\n\n return result.content;\n } catch (error) {\n logger.error('❌ [AliyunOSSProvider] 文件下载失败: ' + (filePath) + ':', error);\n\n if (this.isOSSError(error) && error.code === 'NoSuchKey') {\n throw new StorageProviderError(`文件不存在`);\n }\n\n throw new StorageProviderError(`文件下载失败`);\n }\n }\n\n /**\n * 删除文件\n */\n async delete(filePath: string): Promise<StorageResult> {\n this.ensureInitialized();\n\n logger.info('🗑️ [AliyunOSSProvider] 开始从OSS删除文件: ' + (filePath));\n\n try {\n const result = await this.client?.delete(filePath) as any;\n\n logger.info('✅ [AliyunOSSProvider] 文件删除完成: ' + (filePath));\n\n return {\n success: true,\n data: {\n requestId: result?.res?.rt || 0,\n deletedPath: filePath,\n },\n };\n } catch (error) {\n logger.error('❌ [AliyunOSSProvider] 文件删除失败: ' + (filePath) + ':', error);\n\n // OSS中删除不存在的文件不会报错,但我们统一处理\n if (this.isOSSError(error) && error.code === 'NoSuchKey') {\n logger.warn('⚠️ [AliyunOSSProvider] 文件不存在: ' + (filePath));\n return {\n success: true,\n data: { reason: 'file_not_exists' },\n };\n }\n\n return {\n success: false,\n error: this.formatOSSError(error),\n };\n }\n }\n\n /**\n * 获取文件信息\n */\n async getFileInfo(filePath: string): Promise<StorageResult> {\n this.ensureInitialized();\n\n try {\n const result = await this.client?.head(filePath) as any;\n\n return {\n success: true,\n size: parseInt(String(result.meta['content-length'] || '0')),\n data: {\n etag: result.meta.etag || '',\n lastModified: result.meta['last-modified'] || '',\n contentType: result.meta['content-type'],\n meta: result.meta,\n size: parseInt(String(result.meta['content-length'] || '0')),\n },\n };\n } catch (error) {\n if (this.isOSSError(error) && error.code === 'NoSuchKey') {\n return {\n success: false,\n error: '文件不存在',\n };\n }\n\n return {\n success: false,\n error: this.formatOSSError(error),\n };\n }\n }\n\n /**\n * 生成访问URL\n */\n async getAccessUrl(filePath: string, expiresIn?: number): Promise<string> {\n this.ensureInitialized();\n\n try {\n // 对于图片文件,直接返回公开URL,避免CORS问题\n // 对于其他文件,使用签名URL\n const isImage = /\\.(jpg|jpeg|png|gif|webp|svg|bmp|ico)$/i.test(filePath);\n\n if (isImage) {\n // 图片文件使用公开URL\n return this.generateAccessUrl(filePath);\n } else {\n // 其他文件使用签名URL\n const expires = expiresIn || 3600; // 默认1小时\n const signedUrl = this.client?.signatureUrl(filePath, {\n expires,\n method: 'GET',\n });\n\n return signedUrl || '' ;\n }\n } catch (error) {\n logger.error('❌ [AliyunOSSProvider] 生成访问URL失败: ' + (filePath) + ':', error);\n throw new StorageProviderError(`生成访问URL失败`);\n }\n }\n\n /**\n * 生成预签名上传URL\n */\n async getUploadUrl(filePath: string, expiresIn?: number): Promise<string> {\n this.ensureInitialized();\n\n try {\n const expires = expiresIn || 3600; // 默认1小时\n const signedUrl = this.client?.signatureUrl(filePath, {\n expires,\n method: 'PUT',\n });\n\n return signedUrl || '' ;\n } catch (error) {\n logger.error('❌ [AliyunOSSProvider] 生成上传URL失败: ' + (filePath) + ':', error);\n throw new StorageProviderError(`生成上传URL失败`);\n }\n }\n\n /**\n * 检查文件是否存在\n */\n async exists(filePath: string): Promise<boolean> {\n this.ensureInitialized();\n\n try {\n await this.client?.head(filePath);\n return true;\n } catch (error) {\n if (this.isOSSError(error) && error.code === 'NoSuchKey') {\n return false;\n }\n // 其他错误也视为文件不存在\n logger.warn('⚠️ [AliyunOSSProvider] 检查文件存在性时出错: ' + (filePath) + ':', error);\n return false;\n }\n }\n\n /**\n * 列出文件(详细信息)\n */\n async listFiles(prefix: string, delimiter: string = '/', maxKeys: number = 1000): Promise<{\n files: Array<{\n name: string;\n url: string;\n size: number;\n lastModified: string;\n etag: string;\n type: string;\n }>;\n folders: string[];\n nextMarker?: string;\n }> {\n this.ensureInitialized();\n\n try {\n const options: any = {\n prefix,\n delimiter,\n 'max-keys': String(maxKeys),\n };\n\n const result = await this.client?.list(options, {}) as any;\n\n const files = (result.objects || []).map((obj: any) => ({\n name: obj.name,\n url: this.generateAccessUrl(obj.name),\n size: obj.size,\n lastModified: obj.lastModified,\n etag: obj.etag,\n type: 'file',\n }));\n\n const folders = result.prefixes || [];\n\n return {\n files,\n folders,\n nextMarker: result.nextMarker,\n };\n } catch (error) {\n logger.error('❌ [AliyunOSSProvider] 列出文件失败: ' + (prefix) + ':', error);\n throw new StorageProviderError(`列出文件失败`);\n }\n }\n\n /**\n * 列出文件\n */\n async list(prefix: string, maxKeys?: number): Promise<string[]> {\n this.ensureInitialized();\n\n try {\n const options: any = {\n prefix,\n 'max-keys': String(maxKeys || 1000),\n };\n\n const result = await this.client?.list(options, {}) as any;\n\n return result.objects?.map((obj: any) => obj.name) || [];\n } catch (error) {\n logger.error('❌ [AliyunOSSProvider] 列出文件失败: ' + (prefix) + ':', error);\n return [];\n }\n }\n\n // ============= 私有方法 =============\n\n /**\n * 确保已初始化\n */\n private ensureInitialized(): void {\n if (!this.isInitialized || !this.client || !this.config) {\n throw new StorageProviderError('OSS存储提供者未初始化');\n }\n }\n\n /**\n * 验证配置\n */\n private validateConfig(): void {\n if (!this.config) {\n throw new StorageProviderError('OSS配置为空');\n }\n\n const required = ['region', 'bucket', 'accessKeyId', 'accessKeySecret'];\n const missing = required.filter((key) => !this.config![key as keyof AliyunOSSConfig]);\n\n if (missing.length > 0) {\n throw new StorageProviderError(`OSS配置缺少必需项`);\n }\n }\n\n /**\n * 测试连接\n */\n private async testConnection(): Promise<void> {\n \n }\n\n /**\n * 分片上传大文件\n */\n private async multipartUpload(filePath: string, buffer: Buffer, options: any): Promise<any> {\n logger.info(`📦 [AliyunOSSProvider] 使用多分片上传`);\n\n // 使用OSS的multipartUpload方法\n const result = await this.client?.multipartUpload(filePath, buffer, {\n partSize: 10 * 1024 * 1024, // 10MB per chunk\n parallel: 4, // 并发数\n progress: (p: number) => {\n if (p % 0.1 < 0.01) {\n // 每10%显示一次进度\n logger.info('📦 [AliyunOSSProvider] 上传进度: ' + ((p * 100).toFixed(1)) + '%');\n }\n },\n meta: options.meta,\n headers: options.headers,\n });\n\n return {\n name: result?.name || '',\n url: result?.name || '', // OSS返回的是object名称\n data: result?.data || {},\n res: result?.res || {},\n };\n }\n\n /**\n * 生成公开访问URL\n */\n private generateAccessUrl(filePath: string): string {\n if (!this.config) {\n throw new StorageProviderError('OSS配置为空');\n }\n\n // 确保文件路径不以斜杠开头\n const normalizedPath = filePath.startsWith('/') ? filePath.substring(1) : filePath;\n\n if (this.config.customDomain) {\n // 使用自定义域名\n const protocol = this.config.secure !== false ? 'https' : 'http';\n const url = (protocol) + '://' + (this.config.customDomain) + '/' + (normalizedPath);\n logger.info('🔗 [AliyunOSSProvider] 使用自定义域名: ' + (url));\n return url;\n } else {\n // 使用默认OSS域名\n const protocol = this.config.secure !== false ? 'https' : 'http';\n const url = (protocol) + '://' + (this.config.bucket) + '.' + (this.config.region) + '.aliyuncs.com/' + (normalizedPath);\n logger.info('🔗 [AliyunOSSProvider] 使用默认OSS域名: ' + (url));\n return url;\n }\n }\n\n /**\n * 判断是否为OSS错误\n */\n private isOSSError(\n error: any\n ): error is { code: string; name: string; message: string; requestId?: string } {\n // 安全地检查错误属性,避免生产环境中的压缩问题\n if (!error || typeof error !== 'object') {\n return false;\n }\n\n return typeof error.code === 'string' &&\n typeof (error.name || '') === 'string' &&\n typeof (error.message || '') === 'string';\n }\n\n /**\n * 格式化OSS错误信息\n */\n private formatOSSError(error: any): string {\n // if (this.isOSSError(error)) {\n // const requestId = error.requestId ? ' (RequestId: ' + (error.requestId) + ')' : '';\n // return (error.code) + ': ' + (error.message) + (requestId);\n // }\n if (error instanceof Error) {\n return error.message;\n }\n if (typeof error === 'string') {\n return error;\n }\n return '未知错误';\n }\n\n /**\n * 流式上传(可选实现)\n */\n async uploadStream(\n readableStream: NodeJS.ReadableStream,\n filePath: string,\n contentType?: string,\n contentLength?: number\n ): Promise<StorageResult> {\n this.ensureInitialized();\n\n const startTime = Date.now();\n logger.info('📤 [AliyunOSSProvider] 开始流式上传文件到OSS: ' + (filePath));\n\n try {\n const options: any = {\n timeout: 300000,\n mime: contentType || 'application/octet-stream',\n meta: { uid: 0, pid: 0 },\n callback: { url: '', body: '' },\n headers: {} as any,\n };\n\n if (contentLength) {\n options.headers['Content-Length'] = contentLength.toString();\n }\n\n const result = await this.client?.putStream(filePath, readableStream, options) as any;\n\n const accessUrl = this.generateAccessUrl(filePath);\n\n const uploadTime = Date.now() - startTime;\n logger.info('✅ [AliyunOSSProvider] 流式上传完成: ' + (filePath) + ', 耗时: ' + (uploadTime) + 'ms');\n\n return {\n success: true,\n path: filePath,\n url: accessUrl,\n size: contentLength,\n data: {\n name: result.name,\n requestId: result.res?.rt || 0,\n uploadTime,\n ossUrl: result.url || result.name,\n },\n };\n } catch (error) {\n logger.error('❌ [AliyunOSSProvider] 流式上传失败: ' + (filePath) + ':', error);\n\n return {\n success: false,\n error: this.formatOSSError(error),\n };\n }\n }\n\n /**\n * 批量删除文件\n */\n async batchDelete(filePaths: string[]): Promise<StorageResult> {\n this.ensureInitialized();\n\n logger.info('🗑️ [AliyunOSSProvider] 开始批量删除文件,数量: ' + (filePaths.length));\n\n try {\n const result = await this.client?.deleteMulti(filePaths, {\n quiet: false, // 返回删除结果\n }) as any;\n\n logger.info('✅ [AliyunOSSProvider] 批量删除完成,成功: ' + (result.deleted?.length || 0));\n\n return {\n success: true,\n data: {\n deleted: result.deleted,\n requestId: result.res?.rt || 0,\n },\n };\n } catch (error) {\n logger.error(`❌ [AliyunOSSProvider] 批量删除失败:`, error);\n\n return {\n success: false,\n error: this.formatOSSError(error),\n };\n }\n }\n\n /**\n * 复制文件\n */\n async copy(sourcePath: string, targetPath: string): Promise<StorageResult> {\n this.ensureInitialized();\n\n logger.info('📋 [AliyunOSSProvider] 开始复制文件: ' + (sourcePath) + ' -> ' + (targetPath));\n\n try {\n const result = await this.client?.copy(targetPath, sourcePath) as any;\n\n logger.info('✅ [AliyunOSSProvider] 文件复制完成: ' + (sourcePath) + ' -> ' + (targetPath));\n\n return {\n success: true,\n data: {\n etag: result.data?.etag,\n lastModified: result.data?.lastModified,\n requestId: result.res?.rt || 0,\n },\n };\n } catch (error) {\n logger.error('❌ [AliyunOSSProvider] 文件复制失败: ' + (sourcePath) + ' -> ' + (targetPath) + ':', error);\n\n return {\n success: false,\n error: this.formatOSSError(error),\n };\n }\n }\n\n /**\n * 编码元数据,避免中文字符在HTTP头部中的问题\n */\n private encodeMetadata(metadata: Record<string, any>): Record<string, string> {\n const encoded: Record<string, string> = {};\n\n for (const [key, value] of Object.entries(metadata)) {\n if (value !== null && value !== undefined) {\n // 将值转换为字符串并进行URL编码\n const stringValue = String(value);\n encoded[key] = encodeURIComponent(stringValue);\n }\n }\n\n return encoded;\n }\n}\n\n"]}