@cloudbase/cli 2.0.0 → 2.0.3-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 (206) 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 +6 -4
  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/lib/run/service/deployPackage.js +6 -4
  16. package/lib/utils/net/http-request.js +4 -4
  17. package/lib/utils/tcbrApi/tcbr-cloud-api/request.js +3 -3
  18. package/package.json +17 -17
  19. package/post-install.js +61 -61
  20. package/runtime/nodejs/bootstrap.js +255 -255
  21. package/runtime/nodejs/runtime.js +183 -183
  22. package/src/auth/index.ts +1 -1
  23. package/src/auth/login.ts +91 -91
  24. package/src/auth/logout.ts +7 -7
  25. package/src/commands/account/index.ts +2 -2
  26. package/src/commands/account/login.ts +192 -192
  27. package/src/commands/account/logout.ts +24 -24
  28. package/src/commands/common.ts +246 -246
  29. package/src/commands/env/base.ts +90 -90
  30. package/src/commands/env/create.ts +92 -92
  31. package/src/commands/env/domain.ts +186 -186
  32. package/src/commands/env/index.ts +4 -4
  33. package/src/commands/env/login.ts +235 -235
  34. package/src/commands/framework/index.ts +124 -124
  35. package/src/commands/functions/alias/getRoute.ts +76 -76
  36. package/src/commands/functions/alias/index.ts +2 -2
  37. package/src/commands/functions/alias/setRoute.ts +82 -82
  38. package/src/commands/functions/code-download.ts +100 -100
  39. package/src/commands/functions/code-update.ts +62 -62
  40. package/src/commands/functions/concurrency/delete.ts +45 -45
  41. package/src/commands/functions/concurrency/index.ts +2 -2
  42. package/src/commands/functions/concurrency/list.ts +58 -58
  43. package/src/commands/functions/concurrency/set.ts +47 -47
  44. package/src/commands/functions/config-update.ts +76 -76
  45. package/src/commands/functions/copy.ts +62 -62
  46. package/src/commands/functions/delete.ts +79 -79
  47. package/src/commands/functions/deploy.ts +293 -293
  48. package/src/commands/functions/detail.ts +138 -138
  49. package/src/commands/functions/index.ts +16 -16
  50. package/src/commands/functions/invoke.ts +121 -121
  51. package/src/commands/functions/layer/bind.ts +182 -182
  52. package/src/commands/functions/layer/common.ts +8 -8
  53. package/src/commands/functions/layer/create.ts +49 -49
  54. package/src/commands/functions/layer/delete.ts +73 -73
  55. package/src/commands/functions/layer/download.ts +92 -92
  56. package/src/commands/functions/layer/index.ts +7 -7
  57. package/src/commands/functions/layer/list.ts +94 -94
  58. package/src/commands/functions/layer/sort.ts +76 -76
  59. package/src/commands/functions/list.ts +68 -68
  60. package/src/commands/functions/log.ts +148 -148
  61. package/src/commands/functions/run.ts +249 -249
  62. package/src/commands/functions/trigger-create.ts +79 -79
  63. package/src/commands/functions/trigger-delete.ts +105 -105
  64. package/src/commands/functions/version/index.ts +1 -1
  65. package/src/commands/functions/version/list.ts +73 -73
  66. package/src/commands/functions/version/publish.ts +43 -43
  67. package/src/commands/gateway/create.ts +109 -109
  68. package/src/commands/gateway/delete.ts +81 -81
  69. package/src/commands/gateway/domain.ts +159 -159
  70. package/src/commands/gateway/index.ts +5 -5
  71. package/src/commands/gateway/list.ts +76 -76
  72. package/src/commands/gateway/switch.ts +107 -107
  73. package/src/commands/helpers/index.ts +2 -2
  74. package/src/commands/helpers/init.ts +431 -431
  75. package/src/commands/helpers/new.ts +117 -117
  76. package/src/commands/helpers/open.ts +67 -67
  77. package/src/commands/hosting/hosting.ts +360 -360
  78. package/src/commands/index.ts +13 -13
  79. package/src/commands/lowcode/app.ts +34 -34
  80. package/src/commands/lowcode/comps.ts +322 -322
  81. package/src/commands/lowcode/index.ts +1 -1
  82. package/src/commands/lowcode/utils.ts +24 -24
  83. package/src/commands/run/image/index.ts +4 -4
  84. package/src/commands/run/standalonegateway/common.ts +7 -7
  85. package/src/commands/run/standalonegateway/create.ts +85 -85
  86. package/src/commands/run/standalonegateway/destroy.ts +59 -59
  87. package/src/commands/run/standalonegateway/index.ts +4 -4
  88. package/src/commands/run/standalonegateway/list.ts +53 -53
  89. package/src/commands/run/standalonegateway/package.ts +62 -62
  90. package/src/commands/run/standalonegateway/turn.ts +63 -63
  91. package/src/commands/run/version/index.ts +4 -4
  92. package/src/commands/smart.ts +132 -132
  93. package/src/commands/storage/storage.ts +464 -464
  94. package/src/commands/third/thirdAttach.ts +49 -49
  95. package/src/completion/index.ts +13 -13
  96. package/src/decorators/captureError.ts +25 -25
  97. package/src/decorators/constants.ts +12 -12
  98. package/src/decorators/deprecate.ts +25 -25
  99. package/src/decorators/guard.ts +42 -42
  100. package/src/decorators/index.ts +7 -7
  101. package/src/decorators/injectParams.ts +54 -54
  102. package/src/decorators/params/common.ts +28 -28
  103. package/src/decorators/params/index.ts +35 -35
  104. package/src/env/domain.ts +33 -33
  105. package/src/env/index.ts +63 -63
  106. package/src/env/login.ts +80 -80
  107. package/src/error.ts +36 -36
  108. package/src/function/alias.ts +43 -43
  109. package/src/function/base.ts +253 -253
  110. package/src/function/code.ts +55 -55
  111. package/src/function/concurrency.ts +57 -57
  112. package/src/function/create.ts +78 -78
  113. package/src/function/delete.ts +42 -42
  114. package/src/function/index.ts +10 -10
  115. package/src/function/layer/attach.ts +68 -68
  116. package/src/function/layer/create.ts +63 -63
  117. package/src/function/layer/delete.ts +21 -21
  118. package/src/function/layer/download.ts +54 -54
  119. package/src/function/layer/index.ts +7 -7
  120. package/src/function/layer/list.ts +32 -32
  121. package/src/function/layer/sort.ts +24 -24
  122. package/src/function/trigger.ts +97 -97
  123. package/src/function/update.ts +35 -35
  124. package/src/function/version.ts +38 -38
  125. package/src/function/vpc.ts +22 -22
  126. package/src/gateway/index.ts +137 -137
  127. package/src/hosting.ts +212 -212
  128. package/src/index.ts +13 -13
  129. package/src/logger.ts +17 -17
  130. package/src/run/create.ts +23 -23
  131. package/src/run/delete.ts +15 -15
  132. package/src/run/image/build.ts +36 -36
  133. package/src/run/image/delete.ts +13 -13
  134. package/src/run/image/index.ts +3 -3
  135. package/src/run/image/info.ts +26 -26
  136. package/src/run/list.ts +29 -29
  137. package/src/run/repo.ts +24 -24
  138. package/src/run/service/deployPackage.ts +6 -4
  139. package/src/run/standalonegateway/create.ts +24 -24
  140. package/src/run/standalonegateway/destroy.ts +19 -19
  141. package/src/run/standalonegateway/index.ts +4 -4
  142. package/src/run/standalonegateway/list.ts +74 -74
  143. package/src/run/standalonegateway/package/list.ts +24 -24
  144. package/src/run/standalonegateway/turn/index.ts +1 -1
  145. package/src/run/standalonegateway/turn/off.ts +19 -19
  146. package/src/run/standalonegateway/turn/on.ts +19 -19
  147. package/src/run/version/create.ts +68 -68
  148. package/src/run/version/delete.ts +15 -15
  149. package/src/run/version/index.ts +5 -5
  150. package/src/run/version/list.ts +16 -16
  151. package/src/run/version/modify.ts +16 -16
  152. package/src/run/version/repo.ts +27 -27
  153. package/src/run/version/update.ts +58 -58
  154. package/src/storage.ts +114 -114
  155. package/src/third/index.ts +12 -12
  156. package/src/utils/auth.ts +15 -15
  157. package/src/utils/cli-table.ts +23 -23
  158. package/src/utils/config.ts +39 -39
  159. package/src/utils/env.ts +244 -244
  160. package/src/utils/fs/del.ts +5 -5
  161. package/src/utils/fs/index.ts +71 -71
  162. package/src/utils/function-packer.ts +97 -97
  163. package/src/utils/log.ts +81 -81
  164. package/src/utils/net/cloud-api-request.ts +62 -62
  165. package/src/utils/net/credential.ts +53 -53
  166. package/src/utils/net/http-request.ts +84 -84
  167. package/src/utils/net/index.ts +4 -4
  168. package/src/utils/net/manager-service.ts +36 -36
  169. package/src/utils/net/proxy.ts +6 -6
  170. package/src/utils/notice.ts +28 -28
  171. package/src/utils/output/highlight.ts +5 -5
  172. package/src/utils/output/index.ts +2 -2
  173. package/src/utils/output/link.ts +10 -10
  174. package/src/utils/output/loading.ts +82 -82
  175. package/src/utils/parallel.ts +82 -82
  176. package/src/utils/platform/index.ts +2 -2
  177. package/src/utils/platform/mac.ts +21 -21
  178. package/src/utils/platform/os.ts +64 -64
  179. package/src/utils/platform/port.ts +10 -10
  180. package/src/utils/progress-bar.ts +38 -38
  181. package/src/utils/prompt/select.ts +59 -59
  182. package/src/utils/reporter/agree.ts +20 -20
  183. package/src/utils/reporter/download.ts +26 -26
  184. package/src/utils/reporter/index.ts +3 -3
  185. package/src/utils/reporter/usage.ts +20 -20
  186. package/src/utils/store/auth.ts +49 -49
  187. package/src/utils/store/common.ts +8 -8
  188. package/src/utils/store/db.ts +68 -68
  189. package/src/utils/store/index.ts +4 -4
  190. package/src/utils/store/usage.ts +12 -12
  191. package/src/utils/tcbrApi/tcbr-cloud-api/request.ts +13 -13
  192. package/src/utils/template.ts +170 -170
  193. package/src/utils/tools/encoding.ts +8 -8
  194. package/src/utils/tools/index.ts +4 -4
  195. package/src/utils/tools/object.ts +33 -33
  196. package/src/utils/tools/time.ts +38 -38
  197. package/src/utils/tools/uid.ts +19 -19
  198. package/templates/html/loginFail.html +90 -90
  199. package/templates/html/loginSuccess.html +86 -86
  200. package/templates/server/node/_gitignore +54 -54
  201. package/templates/server/node/cloudbaserc.json +10 -10
  202. package/templates/server/node/index.js +5 -5
  203. package/templates/server/node/package.json +9 -9
  204. package/tsconfig.json +19 -19
  205. package/tsconfig.test.json +13 -13
  206. package/.vscode/settings.json +0 -3
@@ -1,13 +1,13 @@
1
- import { CloudApiService } from '../utils'
2
- const tcbService = CloudApiService.getInstance('tcb')
3
-
4
- // 解除第三方小程序绑定
5
- export async function deleteThirdPartAttach(options) {
6
- const { ThirdPartAppid, TypeFlag } = options
7
- const res: any = await tcbService.request('DeleteThirdPartAttach', {
8
- ThirdPartAppid,
9
- TypeFlag
10
- })
11
-
12
- return res
1
+ import { CloudApiService } from '../utils'
2
+ const tcbService = CloudApiService.getInstance('tcb')
3
+
4
+ // 解除第三方小程序绑定
5
+ export async function deleteThirdPartAttach(options) {
6
+ const { ThirdPartAppid, TypeFlag } = options
7
+ const res: any = await tcbService.request('DeleteThirdPartAttach', {
8
+ ThirdPartAppid,
9
+ TypeFlag
10
+ })
11
+
12
+ return res
13
13
  }
package/src/utils/auth.ts CHANGED
@@ -1,15 +1,15 @@
1
- import _ from 'lodash'
2
- import { AuthSupevisor } from '@cloudbase/toolbox'
3
- import { getProxy } from './net'
4
- import { REQUEST_TIMEOUT } from '../constant'
5
-
6
- export const authSupevisor = AuthSupevisor.getInstance({
7
- cache: true,
8
- proxy: getProxy(),
9
- timeout: REQUEST_TIMEOUT,
10
- throwError: true
11
- })
12
-
13
- export async function getLoginState() {
14
- return authSupevisor.getLoginState()
15
- }
1
+ import _ from 'lodash'
2
+ import { AuthSupevisor } from '@cloudbase/toolbox'
3
+ import { getProxy } from './net'
4
+ import { REQUEST_TIMEOUT } from '../constant'
5
+
6
+ export const authSupevisor = AuthSupevisor.getInstance({
7
+ cache: true,
8
+ proxy: getProxy(),
9
+ timeout: REQUEST_TIMEOUT,
10
+ throwError: true
11
+ })
12
+
13
+ export async function getLoginState() {
14
+ return authSupevisor.getLoginState()
15
+ }
@@ -1,23 +1,23 @@
1
- import Table, { TableOptions, HorizontalTable } from 'cli-table3'
2
-
3
- // 打印水平方向的表格
4
- export function printHorizontalTable(
5
- head: string[],
6
- data: (string | number)[][] = [],
7
- options?: TableOptions
8
- ) {
9
- if (!data?.length) {
10
- console.log('列表数据为空')
11
- }
12
- const table: HorizontalTable = new Table({
13
- head,
14
- style: { head: ['yellow'] },
15
- colAligns: new Array(head.length).fill('center'),
16
- ...options
17
- }) as HorizontalTable
18
-
19
- data.forEach((item: Table.Cell[]) => {
20
- table.push(item)
21
- })
22
- console.log(table.toString())
23
- }
1
+ import Table, { TableOptions, HorizontalTable } from 'cli-table3'
2
+
3
+ // 打印水平方向的表格
4
+ export function printHorizontalTable(
5
+ head: string[],
6
+ data: (string | number)[][] = [],
7
+ options?: TableOptions
8
+ ) {
9
+ if (!data?.length) {
10
+ console.log('列表数据为空')
11
+ }
12
+ const table: HorizontalTable = new Table({
13
+ head,
14
+ style: { head: ['yellow'] },
15
+ colAligns: new Array(head.length).fill('center'),
16
+ ...options
17
+ }) as HorizontalTable
18
+
19
+ data.forEach((item: Table.Cell[]) => {
20
+ table.push(item)
21
+ })
22
+ console.log(table.toString())
23
+ }
@@ -1,39 +1,39 @@
1
- import _ from 'lodash'
2
- import path from 'path'
3
- import yargs, { Arguments } from 'yargs'
4
- import { ConfigParser, ICloudBaseConfig } from '@cloudbase/toolbox'
5
-
6
- export interface IArgs {
7
- envId: string
8
- region: string
9
- verbose: boolean
10
- configPath: string
11
- [x: string]: unknown
12
- }
13
-
14
- export const getArgs = (): Arguments<IArgs> => {
15
- // console.log(yargs.argv)
16
- return yargs.alias('e', 'envId').alias('r', 'region').argv as any
17
- }
18
-
19
- // 获取 cloudbase 配置
20
- export const getCloudBaseConfig = async (configPath?: string): Promise<ICloudBaseConfig> => {
21
- const args = getArgs()
22
-
23
- let specificConfigPath = configPath || args.configPath
24
- specificConfigPath = specificConfigPath ? path.resolve(specificConfigPath) : undefined
25
-
26
- const parser = new ConfigParser({
27
- configPath: specificConfigPath
28
- })
29
- const config: ICloudBaseConfig = await parser.get()
30
-
31
- // 合并默认配置
32
- if (config?.functionDefaultConfig && config?.functions?.length) {
33
- config.functions = config.functions.map((rawConfig) =>
34
- _.merge({}, config.functionDefaultConfig, rawConfig)
35
- )
36
- }
37
-
38
- return config
39
- }
1
+ import _ from 'lodash'
2
+ import path from 'path'
3
+ import yargs, { Arguments } from 'yargs'
4
+ import { ConfigParser, ICloudBaseConfig } from '@cloudbase/toolbox'
5
+
6
+ export interface IArgs {
7
+ envId: string
8
+ region: string
9
+ verbose: boolean
10
+ configPath: string
11
+ [x: string]: unknown
12
+ }
13
+
14
+ export const getArgs = (): Arguments<IArgs> => {
15
+ // console.log(yargs.argv)
16
+ return yargs.alias('e', 'envId').alias('r', 'region').argv as any
17
+ }
18
+
19
+ // 获取 cloudbase 配置
20
+ export const getCloudBaseConfig = async (configPath?: string): Promise<ICloudBaseConfig> => {
21
+ const args = getArgs()
22
+
23
+ let specificConfigPath = configPath || args.configPath
24
+ specificConfigPath = specificConfigPath ? path.resolve(specificConfigPath) : undefined
25
+
26
+ const parser = new ConfigParser({
27
+ configPath: specificConfigPath
28
+ })
29
+ const config: ICloudBaseConfig = await parser.get()
30
+
31
+ // 合并默认配置
32
+ if (config?.functionDefaultConfig && config?.functions?.length) {
33
+ config.functions = config.functions.map((rawConfig) =>
34
+ _.merge({}, config.functionDefaultConfig, rawConfig)
35
+ )
36
+ }
37
+
38
+ return config
39
+ }
package/src/utils/env.ts CHANGED
@@ -1,244 +1,244 @@
1
- import _ from 'lodash'
2
- import open from 'open'
3
- import { prompt } from 'enquirer'
4
- import { getDataFromWeb, isCamRefused } from '@cloudbase/toolbox'
5
- import { CloudBaseError } from '../error'
6
- import { listEnvs, getEnvInfo } from '../env'
7
- import { ENV_STATUS, STATUS_TEXT } from '../constant'
8
- import { CloudApiService, getMangerService } from './net'
9
- import { execWithLoading } from './output'
10
- import { logger } from './log'
11
-
12
- const tcbService = CloudApiService.getInstance('tcb')
13
-
14
- const ENV_INIT_TIP = '环境初始化中,预计需要三分钟'
15
- const CREATE_ENV = 'CREATE'
16
- const consoleUrl = 'https://console.cloud.tencent.com/tcb/env/index?action=CreateEnv&from=cli'
17
-
18
- /**
19
- * 展示环境列表选择器,获取用户选择的环境
20
- */
21
- export async function getSelectedEnv(inputEnvId?: string) {
22
- // 检查是否开通 TCB 服务
23
- const isInitNow = await checkTcbService()
24
-
25
- let envData = []
26
-
27
- // 刚初始化服务,新创建的环境还未就绪
28
- if (isInitNow) {
29
- envData = await execWithLoading(
30
- () => {
31
- // 等待用户完成创建环境的流程
32
- return new Promise((resolve) => {
33
- const timer = setInterval(async () => {
34
- const envs = await listEnvs()
35
- if (envs.length) {
36
- clearInterval(timer)
37
- resolve(envs)
38
- }
39
- }, 2000)
40
- })
41
- },
42
- {
43
- startTip: '获取环境列表中'
44
- }
45
- )
46
- } else {
47
- // 选择环境
48
- envData = await execWithLoading(() => listEnvs(), {
49
- startTip: '获取环境列表中'
50
- })
51
- }
52
-
53
- envData = envData || []
54
-
55
- // 检查输入的环境 Id 是否属于用户
56
- if (envData?.length && inputEnvId) {
57
- const inputEnvIdExist = envData.find((_) => _.EnvId === inputEnvId)
58
- if (!inputEnvIdExist) {
59
- throw new CloudBaseError(`你指定的环境 Id ${inputEnvId} 不存在,请指定正确的环境 Id!`)
60
- }
61
- return inputEnvId
62
- }
63
-
64
- // 按名称排序,构造展示数据
65
- const envs: { name: string; value: string }[] = envData
66
- .map((item) => {
67
- let name = `${item.Alias} - [${item.EnvId}:${item.PackageName || '按量计费'}]`
68
- if (item.Status !== ENV_STATUS.NORMAL) {
69
- name += `(${STATUS_TEXT[item.Status]})`
70
- }
71
-
72
- return {
73
- name,
74
- value: item.EnvId
75
- }
76
- })
77
- .sort((prev, next) => prev.value.charCodeAt(0) - next.value.charCodeAt(0))
78
-
79
- const choices = [
80
- ...envs,
81
- {
82
- name: envData.length ? '创建新环境' : '无可用环境,创建新环境',
83
- value: CREATE_ENV
84
- }
85
- ]
86
-
87
- let { env } = await prompt<any>({
88
- choices,
89
- type: 'select',
90
- name: 'env',
91
- message: '请选择关联环境',
92
- result(choice) {
93
- return this.map(choice)[choice]
94
- }
95
- })
96
-
97
- // 创建新环境
98
- if (env === CREATE_ENV) {
99
- logger.success('已打开控制台,请前往控制台创建环境')
100
- // 从控制台获取创建环境生成的 envId
101
- const { envId } = await getDataFromWeb((port) => `${consoleUrl}&port=${port}`, 'getData')
102
- if (!envId) {
103
- throw new CloudBaseError('接收环境 Id 信息失败,请重新运行 init 命令!')
104
- }
105
- logger.success(`创建环境成功,环境 Id: ${envId}`)
106
- env = envId
107
- }
108
-
109
- // 检查环境状态
110
- await checkEnvStatus(env)
111
-
112
- return env
113
- }
114
-
115
- // 选择地域
116
- export async function getSelectRegion() {
117
- const { region } = await prompt<any>({
118
- choices: [
119
- {
120
- name: '上海',
121
- value: 'ap-shanghai'
122
- },
123
- {
124
- name: '广州',
125
- value: 'ap-guangzhou'
126
- }
127
- ],
128
- type: 'select',
129
- name: 'region',
130
- message: '请选择环境所在地域',
131
- result(choice) {
132
- return this.map(choice)[choice]
133
- }
134
- })
135
-
136
- tcbService.region = region
137
-
138
- return region
139
- }
140
-
141
- // 检查 TCB 服务是否开通
142
- export async function checkTcbService(): Promise<boolean> {
143
- const app = await getMangerService()
144
- let Initialized
145
- try {
146
- Initialized = (await app.env.checkTcbService()).Initialized
147
- } catch (e) {
148
- // 忽略 CAM 权限问题
149
- if (!isCamRefused(e)) {
150
- throw e
151
- }
152
- }
153
-
154
- if (!Initialized) {
155
- const { jump } = await prompt<any>({
156
- type: 'confirm',
157
- name: 'jump',
158
- message:
159
- '你还没有开通云开发服务,是否跳转到控制台开通云开发服务?(取消将无法继续操作)',
160
- initial: true
161
- })
162
-
163
- if (!jump) {
164
- throw new CloudBaseError('init 操作终止,请开通云开发服务后再进行操作!')
165
- }
166
-
167
- // 打开控制台
168
- open(consoleUrl)
169
- logger.success('已打开云开发控制台,请登录并在云开发控制台中开通服务!')
170
-
171
- await execWithLoading(() => waitForServiceEnable(), {
172
- startTip: '等待云开发服务开通中',
173
- successTip: '云开发服务开通成功!'
174
- })
175
-
176
- // 返回一个是否刚初始化服务的标志
177
- return true
178
- }
179
-
180
- return false
181
- }
182
-
183
- // 检查环境的状态,是否可以正常使用
184
- export async function checkEnvStatus(envId: string) {
185
- const env = await getEnvInfo(envId)
186
- if (env.Status === ENV_STATUS.UNAVAILABLE) {
187
- await checkEnvAvaliable(envId)
188
- } else if (env.Status !== ENV_STATUS.NORMAL) {
189
- throw new CloudBaseError('所有环境状态异常')
190
- }
191
- }
192
-
193
- // 检测环境是否可用
194
- export async function checkEnvAvaliable(envId: string) {
195
- let count = 0
196
-
197
- await execWithLoading(
198
- (flush) => {
199
- const increase = setInterval(() => {
200
- flush(`${ENV_INIT_TIP} ${++count}S`)
201
- }, 1000)
202
-
203
- return new Promise<void>((resolve) => {
204
- const timer = setInterval(async () => {
205
- const env = await getEnvInfo(envId)
206
- // 环境初始化中
207
- if (env.Status === ENV_STATUS.NORMAL) {
208
- clearInterval(timer)
209
- clearInterval(increase)
210
- resolve()
211
- }
212
- }, 3000)
213
- })
214
- },
215
- {
216
- startTip: ENV_INIT_TIP,
217
- successTip: `环境 ${envId} 初始化成功`
218
- }
219
- )
220
- }
221
-
222
- // 等待服务开通
223
- async function waitForServiceEnable() {
224
- return new Promise<void>((resolve) => {
225
- const timer = setInterval(async () => {
226
- const app = await getMangerService()
227
- try {
228
- const { Initialized } = await app.env.checkTcbService()
229
- if (Initialized) {
230
- clearInterval(timer)
231
- setTimeout(() => {
232
- // 服务初始化完成到环境创建完成有一定的延迟时间,延迟 5S 返回
233
- resolve()
234
- }, 5000)
235
- }
236
- } catch (e) {
237
- // 忽略 CAM 权限问题
238
- if (!isCamRefused(e)) {
239
- throw e
240
- }
241
- }
242
- }, 3000)
243
- })
244
- }
1
+ import _ from 'lodash'
2
+ import open from 'open'
3
+ import { prompt } from 'enquirer'
4
+ import { getDataFromWeb, isCamRefused } from '@cloudbase/toolbox'
5
+ import { CloudBaseError } from '../error'
6
+ import { listEnvs, getEnvInfo } from '../env'
7
+ import { ENV_STATUS, STATUS_TEXT } from '../constant'
8
+ import { CloudApiService, getMangerService } from './net'
9
+ import { execWithLoading } from './output'
10
+ import { logger } from './log'
11
+
12
+ const tcbService = CloudApiService.getInstance('tcb')
13
+
14
+ const ENV_INIT_TIP = '环境初始化中,预计需要三分钟'
15
+ const CREATE_ENV = 'CREATE'
16
+ const consoleUrl = 'https://console.cloud.tencent.com/tcb/env/index?action=CreateEnv&from=cli'
17
+
18
+ /**
19
+ * 展示环境列表选择器,获取用户选择的环境
20
+ */
21
+ export async function getSelectedEnv(inputEnvId?: string) {
22
+ // 检查是否开通 TCB 服务
23
+ const isInitNow = await checkTcbService()
24
+
25
+ let envData = []
26
+
27
+ // 刚初始化服务,新创建的环境还未就绪
28
+ if (isInitNow) {
29
+ envData = await execWithLoading(
30
+ () => {
31
+ // 等待用户完成创建环境的流程
32
+ return new Promise((resolve) => {
33
+ const timer = setInterval(async () => {
34
+ const envs = await listEnvs()
35
+ if (envs.length) {
36
+ clearInterval(timer)
37
+ resolve(envs)
38
+ }
39
+ }, 2000)
40
+ })
41
+ },
42
+ {
43
+ startTip: '获取环境列表中'
44
+ }
45
+ )
46
+ } else {
47
+ // 选择环境
48
+ envData = await execWithLoading(() => listEnvs(), {
49
+ startTip: '获取环境列表中'
50
+ })
51
+ }
52
+
53
+ envData = envData || []
54
+
55
+ // 检查输入的环境 Id 是否属于用户
56
+ if (envData?.length && inputEnvId) {
57
+ const inputEnvIdExist = envData.find((_) => _.EnvId === inputEnvId)
58
+ if (!inputEnvIdExist) {
59
+ throw new CloudBaseError(`你指定的环境 Id ${inputEnvId} 不存在,请指定正确的环境 Id!`)
60
+ }
61
+ return inputEnvId
62
+ }
63
+
64
+ // 按名称排序,构造展示数据
65
+ const envs: { name: string; value: string }[] = envData
66
+ .map((item) => {
67
+ let name = `${item.Alias} - [${item.EnvId}:${item.PackageName || '按量计费'}]`
68
+ if (item.Status !== ENV_STATUS.NORMAL) {
69
+ name += `(${STATUS_TEXT[item.Status]})`
70
+ }
71
+
72
+ return {
73
+ name,
74
+ value: item.EnvId
75
+ }
76
+ })
77
+ .sort((prev, next) => prev.value.charCodeAt(0) - next.value.charCodeAt(0))
78
+
79
+ const choices = [
80
+ ...envs,
81
+ {
82
+ name: envData.length ? '创建新环境' : '无可用环境,创建新环境',
83
+ value: CREATE_ENV
84
+ }
85
+ ]
86
+
87
+ let { env } = await prompt<any>({
88
+ choices,
89
+ type: 'select',
90
+ name: 'env',
91
+ message: '请选择关联环境',
92
+ result(choice) {
93
+ return this.map(choice)[choice]
94
+ }
95
+ })
96
+
97
+ // 创建新环境
98
+ if (env === CREATE_ENV) {
99
+ logger.success('已打开控制台,请前往控制台创建环境')
100
+ // 从控制台获取创建环境生成的 envId
101
+ const { envId } = await getDataFromWeb((port) => `${consoleUrl}&port=${port}`, 'getData')
102
+ if (!envId) {
103
+ throw new CloudBaseError('接收环境 Id 信息失败,请重新运行 init 命令!')
104
+ }
105
+ logger.success(`创建环境成功,环境 Id: ${envId}`)
106
+ env = envId
107
+ }
108
+
109
+ // 检查环境状态
110
+ await checkEnvStatus(env)
111
+
112
+ return env
113
+ }
114
+
115
+ // 选择地域
116
+ export async function getSelectRegion() {
117
+ const { region } = await prompt<any>({
118
+ choices: [
119
+ {
120
+ name: '上海',
121
+ value: 'ap-shanghai'
122
+ },
123
+ {
124
+ name: '广州',
125
+ value: 'ap-guangzhou'
126
+ }
127
+ ],
128
+ type: 'select',
129
+ name: 'region',
130
+ message: '请选择环境所在地域',
131
+ result(choice) {
132
+ return this.map(choice)[choice]
133
+ }
134
+ })
135
+
136
+ tcbService.region = region
137
+
138
+ return region
139
+ }
140
+
141
+ // 检查 TCB 服务是否开通
142
+ export async function checkTcbService(): Promise<boolean> {
143
+ const app = await getMangerService()
144
+ let Initialized
145
+ try {
146
+ Initialized = (await app.env.checkTcbService()).Initialized
147
+ } catch (e) {
148
+ // 忽略 CAM 权限问题
149
+ if (!isCamRefused(e)) {
150
+ throw e
151
+ }
152
+ }
153
+
154
+ if (!Initialized) {
155
+ const { jump } = await prompt<any>({
156
+ type: 'confirm',
157
+ name: 'jump',
158
+ message:
159
+ '你还没有开通云开发服务,是否跳转到控制台开通云开发服务?(取消将无法继续操作)',
160
+ initial: true
161
+ })
162
+
163
+ if (!jump) {
164
+ throw new CloudBaseError('init 操作终止,请开通云开发服务后再进行操作!')
165
+ }
166
+
167
+ // 打开控制台
168
+ open(consoleUrl)
169
+ logger.success('已打开云开发控制台,请登录并在云开发控制台中开通服务!')
170
+
171
+ await execWithLoading(() => waitForServiceEnable(), {
172
+ startTip: '等待云开发服务开通中',
173
+ successTip: '云开发服务开通成功!'
174
+ })
175
+
176
+ // 返回一个是否刚初始化服务的标志
177
+ return true
178
+ }
179
+
180
+ return false
181
+ }
182
+
183
+ // 检查环境的状态,是否可以正常使用
184
+ export async function checkEnvStatus(envId: string) {
185
+ const env = await getEnvInfo(envId)
186
+ if (env.Status === ENV_STATUS.UNAVAILABLE) {
187
+ await checkEnvAvaliable(envId)
188
+ } else if (env.Status !== ENV_STATUS.NORMAL) {
189
+ throw new CloudBaseError('所有环境状态异常')
190
+ }
191
+ }
192
+
193
+ // 检测环境是否可用
194
+ export async function checkEnvAvaliable(envId: string) {
195
+ let count = 0
196
+
197
+ await execWithLoading(
198
+ (flush) => {
199
+ const increase = setInterval(() => {
200
+ flush(`${ENV_INIT_TIP} ${++count}S`)
201
+ }, 1000)
202
+
203
+ return new Promise<void>((resolve) => {
204
+ const timer = setInterval(async () => {
205
+ const env = await getEnvInfo(envId)
206
+ // 环境初始化中
207
+ if (env.Status === ENV_STATUS.NORMAL) {
208
+ clearInterval(timer)
209
+ clearInterval(increase)
210
+ resolve()
211
+ }
212
+ }, 3000)
213
+ })
214
+ },
215
+ {
216
+ startTip: ENV_INIT_TIP,
217
+ successTip: `环境 ${envId} 初始化成功`
218
+ }
219
+ )
220
+ }
221
+
222
+ // 等待服务开通
223
+ async function waitForServiceEnable() {
224
+ return new Promise<void>((resolve) => {
225
+ const timer = setInterval(async () => {
226
+ const app = await getMangerService()
227
+ try {
228
+ const { Initialized } = await app.env.checkTcbService()
229
+ if (Initialized) {
230
+ clearInterval(timer)
231
+ setTimeout(() => {
232
+ // 服务初始化完成到环境创建完成有一定的延迟时间,延迟 5S 返回
233
+ resolve()
234
+ }, 5000)
235
+ }
236
+ } catch (e) {
237
+ // 忽略 CAM 权限问题
238
+ if (!isCamRefused(e)) {
239
+ throw e
240
+ }
241
+ }
242
+ }, 3000)
243
+ })
244
+ }