sa2kit 1.6.57 → 1.6.59

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