'use strict'; const expect = require('chai').expect; const sinon = require('sinon'); const path = require('path'); const fs = require('fs'); const proxyquire = require('proxyquire'); const AwsProvider = require('../provider/awsProvider'); const Serverless = require('../../../Serverless'); const testUtils = require('../../../../tests/utils'); describe('AwsDeployFunction', () => { let AwsDeployFunction; let serverless; let awsDeployFunction; let cryptoStub; beforeEach(() => { serverless = new Serverless(); serverless.servicePath = true; serverless.service.environment = { vars: {}, stages: { dev: { vars: {}, regions: { 'us-east-1': { vars: {}, }, }, }, }, }; serverless.service.functions = { first: { handler: true, }, }; const options = { stage: 'dev', region: 'us-east-1', function: 'first', functionObj: { name: 'first', }, }; serverless.init(); serverless.setProvider('aws', new AwsProvider(serverless)); cryptoStub = { createHash: function () { return this; }, // eslint-disable-line update: function () { return this; }, // eslint-disable-line digest: sinon.stub(), }; AwsDeployFunction = proxyquire('./index.js', { crypto: cryptoStub, }); awsDeployFunction = new AwsDeployFunction(serverless, options); }); describe('#constructor()', () => { it('should have hooks', () => expect(awsDeployFunction.hooks).to.be.not.empty); it('should set the provider variable to an instance of AwsProvider', () => expect(awsDeployFunction.provider).to.be.instanceof(AwsProvider)); it('should set an empty options object if no options are given', () => { const awsDeployFunctionWithEmptyOptions = new AwsDeployFunction(serverless); expect(awsDeployFunctionWithEmptyOptions.options).to.deep.equal({}); }); }); describe('#checkIfFunctionExists()', () => { let getFunctionStub; beforeEach(() => { getFunctionStub = sinon .stub(awsDeployFunction.provider, 'request') .resolves({ func: { name: 'first' } }); }); afterEach(() => { awsDeployFunction.provider.request.restore(); }); it('it should throw error if function is not provided', () => { serverless.service.functions = null; expect(() => awsDeployFunction.checkIfFunctionExists()).to.throw(Error); }); it('should check if the function is deployed and save the result', () => { awsDeployFunction.serverless.service.functions = { first: { name: 'first', handler: 'handler.first', }, }; return awsDeployFunction.checkIfFunctionExists().then(() => { expect(getFunctionStub.calledOnce).to.be.equal(true); expect(getFunctionStub.calledWithExactly( 'Lambda', 'getFunction', { FunctionName: 'first', }, awsDeployFunction.options.stage, awsDeployFunction.options.region )).to.be.equal(true); expect(awsDeployFunction.serverless.service.provider.remoteFunctionData).to.deep.equal({ func: { name: 'first', }, }); }); }); }); describe('#deployFunction()', () => { let artifactFilePath; let updateFunctionCodeStub; let statSyncStub; let readFileSyncStub; beforeEach(() => { // write a file to disc to simulate that the deployment artifact exists awsDeployFunction.packagePath = testUtils.getTmpDirPath(); artifactFilePath = path.join(awsDeployFunction.packagePath, 'first.zip'); serverless.utils.writeFileSync(artifactFilePath, 'first.zip file content'); updateFunctionCodeStub = sinon .stub(awsDeployFunction.provider, 'request') .resolves(); statSyncStub = sinon .stub(fs, 'statSync') .returns({ size: 1024 }); sinon.spy(awsDeployFunction.serverless.cli, 'log'); readFileSyncStub = sinon .stub(fs, 'readFileSync') .returns(); awsDeployFunction.serverless.service.provider.remoteFunctionData = { Configuration: { CodeSha256: 'remote-hash-zip-file', }, }; }); afterEach(() => { awsDeployFunction.provider.request.restore(); fs.statSync.restore(); fs.readFileSync.restore(); }); it('should deploy the function if the hashes are different', () => { cryptoStub.createHash().update().digest.onCall(0).returns('local-hash-zip-file'); return awsDeployFunction.deployFunction().then(() => { const data = fs.readFileSync(artifactFilePath); expect(updateFunctionCodeStub.calledOnce).to.be.equal(true); expect(readFileSyncStub.called).to.equal(true); expect(updateFunctionCodeStub.calledWithExactly( 'Lambda', 'updateFunctionCode', { FunctionName: 'first', ZipFile: data, }, awsDeployFunction.options.stage, awsDeployFunction.options.region )).to.be.equal(true); expect(readFileSyncStub.calledWithExactly(artifactFilePath)).to.equal(true); }); }); it('should deploy the function if the hashes are same but the "force" option is used', () => { awsDeployFunction.options.force = true; cryptoStub.createHash().update().digest.onCall(0).returns('remote-hash-zip-file'); return awsDeployFunction.deployFunction().then(() => { const data = fs.readFileSync(artifactFilePath); expect(updateFunctionCodeStub.calledOnce).to.be.equal(true); expect(readFileSyncStub.called).to.equal(true); expect(updateFunctionCodeStub.calledWithExactly( 'Lambda', 'updateFunctionCode', { FunctionName: 'first', ZipFile: data, }, awsDeployFunction.options.stage, awsDeployFunction.options.region )).to.be.equal(true); expect(readFileSyncStub.calledWithExactly(artifactFilePath)).to.equal(true); }); }); it('should resolve if the hashes are the same', () => { cryptoStub.createHash().update().digest.onCall(0).returns('remote-hash-zip-file'); return awsDeployFunction.deployFunction().then(() => { const expected = 'Code not changed. Skipping function deployment.'; expect(updateFunctionCodeStub.calledOnce).to.be.equal(false); expect(readFileSyncStub.calledOnce).to.equal(true); expect(awsDeployFunction.serverless.cli.log.calledWithExactly(expected)).to.equal(true); expect(readFileSyncStub.calledWithExactly(artifactFilePath)).to.equal(true); }); }); it('should log artifact size', () => { // awnY7Oi280gp5kTCloXzsqJCO4J766x6hATWqQsN/uM= <-- hash of the local zip file readFileSyncStub.returns(new Buffer('my-service.zip content')); return awsDeployFunction.deployFunction().then(() => { const expected = 'Uploading function: first (1 KB)...'; expect(readFileSyncStub.calledOnce).to.equal(true); expect(statSyncStub.calledOnce).to.equal(true); expect(awsDeployFunction.serverless.cli.log.calledWithExactly(expected)).to.be.equal(true); expect(readFileSyncStub.calledWithExactly(artifactFilePath)).to.equal(true); }); }); describe('when artifact is provided', () => { let getFunctionStub; const artifactZipFile = 'artifact.zip'; beforeEach(() => { getFunctionStub = sinon.stub(serverless.service, 'getFunction').returns({ handler: true, package: { artifact: artifactZipFile, }, }); }); afterEach(() => { serverless.service.getFunction.restore(); }); it('should read the provided artifact', () => awsDeployFunction.deployFunction().then(() => { const data = fs.readFileSync(artifactZipFile); expect(readFileSyncStub).to.have.been.calledWithExactly(artifactZipFile); expect(statSyncStub).to.have.been.calledWithExactly(artifactZipFile); expect(getFunctionStub).to.have.been.calledWithExactly('first'); expect(updateFunctionCodeStub.calledOnce).to.equal(true); expect(updateFunctionCodeStub.calledWithExactly( 'Lambda', 'updateFunctionCode', { FunctionName: 'first', ZipFile: data, }, awsDeployFunction.options.stage, awsDeployFunction.options.region )).to.be.equal(true); })); }); }); });