mirror of
https://github.com/serverless/serverless.git
synced 2025-12-08 19:46:03 +00:00
149 lines
5.0 KiB
JavaScript
149 lines
5.0 KiB
JavaScript
'use strict';
|
|
|
|
const spawn = require('child-process-ext/spawn');
|
|
const fsp = require('fs').promises;
|
|
const fse = require('fs-extra');
|
|
const path = require('path');
|
|
const _ = require('lodash');
|
|
const isPlainObject = require('type/plain-object/is');
|
|
const yaml = require('js-yaml');
|
|
const cloudformationSchema = require('@serverless/utils/cloudformation-schema');
|
|
const { log, progress, style } = require('@serverless/utils/log');
|
|
const ServerlessError = require('../lib/serverless-error');
|
|
const yamlAstParser = require('../lib/utils/yaml-ast-parser');
|
|
const npmCommandDeferred = require('../lib/utils/npm-command-deferred');
|
|
const {
|
|
getPluginInfo,
|
|
getServerlessFilePath,
|
|
validate,
|
|
} = require('../lib/commands/plugin-management');
|
|
|
|
const mainProgress = progress.get('main');
|
|
|
|
module.exports = async ({ configuration, serviceDir, configurationFilename, options }) => {
|
|
const commandRunStartTime = Date.now();
|
|
validate({ serviceDir });
|
|
|
|
const pluginInfo = getPluginInfo(options.name);
|
|
const pluginName = pluginInfo.name;
|
|
const pluginVersion = pluginInfo.version || 'latest';
|
|
const configurationFilePath = getServerlessFilePath({ serviceDir, configurationFilename });
|
|
|
|
const context = { configuration, serviceDir, configurationFilePath, pluginName, pluginVersion };
|
|
mainProgress.notice(
|
|
`Installing plugin "${pluginName}${pluginVersion === 'latest' ? '' : `@${pluginVersion}`}"`,
|
|
{ isMainEvent: true }
|
|
);
|
|
await installPlugin(context);
|
|
// Check if plugin is already added
|
|
const pluginAlreadyPresentInConfig =
|
|
(_.get(configuration, 'plugins.modules') &&
|
|
configuration.plugin.modules.includes(pluginName)) ||
|
|
(configuration.plugins && configuration.plugins.includes(pluginName));
|
|
if (!pluginAlreadyPresentInConfig) {
|
|
await addPluginToServerlessFile(context);
|
|
}
|
|
|
|
log.notice();
|
|
log.notice.success(
|
|
`Plugin "${pluginName}${
|
|
pluginVersion === 'latest' ? '' : `@${pluginVersion}`
|
|
}" installed ${style.aside(`(${Math.floor((Date.now() - commandRunStartTime) / 1000)}s)`)}`
|
|
);
|
|
};
|
|
|
|
const installPlugin = async ({ serviceDir, pluginName, pluginVersion }) => {
|
|
const pluginFullName = `${pluginName}@${pluginVersion}`;
|
|
await npmInstall(pluginFullName, { serviceDir });
|
|
};
|
|
|
|
const addPluginToServerlessFile = async ({ configurationFilePath, pluginName }) => {
|
|
const fileExtension = path.extname(configurationFilePath);
|
|
if (fileExtension === '.js' || fileExtension === '.ts') {
|
|
requestManualUpdate(configurationFilePath);
|
|
return;
|
|
}
|
|
|
|
const checkIsArrayPluginsObject = (pluginsObject) =>
|
|
pluginsObject == null || Array.isArray(pluginsObject);
|
|
// pluginsObject type determined based on the value loaded during the serverless init.
|
|
if (fileExtension === '.json') {
|
|
const serverlessFileObj = await fse.readJson(configurationFilePath);
|
|
const newServerlessFileObj = serverlessFileObj;
|
|
const isArrayPluginsObject = checkIsArrayPluginsObject(newServerlessFileObj.plugins);
|
|
// null modules property is not supported
|
|
let plugins = isArrayPluginsObject
|
|
? newServerlessFileObj.plugins || []
|
|
: newServerlessFileObj.plugins.modules;
|
|
|
|
if (plugins == null) {
|
|
throw new ServerlessError(
|
|
'plugins modules property must be present',
|
|
'PLUGINS_MODULES_MISSING'
|
|
);
|
|
}
|
|
|
|
plugins.push(pluginName);
|
|
plugins = _.sortedUniq(plugins);
|
|
|
|
if (isArrayPluginsObject) {
|
|
newServerlessFileObj.plugins = plugins;
|
|
} else {
|
|
newServerlessFileObj.plugins.modules = plugins;
|
|
}
|
|
|
|
await fse.writeJson(configurationFilePath, newServerlessFileObj);
|
|
return;
|
|
}
|
|
|
|
const serverlessFileObj = yaml.load(await fsp.readFile(configurationFilePath, 'utf8'), {
|
|
filename: configurationFilePath,
|
|
schema: cloudformationSchema,
|
|
});
|
|
if (serverlessFileObj.plugins != null) {
|
|
// Plugins section can be behind veriables, opt-out in such case
|
|
if (isPlainObject(serverlessFileObj.plugins)) {
|
|
if (
|
|
serverlessFileObj.plugins.modules != null &&
|
|
!Array.isArray(serverlessFileObj.plugins.modules)
|
|
) {
|
|
requestManualUpdate(configurationFilePath);
|
|
return;
|
|
}
|
|
} else if (!Array.isArray(serverlessFileObj.plugins)) {
|
|
requestManualUpdate(configurationFilePath);
|
|
return;
|
|
}
|
|
}
|
|
await yamlAstParser.addNewArrayItem(
|
|
configurationFilePath,
|
|
checkIsArrayPluginsObject(serverlessFileObj.plugins) ? 'plugins' : 'plugins.modules',
|
|
pluginName
|
|
);
|
|
};
|
|
|
|
const npmInstall = async (name, { serviceDir }) => {
|
|
const { command, args } = await npmCommandDeferred;
|
|
try {
|
|
await spawn(command, [...args, 'install', '--save-dev', name], {
|
|
cwd: serviceDir,
|
|
stdio: 'pipe',
|
|
// To parse quotes used in module versions. E.g. 'serverless@"^1.60.0 || 2"'
|
|
// https://stackoverflow.com/a/48015470
|
|
shell: true,
|
|
});
|
|
} catch (error) {
|
|
log.error(String(error.stderrBuffer));
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
const requestManualUpdate = (configurationFilePath) => {
|
|
log.notice();
|
|
log.notice.skip(
|
|
`Can't automatically add plugin into "${path.basename(
|
|
configurationFilePath
|
|
)}" file. Please add it manually.`
|
|
);
|
|
};
|