@cloudbase/cli 2.0.3 → 2.0.4-alpha.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (199) hide show
  1. package/.editorconfig +9 -9
  2. package/.eslintignore +7 -7
  3. package/.eslintrc +35 -35
  4. package/.prettierrc.js +29 -29
  5. package/.vscode/launch.json +16 -16
  6. package/LICENSE +5 -5
  7. package/README.md +35 -35
  8. package/bin/cloudbase.js +5 -5
  9. package/bin/tcb.js +0 -0
  10. package/changelog.md +6 -6
  11. package/jest.config.js +17 -17
  12. package/lib/commands/account/login.js +18 -18
  13. package/lib/commands/storage/storage.js +1 -1
  14. package/lib/env/login.js +7 -7
  15. package/package.json +2 -2
  16. package/post-install.js +61 -61
  17. package/runtime/nodejs/bootstrap.js +255 -255
  18. package/runtime/nodejs/runtime.js +183 -183
  19. package/src/auth/index.ts +1 -1
  20. package/src/auth/login.ts +91 -91
  21. package/src/auth/logout.ts +7 -7
  22. package/src/commands/account/index.ts +2 -2
  23. package/src/commands/account/login.ts +192 -192
  24. package/src/commands/account/logout.ts +24 -24
  25. package/src/commands/env/base.ts +90 -90
  26. package/src/commands/env/create.ts +92 -92
  27. package/src/commands/env/domain.ts +186 -186
  28. package/src/commands/env/index.ts +4 -4
  29. package/src/commands/env/login.ts +235 -235
  30. package/src/commands/framework/index.ts +124 -124
  31. package/src/commands/functions/alias/getRoute.ts +76 -76
  32. package/src/commands/functions/alias/index.ts +2 -2
  33. package/src/commands/functions/alias/setRoute.ts +82 -82
  34. package/src/commands/functions/code-download.ts +100 -100
  35. package/src/commands/functions/code-update.ts +62 -62
  36. package/src/commands/functions/concurrency/delete.ts +45 -45
  37. package/src/commands/functions/concurrency/index.ts +2 -2
  38. package/src/commands/functions/concurrency/list.ts +58 -58
  39. package/src/commands/functions/concurrency/set.ts +47 -47
  40. package/src/commands/functions/config-update.ts +76 -76
  41. package/src/commands/functions/copy.ts +62 -62
  42. package/src/commands/functions/delete.ts +79 -79
  43. package/src/commands/functions/deploy.ts +293 -293
  44. package/src/commands/functions/detail.ts +138 -138
  45. package/src/commands/functions/index.ts +16 -16
  46. package/src/commands/functions/invoke.ts +121 -121
  47. package/src/commands/functions/layer/bind.ts +182 -182
  48. package/src/commands/functions/layer/common.ts +8 -8
  49. package/src/commands/functions/layer/create.ts +49 -49
  50. package/src/commands/functions/layer/delete.ts +73 -73
  51. package/src/commands/functions/layer/download.ts +92 -92
  52. package/src/commands/functions/layer/index.ts +7 -7
  53. package/src/commands/functions/layer/list.ts +94 -94
  54. package/src/commands/functions/layer/sort.ts +76 -76
  55. package/src/commands/functions/list.ts +68 -68
  56. package/src/commands/functions/log.ts +148 -148
  57. package/src/commands/functions/run.ts +249 -249
  58. package/src/commands/functions/trigger-create.ts +79 -79
  59. package/src/commands/functions/trigger-delete.ts +105 -105
  60. package/src/commands/functions/version/index.ts +1 -1
  61. package/src/commands/functions/version/list.ts +73 -73
  62. package/src/commands/functions/version/publish.ts +43 -43
  63. package/src/commands/gateway/create.ts +109 -109
  64. package/src/commands/gateway/delete.ts +81 -81
  65. package/src/commands/gateway/domain.ts +159 -159
  66. package/src/commands/gateway/index.ts +5 -5
  67. package/src/commands/gateway/list.ts +76 -76
  68. package/src/commands/gateway/switch.ts +107 -107
  69. package/src/commands/helpers/index.ts +2 -2
  70. package/src/commands/helpers/init.ts +431 -431
  71. package/src/commands/helpers/new.ts +117 -117
  72. package/src/commands/helpers/open.ts +67 -67
  73. package/src/commands/hosting/hosting.ts +360 -360
  74. package/src/commands/index.ts +13 -13
  75. package/src/commands/lowcode/app.ts +34 -34
  76. package/src/commands/lowcode/comps.ts +322 -322
  77. package/src/commands/lowcode/index.ts +1 -1
  78. package/src/commands/lowcode/utils.ts +24 -24
  79. package/src/commands/run/image/index.ts +4 -4
  80. package/src/commands/run/standalonegateway/common.ts +7 -7
  81. package/src/commands/run/standalonegateway/create.ts +85 -85
  82. package/src/commands/run/standalonegateway/destroy.ts +59 -59
  83. package/src/commands/run/standalonegateway/index.ts +4 -4
  84. package/src/commands/run/standalonegateway/list.ts +53 -53
  85. package/src/commands/run/standalonegateway/package.ts +62 -62
  86. package/src/commands/run/standalonegateway/turn.ts +63 -63
  87. package/src/commands/run/version/index.ts +4 -4
  88. package/src/commands/smart.ts +132 -132
  89. package/src/commands/storage/storage.ts +464 -464
  90. package/src/commands/third/thirdAttach.ts +49 -49
  91. package/src/completion/index.ts +13 -13
  92. package/src/decorators/captureError.ts +25 -25
  93. package/src/decorators/constants.ts +12 -12
  94. package/src/decorators/deprecate.ts +25 -25
  95. package/src/decorators/guard.ts +42 -42
  96. package/src/decorators/index.ts +7 -7
  97. package/src/decorators/injectParams.ts +54 -54
  98. package/src/decorators/params/common.ts +28 -28
  99. package/src/decorators/params/index.ts +35 -35
  100. package/src/env/domain.ts +33 -33
  101. package/src/env/index.ts +63 -63
  102. package/src/env/login.ts +80 -80
  103. package/src/error.ts +36 -36
  104. package/src/function/alias.ts +43 -43
  105. package/src/function/base.ts +253 -253
  106. package/src/function/code.ts +55 -55
  107. package/src/function/concurrency.ts +57 -57
  108. package/src/function/create.ts +78 -78
  109. package/src/function/delete.ts +42 -42
  110. package/src/function/index.ts +10 -10
  111. package/src/function/layer/attach.ts +68 -68
  112. package/src/function/layer/create.ts +63 -63
  113. package/src/function/layer/delete.ts +21 -21
  114. package/src/function/layer/download.ts +54 -54
  115. package/src/function/layer/index.ts +7 -7
  116. package/src/function/layer/list.ts +32 -32
  117. package/src/function/layer/sort.ts +24 -24
  118. package/src/function/trigger.ts +97 -97
  119. package/src/function/update.ts +35 -35
  120. package/src/function/version.ts +38 -38
  121. package/src/function/vpc.ts +22 -22
  122. package/src/gateway/index.ts +137 -137
  123. package/src/hosting.ts +212 -212
  124. package/src/index.ts +13 -13
  125. package/src/logger.ts +17 -17
  126. package/src/run/create.ts +23 -23
  127. package/src/run/delete.ts +15 -15
  128. package/src/run/image/build.ts +36 -36
  129. package/src/run/image/delete.ts +13 -13
  130. package/src/run/image/index.ts +3 -3
  131. package/src/run/image/info.ts +26 -26
  132. package/src/run/list.ts +29 -29
  133. package/src/run/repo.ts +24 -24
  134. package/src/run/standalonegateway/create.ts +24 -24
  135. package/src/run/standalonegateway/destroy.ts +19 -19
  136. package/src/run/standalonegateway/index.ts +4 -4
  137. package/src/run/standalonegateway/list.ts +74 -74
  138. package/src/run/standalonegateway/package/list.ts +24 -24
  139. package/src/run/standalonegateway/turn/index.ts +1 -1
  140. package/src/run/standalonegateway/turn/off.ts +19 -19
  141. package/src/run/standalonegateway/turn/on.ts +19 -19
  142. package/src/run/version/create.ts +68 -68
  143. package/src/run/version/delete.ts +15 -15
  144. package/src/run/version/index.ts +5 -5
  145. package/src/run/version/list.ts +16 -16
  146. package/src/run/version/modify.ts +16 -16
  147. package/src/run/version/repo.ts +27 -27
  148. package/src/run/version/update.ts +58 -58
  149. package/src/storage.ts +114 -114
  150. package/src/third/index.ts +12 -12
  151. package/src/utils/auth.ts +15 -15
  152. package/src/utils/cli-table.ts +23 -23
  153. package/src/utils/config.ts +39 -39
  154. package/src/utils/env.ts +244 -244
  155. package/src/utils/fs/del.ts +5 -5
  156. package/src/utils/fs/index.ts +71 -71
  157. package/src/utils/function-packer.ts +97 -97
  158. package/src/utils/log.ts +81 -81
  159. package/src/utils/net/cloud-api-request.ts +62 -62
  160. package/src/utils/net/credential.ts +53 -53
  161. package/src/utils/net/index.ts +4 -4
  162. package/src/utils/net/manager-service.ts +36 -36
  163. package/src/utils/net/proxy.ts +6 -6
  164. package/src/utils/notice.ts +28 -28
  165. package/src/utils/output/highlight.ts +5 -5
  166. package/src/utils/output/index.ts +2 -2
  167. package/src/utils/output/link.ts +10 -10
  168. package/src/utils/output/loading.ts +82 -82
  169. package/src/utils/parallel.ts +82 -82
  170. package/src/utils/platform/index.ts +2 -2
  171. package/src/utils/platform/mac.ts +21 -21
  172. package/src/utils/platform/os.ts +64 -64
  173. package/src/utils/platform/port.ts +10 -10
  174. package/src/utils/progress-bar.ts +38 -38
  175. package/src/utils/prompt/select.ts +59 -59
  176. package/src/utils/reporter/agree.ts +20 -20
  177. package/src/utils/reporter/download.ts +26 -26
  178. package/src/utils/reporter/index.ts +3 -3
  179. package/src/utils/reporter/usage.ts +20 -20
  180. package/src/utils/store/auth.ts +49 -49
  181. package/src/utils/store/common.ts +8 -8
  182. package/src/utils/store/db.ts +68 -68
  183. package/src/utils/store/index.ts +4 -4
  184. package/src/utils/store/usage.ts +12 -12
  185. package/src/utils/template.ts +170 -170
  186. package/src/utils/tools/encoding.ts +8 -8
  187. package/src/utils/tools/index.ts +4 -4
  188. package/src/utils/tools/object.ts +33 -33
  189. package/src/utils/tools/time.ts +38 -38
  190. package/src/utils/tools/uid.ts +19 -19
  191. package/templates/html/loginFail.html +90 -90
  192. package/templates/html/loginSuccess.html +86 -86
  193. package/templates/server/node/_gitignore +54 -54
  194. package/templates/server/node/cloudbaserc.json +10 -10
  195. package/templates/server/node/index.js +5 -5
  196. package/templates/server/node/package.json +9 -9
  197. package/tsconfig.json +19 -19
  198. package/tsconfig.test.json +13 -13
  199. package/.vscode/settings.json +0 -3
@@ -1,464 +1,464 @@
1
- import fs from 'fs'
2
- import path from 'path'
3
- import inquirer from 'inquirer'
4
- import logSymbols from 'log-symbols'
5
- import { StorageService } from '@cloudbase/manager-node/types/storage'
6
- import { Command, ICommand } from '../common'
7
-
8
- import {
9
- loadingFactory,
10
- printHorizontalTable,
11
- formatDate,
12
- createUploadProgressBar,
13
- formateFileSize,
14
- getMangerService,
15
- isDirectory,
16
- checkFullAccess
17
- } from '../../utils'
18
-
19
- import { CloudBaseError } from '../../error'
20
- import { InjectParams, EnvId, ArgsParams, ArgsOptions, Log, Logger } from '../../decorators'
21
-
22
- const AclMap = {
23
- READONLY: '所有用户可读,仅创建者和管理员可写',
24
- PRIVATE: '仅创建者及管理员可读写',
25
- ADMINWRITE: '所有用户可读,仅管理员可写',
26
- ADMINONLY: '仅管理员可读写'
27
- }
28
-
29
- async function getStorageService(envId: string): Promise<StorageService> {
30
- const { storage } = await getMangerService(envId)
31
- return storage
32
- }
33
-
34
- function checkCloudPath(cloudPath: string) {
35
- if (cloudPath?.[0] === '/') {
36
- throw new CloudBaseError('cloudPath 不能以 "/" 开头')
37
- }
38
- }
39
-
40
- @ICommand()
41
- export class UploadCommand extends Command {
42
- get options() {
43
- return {
44
- cmd: 'storage',
45
- childCmd: 'upload <localPath> [cloudPath]',
46
- deprecateCmd: 'storage:upload <localPath> [cloudPath]',
47
- options: [
48
- {
49
- flags: '-e, --envId <envId>',
50
- desc: '环境 Id'
51
- },
52
- { flags: '--times <times>', desc: '设置上传重试次数,默认值为 1' },
53
- {
54
- flags: '--interval <interval>',
55
- desc: '设置上传失败时,重试时间间隔(毫秒),默认值为 500'
56
- }
57
- ],
58
- desc: '上传文件/文件夹'
59
- }
60
- }
61
-
62
- @InjectParams()
63
- async execute(
64
- @EnvId() envId,
65
- @ArgsParams() params,
66
- @ArgsOptions() options,
67
- @Log() log: Logger
68
- ) {
69
- const localPath = params?.[0]
70
- const cloudPath = params?.[1]
71
- const retryCount = options?.times
72
- const retryInterval = options?.interval
73
- const resolveLocalPath = path.resolve(localPath)
74
- if (!checkFullAccess(resolveLocalPath)) {
75
- throw new CloudBaseError('文件未找到!')
76
- }
77
- if (retryCount > 10) {
78
- throw new CloudBaseError('上传重试次数为 0-10 次之间')
79
- }
80
-
81
- const loading = loadingFactory()
82
- loading.start('准备上传中...')
83
- const storageService = await getStorageService(envId)
84
- const isDir = fs.statSync(resolveLocalPath).isDirectory()
85
-
86
- let totalFiles = 0
87
-
88
- if (isDir) {
89
- let files = await storageService.walkLocalDir(resolveLocalPath)
90
- files = files.filter((item) => !isDirectory(item))
91
- totalFiles = files.length
92
- }
93
-
94
- // 上传进度条
95
- const onProgress = createUploadProgressBar(
96
- () => {
97
- !isDir && log.success('上传文件成功!')
98
- },
99
- () => {
100
- loading.stop()
101
- }
102
- )
103
-
104
- const successFiles = []
105
- const failedFiles = []
106
- if (isDir) {
107
- await storageService.uploadDirectory({
108
- localPath: resolveLocalPath,
109
- cloudPath,
110
- onProgress,
111
- onFileFinish: (...args) => {
112
- const error = args[0]
113
- const fileInfo = (args as any)[2]
114
- if (error) {
115
- failedFiles.push(fileInfo.Key)
116
- } else {
117
- successFiles.push(fileInfo.Key)
118
- }
119
- },
120
- retryCount: retryCount || 1,
121
- retryInterval,
122
- parallel: 20
123
- })
124
-
125
- log.success(`文件共计 ${totalFiles} 个`)
126
- log.success(`文件上传成功 ${successFiles.length} 个`)
127
- // 上传成功的文件
128
- if (totalFiles <= 50) {
129
- printHorizontalTable(
130
- ['状态', '文件'],
131
- successFiles.map((item) => [logSymbols.success, item])
132
- )
133
- }
134
-
135
- // 上传失败的文件
136
- log.error(`文件上传失败 ${failedFiles.length} 个`)
137
- if (failedFiles.length) {
138
- if (totalFiles <= 50) {
139
- printHorizontalTable(
140
- ['状态', '文件'],
141
- failedFiles.map((item) => [logSymbols.error, item])
142
- )
143
- } else {
144
- // 写入文件到本地
145
- const errorLogPath = path.resolve('./cloudbase-error.log')
146
- log.error('上传失败文件:')
147
- console.log(errorLogPath)
148
- fs.writeFileSync(errorLogPath, failedFiles.join('\n'))
149
- }
150
- }
151
- } else {
152
- const assignCloudPath = cloudPath || path.parse(resolveLocalPath).base
153
- await storageService.uploadFile({
154
- localPath: resolveLocalPath,
155
- cloudPath: assignCloudPath,
156
- onProgress
157
- })
158
- }
159
- }
160
- }
161
-
162
- @ICommand()
163
- export class DownloadCommand extends Command {
164
- get options() {
165
- return {
166
- cmd: 'storage',
167
- childCmd: 'download <cloudPath> <localPath>',
168
- deprecateCmd: 'storage:download <cloudPath> <localPath>',
169
- options: [
170
- {
171
- flags: '-e, --envId <envId>',
172
- desc: '环境 Id'
173
- },
174
- {
175
- flags: '-d, --dir',
176
- desc: '下载目标是否为文件夹'
177
- }
178
- ],
179
- desc: '下载文件/文件夹,文件夹需指定 --dir 选项'
180
- }
181
- }
182
-
183
- @InjectParams()
184
- async execute(@EnvId() envId, @ArgsOptions() options, @ArgsParams() params) {
185
- const cloudPath = params?.[0]
186
- const localPath = params?.[1]
187
- const storageService = await getStorageService(envId)
188
- const resolveLocalPath = path.resolve(localPath)
189
-
190
- const { dir } = options
191
- const fileText = dir ? '文件夹' : '文件'
192
-
193
- if (dir && !checkFullAccess(resolveLocalPath)) {
194
- throw new CloudBaseError('存储文件夹不存在!')
195
- }
196
-
197
- const loading = loadingFactory()
198
-
199
- loading.start(`下载${fileText}中`)
200
-
201
- if (dir) {
202
- await storageService.downloadDirectory({
203
- localPath: resolveLocalPath,
204
- cloudPath,
205
- parallel: 20
206
- })
207
- } else {
208
- await storageService.downloadFile({
209
- cloudPath,
210
- localPath: resolveLocalPath
211
- })
212
- }
213
-
214
- loading.succeed(`下载${fileText}成功!`)
215
- }
216
- }
217
-
218
- @ICommand()
219
- export class DeleteFileCommand extends Command {
220
- get options() {
221
- return {
222
- cmd: 'storage',
223
- childCmd: 'delete [cloudPath]',
224
- deprecateCmd: 'storage:delete [cloudPath]',
225
- options: [
226
- {
227
- flags: '-e, --envId <envId>',
228
- desc: '环境 Id'
229
- },
230
- {
231
- flags: '-d, --dir',
232
- desc: '下载目标是否为文件夹'
233
- }
234
- ],
235
- desc: '删除文件/文件夹,文件夹需指定 --dir 选项'
236
- }
237
- }
238
-
239
- @InjectParams()
240
- async execute(@EnvId() envId, @ArgsOptions() options, @ArgsParams() params) {
241
- let isDir = options.dir
242
- const cloudPath = params?.[0]
243
- const loading = loadingFactory()
244
- const storageService = await getStorageService(envId)
245
-
246
- // 删除所有文件,危险操作,需要提示
247
- if (!cloudPath) {
248
- const { confirm } = await inquirer.prompt({
249
- type: 'confirm',
250
- name: 'confirm',
251
- message: '指定云端路径为空,将会删除所有文件,是否继续',
252
- default: false
253
- })
254
- if (!confirm) {
255
- throw new CloudBaseError('操作终止!')
256
- }
257
- isDir = true
258
- }
259
-
260
- // cloudPath 为 / 时,只能删除文件夹
261
- if (cloudPath === '/') {
262
- isDir = true
263
- }
264
-
265
- const fileText = isDir ? '文件夹' : '文件'
266
- loading.start(`删除${fileText}中`)
267
-
268
- if (isDir) {
269
- await storageService.deleteDirectory(cloudPath)
270
- } else {
271
- await storageService.deleteFile([cloudPath])
272
- }
273
-
274
- loading.succeed(`删除${fileText}成功!`)
275
- }
276
- }
277
-
278
- @ICommand()
279
- export class StorageListCommand extends Command {
280
- get options() {
281
- return {
282
- cmd: 'storage',
283
- childCmd: 'list [cloudPath]',
284
- deprecateCmd: 'storage:list [cloudPath]',
285
- options: [
286
- {
287
- flags: '-e, --envId <envId>',
288
- desc: '环境 Id'
289
- }
290
- // {
291
- // flags: '-m, --max',
292
- // desc: '传输数据的最大条数'
293
- // },
294
- // {
295
- // flags: '--markder',
296
- // desc: '起始路径名,后(不含)按照 UTF-8 字典序返回条目'
297
- // }
298
- ],
299
- desc: '获取文件存储的文件列表,不指定路径时获取全部文件列表'
300
- }
301
- }
302
-
303
- @InjectParams()
304
- async execute(@EnvId() envId, @ArgsParams() params) {
305
- const cloudPath = params?.[0]
306
- const storageService = await getStorageService(envId)
307
-
308
- const loading = loadingFactory()
309
- loading.start('获取文件列表中...')
310
- const list = await storageService.listDirectoryFiles(cloudPath)
311
- loading.stop()
312
-
313
- const head = ['序号', 'Key', 'LastModified', 'ETag', 'Size(KB)']
314
-
315
- const notDir = (item) => !(Number(item.Size) === 0 && /\/$/g.test(item.Key))
316
-
317
- const tableData = list
318
- .filter(notDir)
319
- .map((item, index) => [
320
- index + 1,
321
- item.Key,
322
- formatDate(item.LastModified, 'yyyy-MM-dd hh:mm:ss'),
323
- item.ETag,
324
- String(formateFileSize(item.Size, 'KB'))
325
- ])
326
- printHorizontalTable(head, tableData)
327
- }
328
- }
329
-
330
- @ICommand()
331
- export class GetUrlCommand extends Command {
332
- get options() {
333
- return {
334
- cmd: 'storage',
335
- childCmd: 'url <cloudPath>',
336
- deprecateCmd: 'storage:url <cloudPath>',
337
- options: [
338
- {
339
- flags: '-e, --envId <envId>',
340
- desc: '环境 Id'
341
- }
342
- ],
343
- desc: '获取文件临时访问地址'
344
- }
345
- }
346
-
347
- @InjectParams()
348
- async execute(@EnvId() envId, @ArgsParams() params, @Log() log: Logger) {
349
- const cloudPath = params?.[0]
350
- const storageService = await getStorageService(envId)
351
-
352
- const fileList = await storageService.getTemporaryUrl([cloudPath])
353
- const { url } = fileList[0]
354
-
355
- log.success(`文件临时访问地址:${url}`)
356
- }
357
- }
358
-
359
- @ICommand()
360
- export class StorageDetailCommand extends Command {
361
- get options() {
362
- return {
363
- cmd: 'storage',
364
- childCmd: 'detail <cloudPath>',
365
- deprecateCmd: 'storage:detail <cloudPath>',
366
- options: [
367
- {
368
- flags: '-e, --envId <envId>',
369
- desc: '环境 Id'
370
- }
371
- ],
372
- desc: '获取文件信息'
373
- }
374
- }
375
-
376
- @InjectParams()
377
- async execute(@EnvId() envId, @ArgsParams() params) {
378
- const cloudPath = params?.[0]
379
- const storageService = await getStorageService(envId)
380
-
381
- const fileInfo = await storageService.getFileInfo(cloudPath)
382
- const date = formatDate(fileInfo.Date, 'yyyy-MM-dd hh:mm:ss')
383
-
384
- const logInfo = `文件大小:${fileInfo.Size}\n文件类型:${fileInfo.Type}\n修改日期:${date}\nETag:${fileInfo.ETag}
385
- `
386
- console.log(logInfo)
387
- }
388
- }
389
-
390
- @ICommand()
391
- export class GetAclCommand extends Command {
392
- get options() {
393
- return {
394
- cmd: 'storage',
395
- childCmd: 'get-acl',
396
- deprecateCmd: 'storage:get-acl',
397
- options: [
398
- {
399
- flags: '-e, --envId <envId>',
400
- desc: '环境 Id'
401
- }
402
- ],
403
- desc: '获取文件存储权限信息'
404
- }
405
- }
406
-
407
- @InjectParams()
408
- async execute(@EnvId() envId) {
409
- const storageService = await getStorageService(envId)
410
- const acl = await storageService.getStorageAcl()
411
-
412
- console.log(`当前权限【${AclMap[acl]}】`)
413
- }
414
- }
415
-
416
- @ICommand()
417
- export class setAclCommand extends Command {
418
- get options() {
419
- return {
420
- cmd: 'storage',
421
- childCmd: 'set-acl',
422
- deprecateCmd: 'storage:set-acl',
423
- options: [
424
- {
425
- flags: '-e, --envId <envId>',
426
- desc: '环境 Id'
427
- }
428
- ],
429
- desc: '设置文件存储权限信息'
430
- }
431
- }
432
-
433
- @InjectParams()
434
- async execute(@EnvId() envId, @Log() log: Logger) {
435
- const storageService = await getStorageService(envId)
436
-
437
- const { acl } = await inquirer.prompt({
438
- type: 'list',
439
- name: 'acl',
440
- message: '选择权限',
441
- choices: [
442
- {
443
- name: '所有用户可读,仅创建者和管理员可写',
444
- value: 'READONLY'
445
- },
446
- {
447
- name: '仅创建者及管理员可读写',
448
- value: 'PRIVATE'
449
- },
450
- {
451
- name: '所有用户可读,仅管理员可写',
452
- value: 'ADMINWRITE'
453
- },
454
- {
455
- name: '仅管理员可读写',
456
- value: 'ADMINONLY'
457
- }
458
- ]
459
- })
460
-
461
- await storageService.setStorageAcl(acl)
462
- log.success('设置存储权限成功!')
463
- }
464
- }
1
+ import fs from 'fs'
2
+ import path from 'path'
3
+ import inquirer from 'inquirer'
4
+ import logSymbols from 'log-symbols'
5
+ import { StorageService } from '@cloudbase/manager-node/types/storage'
6
+ import { Command, ICommand } from '../common'
7
+
8
+ import {
9
+ loadingFactory,
10
+ printHorizontalTable,
11
+ formatDate,
12
+ createUploadProgressBar,
13
+ formateFileSize,
14
+ getMangerService,
15
+ isDirectory,
16
+ checkFullAccess
17
+ } from '../../utils'
18
+
19
+ import { CloudBaseError } from '../../error'
20
+ import { InjectParams, EnvId, ArgsParams, ArgsOptions, Log, Logger } from '../../decorators'
21
+
22
+ const AclMap = {
23
+ READONLY: '所有用户可读,仅创建者和管理员可写',
24
+ PRIVATE: '仅创建者及管理员可读写',
25
+ ADMINWRITE: '所有用户可读,仅管理员可写',
26
+ ADMINONLY: '仅管理员可读写'
27
+ }
28
+
29
+ async function getStorageService(envId: string): Promise<StorageService> {
30
+ const { storage } = await getMangerService(envId)
31
+ return storage
32
+ }
33
+
34
+ function checkCloudPath(cloudPath: string) {
35
+ if (cloudPath?.[0] === '/') {
36
+ throw new CloudBaseError('cloudPath 不能以 "/" 开头')
37
+ }
38
+ }
39
+
40
+ @ICommand()
41
+ export class UploadCommand extends Command {
42
+ get options() {
43
+ return {
44
+ cmd: 'storage',
45
+ childCmd: 'upload <localPath> [cloudPath]',
46
+ deprecateCmd: 'storage:upload <localPath> [cloudPath]',
47
+ options: [
48
+ {
49
+ flags: '-e, --envId <envId>',
50
+ desc: '环境 Id'
51
+ },
52
+ { flags: '--times <times>', desc: '设置上传重试次数,默认值为 1' },
53
+ {
54
+ flags: '--interval <interval>',
55
+ desc: '设置上传失败时,重试时间间隔(毫秒),默认值为 500'
56
+ }
57
+ ],
58
+ desc: '上传文件/文件夹'
59
+ }
60
+ }
61
+
62
+ @InjectParams()
63
+ async execute(
64
+ @EnvId() envId,
65
+ @ArgsParams() params,
66
+ @ArgsOptions() options,
67
+ @Log() log: Logger
68
+ ) {
69
+ const localPath = params?.[0]
70
+ const cloudPath = params?.[1]
71
+ const retryCount = options?.times
72
+ const retryInterval = options?.interval
73
+ const resolveLocalPath = path.resolve(localPath)
74
+ if (!checkFullAccess(resolveLocalPath)) {
75
+ throw new CloudBaseError('文件未找到!')
76
+ }
77
+ if (retryCount > 10) {
78
+ throw new CloudBaseError('上传重试次数为 0-10 次之间')
79
+ }
80
+
81
+ const loading = loadingFactory()
82
+ loading.start('准备上传中...')
83
+ const storageService = await getStorageService(envId)
84
+ const isDir = fs.statSync(resolveLocalPath).isDirectory()
85
+
86
+ let totalFiles = 0
87
+
88
+ if (isDir) {
89
+ let files = await storageService.walkLocalDir(resolveLocalPath)
90
+ files = files.filter((item) => !isDirectory(item))
91
+ totalFiles = files.length
92
+ }
93
+
94
+ // 上传进度条
95
+ const onProgress = createUploadProgressBar(
96
+ () => {
97
+ !isDir && log.success('上传文件成功!')
98
+ },
99
+ () => {
100
+ loading.stop()
101
+ }
102
+ )
103
+
104
+ const successFiles = []
105
+ const failedFiles = []
106
+ if (isDir) {
107
+ await storageService.uploadDirectory({
108
+ localPath: resolveLocalPath,
109
+ cloudPath,
110
+ onProgress,
111
+ onFileFinish: (...args) => {
112
+ const error = args[0]
113
+ const fileInfo = (args as any)[2]
114
+ if (error) {
115
+ failedFiles.push(fileInfo.Key)
116
+ } else {
117
+ successFiles.push(fileInfo.Key)
118
+ }
119
+ },
120
+ retryCount: retryCount || 1,
121
+ retryInterval,
122
+ parallel: 20
123
+ })
124
+
125
+ log.success(`文件共计 ${totalFiles} 个`)
126
+ log.success(`文件上传成功 ${successFiles.length} 个`)
127
+ // 上传成功的文件
128
+ if (totalFiles <= 50) {
129
+ printHorizontalTable(
130
+ ['状态', '文件'],
131
+ successFiles.map((item) => [logSymbols.success, item])
132
+ )
133
+ }
134
+
135
+ // 上传失败的文件
136
+ log.error(`文件上传失败 ${failedFiles.length} 个`)
137
+ if (failedFiles.length) {
138
+ if (totalFiles <= 50) {
139
+ printHorizontalTable(
140
+ ['状态', '文件'],
141
+ failedFiles.map((item) => [logSymbols.error, item])
142
+ )
143
+ } else {
144
+ // 写入文件到本地
145
+ const errorLogPath = path.resolve('./cloudbase-error.log')
146
+ log.error('上传失败文件:')
147
+ console.log(errorLogPath)
148
+ fs.writeFileSync(errorLogPath, failedFiles.join('\n'))
149
+ }
150
+ }
151
+ } else {
152
+ const assignCloudPath = cloudPath || path.parse(resolveLocalPath).base
153
+ await storageService.uploadFile({
154
+ localPath: resolveLocalPath,
155
+ cloudPath: assignCloudPath,
156
+ onProgress
157
+ })
158
+ }
159
+ }
160
+ }
161
+
162
+ @ICommand()
163
+ export class DownloadCommand extends Command {
164
+ get options() {
165
+ return {
166
+ cmd: 'storage',
167
+ childCmd: 'download <cloudPath> <localPath>',
168
+ deprecateCmd: 'storage:download <cloudPath> <localPath>',
169
+ options: [
170
+ {
171
+ flags: '-e, --envId <envId>',
172
+ desc: '环境 Id'
173
+ },
174
+ {
175
+ flags: '-d, --dir',
176
+ desc: '下载目标是否为文件夹'
177
+ }
178
+ ],
179
+ desc: '下载文件/文件夹,文件夹需指定 --dir 选项'
180
+ }
181
+ }
182
+
183
+ @InjectParams()
184
+ async execute(@EnvId() envId, @ArgsOptions() options, @ArgsParams() params) {
185
+ const cloudPath = params?.[0]
186
+ const localPath = params?.[1]
187
+ const storageService = await getStorageService(envId)
188
+ const resolveLocalPath = path.resolve(localPath)
189
+
190
+ const { dir } = options
191
+ const fileText = dir ? '文件夹' : '文件'
192
+
193
+ if (dir && !checkFullAccess(resolveLocalPath)) {
194
+ throw new CloudBaseError('存储文件夹不存在!')
195
+ }
196
+
197
+ const loading = loadingFactory()
198
+
199
+ loading.start(`下载${fileText}中`)
200
+
201
+ if (dir) {
202
+ await storageService.downloadDirectory({
203
+ localPath: resolveLocalPath,
204
+ cloudPath,
205
+ parallel: 20
206
+ })
207
+ } else {
208
+ await storageService.downloadFile({
209
+ cloudPath,
210
+ localPath: resolveLocalPath
211
+ })
212
+ }
213
+
214
+ loading.succeed(`下载${fileText}成功!`)
215
+ }
216
+ }
217
+
218
+ @ICommand()
219
+ export class DeleteFileCommand extends Command {
220
+ get options() {
221
+ return {
222
+ cmd: 'storage',
223
+ childCmd: 'delete [cloudPath]',
224
+ deprecateCmd: 'storage:delete [cloudPath]',
225
+ options: [
226
+ {
227
+ flags: '-e, --envId <envId>',
228
+ desc: '环境 Id'
229
+ },
230
+ {
231
+ flags: '-d, --dir',
232
+ desc: '下载目标是否为文件夹'
233
+ }
234
+ ],
235
+ desc: '删除文件/文件夹,文件夹需指定 --dir 选项'
236
+ }
237
+ }
238
+
239
+ @InjectParams()
240
+ async execute(@EnvId() envId, @ArgsOptions() options, @ArgsParams() params) {
241
+ let isDir = options.dir
242
+ const cloudPath = params?.[0]
243
+ const loading = loadingFactory()
244
+ const storageService = await getStorageService(envId)
245
+
246
+ // 删除所有文件,危险操作,需要提示
247
+ if (!cloudPath) {
248
+ const { confirm } = await inquirer.prompt({
249
+ type: 'confirm',
250
+ name: 'confirm',
251
+ message: '指定云端路径为空,将会删除所有文件,是否继续',
252
+ default: false
253
+ })
254
+ if (!confirm) {
255
+ throw new CloudBaseError('操作终止!')
256
+ }
257
+ isDir = true
258
+ }
259
+
260
+ // cloudPath 为 / 时,只能删除文件夹
261
+ if (cloudPath === '/') {
262
+ isDir = true
263
+ }
264
+
265
+ const fileText = isDir ? '文件夹' : '文件'
266
+ loading.start(`删除${fileText}中`)
267
+
268
+ if (isDir) {
269
+ await storageService.deleteDirectory(cloudPath)
270
+ } else {
271
+ await storageService.deleteFile([cloudPath])
272
+ }
273
+
274
+ loading.succeed(`删除${fileText}成功!`)
275
+ }
276
+ }
277
+
278
+ @ICommand()
279
+ export class StorageListCommand extends Command {
280
+ get options() {
281
+ return {
282
+ cmd: 'storage',
283
+ childCmd: 'list [cloudPath]',
284
+ deprecateCmd: 'storage:list [cloudPath]',
285
+ options: [
286
+ {
287
+ flags: '-e, --envId <envId>',
288
+ desc: '环境 Id'
289
+ }
290
+ // {
291
+ // flags: '-m, --max',
292
+ // desc: '传输数据的最大条数'
293
+ // },
294
+ // {
295
+ // flags: '--markder',
296
+ // desc: '起始路径名,后(不含)按照 UTF-8 字典序返回条目'
297
+ // }
298
+ ],
299
+ desc: '获取文件存储的文件列表,不指定路径时获取全部文件列表'
300
+ }
301
+ }
302
+
303
+ @InjectParams()
304
+ async execute(@EnvId() envId, @ArgsParams() params) {
305
+ const cloudPath = params?.[0]
306
+ const storageService = await getStorageService(envId)
307
+
308
+ const loading = loadingFactory()
309
+ loading.start('获取文件列表中...')
310
+ const list = await storageService.listDirectoryFiles(cloudPath)
311
+ loading.stop()
312
+
313
+ const head = ['序号', 'Key', 'LastModified', 'ETag', 'Size(KB)']
314
+
315
+ const notDir = (item) => !(Number(item.Size) === 0 && /\/$/g.test(item.Key))
316
+
317
+ const tableData = list
318
+ .filter(notDir)
319
+ .map((item, index) => [
320
+ index + 1,
321
+ item.Key,
322
+ formatDate(item.LastModified, 'yyyy-MM-dd hh:mm:ss'),
323
+ item.ETag,
324
+ String(formateFileSize(item.Size, 'KB'))
325
+ ])
326
+ printHorizontalTable(head, tableData)
327
+ }
328
+ }
329
+
330
+ @ICommand()
331
+ export class GetUrlCommand extends Command {
332
+ get options() {
333
+ return {
334
+ cmd: 'storage',
335
+ childCmd: 'url <cloudPath>',
336
+ deprecateCmd: 'storage:url <cloudPath>',
337
+ options: [
338
+ {
339
+ flags: '-e, --envId <envId>',
340
+ desc: '环境 Id'
341
+ }
342
+ ],
343
+ desc: '获取文件临时访问地址'
344
+ }
345
+ }
346
+
347
+ @InjectParams()
348
+ async execute(@EnvId() envId, @ArgsParams() params, @Log() log: Logger) {
349
+ const cloudPath = params?.[0]
350
+ const storageService = await getStorageService(envId)
351
+
352
+ const fileList = await storageService.getTemporaryUrl([cloudPath])
353
+ const { url } = fileList[0]
354
+
355
+ log.success(`文件临时访问地址:${url}`)
356
+ }
357
+ }
358
+
359
+ @ICommand()
360
+ export class StorageDetailCommand extends Command {
361
+ get options() {
362
+ return {
363
+ cmd: 'storage',
364
+ childCmd: 'detail <cloudPath>',
365
+ deprecateCmd: 'storage:detail <cloudPath>',
366
+ options: [
367
+ {
368
+ flags: '-e, --envId <envId>',
369
+ desc: '环境 Id'
370
+ }
371
+ ],
372
+ desc: '获取文件信息'
373
+ }
374
+ }
375
+
376
+ @InjectParams()
377
+ async execute(@EnvId() envId, @ArgsParams() params) {
378
+ const cloudPath = params?.[0]
379
+ const storageService = await getStorageService(envId)
380
+
381
+ const fileInfo = await storageService.getFileInfo(cloudPath)
382
+ const date = formatDate(fileInfo.Date, 'yyyy-MM-dd hh:mm:ss')
383
+
384
+ const logInfo = `文件大小:${fileInfo.Size}\n文件类型:${fileInfo.Type}\n修改日期:${date}\nETag:${fileInfo.ETag}
385
+ `
386
+ console.log(logInfo)
387
+ }
388
+ }
389
+
390
+ @ICommand()
391
+ export class GetAclCommand extends Command {
392
+ get options() {
393
+ return {
394
+ cmd: 'storage',
395
+ childCmd: 'get-acl',
396
+ deprecateCmd: 'storage:get-acl',
397
+ options: [
398
+ {
399
+ flags: '-e, --envId <envId>',
400
+ desc: '环境 Id'
401
+ }
402
+ ],
403
+ desc: '获取文件存储权限信息'
404
+ }
405
+ }
406
+
407
+ @InjectParams()
408
+ async execute(@EnvId() envId) {
409
+ const storageService = await getStorageService(envId)
410
+ const acl = await storageService.getStorageAcl()
411
+
412
+ console.log(`当前权限【${AclMap[acl]}】`)
413
+ }
414
+ }
415
+
416
+ @ICommand()
417
+ export class setAclCommand extends Command {
418
+ get options() {
419
+ return {
420
+ cmd: 'storage',
421
+ childCmd: 'set-acl',
422
+ deprecateCmd: 'storage:set-acl',
423
+ options: [
424
+ {
425
+ flags: '-e, --envId <envId>',
426
+ desc: '环境 Id'
427
+ }
428
+ ],
429
+ desc: '设置文件存储权限信息'
430
+ }
431
+ }
432
+
433
+ @InjectParams()
434
+ async execute(@EnvId() envId, @Log() log: Logger) {
435
+ const storageService = await getStorageService(envId)
436
+
437
+ const { acl } = await inquirer.prompt({
438
+ type: 'list',
439
+ name: 'acl',
440
+ message: '选择权限',
441
+ choices: [
442
+ {
443
+ name: '所有用户可读,仅创建者和管理员可写',
444
+ value: 'READONLY'
445
+ },
446
+ {
447
+ name: '仅创建者及管理员可读写',
448
+ value: 'PRIVATE'
449
+ },
450
+ {
451
+ name: '所有用户可读,仅管理员可写',
452
+ value: 'ADMINWRITE'
453
+ },
454
+ {
455
+ name: '仅管理员可读写',
456
+ value: 'ADMINONLY'
457
+ }
458
+ ]
459
+ })
460
+
461
+ await storageService.setStorageAcl(acl)
462
+ log.success('设置存储权限成功!')
463
+ }
464
+ }