2019-11-20 11:11:20 +01:00

414 lines
14 KiB
JavaScript

'use strict';
const crypto = require('crypto');
const path = require('path');
const chai = require('chai');
const _ = require('lodash');
const AwsProvider = require('../../../provider/awsProvider');
const AwsCompileLayers = require('./index');
const Serverless = require('../../../../../Serverless');
const { getTmpDirPath } = require('../../../../../../tests/utils/fs');
chai.use(require('chai-as-promised'));
const expect = chai.expect;
describe('AwsCompileLayers', () => {
let serverless;
let awsCompileLayers;
const layerName = 'test';
const compiledLayerName = 'TestLambdaLayer';
beforeEach(() => {
const options = {
stage: 'dev',
region: 'us-east-1',
};
serverless = new Serverless(options);
serverless.setProvider('aws', new AwsProvider(serverless, options));
serverless.cli = new serverless.classes.CLI();
serverless.config.servicePath = process.cwd();
awsCompileLayers = new AwsCompileLayers(serverless, options);
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate = {
Resources: {},
Outputs: {},
};
const serviceArtifact = 'new-service.zip';
const individualArtifact = 'test.zip';
awsCompileLayers.packagePath = getTmpDirPath();
// The contents of the test artifacts need to be predictable so the hashes stay the same
serverless.utils.writeFileSync(
path.join(awsCompileLayers.packagePath, serviceArtifact),
'foobar'
);
serverless.utils.writeFileSync(
path.join(awsCompileLayers.packagePath, individualArtifact),
'barbaz'
);
awsCompileLayers.serverless.service.service = 'new-service';
awsCompileLayers.serverless.service.package.artifactDirectoryName = 'somedir';
awsCompileLayers.serverless.service.package.artifact = path.join(
awsCompileLayers.packagePath,
serviceArtifact
);
awsCompileLayers.serverless.service.layers = {};
awsCompileLayers.serverless.service.layers[layerName] = {
name: 'test',
package: {
artifact: path.join(awsCompileLayers.packagePath, individualArtifact),
},
handler: 'handler.hello',
};
});
describe('#constructor()', () => {
it('should set the provider variable to an instance of AwsProvider', () =>
expect(awsCompileLayers.provider).to.be.instanceof(AwsProvider));
});
describe('#compileLayers()', () => {
it('should use layer artifact if individually', () => {
awsCompileLayers.serverless.service.package.individually = true;
return expect(awsCompileLayers.compileLayers()).to.be.fulfilled.then(() => {
const layerResource =
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Resources[
compiledLayerName
];
const s3Folder = awsCompileLayers.serverless.service.package.artifactDirectoryName;
const s3FileName = awsCompileLayers.serverless.service.layers[layerName].package.artifact
.split(path.sep)
.pop();
expect(layerResource.Properties.Content.S3Key).to.deep.equal(`${s3Folder}/${s3FileName}`);
});
});
it('should create a simple layer resource', () => {
const s3Folder = awsCompileLayers.serverless.service.package.artifactDirectoryName;
const s3FileName = awsCompileLayers.serverless.service.layers.test.package.artifact
.split(path.sep)
.pop();
awsCompileLayers.serverless.service.layers = {
test: {
path: 'layer',
},
};
const compiledLayer = {
Type: 'AWS::Lambda::LayerVersion',
Properties: {
Content: {
S3Key: `${s3Folder}/${s3FileName}`,
S3Bucket: { Ref: 'ServerlessDeploymentBucket' },
},
LayerName: 'test',
},
};
const compiledLayerOutput = {
Description: 'Current Lambda layer version',
Value: {
Ref: 'TestLambdaLayer',
},
};
return expect(awsCompileLayers.compileLayers()).to.be.fulfilled.then(() => {
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Resources
.TestLambdaLayer
).to.deep.equal(compiledLayer);
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Outputs
.TestLambdaLayerQualifiedArn
).to.deep.equal(compiledLayerOutput);
});
});
it('should create a layer resource with a retention policy', () => {
const s3Folder = awsCompileLayers.serverless.service.package.artifactDirectoryName;
const s3FileName = awsCompileLayers.serverless.service.layers.test.package.artifact
.split(path.sep)
.pop();
awsCompileLayers.serverless.service.layers = {
test: {
path: 'layer',
retain: true,
},
};
const compiledLayer = {
Type: 'AWS::Lambda::LayerVersion',
Properties: {
Content: {
S3Bucket: { Ref: 'ServerlessDeploymentBucket' },
S3Key: `${s3Folder}/${s3FileName}`,
},
LayerName: 'test',
},
};
const sha = crypto
.createHash('sha1')
.update(JSON.stringify(_.omit(compiledLayer, ['Properties.Content.S3Key'])))
.digest('hex');
compiledLayer.DeletionPolicy = 'Retain';
const compiledLayerOutput = {
Description: 'Current Lambda layer version',
Value: {
Ref: `TestLambdaLayer${sha}`,
},
};
return expect(awsCompileLayers.compileLayers()).to.be.fulfilled.then(() => {
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Resources[
`TestLambdaLayer${sha}`
]
).to.deep.equal(compiledLayer);
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Outputs
.TestLambdaLayerQualifiedArn
).to.deep.equal(compiledLayerOutput);
});
});
it('should create a layer resource with a retention policy, has the same logical id when not modify', () => {
awsCompileLayers.serverless.service.layers = {
test: {
path: 'layer',
retain: true,
},
};
const secondAwsCompileLayers = _.cloneDeep(awsCompileLayers);
secondAwsCompileLayers.serverless.service.package.artifactDirectoryName = 'somedir2';
return expect(
Promise.all([awsCompileLayers.compileLayers(), secondAwsCompileLayers.compileLayers()])
).to.be.fulfilled.then(() => {
expect(
_.findKey(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Resources,
r => r.Type === 'AWS::Lambda::LayerVersion'
)
).to.deep.equal(
_.findKey(
secondAwsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate
.Resources,
r => r.Type === 'AWS::Lambda::LayerVersion'
)
);
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Outputs
.TestLambdaLayerQualifiedArn
).to.deep.equal(
secondAwsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Outputs
.TestLambdaLayerQualifiedArn
);
});
});
it('should create a layer resource with a retention policy, has different logical id when modify', () => {
awsCompileLayers.serverless.service.layers = {
test: {
path: 'layer',
retain: true,
},
};
const secondAwsCompileLayers = _.cloneDeep(awsCompileLayers);
secondAwsCompileLayers.serverless.service.layers.test.description = 'modified description';
return expect(
Promise.all([awsCompileLayers.compileLayers(), secondAwsCompileLayers.compileLayers()])
).to.be.fulfilled.then(() => {
expect(
_.findKey(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Resources,
r => r.Type === 'AWS::Lambda::LayerVersion'
)
).to.not.deep.equal(
_.findKey(
secondAwsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate
.Resources,
r => r.Type === 'AWS::Lambda::LayerVersion'
)
);
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Outputs
.TestLambdaLayerQualifiedArn
).to.not.deep.equal(
secondAwsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Outputs
.TestLambdaLayerQualifiedArn
);
});
});
it('should create a layer resource with permissions', () => {
const s3Folder = awsCompileLayers.serverless.service.package.artifactDirectoryName;
const s3FileName = awsCompileLayers.serverless.service.layers.test.package.artifact
.split(path.sep)
.pop();
awsCompileLayers.serverless.service.layers = {
test: {
path: 'layer',
allowedAccounts: ['*'],
},
};
const compiledLayer = {
Type: 'AWS::Lambda::LayerVersion',
Properties: {
Content: {
S3Key: `${s3Folder}/${s3FileName}`,
S3Bucket: { Ref: 'ServerlessDeploymentBucket' },
},
LayerName: 'test',
},
};
const compiledLayerOutput = {
Description: 'Current Lambda layer version',
Value: {
Ref: 'TestLambdaLayer',
},
};
const compiledLayerVersion = {
Type: 'AWS::Lambda::LayerVersionPermission',
Properties: {
Action: 'lambda:GetLayerVersion',
LayerVersionArn: {
Ref: 'TestLambdaLayer',
},
Principal: '*',
},
};
return expect(awsCompileLayers.compileLayers()).to.be.fulfilled.then(() => {
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Resources
.TestLambdaLayer
).to.deep.equal(compiledLayer);
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Outputs
.TestLambdaLayerQualifiedArn
).to.deep.equal(compiledLayerOutput);
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Resources
.TestWildLambdaLayerPermission
).to.deep.equal(compiledLayerVersion);
});
});
it('should create a layer resource with permissions per account', () => {
const s3Folder = awsCompileLayers.serverless.service.package.artifactDirectoryName;
const s3FileName = awsCompileLayers.serverless.service.layers.test.package.artifact
.split(path.sep)
.pop();
awsCompileLayers.serverless.service.layers = {
test: {
path: 'layer',
allowedAccounts: [1111111, '2222222'],
},
};
const compiledLayer = {
Type: 'AWS::Lambda::LayerVersion',
Properties: {
Content: {
S3Key: `${s3Folder}/${s3FileName}`,
S3Bucket: { Ref: 'ServerlessDeploymentBucket' },
},
LayerName: 'test',
},
};
const compiledLayerOutput = {
Description: 'Current Lambda layer version',
Value: {
Ref: 'TestLambdaLayer',
},
};
const compiledLayerVersionNumber = {
Type: 'AWS::Lambda::LayerVersionPermission',
Properties: {
Action: 'lambda:GetLayerVersion',
LayerVersionArn: {
Ref: 'TestLambdaLayer',
},
Principal: '1111111',
},
};
const compiledLayerVersionString = {
Type: 'AWS::Lambda::LayerVersionPermission',
Properties: {
Action: 'lambda:GetLayerVersion',
LayerVersionArn: {
Ref: 'TestLambdaLayer',
},
Principal: '2222222',
},
};
return expect(awsCompileLayers.compileLayers()).to.be.fulfilled.then(() => {
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Resources
.TestLambdaLayer
).to.deep.equal(compiledLayer);
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Outputs
.TestLambdaLayerQualifiedArn
).to.deep.equal(compiledLayerOutput);
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Resources
.Test1111111LambdaLayerPermission
).to.deep.equal(compiledLayerVersionNumber);
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Resources
.Test2222222LambdaLayerPermission
).to.deep.equal(compiledLayerVersionString);
});
});
it('should create a layer resource with metadata options set', () => {
const s3Folder = awsCompileLayers.serverless.service.package.artifactDirectoryName;
const s3FileName = awsCompileLayers.serverless.service.layers.test.package.artifact
.split(path.sep)
.pop();
awsCompileLayers.serverless.service.layers = {
test: {
path: 'layer',
description: 'desc',
compatibleRuntimes: ['nodejs12.x'],
licenseInfo: 'GPL',
},
};
const compiledLayer = {
Type: 'AWS::Lambda::LayerVersion',
Properties: {
Content: {
S3Key: `${s3Folder}/${s3FileName}`,
S3Bucket: { Ref: 'ServerlessDeploymentBucket' },
},
LayerName: 'test',
Description: 'desc',
CompatibleRuntimes: ['nodejs12.x'],
LicenseInfo: 'GPL',
},
};
const compiledLayerOutput = {
Description: 'Current Lambda layer version',
Value: {
Ref: 'TestLambdaLayer',
},
};
return expect(awsCompileLayers.compileLayers()).to.be.fulfilled.then(() => {
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Resources
.TestLambdaLayer
).to.deep.equal(compiledLayer);
expect(
awsCompileLayers.serverless.service.provider.compiledCloudFormationTemplate.Outputs
.TestLambdaLayerQualifiedArn
).to.deep.equal(compiledLayerOutput);
});
});
});
});