mirror of
https://github.com/serverless/serverless.git
synced 2026-01-18 14:58:43 +00:00
315 lines
9.3 KiB
JavaScript
315 lines
9.3 KiB
JavaScript
'use strict';
|
|
|
|
const chai = require('chai');
|
|
const sinon = require('sinon');
|
|
const path = require('path');
|
|
const proxyquire = require('proxyquire');
|
|
const AwsProvider = require('../provider/awsProvider');
|
|
const Serverless = require('../../../Serverless');
|
|
const { getTmpDirPath } = require('../../../../tests/utils/fs');
|
|
|
|
chai.use(require('chai-as-promised'));
|
|
|
|
const expect = chai.expect;
|
|
|
|
describe('AwsInvoke', () => {
|
|
let AwsInvoke;
|
|
let awsInvoke;
|
|
let serverless;
|
|
let stdinStub;
|
|
const options = {
|
|
stage: 'dev',
|
|
region: 'us-east-1',
|
|
function: 'first',
|
|
};
|
|
|
|
beforeEach(() => {
|
|
stdinStub = sinon.stub().resolves('');
|
|
AwsInvoke = proxyquire('./index', {
|
|
'get-stdin': stdinStub,
|
|
});
|
|
serverless = new Serverless();
|
|
serverless.setProvider('aws', new AwsProvider(serverless, options));
|
|
awsInvoke = new AwsInvoke(serverless, options);
|
|
});
|
|
|
|
describe('#constructor()', () => {
|
|
it('should have hooks', () => expect(awsInvoke.hooks).to.be.not.empty);
|
|
|
|
it('should set the provider variable to an instance of AwsProvider', () =>
|
|
expect(awsInvoke.provider).to.be.instanceof(AwsProvider));
|
|
|
|
it('should run promise chain in order', () => {
|
|
const validateStub = sinon.stub(awsInvoke, 'extendedValidate').resolves();
|
|
const invokeStub = sinon.stub(awsInvoke, 'invoke').resolves();
|
|
const logStub = sinon.stub(awsInvoke, 'log').resolves();
|
|
|
|
return awsInvoke.hooks['invoke:invoke']().then(() => {
|
|
expect(validateStub.calledOnce).to.be.equal(true);
|
|
expect(invokeStub.calledAfter(validateStub)).to.be.equal(true);
|
|
expect(logStub.calledAfter(invokeStub)).to.be.equal(true);
|
|
|
|
awsInvoke.extendedValidate.restore();
|
|
awsInvoke.invoke.restore();
|
|
awsInvoke.log.restore();
|
|
});
|
|
});
|
|
|
|
it('should set an empty options object if no options are given', () => {
|
|
const awsInvokeWithEmptyOptions = new AwsInvoke(serverless);
|
|
|
|
expect(awsInvokeWithEmptyOptions.options).to.deep.equal({});
|
|
});
|
|
});
|
|
|
|
describe('#extendedValidate()', () => {
|
|
let backupIsTTY;
|
|
beforeEach(() => {
|
|
serverless.config.servicePath = true;
|
|
serverless.service.environment = {
|
|
vars: {},
|
|
stages: {
|
|
dev: {
|
|
vars: {},
|
|
regions: {
|
|
'us-east-1': {
|
|
vars: {},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
};
|
|
serverless.service.functions = {
|
|
first: {
|
|
handler: true,
|
|
},
|
|
};
|
|
awsInvoke.options.data = null;
|
|
awsInvoke.options.path = false;
|
|
|
|
// Ensure there's no attempt to read path from stdin
|
|
backupIsTTY = process.stdin.isTTY;
|
|
process.stdin.isTTY = true;
|
|
});
|
|
|
|
afterEach(() => {
|
|
if (backupIsTTY) process.stdin.isTTY = backupIsTTY;
|
|
delete process.stdin.isTTY;
|
|
});
|
|
|
|
it('it should throw error if function is not provided', () => {
|
|
serverless.service.functions = null;
|
|
return expect(awsInvoke.extendedValidate()).to.be.rejected;
|
|
});
|
|
|
|
it('should not throw error when there is no input data', () => {
|
|
awsInvoke.options.data = undefined;
|
|
|
|
return expect(awsInvoke.extendedValidate()).to.be.fulfilled.then(() => {
|
|
expect(awsInvoke.options.data).to.equal('');
|
|
});
|
|
});
|
|
|
|
it('should keep data if it is a simple string', () => {
|
|
awsInvoke.options.data = 'simple-string';
|
|
|
|
return expect(awsInvoke.extendedValidate()).to.be.fulfilled.then(() => {
|
|
expect(awsInvoke.options.data).to.equal('simple-string');
|
|
});
|
|
});
|
|
|
|
it('should parse data if it is a json string', () => {
|
|
awsInvoke.options.data = '{"key": "value"}';
|
|
|
|
return expect(awsInvoke.extendedValidate()).to.be.fulfilled.then(() => {
|
|
expect(awsInvoke.options.data).to.deep.equal({ key: 'value' });
|
|
});
|
|
});
|
|
|
|
it('should skip parsing data if "raw" requested', () => {
|
|
awsInvoke.options.data = '{"key": "value"}';
|
|
awsInvoke.options.raw = true;
|
|
|
|
return expect(awsInvoke.extendedValidate()).to.be.fulfilled.then(() => {
|
|
expect(awsInvoke.options.data).to.deep.equal('{"key": "value"}');
|
|
});
|
|
});
|
|
|
|
it('it should parse file if relative file path is provided', () => {
|
|
serverless.config.servicePath = getTmpDirPath();
|
|
const data = {
|
|
testProp: 'testValue',
|
|
};
|
|
serverless.utils.writeFileSync(
|
|
path.join(serverless.config.servicePath, 'data.json'),
|
|
JSON.stringify(data)
|
|
);
|
|
awsInvoke.options.path = 'data.json';
|
|
|
|
return expect(awsInvoke.extendedValidate()).to.be.fulfilled.then(() => {
|
|
expect(awsInvoke.options.data).to.deep.equal(data);
|
|
});
|
|
});
|
|
|
|
it('it should parse file if absolute file path is provided', () => {
|
|
serverless.config.servicePath = getTmpDirPath();
|
|
const data = {
|
|
testProp: 'testValue',
|
|
};
|
|
const dataFile = path.join(serverless.config.servicePath, 'data.json');
|
|
serverless.utils.writeFileSync(dataFile, JSON.stringify(data));
|
|
awsInvoke.options.path = dataFile;
|
|
|
|
return expect(awsInvoke.extendedValidate()).to.be.fulfilled.then(() => {
|
|
expect(awsInvoke.options.data).to.deep.equal(data);
|
|
});
|
|
});
|
|
|
|
it('it should parse a yaml file if file path is provided', () => {
|
|
serverless.config.servicePath = getTmpDirPath();
|
|
const yamlContent = 'testProp: testValue';
|
|
|
|
serverless.utils.writeFileSync(
|
|
path.join(serverless.config.servicePath, 'data.yml'),
|
|
yamlContent
|
|
);
|
|
awsInvoke.options.path = 'data.yml';
|
|
|
|
return expect(awsInvoke.extendedValidate()).to.be.fulfilled.then(() => {
|
|
expect(awsInvoke.options.data).to.deep.equal({
|
|
testProp: 'testValue',
|
|
});
|
|
});
|
|
});
|
|
|
|
it('it should throw error if service path is not set', () => {
|
|
serverless.config.servicePath = false;
|
|
return expect(awsInvoke.extendedValidate()).to.be.rejected;
|
|
});
|
|
|
|
it('it should throw error if file path does not exist', () => {
|
|
serverless.config.servicePath = getTmpDirPath();
|
|
awsInvoke.options.path = 'some/path';
|
|
|
|
return expect(awsInvoke.extendedValidate()).to.be.rejectedWith(
|
|
'The file you provided does not exist.'
|
|
);
|
|
});
|
|
|
|
it('should resolve if path is not given', () => {
|
|
awsInvoke.options.path = false;
|
|
return expect(awsInvoke.extendedValidate()).to.be.fulfilled;
|
|
});
|
|
});
|
|
|
|
describe('#invoke()', () => {
|
|
let invokeStub;
|
|
beforeEach(() => {
|
|
invokeStub = sinon.stub(awsInvoke.provider, 'request').resolves();
|
|
awsInvoke.serverless.service.service = 'new-service';
|
|
awsInvoke.options = {
|
|
stage: 'dev',
|
|
function: 'first',
|
|
functionObj: {
|
|
name: 'customName',
|
|
},
|
|
};
|
|
});
|
|
|
|
it('should invoke with correct params', () =>
|
|
awsInvoke.invoke().then(() => {
|
|
expect(invokeStub.calledOnce).to.be.equal(true);
|
|
expect(
|
|
invokeStub.calledWithExactly('Lambda', 'invoke', {
|
|
FunctionName: 'customName',
|
|
InvocationType: 'RequestResponse',
|
|
LogType: 'None',
|
|
Payload: new Buffer(JSON.stringify({})),
|
|
})
|
|
).to.be.equal(true);
|
|
awsInvoke.provider.request.restore();
|
|
}));
|
|
|
|
it('should invoke and log', () => {
|
|
awsInvoke.options.log = true;
|
|
|
|
return awsInvoke.invoke().then(() => {
|
|
expect(invokeStub.calledOnce).to.be.equal(true);
|
|
expect(
|
|
invokeStub.calledWithExactly('Lambda', 'invoke', {
|
|
FunctionName: 'customName',
|
|
InvocationType: 'RequestResponse',
|
|
LogType: 'Tail',
|
|
Payload: new Buffer(JSON.stringify({})),
|
|
})
|
|
).to.be.equal(true);
|
|
awsInvoke.provider.request.restore();
|
|
});
|
|
});
|
|
|
|
it('should invoke with other invocation type', () => {
|
|
awsInvoke.options.type = 'OtherType';
|
|
|
|
return awsInvoke.invoke().then(() => {
|
|
expect(invokeStub.calledOnce).to.be.equal(true);
|
|
expect(
|
|
invokeStub.calledWithExactly('Lambda', 'invoke', {
|
|
FunctionName: 'customName',
|
|
InvocationType: 'OtherType',
|
|
LogType: 'None',
|
|
Payload: new Buffer(JSON.stringify({})),
|
|
})
|
|
).to.be.equal(true);
|
|
awsInvoke.provider.request.restore();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('#log()', () => {
|
|
let consoleLogStub;
|
|
|
|
beforeEach(() => {
|
|
consoleLogStub = sinon.stub(awsInvoke, 'consoleLog');
|
|
});
|
|
|
|
afterEach(() => {
|
|
awsInvoke.consoleLog.restore();
|
|
});
|
|
|
|
it('should log payload', () => {
|
|
const invocationReplyMock = {
|
|
Payload: `
|
|
{
|
|
"testProp": "testValue"
|
|
}
|
|
`,
|
|
LogResult: 'test',
|
|
};
|
|
|
|
return awsInvoke.log(invocationReplyMock).then(() => {
|
|
const expectedPayloadMessage = '{\n "testProp": "testValue"\n}';
|
|
|
|
expect(consoleLogStub.calledWith(expectedPayloadMessage)).to.equal(true);
|
|
});
|
|
});
|
|
|
|
it('rejects the promise for failed invocations', () => {
|
|
const invocationReplyMock = {
|
|
Payload: `
|
|
{
|
|
"testProp": "testValue"
|
|
}
|
|
`,
|
|
LogResult: 'test',
|
|
FunctionError: true,
|
|
};
|
|
|
|
return awsInvoke.log(invocationReplyMock).catch(err => {
|
|
expect(err)
|
|
.to.and.be.instanceof(Error)
|
|
.and.have.property('message', 'Invoked function failed');
|
|
});
|
|
});
|
|
});
|
|
});
|