mirror of
https://github.com/serverless/serverless.git
synced 2026-01-18 14:58:43 +00:00
554 lines
15 KiB
JavaScript
554 lines
15 KiB
JavaScript
'use strict';
|
|
|
|
/**
|
|
* JAWS Command: project create
|
|
* - Asks the user for information about their new JAWS project
|
|
* - Creates a new project in the current working directory
|
|
* - Creates IAM resources via CloudFormation
|
|
*/
|
|
|
|
const GlobalCmd = require('./GlobalCmd'),
|
|
JawsError = require('../jaws-error'),
|
|
JawsCLI = require('../utils/cli'),
|
|
Promise = require('bluebird'),
|
|
path = require('path'),
|
|
os = require('os'),
|
|
AWSUtils = require('../utils/aws'),
|
|
utils = require('../utils'),
|
|
shortid = require('shortid');
|
|
|
|
let fs = require('fs');
|
|
Promise.promisifyAll(fs);
|
|
|
|
function generateShortId(maxLen) {
|
|
return shortid.generate().replace(/\W+/g, '').substring(0, maxLen).replace(/[_-]/g, '');
|
|
}
|
|
|
|
/**
|
|
* CMD Class
|
|
* @param name
|
|
* @param stage
|
|
* @param domain
|
|
* @param notificationEmail
|
|
* @param region
|
|
* @param profile
|
|
* @param noCf
|
|
* @param runtime
|
|
* @constructor
|
|
*/
|
|
|
|
const CMD = class ProjectCreate extends GlobalCmd {
|
|
constructor(name, stage, region, domain, notificationEmail, profile, noCf, runtime) {
|
|
super();
|
|
|
|
// Defaults
|
|
this._name = name ? name : null;
|
|
this._domain = domain ? domain : null;
|
|
this._stage = stage ? stage.toLowerCase().replace(/\W+/g, '').substring(0, 15) : null;
|
|
this._notificationEmail = notificationEmail;
|
|
this._region = region;
|
|
this._profile = profile;
|
|
this._runtime = runtime ? runtime : 'nodejs';
|
|
this._noCf = noCf;
|
|
this._prompts = {
|
|
properties: {},
|
|
};
|
|
this.Prompter = JawsCLI.prompt();
|
|
this.Prompter.override = {};
|
|
this._spinner = null;
|
|
|
|
}
|
|
|
|
/**
|
|
* Run
|
|
* @returns {*}
|
|
*/
|
|
|
|
run() {
|
|
let _this = this;
|
|
|
|
return Promise.try(function() {
|
|
|
|
// ASCII Greeting
|
|
JawsCLI.ascii();
|
|
|
|
})
|
|
.bind(_this)
|
|
.then(_this._prompt)
|
|
.then(_this._prepareProjectData)
|
|
.then(_this._createProjectDirectory)
|
|
.then(function() {
|
|
if (_this._noCf) {
|
|
let stackResourcesName = AWSUtils.cfGetResourcesStackName(
|
|
_this._stage,
|
|
_this._name
|
|
);
|
|
|
|
JawsCLI.log('Remember to run CloudFormation manually');
|
|
JawsCLI.log(`MAKE SURE to create stack with name: ${stackResourcesName}`);
|
|
JawsCLI.log('After creating CF stack, remember to put the IAM role outputs and jawsBucket in your '
|
|
+ 'project jaws.json in the right stage/region.');
|
|
return false;
|
|
} else {
|
|
return _this._createCfStack()
|
|
.bind(_this)
|
|
.then(function(cfData) {
|
|
if (_this._spinner) {
|
|
_this._spinner.stop(true);
|
|
}
|
|
_this._cfData = cfData;
|
|
})
|
|
.then(_this._putEnvFile)
|
|
.then(_this._putCfFile);
|
|
}
|
|
})
|
|
.then(_this._createProjectJson)
|
|
.then(_this._initRuntime)
|
|
.then(function() {
|
|
JawsCLI.log('Your project "' + _this._name
|
|
+ '" has been successfully created in the current directory.');
|
|
});
|
|
}
|
|
|
|
|
|
/**
|
|
* Prompt
|
|
* @private
|
|
*/
|
|
|
|
_prompt() {
|
|
|
|
utils.jawsDebug('Prompting for new project information');
|
|
let _this = this;
|
|
|
|
let nameDescription = 'Enter a project name: ';
|
|
|
|
// Prompt: name (project name)
|
|
_this.Prompter.override.name = _this._name;
|
|
_this._prompts.properties.name = {
|
|
description: nameDescription.yellow,
|
|
default: 'jaws-' + generateShortId(19),
|
|
message: 'Name must be only letters, numbers or dashes',
|
|
conform: function(name) {
|
|
let re = /^[a-zA-Z0-9-_]+$/;
|
|
return re.test(name);
|
|
},
|
|
};
|
|
|
|
// Prompt: domain - for AWS hosted zone and more
|
|
_this.Prompter.override.domain = _this._domain;
|
|
|
|
let domainDescription = 'Enter a project domain (You can change this at any time: ';
|
|
|
|
_this._prompts.properties.domain = {
|
|
description: domainDescription.yellow,
|
|
default: 'myapp.com',
|
|
message: 'Domain must only contain lowercase letters, numbers, periods and dashes',
|
|
conform: function(bucket) {
|
|
let re = /^[a-z0-9-.]+$/;
|
|
return re.test(bucket);
|
|
},
|
|
};
|
|
|
|
// Prompt: notification email - for AWS alerts
|
|
_this.Prompter.override.notificationEmail = _this._notificationEmail;
|
|
_this._prompts.properties.notificationEmail = {
|
|
description: 'Enter an email to use for AWS alarms: '.yellow,
|
|
required: true,
|
|
message: 'Please enter a valid email',
|
|
default: 'me@myapp.com',
|
|
conform: function(email) {
|
|
if (!email) return false;
|
|
return true;
|
|
},
|
|
};
|
|
|
|
// Prompt: stage
|
|
_this.Prompter.override.stage = _this._stage;
|
|
|
|
let stageDescription = 'Enter a stage for this project: ';
|
|
|
|
_this._prompts.properties.stage = {
|
|
description: stageDescription.yellow,
|
|
default: 'dev',
|
|
message: 'Stage must be letters only',
|
|
conform: function(stage) {
|
|
let re = /^[a-zA-Z]+$/;
|
|
return re.test(stage);
|
|
},
|
|
};
|
|
|
|
// Prompt: notification email - for AWS alerts
|
|
_this.Prompter.override.notificationEmail = _this._notificationEmail;
|
|
|
|
let notificationEmailDescription = 'Enter an email to use for AWS alarms: ';
|
|
|
|
_this._prompts.properties.notificationEmail = {
|
|
description: notificationEmailDescription.yellow,
|
|
required: true,
|
|
message: 'Please enter a valid email',
|
|
default: 'you@yourapp.com',
|
|
conform: function(email) {
|
|
if (!email) return false;
|
|
return true;
|
|
},
|
|
};
|
|
|
|
// Prompt: API Keys - Create an AWS profile by entering API keys
|
|
if (!utils.fileExistsSync(path.join(AWSUtils.getConfigDir(), 'credentials'))) {
|
|
|
|
_this.Prompter.override.awsAdminKeyId = _this._awsAdminKeyId;
|
|
|
|
let apiKeyDescription = 'Enter the ACCESS KEY ID for your Admin AWS IAM User: ';
|
|
|
|
_this._prompts.properties.awsAdminKeyId = {
|
|
description: apiKeyDescription.yellow,
|
|
required: true,
|
|
message: 'Please enter a valid access key ID',
|
|
conform: function(key) {
|
|
if (!key) return false;
|
|
return true;
|
|
},
|
|
};
|
|
_this.Prompter.override.awsAdminSecretKey = _this._awsAdminSecretKey;
|
|
|
|
let apiSecretDescription = 'Enter the SECRET ACCESS KEY for your Admin AWS IAM User: ';
|
|
|
|
_this._prompts.properties.awsAdminSecretKey = {
|
|
description: apiSecretDescription.yellow,
|
|
required: true,
|
|
message: 'Please enter a valid secret access key',
|
|
conform: function(key) {
|
|
if (!key) return false;
|
|
return true;
|
|
},
|
|
};
|
|
}
|
|
|
|
// Show Prompts
|
|
return _this.Prompter.getAsync(_this._prompts)
|
|
.then(function(answers) {
|
|
_this._name = answers.name;
|
|
_this._domain = answers.domain;
|
|
_this._stage = answers.stage.toLowerCase();
|
|
_this._notificationEmail = answers.notificationEmail;
|
|
_this._awsAdminKeyId = answers.awsAdminKeyId;
|
|
_this._awsAdminSecretKey = answers.awsAdminSecretKey;
|
|
|
|
// If region exists, skip select prompt
|
|
if (_this._region) return;
|
|
|
|
// Prompt: region select
|
|
let choices = [];
|
|
AWSUtils.validLambdaRegions.forEach(function(r) {
|
|
choices.push({
|
|
key: '',
|
|
value: r,
|
|
label: r,
|
|
});
|
|
});
|
|
|
|
return JawsCLI.select('Select a region for your project: ', choices, false)
|
|
.then(function(results) {
|
|
_this._region = results[0].value;
|
|
});
|
|
})
|
|
.then(function() {
|
|
|
|
// If profile exists, skip select prompt
|
|
if (_this._profile) return Promise.resolve();
|
|
|
|
// If aws credentials were passed, skip select prompt
|
|
if (_this._awsAdminKeyId && _this._awsAdminSecretKey) return Promise.resolve();
|
|
|
|
// Prompt: profile select
|
|
let profilesList = AWSUtils.profilesMap(),
|
|
profiles = Object.keys(profilesList),
|
|
choices = [];
|
|
|
|
for (let i = 0; i < profiles.length; i++) {
|
|
choices.push({
|
|
key: '',
|
|
value: profiles[i],
|
|
label: profiles[i],
|
|
});
|
|
}
|
|
|
|
return JawsCLI.select('Select an AWS profile for your project: ', choices, false)
|
|
.then(function(results) {
|
|
_this._profile = results[0].value;
|
|
});
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Prepare Project Data
|
|
* @private
|
|
*/
|
|
|
|
_prepareProjectData() {
|
|
|
|
let _this = this;
|
|
|
|
// Validate: Ensure stage isn't "local"
|
|
if (_this._stage.toLowerCase() == 'local') {
|
|
throw new JawsError('Stage ' + _this._stage + ' is reserved');
|
|
}
|
|
|
|
// Validate: AWS only allows Alphanumeric and - in name
|
|
let nameOk = /^([a-zA-Z0-9-]+)$/.exec(_this._name);
|
|
if (!nameOk) {
|
|
throw new JawsError('Project names can only be alphanumeric and -');
|
|
}
|
|
|
|
// Append unique id if name is in use
|
|
if (utils.dirExistsSync(path.join(process.cwd(), _this._name))) {
|
|
_this._name = _this._name + '-' + generateShortId(19);
|
|
}
|
|
|
|
// Append unique id if domain is default
|
|
if (_this._domain === 'myapp.com') {
|
|
_this._domain = 'myapp-' + generateShortId(8) + '.com';
|
|
}
|
|
|
|
// Set JAWS Bucket
|
|
_this._jawsBucket = utils.generateJawsBucketName(_this._stage, _this._region, _this._domain);
|
|
|
|
// Validate: If no profile, ensure access keys, create profile
|
|
if (!_this._profile) {
|
|
|
|
if (!_this._awsAdminKeyId) {
|
|
throw new JawsError(
|
|
'An AWS Access Key ID is required',
|
|
JawsError.errorCodes.MISSING_AWS_CREDS);
|
|
}
|
|
|
|
if (!_this._awsAdminSecretKey) {
|
|
throw new JawsError(
|
|
'An AWS Secret Key is required',
|
|
JawsError.errorCodes.MISSING_AWS_CREDS);
|
|
}
|
|
|
|
// Set profile
|
|
AWSUtils.profilesSet('default', _this._region, _this._awsAdminKeyId, _this._awsAdminSecretKey);
|
|
_this._profile = 'default';
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Create Project Directory
|
|
* @private
|
|
*/
|
|
|
|
_createProjectDirectory() {
|
|
let _this = this;
|
|
|
|
_this._projectRootPath = path.resolve(path.join(path.dirname('.'), _this._name));
|
|
|
|
// Prepare admin.env
|
|
let adminEnv = 'ADMIN_AWS_PROFILE=' + _this._profile + os.EOL;
|
|
|
|
// Prepare README.md
|
|
let readme = '#' + _this._name;
|
|
|
|
// Create Project Scaffolding
|
|
return utils.writeFile(
|
|
path.join(_this._projectRootPath, '.env'),
|
|
'JAWS_STAGE=' + _this._stage
|
|
+ '\nJAWS_DATA_MODEL_STAGE=' + _this._stage
|
|
)
|
|
.then(function() {
|
|
return Promise.all([
|
|
fs.mkdirAsync(path.join(_this._projectRootPath, 'tests')),
|
|
fs.mkdirAsync(path.join(_this._projectRootPath, 'lib')),
|
|
fs.mkdirAsync(path.join(_this._projectRootPath, 'aws_modules')),
|
|
utils.writeFile(path.join(_this._projectRootPath, 'admin.env'), adminEnv),
|
|
utils.writeFile(path.join(_this._projectRootPath, 'README.md'), readme),
|
|
utils.generateResourcesCf(
|
|
_this._projectRootPath,
|
|
_this._name,
|
|
_this._domain,
|
|
_this._stage,
|
|
_this._region,
|
|
_this._notificationEmail,
|
|
_this._jawsBucket
|
|
),
|
|
fs.writeFileAsync(path.join(_this._projectRootPath, '.gitignore'), fs.readFileSync(__dirname + '/../templates/gitignore')),
|
|
]);
|
|
});
|
|
}
|
|
|
|
|
|
/**
|
|
* Put ENV File
|
|
* - Creates ENV file in JAWS stage/region bucket
|
|
* @private
|
|
*/
|
|
|
|
_putEnvFile() {
|
|
|
|
let _this = this,
|
|
stage = this.stage;
|
|
|
|
let envFileContents = `JAWS_STAGE=${stage}
|
|
JAWS_DATA_MODEL_STAGE=${stage}`;
|
|
|
|
return AWSUtils.putEnvFile(
|
|
_this._profile,
|
|
_this._region,
|
|
_this._jawsBucket,
|
|
_this._name,
|
|
_this._stage,
|
|
envFileContents);
|
|
}
|
|
|
|
|
|
/**
|
|
* Put CF File
|
|
* @returns {*}
|
|
* @private
|
|
*/
|
|
|
|
_putCfFile() {
|
|
|
|
let _this = this;
|
|
|
|
return AWSUtils.putCfFile(
|
|
_this._profile,
|
|
_this._projectRootPath,
|
|
_this._region,
|
|
_this._jawsBucket,
|
|
_this._name,
|
|
_this._stage,
|
|
'resources');
|
|
|
|
}
|
|
|
|
/**
|
|
* Create CloudFormation Stack
|
|
* @private
|
|
*/
|
|
|
|
_createCfStack() {
|
|
|
|
let _this = this;
|
|
|
|
JawsCLI.log('Creating CloudFormation Stack for your new project (~5 mins)...');
|
|
_this._spinner = JawsCLI.spinner();
|
|
_this._spinner.start();
|
|
|
|
// Create CF stack
|
|
return AWSUtils.cfCreateResourcesStack(
|
|
_this._profile,
|
|
_this._region,
|
|
_this._projectRootPath,
|
|
_this._name,
|
|
_this._stage,
|
|
_this._domain,
|
|
_this._notificationEmail,
|
|
_this._jawsBucket
|
|
)
|
|
.then(function(cfData) {
|
|
return AWSUtils.monitorCf(cfData, _this._profile, _this._region, 'create');
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Create Project JSON
|
|
* @private
|
|
*/
|
|
|
|
_createProjectJson() {
|
|
|
|
let _this = this,
|
|
iamRoleArnLambda,
|
|
iamRoleArnApiGateway;
|
|
|
|
if (_this._cfData) {
|
|
for (let i = 0; i < _this._cfData.Outputs.length; i++) {
|
|
if (_this._cfData.Outputs[i].OutputKey === 'IamRoleArnLambda') {
|
|
iamRoleArnLambda = _this._cfData.Outputs[i].OutputValue;
|
|
}
|
|
|
|
if (_this._cfData.Outputs[i].OutputKey === 'IamRoleArnApiGateway') {
|
|
iamRoleArnApiGateway = _this._cfData.Outputs[i].OutputValue;
|
|
}
|
|
}
|
|
}
|
|
|
|
let templatesPath = path.join(__dirname, '..', 'templates'),
|
|
jawsJson = utils.readAndParseJsonSync(path.join(templatesPath, 'jaws.json'));
|
|
|
|
jawsJson.stages[_this._stage] = [{
|
|
region: _this._region,
|
|
iamRoleArnLambda: iamRoleArnLambda || '',
|
|
iamRoleArnApiGateway: iamRoleArnApiGateway || '',
|
|
jawsBucket: _this._jawsBucket,
|
|
}];
|
|
|
|
jawsJson.name = _this._name;
|
|
jawsJson.domain = _this._domain;
|
|
|
|
fs.writeFileSync(path.join(_this._projectRootPath, 'jaws.json'),
|
|
JSON.stringify(jawsJson, null, 2));
|
|
|
|
return jawsJson;
|
|
}
|
|
|
|
/**
|
|
* Init Runtime
|
|
* @private
|
|
*/
|
|
|
|
_initRuntime() {
|
|
|
|
let _this = this;
|
|
|
|
JawsCLI.log('Preparing your runtime and installing jaws-core module...');
|
|
|
|
if (_this._runtime === 'nodejs') {
|
|
let packageJsonTemplate = utils.readAndParseJsonSync(path.join(__dirname, '..', 'templates', 'nodejs', 'package.json'));
|
|
packageJsonTemplate.name = _this._name;
|
|
return fs.writeFileAsync(path.join(_this._projectRootPath, 'package.json'), JSON.stringify(packageJsonTemplate, null, 2))
|
|
.then(function() {
|
|
utils.jawsDebug('test_utils', 'Running NPM install...');
|
|
utils.npmInstall(_this._projectRootPath);
|
|
});
|
|
} else {
|
|
throw new JawsError('Unsupported runtime "' + _this.runtime + '"', JawsError.errorCodes.UNKNOWN);
|
|
}
|
|
}
|
|
};
|
|
|
|
/**************************************
|
|
* EXPORTS
|
|
**************************************/
|
|
|
|
/**
|
|
* Run
|
|
* @param name
|
|
* @param stage
|
|
* @param domain
|
|
* @param region
|
|
* @param notificationEmail
|
|
* @param profile
|
|
* @param noCf
|
|
* @param runtime defaults to 'nodejs'
|
|
* @returns {Promise}
|
|
*/
|
|
|
|
exports.run = function(name, stage, region, domain, notificationEmail, profile, noCf, runtime) {
|
|
utils.jawsDebug('Running new project:', name);
|
|
let command = new CMD(
|
|
name,
|
|
stage,
|
|
region,
|
|
domain,
|
|
notificationEmail,
|
|
profile,
|
|
noCf,
|
|
runtime);
|
|
return command.run();
|
|
};
|