mirror of
https://github.com/serverless/serverless.git
synced 2026-01-18 14:58:43 +00:00
227 lines
6.2 KiB
JavaScript
227 lines
6.2 KiB
JavaScript
'use strict';
|
|
|
|
const SError = require('./ServerlessError'),
|
|
SUtils = require('./utils/index'),
|
|
SCli = require('./utils/cli'),
|
|
BbPromise = require('bluebird');
|
|
|
|
/**
|
|
* This is the base class that all Serverless Plugins should extend.
|
|
*/
|
|
|
|
class ServerlessPlugin {
|
|
|
|
/**
|
|
* Constructor
|
|
*/
|
|
|
|
constructor(S, config) {
|
|
this.S = S;
|
|
this.config = config;
|
|
}
|
|
|
|
/**
|
|
* Define your plugins name
|
|
*/
|
|
|
|
static getName() {
|
|
return 'com.yourdomain.' + ServerlessPlugin.name;
|
|
}
|
|
|
|
/**
|
|
* Register Actions
|
|
*/
|
|
|
|
registerActions() {
|
|
return BbPromise.resolve();
|
|
}
|
|
|
|
/**
|
|
* Register Hooks
|
|
*/
|
|
|
|
registerHooks() {
|
|
return BbPromise.resolve();
|
|
}
|
|
|
|
/**
|
|
* CLI Prompt Input
|
|
* - Handy CLI Prompt Input function for Plugins
|
|
* @param promptSchema @see https://github.com/flatiron/prompt#prompting-with-validation-default-values-and-more-complex-properties
|
|
* @param overrides map {key: 'overrideValue'}
|
|
* @returns {Promise} containing answers by key
|
|
*/
|
|
|
|
cliPromptInput(promptSchema, overrides) {
|
|
if (this.S._interactive) { //CLI
|
|
let Prompter = SCli.prompt();
|
|
Prompter.override = overrides;
|
|
return Prompter.getAsync(promptSchema);
|
|
} else {
|
|
return BbPromise.resolve(); //in non interactive mode. All options must be set programatically
|
|
}
|
|
}
|
|
|
|
/**
|
|
* CLI Prompt Select
|
|
* - Handy CLI Select Input function for Plugins
|
|
* @param message string
|
|
* @param choices [{key:"",value:"",label:""}]
|
|
* @param multi boolean
|
|
* @param doneLabel string optional
|
|
* @returns {Promise} containing [{value:'blah'},..]
|
|
*/
|
|
|
|
cliPromptSelect(message, choices, multi, doneLabel) {
|
|
if (this.S._interactive) { //CLI
|
|
return SCli.select(message, choices, multi, doneLabel);
|
|
} else if (this.S.isWebInterface) {
|
|
//TODO: implement
|
|
return BbPromise.reject(new SError('Not implemented', SError.errorCodes.UNKNOWN));
|
|
} else {
|
|
return BbPromise.reject(new SError('You must specify all necessary options when in a non-interactive mode', SError.errorCodes.UNKNOWN));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* CLI Prompt Select Functions
|
|
* - Prompt the user to select a function in the current working directory
|
|
*/
|
|
|
|
cliPromptSelectFunctions(cwd, message, multi, skipSingles) {
|
|
|
|
let _this = this;
|
|
|
|
// If not interactive, throw error
|
|
if (!this.S._interactive) {
|
|
return BbPromise.reject(new SError('Sorry, this is only available in interactive mode'));
|
|
}
|
|
|
|
// Get Functions
|
|
return SUtils.getFunctions(cwd, null)
|
|
.then(function(functions) {
|
|
|
|
// If no functions found, return
|
|
if (!functions.length) {
|
|
return [];
|
|
}
|
|
|
|
// Skip if only a single option is available
|
|
if (functions.length === 1 && skipSingles) {
|
|
return functions;
|
|
}
|
|
|
|
// Prepare function choices
|
|
let choices = [];
|
|
for (let i = 0; i < functions.length; i++) choices.push({
|
|
key: ' ',
|
|
value: functions[i],
|
|
label: functions[i].name,
|
|
moduleName: functions[i].moduleName
|
|
});
|
|
|
|
// Order functions by module
|
|
choices.sort(function(a,b) {
|
|
if (a.moduleName < b.moduleName) return -1;
|
|
if (a.moduleName > b.moduleName) return 1;
|
|
return 0;
|
|
});
|
|
|
|
// Add spacers between modules
|
|
let lastModule;
|
|
for (let i = 0; i < choices.length; i++) {
|
|
if (choices[i].moduleName !== lastModule) {
|
|
lastModule = choices[i].moduleName;
|
|
choices.splice(i, 0, { spacer: lastModule });
|
|
}
|
|
}
|
|
|
|
// Show select input
|
|
return _this.cliPromptSelect(message, choices, multi, 'Deploy')
|
|
.then(function(selected) {
|
|
|
|
let selectedFunctions = [];
|
|
for (let i = 0; i < selected.length; i++) {
|
|
if (selected[i].toggled) selectedFunctions.push(selected[i].value);
|
|
}
|
|
|
|
return selectedFunctions;
|
|
});
|
|
});
|
|
}
|
|
|
|
/**
|
|
* CLI Prompt Select Endpoints
|
|
* - Prompt the user to select an endpoint in the current working directory
|
|
*/
|
|
|
|
cliPromptSelectEndpoints(cwd, message, multi, skipSingles) {
|
|
|
|
let _this = this;
|
|
|
|
// If not interactive, throw error
|
|
if (!this.S._interactive) {
|
|
return BbPromise.reject(new SError('Sorry, this is only available in interactive mode'));
|
|
}
|
|
|
|
// Get Functions
|
|
return SUtils.getEndpoints(cwd, null)
|
|
.then(function(endpoints) {
|
|
|
|
// If no endpoints found, return
|
|
if (!endpoints.length) {
|
|
return [];
|
|
}
|
|
|
|
// Skip if only a single option is available
|
|
if (endpoints.length === 1 && skipSingles) {
|
|
return endpoints;
|
|
}
|
|
|
|
// Prepare endpoints choices
|
|
let choices = [];
|
|
for (let i = 0; i < endpoints.length; i++) choices.push({
|
|
key: ' ',
|
|
value: endpoints[i],
|
|
label: endpoints[i].method + ' - ' + endpoints[i].path,
|
|
});
|
|
|
|
// Order functions by module
|
|
choices.sort(function(a,b) {
|
|
|
|
let valueA = a.value.moduleName + a.value.functionName;
|
|
let valueB = b.value.moduleName + b.value.functionName;
|
|
|
|
if (valueA < valueA) return -1;
|
|
if (valueB > valueB) return 1;
|
|
return 0;
|
|
});
|
|
|
|
// Add spacers between modules
|
|
let last;
|
|
for (let i = 0; i < choices.length; i++) {
|
|
|
|
let current = choices[i].value.moduleName + ' - ' + choices[i].value.function.name;
|
|
|
|
if (current !== last) {
|
|
last = current;
|
|
choices.splice(i, 0, { spacer: current });
|
|
}
|
|
}
|
|
|
|
// Show select input
|
|
return _this.cliPromptSelect(message, choices, multi, 'Deploy')
|
|
.then(function(selected) {
|
|
|
|
let selectedEndpoints = [];
|
|
for (let i = 0; i < selected.length; i++) {
|
|
if (selected[i].toggled) selectedEndpoints.push(selected[i].value);
|
|
}
|
|
|
|
return selectedEndpoints;
|
|
});
|
|
});
|
|
}
|
|
}
|
|
|
|
module.exports = ServerlessPlugin; |