serverless/lib/Serverless.test.js
Mariusz Nowak 4944f471b1
Prettify
2019-06-26 12:43:01 +02:00

291 lines
9.9 KiB
JavaScript

'use strict';
const expect = require('chai').expect;
const Serverless = require('./Serverless');
const semverRegex = require('semver-regex');
const path = require('path');
const YAML = require('js-yaml');
const sinon = require('sinon');
const BbPromise = require('bluebird');
const YamlParser = require('../lib/classes/YamlParser');
const PluginManager = require('../lib/classes/PluginManager');
const Utils = require('../lib/classes/Utils');
const Service = require('../lib/classes/Service');
const CLI = require('../lib/classes/CLI');
const { ServerlessError } = require('../lib/classes/Error');
const { getTmpDirPath } = require('../tests/utils/fs');
describe('Serverless', () => {
let serverless;
beforeEach(() => {
serverless = new Serverless();
});
describe('#constructor()', () => {
it('should set the correct config if a config object is passed', () => {
const configObj = { some: 'config' };
const serverlessWithConfig = new Serverless(configObj);
expect(serverlessWithConfig.config.some).to.equal('config');
});
it('should set an empty config object if no config object passed', () => {
// we're only expecting to have "serverless", "serverlessPath" and "servicePath"
expect(Object.keys(serverless.config).length).to.equal(3);
expect(Object.keys(serverless.config)).to.include('serverless');
expect(Object.keys(serverless.config)).to.include('serverlessPath');
expect(Object.keys(serverless.config)).to.include('servicePath');
});
it('should set an empty providers object', () => {
expect(serverless.providers).to.deep.equal({});
});
it('should set the Serverless version', () => {
expect(serverless.version.length).to.be.at.least(1);
});
it('should set the YamlParser class instance', () => {
expect(serverless.yamlParser).to.be.instanceof(YamlParser);
});
it('should set the PluginManager class instance', () => {
expect(serverless.pluginManager).to.be.instanceof(PluginManager);
});
it('should set the Utils class instance', () => {
expect(serverless.utils).to.be.instanceof(Utils);
});
it('should set the Service class instance', () => {
expect(serverless.service).to.be.instanceof(Service);
});
it('should set the servicePath property if it was set in the config object', () => {
const configObj = { servicePath: 'some/path' };
const serverlessWithConfig = new Serverless(configObj);
expect(serverlessWithConfig.config.servicePath).to.equal('some/path');
});
// note: we only test if the property is there
// the test if the correct servicePath is set is done in the Utils class test file
it('should set the servicePath property if no config object is given', () => {
expect(serverless.config.servicePath).to.not.equal(undefined);
});
it('should have a config object', () => {
expect(serverless.config).to.not.equal(undefined);
});
it('should have a classes object', () => {
expect(serverless.classes).to.not.equal(undefined);
});
it('should store the CLI class inside the classes object', () => {
expect(serverless.classes.CLI).to.deep.equal(CLI);
});
it('should store the YamlParser class inside the classes object', () => {
expect(serverless.classes.YamlParser).to.deep.equal(YamlParser);
});
it('should store the PluginManager class inside the classes object', () => {
expect(serverless.classes.PluginManager).to.deep.equal(PluginManager);
});
it('should store the Utils class inside the classes object', () => {
expect(serverless.classes.Utils).to.deep.equal(Utils);
});
it('should store the Service class inside the classes object', () => {
expect(serverless.classes.Service).to.deep.equal(Service);
});
it('should store the Error class inside the classes object', () => {
expect(serverless.classes.Error).to.deep.equal(ServerlessError);
});
});
describe('#init()', () => {
let loadAllPluginsStub;
let asyncPluginInitStub;
let updateAutocompleteCacheFileStub;
beforeEach(() => {
loadAllPluginsStub = sinon.stub(serverless.pluginManager, 'loadAllPlugins').returns();
asyncPluginInitStub = sinon
.stub(serverless.pluginManager, 'asyncPluginInit')
.returns(BbPromise.resolve());
updateAutocompleteCacheFileStub = sinon
.stub(serverless.pluginManager, 'updateAutocompleteCacheFile')
.resolves();
});
afterEach(() => {
serverless.pluginManager.asyncPluginInit.restore();
serverless.pluginManager.loadAllPlugins.restore();
serverless.pluginManager.updateAutocompleteCacheFile.restore();
});
it('should set an instanceId', () =>
serverless.init().then(() => {
expect(serverless.instanceId).to.match(/\d/);
}));
it('should create a new CLI instance', () =>
serverless.init().then(() => {
expect(serverless.cli).to.be.instanceof(CLI);
}));
it('should allow a custom CLI instance', () => {
class CustomCLI extends CLI {}
serverless.classes.CLI = CustomCLI;
return serverless.init().then(() => {
expect(serverless.cli).to.be.instanceof(CLI);
expect(serverless.cli.constructor.name).to.equal('CustomCLI');
});
});
// note: we just test that the processedInput variable is set (not the content of it)
// the test for the correct input is done in the CLI class test file
it('should receive the processed input form the CLI instance', () =>
serverless.init().then(() => {
expect(serverless.processedInput).to.not.deep.equal({});
}));
it('should resolve after loading the service', () => {
const SUtils = new Utils();
const tmpDirPath = getTmpDirPath();
const serverlessYml = {
service: 'new-service',
provider: 'aws',
custom: {},
plugins: ['testPlugin'],
functions: {
functionA: {},
},
resources: {},
package: {
exclude: ['exclude-me'],
include: ['include-me'],
artifact: 'some/path/foo.zip',
},
};
SUtils.writeFileSync(path.join(tmpDirPath, 'serverless.yml'), YAML.dump(serverlessYml));
serverless.config.update({ servicePath: tmpDirPath });
serverless.pluginManager.cliOptions = {
stage: 'dev',
};
return serverless.init().then(() => {
expect(loadAllPluginsStub.calledOnce).to.equal(true);
expect(asyncPluginInitStub.calledOnce).to.equal(true);
expect(updateAutocompleteCacheFileStub.calledOnce).to.equal(true);
});
});
});
describe('#run()', () => {
let logStatStub;
let displayHelpStub;
let validateCommandStub;
let populateServiceStub;
let runStub;
beforeEach(() => {
serverless.cli = new CLI(serverless);
serverless.processedInput = { commands: [], options: {} };
// setup default stubs
logStatStub = sinon.stub(serverless.utils, 'logStat').resolves();
displayHelpStub = sinon.stub(serverless.cli, 'displayHelp').returns(false);
validateCommandStub = sinon.stub(serverless.pluginManager, 'validateCommand').returns();
populateServiceStub = sinon.stub(serverless.variables, 'populateService').resolves();
runStub = sinon.stub(serverless.pluginManager, 'run').resolves();
});
afterEach(() => {
serverless.utils.logStat.restore();
serverless.cli.displayHelp.restore();
serverless.pluginManager.validateCommand.restore();
serverless.variables.populateService.restore();
serverless.pluginManager.run.restore();
});
it('should resolve if the stats logging call throws an error / is rejected', () => {
// overwrite logStatStub
logStatStub.returns(BbPromise.reject());
return serverless.run().then(() => {
expect(logStatStub.calledOnce).to.equal(true);
expect(displayHelpStub.calledOnce).to.equal(true);
expect(validateCommandStub.calledOnce).to.equal(true);
expect(populateServiceStub.calledOnce).to.equal(true);
expect(runStub.calledOnce).to.equal(true);
return BbPromise.resolve();
});
});
it('should resolve if help is displayed or no commands are entered', () => {
// overwrite displayHelpStub
displayHelpStub.returns(true);
return serverless.run().then(() => {
expect(logStatStub.calledOnce).to.equal(true);
expect(displayHelpStub.calledOnce).to.equal(true);
expect(validateCommandStub.calledOnce).to.equal(false);
expect(populateServiceStub.calledOnce).to.equal(false);
expect(runStub.calledOnce).to.equal(false);
});
});
it('should run all the needed functions', () =>
serverless.run().then(() => {
expect(logStatStub.calledOnce).to.equal(true);
expect(displayHelpStub.calledOnce).to.equal(true);
expect(validateCommandStub.calledOnce).to.equal(true);
expect(populateServiceStub.calledOnce).to.equal(true);
expect(runStub.calledOnce).to.equal(true);
}));
});
describe('#setProvider()', () => {
class ProviderMock {}
it('should set the provider object in the provider object', () => {
const myProvider = new ProviderMock();
serverless.setProvider('myProvider', myProvider);
expect(serverless.providers.myProvider).to.equal(myProvider);
});
});
describe('#getProvider()', () => {
class ProviderMock {}
let myProvider;
beforeEach(() => {
myProvider = new ProviderMock();
serverless.setProvider('myProvider', myProvider);
});
it('should return the provider object', () => {
const retrivedProvider = serverless.getProvider('myProvider');
expect(retrivedProvider).to.deep.equal(myProvider);
});
});
describe('#getVersion()', () => {
it('should get the correct Serverless version', () => {
expect(semverRegex().test(serverless.getVersion())).to.equal(true);
});
});
});