diff --git a/packages/downloads/package.json b/packages/downloads/package.json index 529fd067..c23bb216 100644 --- a/packages/downloads/package.json +++ b/packages/downloads/package.json @@ -1,6 +1,6 @@ { "name": "@serverless-devs/downloads", - "version": "0.0.5", + "version": "0.0.6", "description": "download for serverless-devs", "main": "lib/index.js", "scripts": { diff --git a/packages/downloads/src/index.ts b/packages/downloads/src/index.ts index 92885529..2084d454 100644 --- a/packages/downloads/src/index.ts +++ b/packages/downloads/src/index.ts @@ -79,6 +79,22 @@ class Download { // ignore error in windows } } + // support github zip + const items = fs.readdirSync(dest as string, { withFileTypes: true }); + const directories = items.filter(item => item.isDirectory()); + // only one directory, move + if (directories.length === 1 && items.length === 1) { + try { + const directoryName = directories[0].name; + const directoryPath = path.join(dest as string, directoryName); + const tmpFilePath = path.join(dest as string, `../${filePath}-${Date.now()}`); + fs.moveSync(directoryPath, tmpFilePath, { overwrite: true }); + fs.moveSync(tmpFilePath, dest as string, { overwrite: true }); + fs.removeSync(tmpFilePath); + } catch(e) { + throw e; + } + } } private async doDownload(url: string): Promise { const { headers, logger } = this.options; diff --git a/packages/engine/__tests__/index.test.ts b/packages/engine/__tests__/index.test.ts index 91b39c1f..90f2d075 100644 --- a/packages/engine/__tests__/index.test.ts +++ b/packages/engine/__tests__/index.test.ts @@ -3,6 +3,7 @@ import Engine from '../src'; import path from 'path'; import { AssertionError } from 'assert'; import { get } from 'lodash'; +import { DevsError } from '@serverless-devs/utils'; test('指定 template 不存在', async () => { const engine = new Engine({ @@ -330,7 +331,7 @@ test('validate projectName', async () => { }); const context = await engine.start(); console.log(context); - expect(get(context, 'error[0]')).toBeInstanceOf(AssertionError); + expect(get(context, 'error[0]')).toBeInstanceOf(DevsError); expect(get(context, 'error[0].message')).toBe(`The name of the project [deploy] overlaps with a command, please change it's name`); expect(get(context, 'error[0].code')).toBe('ERR_ASSERTION'); }); diff --git a/packages/engine/package.json b/packages/engine/package.json index b03270b6..bf5fc2bc 100644 --- a/packages/engine/package.json +++ b/packages/engine/package.json @@ -1,6 +1,6 @@ { "name": "@serverless-devs/engine", - "version": "0.1.2-beta.4", + "version": "0.1.2-beta.9", "description": "a engine lib for serverless-devs", "main": "lib/index.js", "scripts": { diff --git a/packages/engine/src/actions/index.ts b/packages/engine/src/actions/index.ts index 3cc013de..d70bb8b9 100644 --- a/packages/engine/src/actions/index.ts +++ b/packages/engine/src/actions/index.ts @@ -18,7 +18,7 @@ interface IRecord { magic: Record; // 记录魔法变量 componentProps: Record; // 记录组件的inputs pluginOutput: Record; // 记录plugin的outputs - lable: string; // 记录执行的label + label: string; // 记录执行的label step: IStepOptions; // 记录当前step allowFailure: boolean | IAllowFailure; // step allow_failure > action allow_failure command: string; // 记录当前执行的command @@ -106,8 +106,8 @@ You can still use them now, but we suggest to modify them.`) const hooks = filter(this.actions, item => item.hookType === hookType); if (isEmpty(hooks)) return {}; this.record.startTime = Date.now(); - this.record.lable = this.option.hookLevel === IActionLevel.PROJECT ? `[${this.option.projectName}]` : IActionLevel.GLOBAL; - this.logger.debug(`Start executing the ${hookType}-action in ${this.record.lable}`); + this.record.label = this.option.hookLevel === IActionLevel.PROJECT ? `[${this.option.projectName}]` : IActionLevel.GLOBAL; + this.logger.debug(`Start executing the ${hookType}-action in ${this.record.label}`); // 确保 hooks 中的变量均为解析过后的真实值 const newHooks = getInputs(hooks, this.record.magic); // post-action应获取componentProps, 先清空pluginOutput @@ -128,7 +128,7 @@ You can still use them now, but we suggest to modify them.`) await this.component(hook); } } - this.logger.debug(`The ${hookType}-action successfully to execute in ${this.record.lable}`); + this.logger.debug(`The ${hookType}-action successfully to execute in ${this.record.label}`); if (this.option.hookLevel === IActionLevel.GLOBAL) { this.logger.write(`${chalk.green('✔')} ${chalk.gray(`${IActionLevel.GLOBAL} ${hookType}-action completed (${getProcessTime(this.record.startTime)})`)}`); @@ -207,7 +207,7 @@ You can still use them now, but we suggest to modify them.`) data: get(e, 'data'), stack: error.stack, exitCode: EXIT_CODE.RUN, - prefix: `${this.record.lable} ${hook.hookType}-action failed to [${this.record.command}]:`, + prefix: `${this.record.label} ${hook.hookType}-action failed to [${this.record.command}]:`, trackerType: ETrackerType.runtimeException, }); } @@ -221,7 +221,7 @@ You can still use them now, but we suggest to modify them.`) if (useAllowFailure) return; throw new DevsError(`The ${hook.path} directory does not exist.`, { exitCode: EXIT_CODE.DEVS, - prefix: `${this.record.lable} ${hook.hookType}-action failed to [${this.record.command}]:`, + prefix: `${this.record.label} ${hook.hookType}-action failed to [${this.record.command}]:`, trackerType: ETrackerType.parseException, }); } @@ -257,7 +257,7 @@ You can still use them now, but we suggest to modify them.`) data: get(e, 'data'), stack: error.stack, exitCode: EXIT_CODE.PLUGIN, - prefix: `${this.record.lable} ${hook.hookType}-action failed to [${this.record.command}]:`, + prefix: `${this.record.label} ${hook.hookType}-action failed to [${this.record.command}]:`, trackerType: ETrackerType.runtimeException, }); } @@ -291,11 +291,12 @@ You can still use them now, but we suggest to modify them.`) // 方法存在,执行报错,退出码101 const newInputs = { ...this.record.componentProps, - argv: filter(argv.slice(2), o => !includes([componentName, command], o)), + args: filter(argv.slice(2), o => !includes([componentName, command], o)), }; try { // Execute the command for the component with the prepared inputs. - return await instance[command](newInputs); + await instance[command](newInputs); + return; } catch (e) { const error = e as Error; // Check if the failure is allowed based on the record's allowFailure setting. @@ -308,7 +309,7 @@ You can still use them now, but we suggest to modify them.`) data: get(e, 'data'), stack: error.stack, exitCode: EXIT_CODE.COMPONENT, - prefix: `${this.record.lable} ${hook.hookType}-action failed to [${this.record.command}]:`, + prefix: `${this.record.label} ${hook.hookType}-action failed to [${this.record.command}]:`, trackerType: ETrackerType.runtimeException, }); } @@ -323,9 +324,9 @@ You can still use them now, but we suggest to modify them.`) // 方法不存在,此时系统将会认为是未找到组件方法,系统的exit code为100; throw new DevsError(`The [${command}] command was not found.`, { exitCode: EXIT_CODE.DEVS, - prefix: `${this.record.lable} ${hook.hookType}-action failed to [${this.record.command}]:`, + prefix: `${this.record.label} ${hook.hookType}-action failed to [${this.record.command}]:`, tips: `Please check the component ${componentName} has the ${command} command. Serverless Devs documents:${chalk.underline( - 'https://github.com/Serverless-Devs/Serverless-Devs/blob/master/docs/zh/command', + 'https://manual.serverless-devs.com/', )}`, trackerType: ETrackerType.parseException, }); diff --git a/packages/engine/src/index.ts b/packages/engine/src/index.ts index 7dab95f4..71daaa97 100644 --- a/packages/engine/src/index.ts +++ b/packages/engine/src/index.ts @@ -12,6 +12,7 @@ import Logger, { ILoggerInstance } from '@serverless-devs/logger'; import { DevsError, ETrackerType, emoji, getAbsolutePath, getRootHome, getUserAgent, traceid } from '@serverless-devs/utils'; import { EXIT_CODE } from './constants'; import assert from 'assert'; +import Ajv from 'ajv'; export * from './types'; export { verify, preview } from './utils'; @@ -91,7 +92,7 @@ class Engine { return this.context; } const { steps: _steps, yaml, command, access = yaml.access } = this.spec; - this.logger.write(`${emoji('⌛')} Steps for [${command}] of [${get(this.spec, 'yaml.appName')}]\n${chalk.gray('====================')}`); + this.logger.write(chalk.gray(`${emoji('⌛')} Steps for [${command}] of [${get(this.spec, 'yaml.appName')}]\n${chalk.gray('====================')}`)); // 初始化全局的 action this.globalActionInstance = new Actions(yaml.actions, { hookLevel: IActionLevel.GLOBAL, @@ -206,14 +207,54 @@ class Engine { */ private async validate() { const { steps, command, projectName } = this.spec; + let errorsList: any[] = []; + const ajv = new Ajv({ allErrors: true }); + assert(!isEmpty(steps), 'Step is required'); for (const step of steps) { - const instance = await loadComponent(step.component, { engineLogger: this.logger }); + const instance = await loadComponent(step.component, { logger: this.logger, engineLogger: this.logger }); if (projectName && keys(get(instance, 'commands')).includes(projectName)) { - assert(!projectName, `The name of the project [${projectName}] overlaps with a command, please change it's name`); + throw new DevsError(`The name of the project [${projectName}] overlaps with a command, please change it's name.`, { + exitCode: EXIT_CODE.DEVS, + trackerType: ETrackerType.parseException, + prefix: `[${projectName}] failed to [${command}]:` + }); + } + // schema validation + if (get(this.spec, 'yaml.use3x') && get(this.spec, 'yaml.content.validation')) { + const schema = await this.getSchemaByInstance(instance); + if (isEmpty(schema)) continue; + const validate = ajv.compile(JSON.parse(schema)); + if (!validate(step.props)) { + const errors = validate.errors; + if (!errors) continue; + for (const j of errors) { + j.instancePath = step.projectName + '/props' + j.instancePath; + if (j.keyword === 'enum') { + j.message = j.message + ': ' + j.params.allowedValues.join(', '); + } + } + errorsList = errorsList.concat(errors); + } } } - assert(!isEmpty(steps), 'Step is required'); assert(command, 'Command is required'); + if (!isEmpty(errorsList)) { + throw new DevsError(`${ajv.errorsText(errorsList, { dataVar: '', separator: '\n' })}`, { + exitCode: EXIT_CODE.DEVS, + trackerType: ETrackerType.parseException, + prefix: 'Function props validation error:' + }); + } + } + + /** + * Get schema by existing instance, avoid loading components. + * @param instance loadComponent instance + * @param logger Logger + */ + private getSchemaByInstance(instance: any) { + if (!instance || !instance.getSchema) return null; + return instance.getSchema(); } /** @@ -298,7 +339,7 @@ class Engine { cwd: path.dirname(this.spec.yaml.path), vars: this.spec.yaml.vars, resources: {}, - __runtime: this.options.verify ? 'enigne' : 'parse', + __runtime: this.options.verify ? 'engine' : 'parse', __steps: this.context.steps, } as Record; for (const obj of this.context.steps) { @@ -599,7 +640,7 @@ class Engine { throw new DevsError(`The [${command}] command was not found.`, { exitCode: EXIT_CODE.DEVS, tips: `Please check the component ${item.component} has the ${command} command. Serverless Devs documents:${chalk.underline( - 'https://github.com/Serverless-Devs/Serverless-Devs/blob/master/docs/zh/command', + 'https://manual.serverless-devs.com/', )}`, prefix: `[${item.projectName}] failed to [${command}]:`, trackerType: ETrackerType.parseException, @@ -629,7 +670,7 @@ class Engine { // 方法不存在,进行警告,但是并不会报错,最终的exit code为0; this.logger.tips( `The [${command}] command was not found.`, - `Please check the component ${item.component} has the ${command} command. Serverless Devs documents:https://github.com/Serverless-Devs/Serverless-Devs/blob/master/docs/zh/command`, + `Please check the component ${item.component} has the ${command} command. Serverless Devs documents:https://manual.serverless-devs.com/`, ); } diff --git a/packages/load-application/package.json b/packages/load-application/package.json index b28de1c9..fc93a5c3 100644 --- a/packages/load-application/package.json +++ b/packages/load-application/package.json @@ -1,6 +1,6 @@ { "name": "@serverless-devs/load-application", - "version": "0.0.13-beta.1", + "version": "0.0.13-beta.6", "description": "load application for serverless-devs", "main": "lib/index.js", "scripts": { diff --git a/packages/load-application/src/constant.ts b/packages/load-application/src/constant.ts index d2bbe74e..89f46e84 100644 --- a/packages/load-application/src/constant.ts +++ b/packages/load-application/src/constant.ts @@ -1,12 +1,15 @@ import chalk from 'chalk'; +import { getGlobalConfig } from '@serverless-devs/utils'; export const gray = chalk.hex('#8c8d91'); export const RANDOM_PATTERN = '${default-suffix}'; export const DEVSAPP = 'devsapp'; +export const GITHUB_REGISTRY = 'https://api.github.com/repos'; export const REGISTRY = { V2: 'https://registry.devsapp.cn/simple', V3: 'https://api.devsapp.cn/v3', + CUSTOM_URL: getGlobalConfig('registry'), }; export const CONFIGURE_LATER = 'configure_later'; diff --git a/packages/load-application/src/index.ts b/packages/load-application/src/index.ts index b27bb555..76fa34cd 100644 --- a/packages/load-application/src/index.ts +++ b/packages/load-application/src/index.ts @@ -2,21 +2,24 @@ import V3 from './v3'; import V2 from './v2'; import assert from 'assert'; import { IOptions } from './types'; -import { includes } from 'lodash'; +import { includes, get } from 'lodash'; +import { REGISTRY } from './constant'; const debug = require('@serverless-cd/debug')('serverless-devs:load-appliaction'); export default async (template: string, options: IOptions = {}) => { debug(`load application, template: ${template}, options: ${JSON.stringify(options)}`); + const { logger } = options; if (options.uri) { return await v3(template, options); } assert(template, 'template is required'); - if (includes(template, '/')) { + if ((includes(template, '/') && REGISTRY.CUSTOM_URL === REGISTRY.V3) || (REGISTRY.CUSTOM_URL === REGISTRY.V2)) { return await v2(template, options); } try { return await v3(template, options); } catch (error) { + logger.warn(get(error, 'message') + ', try to load from v2 registry.'); debug(`v3 error, ${error}`); return await v2(template, options); } diff --git a/packages/load-application/src/utils/index.ts b/packages/load-application/src/utils/index.ts index 5ab6650f..4e540db1 100644 --- a/packages/load-application/src/utils/index.ts +++ b/packages/load-application/src/utils/index.ts @@ -1,5 +1,5 @@ -import { endsWith, keys, replace } from 'lodash'; -import { RANDOM_PATTERN, REGISTRY } from '../constant'; +import { keys, replace, split } from 'lodash'; +import { RANDOM_PATTERN, REGISTRY, GITHUB_REGISTRY } from '../constant'; import Credential from '@serverless-devs/credential'; export { default as getInputs } from './get-inputs'; @@ -10,7 +10,15 @@ export const tryfun = async (fn: Function, ...args: any[]) => { } catch (ex) {} }; -export const getUrlWithLatest = (name: string) => `${REGISTRY.V3}/packages/${name}/release/latest`; +export const getUrlWithLatest = (name: string) => { + if (REGISTRY.CUSTOM_URL === GITHUB_REGISTRY) { + if (split(name, '/').length === 1) { + return `${REGISTRY.CUSTOM_URL}/devsapp/${name}`; + } + return `${REGISTRY.CUSTOM_URL}/${name}`; + } + return `${REGISTRY.V3}/packages/${name}/release/latest` +}; export const getUrlWithVersion = (name: string, versionId: string) => `${REGISTRY.V3}/packages/${name}/release/tags/${versionId}`; export const randomId = () => Math.random().toString(36).substring(2, 6); @@ -22,5 +30,5 @@ export const getAllCredential = async ({ logger }: any) => { export const getDefaultValue = (value: any) => { if (typeof value !== 'string') return; - return endsWith(value, RANDOM_PATTERN) ? replace(value, RANDOM_PATTERN, randomId()) : value; + return replace(value, RANDOM_PATTERN, randomId()); }; diff --git a/packages/load-application/src/v2.ts b/packages/load-application/src/v2.ts index 793515bc..253b3375 100644 --- a/packages/load-application/src/v2.ts +++ b/packages/load-application/src/v2.ts @@ -4,7 +4,7 @@ import axios from 'axios'; import download from '@serverless-devs/downloads'; import artTemplate from 'art-template'; import { getYamlContent, isCiCdEnvironment, getYamlPath } from '@serverless-devs/utils'; -import { isEmpty, includes, split, get, has, set, sortBy, map, concat, keys, find } from 'lodash'; +import { isEmpty, includes, split, get, has, set, sortBy, map, concat, keys, find, startsWith } from 'lodash'; import parse from './parse'; import { IProvider, IOptions } from './types'; import { CONFIGURE_LATER, DEFAULT_MAGIC_ACCESS, REGISTRY } from './constant'; @@ -374,13 +374,31 @@ class LoadApplication { private async doLoad() { const { logger } = this.options; const zipball_url = this.version ? await this.doZipballUrlWithVersion() : await this.doZipballUrl(); - await download(zipball_url, { - dest: this.tempPath, - logger, - extract: true, - strip: 1, - filename: this.name, - }); + try { + await download(zipball_url, { + dest: this.tempPath, + logger, + extract: true, + strip: 1, + filename: this.name, + }); + } catch(e) { + logger.debug(e); + // if https, try http + if (startsWith(zipball_url, 'https')) { + logger.debug('https error, try http'); + const newZipballUrl = zipball_url.replace('https://', 'http://'); + await download(newZipballUrl, { + dest: this.tempPath, + logger, + extract: true, + strip: 1, + filename: this.name, + }); + } else { + throw e; + } + } } private async doZipballUrl() { const maps = { diff --git a/packages/load-application/src/v3.ts b/packages/load-application/src/v3.ts index bbcdf0bf..434eb9a9 100644 --- a/packages/load-application/src/v3.ts +++ b/packages/load-application/src/v3.ts @@ -4,17 +4,16 @@ import download from '@serverless-devs/downloads'; import _artTemplate from 'art-template'; import _devsArtTemplate from '@serverless-devs/art-template'; import { getYamlContent, registry, isCiCdEnvironment, getYamlPath } from '@serverless-devs/utils'; -import { isEmpty, includes, split, get, has, set, sortBy, map, concat, keys } from 'lodash'; +import { isEmpty, includes, split, get, has, set, sortBy, map, concat, keys, startsWith, merge } from 'lodash'; import axios from 'axios'; import parse from './parse'; import { IOptions } from './types'; import { getInputs, getUrlWithLatest, getUrlWithVersion, getAllCredential, getDefaultValue } from './utils'; -import assert from 'assert'; import YAML from 'yaml'; import inquirer from 'inquirer'; import chalk from 'chalk'; import Credential from '@serverless-devs/credential'; -import { CONFIGURE_LATER, DEFAULT_MAGIC_ACCESS, gray } from './constant'; +import { CONFIGURE_LATER, DEFAULT_MAGIC_ACCESS, GITHUB_REGISTRY, gray } from './constant'; const debug = require('@serverless-cd/debug')('serverless-devs:load-appliaction'); class LoadApplication { @@ -52,7 +51,7 @@ class LoadApplication { */ private secretList: string[] = []; constructor(private template: string, private options: IOptions = {}) { - assert(!includes(this.template, '/'), `The component name ${this.template} cannot contain /`); + // assert(!includes(this.template, '/'), `The component name ${this.template} cannot contain /`); this.options.dest = this.options.dest || process.cwd(); this.options.logger = this.options.logger || console; const [name, version] = split(this.template, '@'); @@ -213,6 +212,7 @@ class LoadApplication { const properties = get(publishData, 'Parameters.properties'); const requiredList = get(publishData, 'Parameters.required'); const promptList = []; + const tmpResult: any = {}; if (properties) { let rangeList = []; for (const key in properties) { @@ -233,8 +233,11 @@ class LoadApplication { } return true; }; - // 布尔类型 - if (item.type === 'boolean') { + if (item.input === 'false' || item.input === false) { + // 不手动输入 + tmpResult[name] = getDefaultValue(item.default) || ''; + } else if (item.type === 'boolean') { + // 布尔类型 promptList.push({ type: 'confirm', name, @@ -314,6 +317,7 @@ class LoadApplication { result.access = DEFAULT_MAGIC_ACCESS; } } + result = merge(tmpResult, result); return result; } private async getCredentialDirectly() { @@ -372,22 +376,52 @@ class LoadApplication { const { logger } = this.options; const zipball_url = this.options.uri || (await this.getZipballUrl()); debug(`zipball_url: ${zipball_url}`); - await download(zipball_url, { - dest: this.tempPath, - logger, - extract: true, - headers: { - ...registry.getSignHeaders(), - devs_mock_env: process.env.DEVS_MOCK_ENV || 'false', - }, - filename: this.name, - }); + try { + await download(zipball_url, { + dest: this.tempPath, + logger, + extract: true, + headers: { + 'User-Agent': 'Serverless-Devs (https://github.com/Serverless-Devs/Serverless-Devs)', + ...registry.getSignHeaders(), + devs_mock_env: process.env.DEVS_MOCK_ENV || 'false', + }, + // use final element as filename + filename: split(this.name, '/')[-1], + }); + } catch(e) { + logger.debug(e); + // if https, try http + if (startsWith(zipball_url, 'https')) { + logger.debug('https error, try http'); + const newZipballUrl = zipball_url.replace('https://', 'http://'); + await download(zipball_url, { + dest: this.tempPath, + logger, + extract: true, + headers: { + 'User-Agent': 'Serverless-Devs (https://github.com/Serverless-Devs/Serverless-Devs)', + ...registry.getSignHeaders(), + devs_mock_env: process.env.DEVS_MOCK_ENV || 'false', + }, + // use final element as filename + filename: split(this.name, '/')[-1], + }); + } else { + throw e; + } + } } private getZipballUrl = async () => { const url = this.version ? getUrlWithVersion(this.name, this.version) : getUrlWithLatest(this.name); debug(`url: ${url}`); const res = await axios.get(url, { headers: registry.getSignHeaders() }); debug(`res: ${JSON.stringify(res.data)}`); + if (startsWith(url, GITHUB_REGISTRY)) { + const defaultBranch = get(res, 'data.default_branch'); + if (isEmpty(defaultBranch)) throw new Error(`Default branch is not found`); + return url + `/zipball/${defaultBranch}`; + } const zipball_url = get(res, 'data.body.zipball_url'); const template = this.version ? `${this.name}@${this.version}` : this.name; if (isEmpty(zipball_url)) throw new Error(`Application ${template} is not found`); diff --git a/packages/load-component/package.json b/packages/load-component/package.json index 35cbde73..48827b22 100644 --- a/packages/load-component/package.json +++ b/packages/load-component/package.json @@ -1,6 +1,6 @@ { "name": "@serverless-devs/load-component", - "version": "0.0.7-beta.2", + "version": "0.0.7-beta.4", "description": "request for serverless-devs", "main": "lib/index.js", "scripts": { diff --git a/packages/load-component/src/utils/index.ts b/packages/load-component/src/utils/index.ts index 3040cbe0..84a3fc93 100644 --- a/packages/load-component/src/utils/index.ts +++ b/packages/load-component/src/utils/index.ts @@ -42,6 +42,10 @@ const getEntryFile = async (componentPath: string) => { export const buildComponentInstance = async (componentPath: string, params?: any) => { const requirePath = await getEntryFile(componentPath); + // bug: `- component: fc invoke` timeout. Delete require cache + try { + delete require.cache[requirePath]; + } catch {} const baseChildComponent = await require(requirePath); const ChildComponent = baseChildComponent.default || baseChildComponent; diff --git a/packages/logger/package.json b/packages/logger/package.json index 064ad03b..d4bb37d0 100644 --- a/packages/logger/package.json +++ b/packages/logger/package.json @@ -1,6 +1,6 @@ { "name": "@serverless-devs/logger", - "version": "0.0.5-beta.1", + "version": "0.0.5-beta.2", "description": "", "main": "lib/index.js", "scripts": { diff --git a/packages/logger/src/engine-logger/index.ts b/packages/logger/src/engine-logger/index.ts index 8a708f66..b43e1c96 100644 --- a/packages/logger/src/engine-logger/index.ts +++ b/packages/logger/src/engine-logger/index.ts @@ -13,6 +13,9 @@ export default class EngineLogger extends Logger { private eol: string; private key: string; private level: LoggerLevel; + private writeMsg: string; + private tipsMsg: string; + private warnMsg: string; constructor(props: IOptions) { super({} as EggLoggerOptions); @@ -29,6 +32,9 @@ export default class EngineLogger extends Logger { this.level = level || get(props, 'level', 'INFO'); this.eol = eol; this.key = key; + this.writeMsg = ''; + this.tipsMsg = ''; + this.warnMsg = ''; const consoleTransport = new ConsoleTransport({ level: level || get(props, 'level', 'INFO'), @@ -81,10 +87,29 @@ export default class EngineLogger extends Logger { this.write(msg); } + tipsOnce(message: string, tips?: string) { + if (this.tipsMsg === message + tips) return; + this.tipsMsg = message + tips; + this.tips(message, tips); + } + write(msg: string) { super.write(transport.transportSecrets(msg)); } + // 多个相同write输出的时候,只保留第一个 + writeOnce(msg: string) { + if (msg === this.writeMsg) return; + this.writeMsg = msg; + this.write(msg); + } + + warnOnce(msg: string) { + if (msg === this.warnMsg) return; + this.warnMsg = msg; + super.warn(msg); + } + private setEol(eol: string = os.EOL) { const c = this.get('console') as object; const f = this.get('file'); diff --git a/packages/parse-spec/package.json b/packages/parse-spec/package.json index 2df081ed..abf1a9e9 100644 --- a/packages/parse-spec/package.json +++ b/packages/parse-spec/package.json @@ -1,6 +1,6 @@ { "name": "@serverless-devs/parse-spec", - "version": "0.0.26-beta.1", + "version": "0.0.26-beta.3", "description": "a parse yaml spec lib for serverless-devs", "main": "lib/index.js", "scripts": { diff --git a/packages/parse-spec/src/index.ts b/packages/parse-spec/src/index.ts index 8d4da1eb..263e92ee 100644 --- a/packages/parse-spec/src/index.ts +++ b/packages/parse-spec/src/index.ts @@ -128,39 +128,32 @@ class ParseSpec { if (has(this.yaml.content, ENVIRONMENT_KEY)) { const envPath: string = utils.getAbsolutePath(get(this.yaml.content, ENVIRONMENT_KEY), path.dirname(this.yaml.path)); const envYamlContent = utils.getYamlContent(envPath); + // 若存在环境变量,默认项目为devsProject + const devsProject = process.env.ALIYUN_DEVS_REMOTE_PROJECT_NAME; + const project = devsProject ? devsProject : get(this.yaml.content, 'name'); // env.yaml is not exist if (isEmpty(envYamlContent)) { - throw new DevsError(`Environment file [${envPath}] is not found`, { - tips: 'You can create a new environment file by running `s env init`', - trackerType: ETrackerType.parseException, - }); + this.options.logger.warnOnce(`Environment file [${envPath}] is not found, run without environment.`); + return {}; } // default-env.json is not exist if (!fs.existsSync(ENVIRONMENT_FILE_PATH)) { - throw new DevsError('Default env is not found', { - tips: 'You can set a default environment by running `s env default`', - trackerType: ETrackerType.parseException, - }); + this.options.logger.warnOnce(`Default env config file [${ENVIRONMENT_FILE_PATH}] is not found, run without environment.`); + return {}; } const { environments } = envYamlContent; - // 若存在环境变量,默认项目为devsProject - const devsProject = process.env.ALIYUN_DEVS_REMOTE_PROJECT_NAME; - const project = devsProject ? devsProject : get(this.yaml.content, 'name'); const defaultEnvContent = require(ENVIRONMENT_FILE_PATH); const defaultEnv = get(find(defaultEnvContent, { project: project }), 'default'); // project is not found in default-env.json if (!defaultEnv) { - throw new DevsError('Default env is not found', { - tips: 'You can set a default environment by running `s env default`', - trackerType: ETrackerType.parseException, - }); + this.options.logger.warnOnce(`Default env is not set, run without environment.`); + return {}; } const environment = find(environments, item => item.name === defaultEnv); // default env is not found in env.yaml if (isEmpty(environment)) { - throw new DevsError(`Default env [${defaultEnv}] was not found`, { - trackerType: ETrackerType.parseException, - }); + this.options.logger.warnOnce(`Default env [${defaultEnv}] is not found, run without environment.`); + return {}; } return { project, environment }; } @@ -178,9 +171,8 @@ class ParseSpec { const envYamlContent = utils.getYamlContent(envPath); // env file is not exist if (isEmpty(envYamlContent)) { - throw new DevsError(`Environment file [${envPath}] is not exist`, { - trackerType: ETrackerType.parseException, - }); + this.options.logger.warnOnce(`Environment file [${envPath}] is not found, run without environment.`); + return {}; } debug(`environment content: ${JSON.stringify(envYamlContent)}`); const { environments } = envYamlContent; @@ -190,10 +182,8 @@ class ParseSpec { const environment = find(environments, item => item.name === this.record.env); // env name is not found if (isEmpty(environment)) { - // TODO: @封崇 - throw new DevsError(`Env [${this.record.env}] was not found`, { - trackerType: ETrackerType.parseException, - }); + this.options.logger.warnOnce(`Env [${this.record.env}] was not found, run without environment.`); + return {}; } return { project, environment }; } diff --git a/packages/registry/package.json b/packages/registry/package.json index 14afd4db..83a39cbf 100644 --- a/packages/registry/package.json +++ b/packages/registry/package.json @@ -1,6 +1,6 @@ { "name": "@serverless-devs/registry", - "version": "0.0.8", + "version": "0.0.9", "description": "request for serverless-devs", "main": "lib/index.js", "scripts": { diff --git a/packages/registry/src/actions/constant.ts b/packages/registry/src/actions/constant.ts index a96ef39c..0b0827d8 100644 --- a/packages/registry/src/actions/constant.ts +++ b/packages/registry/src/actions/constant.ts @@ -44,6 +44,7 @@ export const publishSchema = { "Google Cloud Platform", "专有云", "其它", + "火山引擎", "Alibaba Cloud", "Tencent Cloud", "Huawei Cloud",