UNPKG

jaws-framework

Version:

JAWS is the serverless framework powered by Amazon Web Services.

541 lines (445 loc) 15 kB
'use strict'; /** * JAWS Command: new project * - Asks the user for information about their new JAWS project * - Creates a new project in the current working directory * - Creates IAM resources via CloudFormation */ // Defaults var JawsError = require('../jaws-error'), JawsCLI = require('../utils/cli'), Promise = require('bluebird'), fs = require('fs'), path = require('path'), os = require('os'), AWSUtils = require('../utils/aws'), utils = require('../utils'), shortid = require('shortid'); Promise.promisifyAll(fs); function generateShortId(maxLen) { return shortid.generate().replace(/\W+/g, '').substring(0, maxLen).replace(/[_-]/g, ''); } /** * Run * @param name * @param stage * @param domain * @param region * @param notificationEmail * @param profile * @param noCf * @param runtime defaults to 'nodejs' * @returns {*} */ module.exports.run = function(name, stage, region, domain, notificationEmail, profile, noCf, runtime) { utils.jawsDebug('Running new project:', name); var command = new CMD( name, stage, region, domain, notificationEmail, profile, noCf, runtime || 'nodejs'); return command.run(); }; /** * CMD Class * @param name * @param stage * @param domain * @param notificationEmail * @param region * @param profile * @param noCf * @param runtime * @constructor */ function CMD(name, stage, region, domain, notificationEmail, profile, noCf, runtime) { // 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; this._noCf = noCf; this._prompts = { properties: {}, }; this.Prompter = JawsCLI.prompt(); this.Prompter.override = {}; this._spinner = null; } /** * CMD: Run */ CMD.prototype.run = Promise.method(function() { var _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) { JawsCLI.log('Remember to run CloudFormation manually'); JawsCLI.log('!!MAKE SURE!! to create stack with name: ' + AWSUtils.cfGetResourcesStackName( _this._stage, _this._name )); 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.'); }); }); /** * CMD: Prompt */ CMD.prototype._prompt = Promise.method(function() { utils.jawsDebug('Prompting for new project information'); var _this = this; var 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) { var re = /^[a-zA-Z0-9-_]+$/; return re.test(name); }, }; // Prompt: domain - for AWS hosted zone and more _this.Prompter.override.domain = _this._domain; var 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) { var 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; var 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) { var re = /^[a-zA-Z]+$/; return re.test(stage); }, }; // Prompt: notification email - for AWS alerts _this.Prompter.override.notificationEmail = _this._notificationEmail; var 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; var 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; var 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 var 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 var profilesList = AWSUtils.profilesMap(), profiles = Object.keys(profilesList), choices = []; for (var 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; }); }); }); /** * CMD: Prepare Project Data * @returns {Promise} * @private */ CMD.prototype._prepareProjectData = Promise.method(function() { var _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 var 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'; } }); /** * CMD: Create Project Directory * @returns {Promise} * @private */ CMD.prototype._createProjectDirectory = Promise.method(function() { var _this = this; _this._projectRootPath = path.resolve(path.join(path.dirname('.'), _this._name)); // Prepare admin.env var adminEnv = 'ADMIN_AWS_PROFILE=' + _this._profile + os.EOL; // Prepare README.md var 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')), ]); }); }); /** * CMD Put ENV File * - Uploads .env file to jawsbucket */ CMD.prototype._putEnvFile = Promise.method(function() { var _this = this; var envFileContents = 'JAWS_STAGE=' + _this._stage + '\nJAWS_DATA_MODEL_STAGE=' + _this._stage; return AWSUtils.putEnvFile( _this._profile, _this._region, _this._jawsBucket, _this._name, _this._stage, envFileContents); }); /** * CMD: Put CF File * - Uploads timestamped CF file to jawsbucket */ CMD.prototype._putCfFile = Promise.method(function() { var _this = this; return AWSUtils.putCfFile( _this._profile, _this._projectRootPath, _this._region, _this._jawsBucket, _this._name, _this._stage, 'resources'); }); /** * CMD: Create CloudFormation Stack */ CMD.prototype._createCfStack = Promise.method(function() { var _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'); }); }); /** * CMD: Create Project JSON * * @param cfOutputs. Optional * @returns {Promise} jaws json js obj * @private */ CMD.prototype._createProjectJson = Promise.method(function(cfData) { var _this = this, iamRoleArnLambda, iamRoleArnApiGateway; if (_this._cfData) { for (var 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; } } } var 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; }); /** * CMD: Init Runtime */ CMD.prototype._initRuntime = Promise.method(function() { var _this = this; JawsCLI.log('Preparing your runtime and installing jaws-core module...'); if (_this._runtime === 'nodejs') { var 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); } });