serverless/lib/plugins/aws/deploy/tests/deployFunctions.js
2016-06-16 21:24:09 +02:00

353 lines
12 KiB
JavaScript

'use strict';
const expect = require('chai').expect;
const path = require('path');
const os = require('os');
const deployFunctions = require('../lib/deployFunctions');
const Serverless = require('../../../../Serverless');
const Zip = require('node-zip');
const sinon = require('sinon');
const AWS = require('aws-sdk');
const BbPromise = require('bluebird');
describe('deployFunctions', () => {
let serverless;
let awsDeployMock;
let zip;
class AwsDeployMock {
constructor(serverlessInstance) {
Object.assign(this, deployFunctions);
this.serverless = serverlessInstance;
this.options = {};
this.deployedFunctions = [];
}
}
const functionsObjectMock = {
name_template: 'name-template-name',
first: {
handler: 'first.function.handler',
exclude: [
'foo',
'bar.js',
],
include: [
'bar.js', // should be included even if it's excluded
'includeme',
],
},
second: {
handler: 'second.function.handler',
exclude: [
'baz',
'qux.js',
],
include: [
'qux.js', // should be included even if it's excluded
'includeme',
],
},
};
const simpleDeployedFunctionsArrayMock = [
{
name: 'function',
handler: 'function.handler',
exclude: [
'foo',
'bar.js',
],
},
];
const nestedDeployedFunctionsArrayMock = [
{
name: 'function',
handler: 'nested/function.handler',
exclude: [
'foo',
'bar.js',
],
},
];
const includeExcludedFileDeployedFunctionsArrayMock = [
{
name: 'function',
handler: 'function.handler',
exclude: [
'bar.js',
],
include: [
'bar.js',
],
},
];
const functionCodeMock = `
'use strict';
module.exports.handler = function(event, context, cb) {
return cb(null, {
message: 'First function'
});
};
`;
beforeEach(() => {
serverless = new Serverless();
serverless.init();
awsDeployMock = new AwsDeployMock(serverless);
zip = new Zip();
});
describe('#extractFunctionHandlers()', () => {
beforeEach(() => {
serverless.service.functions = functionsObjectMock;
});
it('should extract all the handlers in the function definitions', () => awsDeployMock
.extractFunctionHandlers().then(() => {
expect(
awsDeployMock.deployedFunctions[0].handler
).to.equal(functionsObjectMock.first.handler);
expect(
awsDeployMock.deployedFunctions[1].handler
).to.equal(functionsObjectMock.second.handler);
})
);
it('should extract the exclude array in the function definitions', () => awsDeployMock
.extractFunctionHandlers().then(() => {
expect(
awsDeployMock.deployedFunctions[0].exclude
).to.include('foo');
expect(
awsDeployMock.deployedFunctions[0].exclude
).to.include('bar.js');
expect(
awsDeployMock.deployedFunctions[1].exclude
).to.include('baz');
expect(
awsDeployMock.deployedFunctions[1].exclude
).to.include('qux.js');
})
);
it('should extract the include array in the functions definitions', () => awsDeployMock
.extractFunctionHandlers().then(() => {
expect(
awsDeployMock.deployedFunctions[0].include
).to.include('bar.js');
expect(
awsDeployMock.deployedFunctions[0].include
).to.include('includeme');
expect(
awsDeployMock.deployedFunctions[1].include
).to.include('qux.js');
expect(
awsDeployMock.deployedFunctions[1].include
).to.include('includeme');
})
);
});
describe('#zipFunctions()', () => {
it('should zip a simple function', () => {
awsDeployMock.deployedFunctions = simpleDeployedFunctionsArrayMock;
const functionFileNameBase = 'function';
// create a function in a temporary directory
const tmpDirPath = path.join(os.tmpdir(), (new Date).getTime().toString());
const tmpFilePath = path.join(tmpDirPath, `${functionFileNameBase}.js`);
serverless.utils.writeFileSync(tmpFilePath, functionCodeMock);
// set the servicePath
serverless.config.servicePath = tmpDirPath;
return awsDeployMock.zipFunctions().then(() => {
expect(awsDeployMock.deployedFunctions[0].zipFileData).to.be.not.empty;
// look into the zippedFileData
const unzippedFileData = zip.load(awsDeployMock.deployedFunctions[0].zipFileData);
expect(unzippedFileData.files[`${functionFileNameBase}.js`].name)
.to.equal(`${functionFileNameBase}.js`);
expect(unzippedFileData.files[`${functionFileNameBase}.js`].dir).to.equal(false);
});
});
it('should zip nested code', () => {
// set the deployedFunctions array
awsDeployMock.deployedFunctions = nestedDeployedFunctionsArrayMock;
const functionFileNameBase = 'function';
// create a function in a temporary directory --> nested/function.js
const tmpDirPath = path.join(os.tmpdir(), (new Date).getTime().toString(), 'nested');
const tmpFilePath = path.join(tmpDirPath, `${functionFileNameBase}.js`);
serverless.utils.writeFileSync(tmpFilePath, functionCodeMock);
// add a lib directory on the same level where the "nested" directory lives --> lib/some-file
const libDirectory = path.join(tmpDirPath, '..', 'lib');
serverless.utils.writeFileSync(path.join(libDirectory, 'some-file'), 'content');
// set the servicePath
serverless.config.servicePath = tmpDirPath;
return awsDeployMock.zipFunctions().then(() => {
expect(awsDeployMock.deployedFunctions[0].zipFileData).to.be.not.empty;
// look into the zippedFileData
const unzippedFileData = zip.load(awsDeployMock.deployedFunctions[0].zipFileData);
expect(unzippedFileData.files[`nested/${functionFileNameBase}.js`].name)
.to.equal(`nested/${functionFileNameBase}.js`);
expect(unzippedFileData.files[`nested/${functionFileNameBase}.js`].dir)
.to.equal(false);
expect(unzippedFileData.files['lib/some-file'].name)
.to.equal('lib/some-file');
expect(unzippedFileData.files['lib/some-file'].dir)
.to.equal(false);
});
});
it('should exclude defined files and folders', () => {
awsDeployMock.deployedFunctions = simpleDeployedFunctionsArrayMock;
const functionFileNameBase = 'function';
// create a function in a temporary directory
const tmpDirPath = path.join(os.tmpdir(), (new Date).getTime().toString());
const tmpFilePath = path.join(tmpDirPath, `${functionFileNameBase}.js`);
serverless.utils.writeFileSync(tmpFilePath, functionCodeMock);
// create a folder with the name "foo" which also includes a file --> foo/baz.txt
serverless.utils.writeFileSync(path.join(tmpDirPath, 'foo', 'baz.txt'), 'content');
// create a file with the name "bar.js" --> bar.js
serverless.utils.writeFileSync(path.join(tmpDirPath, 'bar.js'), 'content');
// set the servicePath
serverless.config.servicePath = tmpDirPath;
return awsDeployMock.zipFunctions().then(() => {
// look into the zippedFileData
const unzippedFileData = zip.load(awsDeployMock.deployedFunctions[0].zipFileData);
expect(unzippedFileData.files[`${functionFileNameBase}.js`].name)
.to.equal(`${functionFileNameBase}.js`);
expect(unzippedFileData.files[`${functionFileNameBase}.js`].dir).to.equal(false);
expect(unzippedFileData.files['foo/baz.txt']).to.equal(undefined);
expect(unzippedFileData.files['bar.js']).to.equal(undefined);
});
});
it('should exclude predefined files and folders (e.g. like .git)', () => {
awsDeployMock.deployedFunctions = simpleDeployedFunctionsArrayMock;
const functionFileNameBase = 'function';
// create a function in a temporary directory
const tmpDirPath = path.join(os.tmpdir(), (new Date).getTime().toString());
const tmpFilePath = path.join(tmpDirPath, `${functionFileNameBase}.js`);
serverless.utils.writeFileSync(tmpFilePath, functionCodeMock);
// create the files and folder which should be ignored
// .gitignore
const gitignoreFilePath = path.join(tmpDirPath, '.gitignore');
serverless.utils.writeFileSync(gitignoreFilePath, 'content');
// .DS_Store
const dsStoreFilePath = path.join(tmpDirPath, '.DS_Store');
serverless.utils.writeFileSync(dsStoreFilePath, 'content');
// serverless.yaml
const serverlessYamlFilePath = path.join(tmpDirPath, 'serverless.yaml');
serverless.utils.writeFileSync(serverlessYamlFilePath, 'content');
// serverless.env.yaml
const serverlessEnvYamlFilePath = path.join(tmpDirPath, 'serverless.env.yaml');
serverless.utils.writeFileSync(serverlessEnvYamlFilePath, 'content');
const gitFilePath = path.join(path.join(tmpDirPath, '.git'), 'some-random-git-file');
serverless.utils.writeFileSync(gitFilePath, 'content');
// set the servicePath
serverless.config.servicePath = tmpDirPath;
return awsDeployMock.zipFunctions().then(() => {
// look into the zippedFileData
const unzippedFileData = zip.load(awsDeployMock.deployedFunctions[0].zipFileData);
expect(unzippedFileData.files[`${functionFileNameBase}.js`].name)
.to.equal(`${functionFileNameBase}.js`);
expect(unzippedFileData.files[`${functionFileNameBase}.js`].dir).to.equal(false);
expect(unzippedFileData.files['.gitignore']).to.equal(undefined);
expect(unzippedFileData.files['.DS_Store']).to.equal(undefined);
expect(unzippedFileData.files['.serverless.yaml']).to.equal(undefined);
expect(unzippedFileData.files['.serverless.env.yaml']).to.equal(undefined);
expect(unzippedFileData.files['.git']).to.equal(undefined);
});
});
it('should include a previously excluded file', () => {
awsDeployMock.deployedFunctions = includeExcludedFileDeployedFunctionsArrayMock;
const functionFileNameBase = 'function';
// create a function in a temporary directory
const tmpDirPath = path.join(os.tmpdir(), (new Date).getTime().toString());
const tmpFilePath = path.join(tmpDirPath, `${functionFileNameBase}.js`);
serverless.utils.writeFileSync(tmpFilePath, functionCodeMock);
// create a file with the name "bar.js" --> bar.js
serverless.utils.writeFileSync(path.join(tmpDirPath, 'bar.js'), 'content');
// set the servicePath
serverless.config.servicePath = tmpDirPath;
return awsDeployMock.zipFunctions().then(() => {
// look into the zippedFileData
const unzippedFileData = zip.load(awsDeployMock.deployedFunctions[0].zipFileData);
expect(unzippedFileData.files[`${functionFileNameBase}.js`].name)
.to.equal(`${functionFileNameBase}.js`);
expect(unzippedFileData.files[`${functionFileNameBase}.js`].dir).to.equal(false);
expect(unzippedFileData.files['bar.js'].name).to.equal('bar.js');
expect(unzippedFileData.files['bar.js'].dir).to.equal(false);
});
});
});
describe('#uploadZipFilesToS3Bucket()', () => {
it('should upload the zip files to the S3 bucket', () => {
awsDeployMock.deployedFunctions = [
{
zipFileKey: true,
zipFileData: true,
},
];
awsDeployMock.S3 = new AWS.S3();
BbPromise.promisifyAll(awsDeployMock.S3, { suffix: 'Promised' });
const putObjectStub = sinon.stub(awsDeployMock.S3, 'putObjectPromised');
return awsDeployMock.uploadZipFilesToS3Bucket().then(() => {
expect(putObjectStub.calledOnce).to.be.equal(true);
awsDeployMock.S3.putObjectPromised.restore();
});
});
});
});