/*globals describe, it, before, beforeEach, afterEach, after */ /*jshint expr:true*/ var should = require('should'), sinon = require('sinon'), Promise = require('bluebird'), path = require('path'), fs = require('fs'), _ = require('lodash'), rewire = require('rewire'), testUtils = require('../utils'), // Thing we are testing defaultConfig = require('../../../config.example')[process.env.NODE_ENV], config = require('../../server/config'), origConfig = _.cloneDeep(config), // storing current environment currentEnv = process.env.NODE_ENV; // To stop jshint complaining should.equal(true, true); function resetConfig() { config.set(_.merge({}, origConfig, defaultConfig)); } describe('Config', function () { after(function () { resetConfig(); }); describe('Theme', function () { beforeEach(function () { config.set({ url: 'http://my-ghost-blog.com', theme: { title: 'casper', description: 'casper', logo: 'casper', cover: 'casper' } }); }); afterEach(function () { resetConfig(); }); it('should have exactly the right keys', function () { var themeConfig = config.theme; // This will fail if there are any extra keys themeConfig.should.have.keys('url', 'title', 'description', 'logo', 'cover'); }); it('should have the correct values for each key', function () { var themeConfig = config.theme; // Check values are as we expect themeConfig.should.have.property('url', 'http://my-ghost-blog.com'); themeConfig.should.have.property('title', 'casper'); themeConfig.should.have.property('description', 'casper'); themeConfig.should.have.property('logo', 'casper'); themeConfig.should.have.property('cover', 'casper'); }); }); describe('Index', function () { afterEach(function () { // Make a copy of the default config file // so we can restore it after every test. // Using _.merge to recursively apply every property. resetConfig(); }); it('should have exactly the right keys', function () { var pathConfig = config.paths; // This will fail if there are any extra keys pathConfig.should.have.keys( 'appRoot', 'subdir', 'config', 'configExample', 'storage', 'contentPath', 'corePath', 'themePath', 'appPath', 'imagesPath', 'imagesRelPath', 'adminViews', 'helperTemplates', 'exportPath', 'lang', 'availableThemes', 'availableApps', 'clientAssets' ); }); it('should have the correct values for each key', function () { var pathConfig = config.paths, appRoot = path.resolve(__dirname, '../../../'); pathConfig.should.have.property('appRoot', appRoot); pathConfig.should.have.property('subdir', ''); }); it('should not return a slash for subdir', function () { config.set({url: 'http://my-ghost-blog.com'}); config.paths.should.have.property('subdir', ''); config.set({url: 'http://my-ghost-blog.com/'}); config.paths.should.have.property('subdir', ''); }); it('should handle subdirectories properly', function () { config.set({url: 'http://my-ghost-blog.com/blog'}); config.paths.should.have.property('subdir', '/blog'); config.set({url: 'http://my-ghost-blog.com/blog/'}); config.paths.should.have.property('subdir', '/blog'); config.set({url: 'http://my-ghost-blog.com/my/blog'}); config.paths.should.have.property('subdir', '/my/blog'); config.set({url: 'http://my-ghost-blog.com/my/blog/'}); config.paths.should.have.property('subdir', '/my/blog'); }); it('should allow specific properties to be user defined', function () { var contentPath = path.join(config.paths.appRoot, 'otherContent', '/'), configFile = 'configFileDanceParty.js'; config.set({ config: configFile, paths: { contentPath: contentPath } }); config.should.have.property('config', configFile); config.paths.should.have.property('contentPath', contentPath); config.paths.should.have.property('themePath', contentPath + 'themes'); config.paths.should.have.property('appPath', contentPath + 'apps'); config.paths.should.have.property('imagesPath', contentPath + 'images'); }); }); describe('Storage', function () { afterEach(function () { resetConfig(); }); it('should default to local-file-store', function () { var storagePath = path.join(config.paths.corePath, '/server/storage/', 'local-file-store'); config.paths.should.have.property('storage', storagePath); config.storage.should.have.property('active', 'local-file-store'); }); it('should allow setting a custom active storage', function () { var storagePath = path.join(config.paths.contentPath, 'storage', 's3'); config.set({ storage: { active: 's3', s3: {} } }); config.paths.should.have.property('storage', storagePath); config.storage.should.have.property('active', 's3'); config.storage.should.have.property('s3', {}); }); }); describe('Url', function () { describe('urlJoin', function () { before(function () { resetConfig(); }); afterEach(function () { resetConfig(); }); it('should deduplicate slashes', function () { config.set({url: 'http://my-ghost-blog.com/'}); config.urlJoin('/', '/my/', '/blog/').should.equal('/my/blog/'); config.urlJoin('/', '//my/', '/blog/').should.equal('/my/blog/'); config.urlJoin('/', '/', '/').should.equal('/'); }); it('should not deduplicate slashes in protocol', function () { config.set({url: 'http://my-ghost-blog.com/'}); config.urlJoin('http://myurl.com', '/rss').should.equal('http://myurl.com/rss'); config.urlJoin('https://myurl.com/', '/rss').should.equal('https://myurl.com/rss'); }); it('should permit schemeless protocol', function () { config.set({url: 'http://my-ghost-blog.com/'}); config.urlJoin('/', '/').should.equal('/'); config.urlJoin('//myurl.com', '/rss').should.equal('//myurl.com/rss'); config.urlJoin('//myurl.com/', '/rss').should.equal('//myurl.com/rss'); config.urlJoin('//myurl.com//', 'rss').should.equal('//myurl.com/rss'); config.urlJoin('', '//myurl.com', 'rss').should.equal('//myurl.com/rss'); }); it('should deduplicate subdir', function () { config.set({url: 'http://my-ghost-blog.com/blog'}); config.urlJoin('blog', 'blog/about').should.equal('blog/about'); config.urlJoin('blog/', 'blog/about').should.equal('blog/about'); }); }); describe('urlFor', function () { before(function () { resetConfig(); }); afterEach(function () { resetConfig(); }); it('should return the home url with no options', function () { config.urlFor().should.equal('/'); config.set({url: 'http://my-ghost-blog.com/blog'}); config.urlFor().should.equal('/blog/'); config.set({url: 'http://my-ghost-blog.com/blog/'}); config.urlFor().should.equal('/blog/'); }); it('should return home url when asked for', function () { var testContext = 'home'; config.set({url: 'http://my-ghost-blog.com'}); config.urlFor(testContext).should.equal('/'); config.urlFor(testContext, true).should.equal('http://my-ghost-blog.com/'); config.set({url: 'http://my-ghost-blog.com/'}); config.urlFor(testContext).should.equal('/'); config.urlFor(testContext, true).should.equal('http://my-ghost-blog.com/'); config.set({url: 'http://my-ghost-blog.com/blog'}); config.urlFor(testContext).should.equal('/blog/'); config.urlFor(testContext, true).should.equal('http://my-ghost-blog.com/blog/'); config.set({url: 'http://my-ghost-blog.com/blog/'}); config.urlFor(testContext).should.equal('/blog/'); config.urlFor(testContext, true).should.equal('http://my-ghost-blog.com/blog/'); }); it('should return rss url when asked for', function () { var testContext = 'rss'; config.set({url: 'http://my-ghost-blog.com'}); config.urlFor(testContext).should.equal('/rss/'); config.urlFor(testContext, true).should.equal('http://my-ghost-blog.com/rss/'); config.set({url: 'http://my-ghost-blog.com/blog'}); config.urlFor(testContext).should.equal('/blog/rss/'); config.urlFor(testContext, true).should.equal('http://my-ghost-blog.com/blog/rss/'); }); it('should return url for a random path when asked for', function () { var testContext = {relativeUrl: '/about/'}; config.set({url: 'http://my-ghost-blog.com'}); config.urlFor(testContext).should.equal('/about/'); config.urlFor(testContext, true).should.equal('http://my-ghost-blog.com/about/'); config.set({url: 'http://my-ghost-blog.com/blog'}); config.urlFor(testContext).should.equal('/blog/about/'); config.urlFor(testContext, true).should.equal('http://my-ghost-blog.com/blog/about/'); }); it('should deduplicate subdirectories in paths', function () { var testContext = {relativeUrl: '/blog/about/'}; config.set({url: 'http://my-ghost-blog.com'}); config.urlFor(testContext).should.equal('/blog/about/'); config.urlFor(testContext, true).should.equal('http://my-ghost-blog.com/blog/about/'); config.set({url: 'http://my-ghost-blog.com/blog'}); config.urlFor(testContext).should.equal('/blog/about/'); config.urlFor(testContext, true).should.equal('http://my-ghost-blog.com/blog/about/'); config.set({url: 'http://my-ghost-blog.com/blog/'}); config.urlFor(testContext).should.equal('/blog/about/'); config.urlFor(testContext, true).should.equal('http://my-ghost-blog.com/blog/about/'); }); it('should return url for a post from post object', function () { var testContext = 'post', testData = {post: testUtils.DataGenerator.Content.posts[2]}; // url is now provided on the postmodel, permalinkSetting tests are in the model_post_spec.js test testData.post.url = '/short-and-sweet/'; config.set({url: 'http://my-ghost-blog.com'}); config.urlFor(testContext, testData).should.equal('/short-and-sweet/'); config.urlFor(testContext, testData, true).should.equal('http://my-ghost-blog.com/short-and-sweet/'); config.set({url: 'http://my-ghost-blog.com/blog'}); config.urlFor(testContext, testData).should.equal('/blog/short-and-sweet/'); config.urlFor(testContext, testData, true).should.equal('http://my-ghost-blog.com/blog/short-and-sweet/'); }); it('should return url for a tag when asked for', function () { var testContext = 'tag', testData = {tag: testUtils.DataGenerator.Content.tags[0]}; config.set({url: 'http://my-ghost-blog.com'}); config.urlFor(testContext, testData).should.equal('/tag/kitchen-sink/'); config.urlFor(testContext, testData, true).should.equal('http://my-ghost-blog.com/tag/kitchen-sink/'); config.set({url: 'http://my-ghost-blog.com/blog'}); config.urlFor(testContext, testData).should.equal('/blog/tag/kitchen-sink/'); config.urlFor(testContext, testData, true).should.equal('http://my-ghost-blog.com/blog/tag/kitchen-sink/'); }); it('should return url for an author when asked for', function () { var testContext = 'author', testData = {author: testUtils.DataGenerator.Content.users[0]}; config.set({url: 'http://my-ghost-blog.com'}); config.urlFor(testContext, testData).should.equal('/author/joe-bloggs/'); config.urlFor(testContext, testData, true).should.equal('http://my-ghost-blog.com/author/joe-bloggs/'); config.set({url: 'http://my-ghost-blog.com/blog'}); config.urlFor(testContext, testData).should.equal('/blog/author/joe-bloggs/'); config.urlFor(testContext, testData, true).should.equal('http://my-ghost-blog.com/blog/author/joe-bloggs/'); }); it('should return url for an image when asked for', function () { var testContext = 'image', testData; config.set({url: 'http://my-ghost-blog.com'}); testData = {image: '/content/images/my-image.jpg'}; config.urlFor(testContext, testData).should.equal('/content/images/my-image.jpg'); config.urlFor(testContext, testData, true).should.equal('http://my-ghost-blog.com/content/images/my-image.jpg'); testData = {image: 'http://placekitten.com/500/200'}; config.urlFor(testContext, testData).should.equal('http://placekitten.com/500/200'); config.urlFor(testContext, testData, true).should.equal('http://placekitten.com/500/200'); testData = {image: '/blog/content/images/my-image2.jpg'}; config.urlFor(testContext, testData).should.equal('/blog/content/images/my-image2.jpg'); // We don't make image urls absolute if they don't look like images relative to the image path config.urlFor(testContext, testData, true).should.equal('/blog/content/images/my-image2.jpg'); config.set({url: 'http://my-ghost-blog.com/blog/'}); testData = {image: '/content/images/my-image3.jpg'}; config.urlFor(testContext, testData).should.equal('/content/images/my-image3.jpg'); // We don't make image urls absolute if they don't look like images relative to the image path config.urlFor(testContext, testData, true).should.equal('/content/images/my-image3.jpg'); testData = {image: '/blog/content/images/my-image4.jpg'}; config.urlFor(testContext, testData).should.equal('/blog/content/images/my-image4.jpg'); config.urlFor(testContext, testData, true).should.equal('http://my-ghost-blog.com/blog/content/images/my-image4.jpg'); }); it('should return a url for a nav item when asked for it', function () { var testContext = 'nav', testData; config.set({url: 'http://my-ghost-blog.com', urlSSL: 'https://my-ghost-blog.com'}); testData = {nav: {url: 'http://my-ghost-blog.com/short-and-sweet/'}}; config.urlFor(testContext, testData).should.equal('http://my-ghost-blog.com/short-and-sweet/'); testData = {nav: {url: 'http://my-ghost-blog.com/short-and-sweet/'}, secure: true}; config.urlFor(testContext, testData).should.equal('https://my-ghost-blog.com/short-and-sweet/'); testData = {nav: {url: 'http://sub.my-ghost-blog.com/'}}; config.urlFor(testContext, testData).should.equal('http://sub.my-ghost-blog.com/'); config.set({url: 'http://my-ghost-blog.com/blog'}); testData = {nav: {url: 'http://my-ghost-blog.com/blog/short-and-sweet/'}}; config.urlFor(testContext, testData).should.equal('http://my-ghost-blog.com/blog/short-and-sweet/'); config.set({url: 'http://my-ghost-blog.com/'}); testData = {nav: {url: 'mailto:marshmallow@my-ghost-blog.com'}}; config.urlFor(testContext, testData).should.equal('mailto:marshmallow@my-ghost-blog.com'); }); it('should return other known paths when requested', function () { config.set({url: 'http://my-ghost-blog.com'}); config.urlFor('sitemap_xsl').should.equal('/sitemap.xsl'); config.urlFor('sitemap_xsl', true).should.equal('http://my-ghost-blog.com/sitemap.xsl'); config.urlFor('api').should.equal('/ghost/api/v0.1'); config.urlFor('api', true).should.equal('http://my-ghost-blog.com/ghost/api/v0.1'); }); }); describe('urlPathForPost', function () { it('should output correct url for post', function () { var permalinkSetting = '/:slug/', /*jshint unused:false*/ testData = testUtils.DataGenerator.Content.posts[2], postLink = '/short-and-sweet/'; // next test config.urlPathForPost(testData, permalinkSetting).should.equal(postLink); }); it('should output correct url for post with date permalink', function () { var permalinkSetting = '/:year/:month/:day/:slug/', /*jshint unused:false*/ testData = testUtils.DataGenerator.Content.posts[2], today = testData.published_at, dd = ('0' + today.getDate()).slice(-2), mm = ('0' + (today.getMonth() + 1)).slice(-2), yyyy = today.getFullYear(), postLink = '/' + yyyy + '/' + mm + '/' + dd + '/short-and-sweet/'; // next test config.urlPathForPost(testData, permalinkSetting).should.equal(postLink); }); it('should output correct url for page with date permalink', function () { var permalinkSetting = '/:year/:month/:day/:slug/', /*jshint unused:false*/ testData = testUtils.DataGenerator.Content.posts[5], postLink = '/static-page-test/'; // next test config.urlPathForPost(testData, permalinkSetting).should.equal(postLink); }); it('should output correct url for post with complex permalink', function () { var permalinkSetting = '/:year/:id/:author/', /*jshint unused:false*/ testData = _.extend( {}, testUtils.DataGenerator.Content.posts[2], {id: 3}, {author: {slug: 'joe-bloggs'}} ), today = testData.published_at, yyyy = today.getFullYear(), postLink = '/' + yyyy + '/3/joe-bloggs/'; // next test config.urlPathForPost(testData, permalinkSetting).should.equal(postLink); }); }); }); describe('File', function () { var sandbox, originalConfig, readFileStub, overrideConfig = function (newConfig) { readFileStub.returns( _.extend({}, defaultConfig, newConfig) ); }, expectedError = new Error('expected bootstrap() to throw error but none thrown'); before(function () { originalConfig = _.cloneDeep(rewire('../../server/config')._config); }); beforeEach(function () { sandbox = sinon.sandbox.create(); readFileStub = sandbox.stub(config, 'readFile'); }); afterEach(function () { config = rewire('../../server/config'); resetConfig(); sandbox.restore(); }); it('loads the config file if one exists', function (done) { // We actually want the real method here. readFileStub.restore(); // the test infrastructure is setup so that there is always config present, // but we want to overwrite the test to actually load config.example.js, so that any local changes // don't break the tests config.set({ paths: { appRoot: path.join(originalConfig.paths.appRoot, 'config.example.js') } }); config.load().then(function (config) { config.url.should.equal(defaultConfig.url); config.database.client.should.equal(defaultConfig.database.client); config.database.connection.should.eql(defaultConfig.database.connection); config.server.host.should.equal(defaultConfig.server.host); config.server.port.should.equal(defaultConfig.server.port); done(); }).catch(done); }); it('uses the passed in config file location', function (done) { // We actually want the real method here. readFileStub.restore(); config.load(path.join(originalConfig.paths.appRoot, 'config.example.js')).then(function (config) { config.url.should.equal(defaultConfig.url); config.database.client.should.equal(defaultConfig.database.client); config.database.connection.should.eql(defaultConfig.database.connection); config.server.host.should.equal(defaultConfig.server.host); config.server.port.should.equal(defaultConfig.server.port); done(); }).catch(done); }); it('creates the config file if one does not exist', function (done) { // trick bootstrap into thinking that the config file doesn't exist yet var existsStub = sandbox.stub(fs, 'stat', function (file, cb) { return cb(true); }), // ensure that the file creation is a stub, the tests shouldn't really create a file writeFileStub = sandbox.stub(config, 'writeFile').returns(Promise.resolve()), validateStub = sandbox.stub(config, 'validate').returns(Promise.resolve()); config.load().then(function () { existsStub.calledOnce.should.be.true; writeFileStub.calledOnce.should.be.true; validateStub.calledOnce.should.be.true; done(); }).catch(done); }); it('accepts urls with a valid scheme', function (done) { // replace the config file with invalid data overrideConfig({url: 'http://testurl.com'}); config.load().then(function (localConfig) { localConfig.url.should.equal('http://testurl.com'); // Next test overrideConfig({url: 'https://testurl.com'}); return config.load(); }).then(function (localConfig) { localConfig.url.should.equal('https://testurl.com'); // Next test overrideConfig({url: 'http://testurl.com/blog/'}); return config.load(); }).then(function (localConfig) { localConfig.url.should.equal('http://testurl.com/blog/'); // Next test overrideConfig({url: 'http://testurl.com/ghostly/'}); return config.load(); }).then(function (localConfig) { localConfig.url.should.equal('http://testurl.com/ghostly/'); done(); }).catch(done); }); it('rejects a fqdn without a scheme', function (done) { overrideConfig({url: 'example.com'}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('rejects a hostname without a scheme', function (done) { overrideConfig({url: 'example'}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('rejects a hostname with a scheme', function (done) { overrideConfig({url: 'https://example'}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('rejects a url with an unsupported scheme', function (done) { overrideConfig({url: 'ftp://example.com'}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('rejects a url with a protocol relative scheme', function (done) { overrideConfig({url: '//example.com'}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('does not permit the word ghost as a url path', function (done) { overrideConfig({url: 'http://example.com/ghost/'}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('does not permit the word ghost to be a component in a url path', function (done) { overrideConfig({url: 'http://example.com/blog/ghost/'}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('does not permit the word ghost to be a component in a url path', function (done) { overrideConfig({url: 'http://example.com/ghost/blog/'}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('does not permit database config to be falsy', function (done) { // replace the config file with invalid data overrideConfig({database: false}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('does not permit database config to be empty', function (done) { // replace the config file with invalid data overrideConfig({database: {}}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('requires server to be present', function (done) { overrideConfig({server: false}); config.load().then(function (localConfig) { /*jshint unused:false*/ done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('allows server to use a socket', function (done) { overrideConfig({server: {socket: 'test'}}); config.load().then(function () { var socketConfig = config.getSocket(); socketConfig.should.be.an.Object; socketConfig.path.should.equal('test'); socketConfig.permissions.should.equal('660'); done(); }).catch(done); }); it('allows server to use a socket and user-defined permissions', function (done) { overrideConfig({ server: { socket: { path: 'test', permissions: '666' } } }); config.load().then(function () { var socketConfig = config.getSocket(); socketConfig.should.be.an.Object; socketConfig.path.should.equal('test'); socketConfig.permissions.should.equal('666'); done(); }).catch(done); }); it('allows server to have a host and a port', function (done) { overrideConfig({server: {host: '127.0.0.1', port: '2368'}}); config.load().then(function (localConfig) { should.exist(localConfig); localConfig.server.host.should.equal('127.0.0.1'); localConfig.server.port.should.equal('2368'); done(); }).catch(done); }); it('rejects server if there is a host but no port', function (done) { overrideConfig({server: {host: '127.0.0.1'}}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('rejects server if there is a port but no host', function (done) { overrideConfig({server: {port: '2368'}}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); it('rejects server if configuration is empty', function (done) { overrideConfig({server: {}}); config.load().then(function () { done(expectedError); }).catch(function (err) { should.exist(err); err.should.be.an.Error; done(); }).catch(done); }); }); describe('Check for deprecation messages:', function () { var logStub, // Can't use afterEach here, because mocha uses console.log to output the checkboxes // which we've just stubbed, so we need to restore it before the test ends to see ticks. resetEnvironment = function () { logStub.restore(); process.env.NODE_ENV = currentEnv; }; beforeEach(function () { logStub = sinon.stub(console, 'log'); process.env.NODE_ENV = 'development'; }); afterEach(function () { logStub.restore(); config = rewire('../../server/config'); }); it('doesn\'t display warning when deprecated options not set', function () { config.checkDeprecated(); logStub.calledOnce.should.be.false; // Future tests: This is important here! resetEnvironment(); }); it('displays warning when updateCheck exists and is truthy', function () { config.set({ updateCheck: 'foo' }); // Run the test code config.checkDeprecated(); logStub.calledOnce.should.be.true; logStub.calledWithMatch('updateCheck').should.be.true; // Future tests: This is important here! resetEnvironment(); }); it('displays warning when updateCheck exists and is falsy', function () { config.set({ updateCheck: false }); // Run the test code config.checkDeprecated(); logStub.calledOnce.should.be.true; logStub.calledWithMatch('updateCheck').should.be.true; // Future tests: This is important here! resetEnvironment(); }); it('displays warning when mail.fromaddress exists and is truthy', function () { config.set({ mail: { fromaddress: 'foo' } }); // Run the test code config.checkDeprecated(); logStub.calledOnce.should.be.true; logStub.calledWithMatch('mail.fromaddress').should.be.true; // Future tests: This is important here! resetEnvironment(); }); it('displays warning when mail.fromaddress exists and is falsy', function () { config.set({ mail: { fromaddress: false } }); // Run the test code config.checkDeprecated(); logStub.calledOnce.should.be.true; logStub.calledWithMatch('mail.fromaddress').should.be.true; // Future tests: This is important here! resetEnvironment(); }); it('doesn\'t display warning when only part of a deprecated option is set', function () { config.set({ mail: { notfromaddress: 'foo' } }); config.checkDeprecated(); logStub.calledOnce.should.be.false; // Future tests: This is important here! resetEnvironment(); }); it('can not modify the deprecatedItems on the config object', function () { config.set({ deprecatedItems: ['foo'] }); config.deprecatedItems.should.not.equal(['foo']); resetEnvironment(); }); }); });