diff --git a/.changeset/beige-rocks-cheat.md b/.changeset/beige-rocks-cheat.md new file mode 100644 index 000000000..60433c2ca --- /dev/null +++ b/.changeset/beige-rocks-cheat.md @@ -0,0 +1,5 @@ +--- +'@koopjs/featureserver': minor +--- + +- add owningSystemUrl to restInfo response diff --git a/.changeset/fair-mirrors-rescue.md b/.changeset/fair-mirrors-rescue.md new file mode 100644 index 000000000..be2b42df1 --- /dev/null +++ b/.changeset/fair-mirrors-rescue.md @@ -0,0 +1,7 @@ +--- +'@koopjs/koop-core': major +--- + +- route change in geoservices +- generic plugins and file-system plugins no longer supported +- add option to skip default geoservices plugin diff --git a/.changeset/perfect-feet-remember.md b/.changeset/perfect-feet-remember.md new file mode 100644 index 000000000..309f297aa --- /dev/null +++ b/.changeset/perfect-feet-remember.md @@ -0,0 +1,7 @@ +--- +'@koopjs/output-geoservices': major +--- + +- change generateToken route so it matches latest pattern in ArcGIS +- add option "useHttpForTokenUrl" to use http protocol on the authInfo.tokenServicesUrl returned by rest/info route + diff --git a/.nycrc b/.nycrc index 8ce00725e..d66ed279b 100644 --- a/.nycrc +++ b/.nycrc @@ -6,7 +6,7 @@ "packages/cache-memory/src/**/*.js", "packages/logger/src/**/*.js", "packages/output-geoservices/src/**/*.js", - "packages/koop-core/src/**/*.js" + "packages/core/src/**/*.js" ], "exclude": ["**/*.spec.js"], "includeAllSources": true diff --git a/demo/index.js b/demo/index.js index a27caa5ac..5f8c17ca7 100644 --- a/demo/index.js +++ b/demo/index.js @@ -1,6 +1,12 @@ const Koop = require('@koopjs/koop-core'); const provider = require('@koopjs/provider-file-geojson'); - const koop = new Koop({ logLevel: 'debug' }); + +// const auth = require('@koopjs/auth-direct-file')( +// 'pass-in-your-secret', +// `${__dirname}/user-store.json` +// ); + +// koop.register(auth); koop.register(provider, { dataDir: './demo/provider-data'}); -koop.server.listen(8080); \ No newline at end of file +koop.server.listen(process.env.PORT || 8080); \ No newline at end of file diff --git a/demo/user-store.json b/demo/user-store.json new file mode 100644 index 000000000..39bd2082b --- /dev/null +++ b/demo/user-store.json @@ -0,0 +1,4 @@ +[{ + "username": "helloworld", + "password": "foobar" +}] \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index a4789f8e6..8635cb7dd 100644 --- a/package-lock.json +++ b/package-lock.json @@ -21,7 +21,7 @@ "@changesets/changelog-git": "^0.1.14", "@changesets/cli": "^2.26.2", "@commitlint/config-conventional": "^18.4.2", - "@koopjs/auth-direct-file": "^2.0.2", + "@koopjs/auth-direct-file": "^3.0.0-alpha.0", "@koopjs/provider-file-geojson": "^2.2.0", "await-spawn": "^4.0.2", "byline": "^5.0.0", @@ -2917,9 +2917,10 @@ } }, "node_modules/@koopjs/auth-direct-file": { - "version": "2.0.2", + "version": "3.0.0-alpha.0", + "resolved": "https://registry.npmjs.org/@koopjs/auth-direct-file/-/auth-direct-file-3.0.0-alpha.0.tgz", + "integrity": "sha512-CM0gXbrlqL2Qc1FhnB5WtCyJSnGnCl5TsMK9kw2AbR8hmD3rZN9UxtmumqggciolgtbOB9SGc14ycvmB8qLEug==", "dev": true, - "license": "Apache-2.0", "dependencies": { "fs-extra": "^11.1.1", "joi": "^17.9.2", @@ -23210,7 +23211,9 @@ } }, "@koopjs/auth-direct-file": { - "version": "2.0.2", + "version": "3.0.0-alpha.0", + "resolved": "https://registry.npmjs.org/@koopjs/auth-direct-file/-/auth-direct-file-3.0.0-alpha.0.tgz", + "integrity": "sha512-CM0gXbrlqL2Qc1FhnB5WtCyJSnGnCl5TsMK9kw2AbR8hmD3rZN9UxtmumqggciolgtbOB9SGc14ycvmB8qLEug==", "dev": true, "requires": { "fs-extra": "^11.1.1", diff --git a/package.json b/package.json index 3e2e340df..a03992c26 100644 --- a/package.json +++ b/package.json @@ -47,7 +47,7 @@ "@changesets/changelog-git": "^0.1.14", "@changesets/cli": "^2.26.2", "@commitlint/config-conventional": "^18.4.2", - "@koopjs/auth-direct-file": "^2.0.2", + "@koopjs/auth-direct-file": "^3.0.0-alpha.0", "@koopjs/provider-file-geojson": "^2.2.0", "await-spawn": "^4.0.2", "byline": "^5.0.0", diff --git a/packages/core/README.md b/packages/core/README.md index 690e19df6..00e97fb7a 100644 --- a/packages/core/README.md +++ b/packages/core/README.md @@ -94,6 +94,16 @@ const options = { } ``` +#### skipGeoservicesRegistration +By default, Koop will register the GeoServices output-plugin (a.k.a. FeatureServer). If you do not want this plugin registered or want to register a specific version, you can skip the default registration by setting the option to `true`: + +```js +const options = { + skipGeoservicesRegistration: true +} +``` + + #### geoservicesDefaults Koop registers the Geoservices output plugin (FeatureServer) by default. This plugin takes its own options including those to set server and layer metadata (e.g., FeatureServer version, copyrightText, maxRecordCount, etc). These are useful for overriding defaults set in the FeatureServer codebase. You can have Koop set these options at start-up by passing the `geoservicesDefaults` option. It should be a JSON object with the specification described in the [FeatureServer documentation](packages/featureserver#featureserver.setdefaults). diff --git a/packages/core/coverage.svg b/packages/core/coverage.svg index df5fb6bbe..4354a36ca 100644 --- a/packages/core/coverage.svg +++ b/packages/core/coverage.svg @@ -1,20 +1,20 @@ - - coverage: 95.2% + + coverage: 100% - + - - + + \ No newline at end of file diff --git a/packages/core/src/data-provider/extend-model.js b/packages/core/src/data-provider/extend-model.js index 6016c4d07..4905f77dd 100644 --- a/packages/core/src/data-provider/extend-model.js +++ b/packages/core/src/data-provider/extend-model.js @@ -35,6 +35,11 @@ module.exports = function extendModel ({ ProviderModel, namespace, logger, cache } async pull (req, callback) { + const { error } = await this.#authorizeRequest(req); + if (error) { + return callback(error); + } + const key = this.#createCacheKey(req); try { @@ -63,6 +68,11 @@ module.exports = function extendModel ({ ProviderModel, namespace, logger, cache // TODO: the pullLayer() and the pullCatalog() are very similar to the pull() // function. We may consider to merging them in the future. async pullLayer (req, callback) { + const { error } = await this.#authorizeRequest(req); + if (error) { + return callback(error); + } + if (!this.#getLayer) { callback(new Error(`getLayer() method is not implemented in the ${this.namespace} provider.`)); } @@ -91,6 +101,11 @@ module.exports = function extendModel ({ ProviderModel, namespace, logger, cache } async pullCatalog (req, callback) { + const { error } = await this.#authorizeRequest(req); + if (error) { + return callback(error); + } + if (!this.#getCatalog) { callback(new Error(`getCatalog() method is not implemented in the ${this.namespace} provider.`)); } @@ -119,6 +134,12 @@ module.exports = function extendModel ({ ProviderModel, namespace, logger, cache } async pullStream (req) { + const { error } = await this.#authorizeRequest(req); + + if (error) { + throw error; + } + if (this.getStream) { await this.#before(req); const providerStream = await this.getStream(req); @@ -135,20 +156,48 @@ module.exports = function extendModel ({ ProviderModel, namespace, logger, cache } return hasher(req.url).toString(); } + + async #authorizeRequest (req) { + try { + await this.authorize(req); + } catch (error) { + error.code = 401; + return { error }; + } + + return { error: null }; + } } - // Add auth methods if auth plugin registered with Koop - if (authModule) { - const { - authenticationSpecification, - authenticate, - authorize - } = authModule; - - Model.prototype.authenticationSpecification = Object.assign({}, authenticationSpecification(namespace), { provider: namespace }); - Model.prototype.authenticate = authenticate; - Model.prototype.authorize = authorize; + // If provider does not have auth-methods, + // check for global auth-module. if exists, use it, + // otherwise use dummy methods + + if (typeof ProviderModel.prototype.authorize !== 'function') { + Model.prototype.authorize = typeof authModule?.authorize === 'function' ? authModule.authorize : async () => {}; } + + if (typeof ProviderModel.prototype.authenticate !== 'function') { + Model.prototype.authenticate = typeof authModule?.authenticate === 'function' ? authModule?.authenticate : async () => { return {}; }; + } + + if(typeof authModule?.authenticationSpecification === 'function') { + logger.warn('Use of "authenticationSpecification" is deprecated. It will be removed in a future release.'); + Model.prototype.authenticationSpecification = authModule?.authenticationSpecification; + } + // Add auth methods if auth plugin registered with Koop + // if (authModule) { + // const { + // authenticationSpecification, + // authenticate, + // authorize + // } = authModule; + + // Model.prototype.authenticationSpecification = Object.assign({}, authenticationSpecification(namespace), { provider: namespace }); + // Model.prototype.authenticate = authenticate; + // Model.prototype.authorize = authorize; + // } + return new Model({ logger, cache }, options); }; diff --git a/packages/core/src/data-provider/extend-model.spec.js b/packages/core/src/data-provider/extend-model.spec.js index 9816ec029..20a80fd6f 100644 --- a/packages/core/src/data-provider/extend-model.spec.js +++ b/packages/core/src/data-provider/extend-model.spec.js @@ -16,6 +16,7 @@ const mockCache = { const mockLogger = { debug: sinon.spy(), + warn: sinon.spy(), info: () => {}, }; @@ -325,6 +326,34 @@ describe('Tests for extend-model', function () { mockCache.insert.notCalled.should.equal(true); }); + it('should pass authorization error in callback', async () => { + const mockCache = { + retrieve: sinon.spy((key, query, callback) => { + callback(null); + }), + insert: sinon.spy(() => {}), + }; + + class Model extends MockModel {} + Model.prototype.authorize = async () => { + throw new Error('unauthorized'); + }; + + const model = extendModel({ + ProviderModel: Model, + logger: mockLogger, + cache: mockCache, + }); + const pullData = promisify(model.pull).bind(model); + + try { + await pullData({ url: 'domain/test-provider', params: {}, query: {} }); + should.fail(); + } catch (err) { + err.message.should.equal('unauthorized'); + } + }); + it('should send error in callback', async () => { const mockCache = { retrieve: sinon.spy((key, query, callback) => { @@ -390,24 +419,76 @@ describe('Tests for extend-model', function () { }); describe('auth methods', () => { - it('should attach auth methods when auth plugin is registered with Koop', () => { + it('should attach auth methods from authModule if provider does not already define them', async () => { const model = extendModel({ ProviderModel: MockModel, namespace: 'test-provider', logger: mockLogger, cache: mockCache, authModule: { - authenticate: () => {}, - authorize: () => {}, - authenticationSpecification: sinon.spy(), + authenticate: () => { + return 'from auth-module'; + }, + authorize: () => { + return 'from auth-module'; + }, + authenticationSpecification: () => { + return 'from auth-module'; + } + + }, + }); + const authenticateResult = await model.authenticate(); + authenticateResult.should.equal('from auth-module'); + const authorizeResult = await model.authorize(); + authorizeResult.should.equal('from auth-module'); + const specResult = await model.authenticationSpecification(); + specResult.should.equal('from auth-module'); + }); + + it('should use provider auth methods if defined', async () => { + class Model extends MockModel {} + Model.prototype.authenticate = async() => { + return 'from provider'; + }; + + Model.prototype.authorize = async() => { + return 'from provider'; + }; + + const model = extendModel({ + ProviderModel: Model, + namespace: 'test-provider', + logger: mockLogger, + cache: mockCache, + authModule: { + authenticate: () => { + return 'from auth-module'; + }, + authorize: () => { + return 'from auth-module'; + } }, }); - model.should.have.property('authorize').and.be.a.Function(); - model.should.have.property('authenticate').and.be.a.Function(); - model.should.have - .property('authenticationSpecification') - .and.deepEqual({ provider: 'test-provider' }); + const authenticateResult = await model.authenticate(); + authenticateResult.should.equal('from provider'); + const authorizeResult = await model.authorize(); + authorizeResult.should.equal('from provider'); + }); + + it('should use dummy auth methods', async () => { + const model = extendModel({ + ProviderModel: MockModel, + namespace: 'test-provider', + logger: mockLogger, + cache: mockCache, + }); + const authenticateResult = await model.authenticate(); + authenticateResult.should.deepEqual({}); + const authorizeResult = await model.authorize(); + should(authorizeResult).deepEqual(undefined); }); + }); describe('transformation functions', function () { @@ -520,6 +601,34 @@ describe('Tests for extend-model', function () { extendModel.prototype.createKey = undefined; }); + it('should pass authorization error in callback', async () => { + const mockCache = { + retrieve: sinon.spy((key, query, callback) => { + callback(null); + }), + insert: sinon.spy(() => {}), + }; + + class Model extends MockModel {} + Model.prototype.authorize = async () => { + throw new Error('unauthorized'); + }; + + const model = extendModel({ + ProviderModel: Model, + logger: mockLogger, + cache: mockCache, + }); + const pullLayer = promisify(model.pullLayer).bind(model); + + try { + await pullLayer({ url: 'domain/test-provider', params: {}, query: {} }); + should.fail(); + } catch (err) { + err.message.should.equal('unauthorized'); + } + }); + it('should throw an error if the getLayer() function is not implemented', async () => { class Model extends MockModel {} Model.prototype.getLayer = undefined; @@ -907,6 +1016,34 @@ describe('Tests for extend-model', function () { err.message.should.equal('err in getCatalog'); } }); + + it('should pass authorization error in callback', async () => { + const mockCache = { + retrieve: sinon.spy((key, query, callback) => { + callback(null); + }), + insert: sinon.spy(() => {}), + }; + + class Model extends MockModel {} + Model.prototype.authorize = async () => { + throw new Error('unauthorized'); + }; + + const model = extendModel({ + ProviderModel: Model, + logger: mockLogger, + cache: mockCache, + }); + const pullCatalog = promisify(model.pullCatalog).bind(model); + + try { + await pullCatalog({ url: 'domain/test-provider', params: {}, query: {} }); + should.fail(); + } catch (err) { + err.message.should.equal('unauthorized'); + } + }); }); describe('model pullStream method', function () { @@ -972,5 +1109,32 @@ describe('Tests for extend-model', function () { err.should.be.an.Error(); } }); + + it('should throw authorization error', async () => { + const mockCache = { + retrieve: sinon.spy((key, query, callback) => { + callback(null); + }), + insert: sinon.spy(() => {}), + }; + + class Model extends MockModel {} + Model.prototype.authorize = async () => { + throw new Error('unauthorized'); + }; + + const model = extendModel({ + ProviderModel: Model, + logger: mockLogger, + cache: mockCache, + }); + + try { + await model.pullStream({ some: 'options' }); + should.fail(); + } catch (err) { + err.message.should.equal('unauthorized'); + } + }); }); }); diff --git a/packages/core/src/index.js b/packages/core/src/index.js index ff7dea472..7015f4c0c 100644 --- a/packages/core/src/index.js +++ b/packages/core/src/index.js @@ -12,11 +12,12 @@ const DataProvider = require('./data-provider'); const geoservices = require('@koopjs/output-geoservices'); class Koop extends Events { + #authModule; + constructor(options) { super(); this.version = pkg.version; - // TODO: remove usage of "config" module this.config = { ...options, ...config }; this.server = initServer(this.config); this.log = this.config.logger || new Logger(this.config); @@ -29,11 +30,13 @@ class Koop extends Events { const { geoservicesDefaults } = this.config; - this.register(geoservices, { - logger: this.log, - authInfo: this.config.authInfo, - defaults: geoservicesDefaults - }); + if (this.config.skipGeoservicesRegistration !== true) { + this.register(geoservices, { + logger: this.log, + authInfo: this.config.authInfo, + defaults: geoservicesDefaults + }); + } this.server .on('mount', () => { @@ -46,7 +49,7 @@ class Koop extends Events { })); } - register(plugin = {}, options) { + register(plugin, options) { if (!plugin) { throw new Error('Plugin registration failed: plugin undefined'); } @@ -63,18 +66,10 @@ class Koop extends Events { return this.#registerOutput(plugin, options); } - if (plugin.type === 'filesystem') { - return this.#registerFilesystem(plugin, options); - } - if (plugin.type === 'auth') { return this.#registerAuth(plugin, options); } - if (plugin.type === 'plugin') { - return this.#registerPlugin(plugin, options); - } - this.log.warn( 'Unrecognized plugin type: "' + plugin.type + @@ -87,7 +82,7 @@ class Koop extends Events { const dataProvider = new DataProvider({ logger: this.log, cache: this.cache, - authModule: this._authModule, + authModule: this.#authModule, pluginDefinition, outputPlugins: this.outputs, options, @@ -112,33 +107,9 @@ class Koop extends Events { } #registerAuth (auth) { - this._authModule = auth; + this.#authModule = auth; this.log.info(`registered auth module: ${auth.name} v${auth.version}`); } - - #registerFilesystem (Filesystem) { - this.fs = new Filesystem(); - this.log.info( - `registered filesystem: ${Filesystem.pluginName || Filesystem.plugin_name || Filesystem.name} v${Filesystem.version}` - ); - } - - #registerPlugin (Plugin) { - const name = Plugin.pluginName || Plugin.plugin_name || Plugin.name; - if (!name) { - throw new Error('Plugin is missing name'); - } - - let dependencies; - if (Array.isArray(Plugin.dependencies) && Plugin.dependencies.length) { - dependencies = Plugin.dependencies.reduce((deps, dep) => { - deps[dep] = this[dep]; - return deps; - }, {}); - } - this[name] = new Plugin(dependencies); - this.log.info('registered plugin:', name, Plugin.version); - } } /** @@ -156,12 +127,12 @@ function initServer(options) { .use(express.static(path.join(__dirname, '/public'))); // Use CORS unless explicitly disabled in the config - if (!options.disableCors) { + if (options.disableCors !== true) { app.use(cors()); } // Use compression unless explicitly disable in the config - if (!options.disableCompression) { + if (options.disableCompression !== true) { app.use(compression()); } diff --git a/packages/core/src/index.spec.js b/packages/core/src/index.spec.js index 9179d9777..ff2cb9719 100644 --- a/packages/core/src/index.spec.js +++ b/packages/core/src/index.spec.js @@ -4,6 +4,28 @@ const _ = require('lodash'); const DataProvider = require('./data-provider'); const providerConstructorSpy = sinon.spy(); + +const mockApp = { + use: sinon.spy(() => mockApp), + disable: sinon.spy(() => mockApp), + set: sinon.spy(() => mockApp), + on: sinon.spy((event, callback) => { + callback(); + return mockApp; + }), + get: sinon.spy((route, handler) => { + if( route === '/status') { + handler({}, { json: () => {} }); + } + return mockApp; + }), + post: sinon.spy(() => mockApp), +}; + +const mockExpress = () => { + return mockApp; +}; + class mockDataProviderModule extends DataProvider { constructor(params) { providerConstructorSpy(params); @@ -11,8 +33,17 @@ class mockDataProviderModule extends DataProvider { } } +class MockLogger { + warn () {} + log () {} + error () {} + info () {} + silly () {} +} const Koop = proxyquire('./', { - './data-provider': mockDataProviderModule + './data-provider': mockDataProviderModule, + '@koopjs/logger': MockLogger, + 'express': mockExpress }); class MockProviderPluginController { @@ -49,84 +80,95 @@ const mockProviderDefinition = { Model: MockModel }; -const Geoservices = require('@koopjs/output-geoservices'); - const should = require('should') // eslint-disable-line -const geoservicesFixtureRoutes = Geoservices.routes.reduce((acc, route) => { - return acc + route.methods.length; -}, 0); -const providerFixtureRoutes = mockProviderDefinition.routes.reduce((acc, route) => { - return acc + route.methods.length; -}, 0); - describe('Index tests', function () { + beforeEach(() => { + sinon.reset(); + }); describe('Koop instantiation', function () { + it('should instantiate Koop without options', function () { + const koop = new Koop(); + koop.config.should.be.empty(); + mockApp.use.callCount.should.equal(5); + }); + it('should instantiate Koop with options', function () { const koop = new Koop({ foo: 'bar', logLevel: 'error' }); koop.config.should.have.property('foo', 'bar'); }); - it('should instantiate Koop without options', function () { - const koop = new Koop(); - koop.config.should.be.empty(); + it('should skip geoservices registration', function () { + const koop = new Koop({ skipGeoservicesRegistration: true}); + koop.register(mockProviderDefinition, { routePrefix: 'watermelon' }); + koop.outputs.length.should.equal(0); + }); + + it('should disable CORS and compression', function () { + new Koop({ disableCors: true, disableCompression: true }); + + mockApp.use.callCount.should.equal(3); }); }); - describe('Provider registration', function () { + describe('Plugin registration', function () { + it('should fail if no plugin', () => { + try { + const koop = new Koop({ logLevel: 'error' }); + koop.register(); + throw new Error('should have thrown'); + } catch (error) { + error.should.have.property('message', 'Plugin registration failed: plugin undefined'); + } + }); + it('should register provider and add output and provider routes to router stack', function () { const koop = new Koop({ logLevel: 'error' }); koop.register(mockProviderDefinition, { routePrefix: 'watermelon' }); koop.providers.length.should.equal(1); - koop.providers[0].should.have.property('namespace', 'test-provider'); - // Check that the stack includes routes with the provider name in the path - const providerRoutes = koop.server._router.stack - .filter((layer) => { - return layer?.route?.path.includes('watermelon'); - }) - .map(layer => { - return _.get(layer, 'route.path'); - }); - providerRoutes.length.should.equal(geoservicesFixtureRoutes + providerFixtureRoutes); }); - }); -}); -describe('Auth plugin registration', function () { - const mockAuthPlugin = { - type: 'auth', - authenticationSpecification: function () { - return function () { }; - }, - authenticate: function () {}, - authorize: function () {} - }; - - it('should register successfully', function () { - const koop = new Koop({ logLevel: 'error' }); - koop.register(mockAuthPlugin); - koop._authModule.should.be.instanceOf(Object); - koop._authModule.authenticate.should.be.instanceOf(Function); - koop._authModule.authorize.should.be.instanceOf(Function); - koop._authModule.authenticationSpecification.should.be.instanceOf(Function); - }); -}); + it('should register unknown plugin type as provider and add output and provider routes to router stack', function () { + const koop = new Koop({ logLevel: 'error' }); + const unknownPlugin = _.cloneDeep(mockProviderDefinition); + delete unknownPlugin.type; -describe('Generic plugin registration', function () { - class fakePlugin { - static type = 'plugin'; - static version = '0.0.0'; + koop.register(unknownPlugin, { routePrefix: 'watermelon' }); + koop.providers.length.should.equal(1); + koop.providers[0].should.have.property('namespace', 'test-provider'); + }); - testFunc () { - return true; - } - } - it('should register successfully', function () { - const koop = new Koop({ logLevel: 'error' }); - koop.register(fakePlugin); - koop.fakePlugin.should.be.instanceOf(Object); - koop.fakePlugin.testFunc.should.be.instanceOf(Function); - koop.fakePlugin.testFunc().should.equal(true); + it('should register auth-plugin successfully', function () { + const mockAuthPlugin = { + type: 'auth', + authenticate: function () {}, + authorize: function () {} + }; + + const koop = new Koop({ logLevel: 'error' }); + try { + koop.register(mockAuthPlugin); + } catch (error) { + (error).should.equal(undefined); + } + + }); + + it('should register cache-plugin successfully', function () { + const mockCachePlugin = class MockCache { + static type = 'cache'; + }; + + const koop = new Koop({ logLevel: 'error' }); + try { + koop.register(mockCachePlugin); + } catch (error) { + (error).should.equal(undefined); + } + + }); }); + }); + diff --git a/packages/featureserver/README.md b/packages/featureserver/README.md index 99bb62f2c..8ed08b81a 100644 --- a/packages/featureserver/README.md +++ b/packages/featureserver/README.md @@ -35,13 +35,11 @@ routes.forEach(route => { ## API * [FeatureServer.route](#featureserver.route) * [FeatureServer.query](#featureserver.query) +* [FeatureServer.restInfo](#featureserver.serverInfo) * [FeatureServer.serverInfo](#featureserver.serverInfo) * [FeatureServer.layerInfo](#featureserver.layerInfo) * [FeatureServer.layers](#featureserver.layers) * [FeatureServer.generateRenderer](#featureserver.generateRenderer) -* [FeatureServer.authenticate](#featureserver.authenticate) -* [FeatureServer.error.authorize](#featureserver.error.authorize) -* [FeatureServer.authenticate](#featureserver.error.authenticate) * [FeatureServer.queryRelatedRecords](#featureserver.queryRelatedRecords) * [FeatureServer.setDefaults](#featureserver.setDefaults) @@ -157,8 +155,22 @@ const options = { FeatureServer.query(geojson, options) ``` +### FeatureServer.restInfo +Pass in a `data` object and the request object and return a response object that adheres to the specification of the `rest/info` response. The `data` object may contain the `owningSystemUrl` and the `authInfo` object: +```js +{ + owningSystemUrl: 'https://domain.com/some/path' + authInfo: { + isTokenBasedSecurity: true, + tokenServicesUrl: 'https://url/that/will/generate/a/token' + } +} +``` + +The response will include the above information as well as the FeatureServer version numbers. + ### FeatureServer.serverInfo -Generate version `10.51` Geoservices server info +Generate Geoservices server info ```js const server = { @@ -214,7 +226,7 @@ FeatureServer.serverInfo(server) ``` ### FeatureServer.layerInfo -Generate version `10.51` Geoservices information about a single layer +Generate Geoservices information about a single layer ```js FeatureServer.layerInfo(geojson, options) ``` @@ -264,7 +276,7 @@ const metadata = { ``` ### FeatureServer.layers -Generate version `10.51` Geoservices information about one or many layers +Generate Geoservices information about one or many layers Can pass a single geojson object or an array of geojson objects ```js @@ -419,59 +431,6 @@ Output: ] ``` -### FeatureServer.authenticate -Pass in an outgoing response object and an authentication success object and this function will route and return a formatted authentication success response. - - FeatureServer.authenticate(res, auth, ssl = false) - -* `auth` is the result of a successful authentication attempt that returns a token and expiration time -* `ssl` is a boolean flag indicating if token should always be passed back via HTTPS. Defaults to `false` - -e.g., - - const auth = { - "token":"elS39KU4bMmZQgMXDuswgA14vavIp4mfpiqcWSr0qM6q4dFguTnnHddWqbpK5Mc3HsCN8XghlwawUUYApOOcxKNyg_9WqTofChJXxxD058_rL1HZkM5PDhUOh9YYQn1K", - "expires":1524508236322 - } - - FeatureServer.authenticate(res, auth) - - { - "token":"elS39KU4bMmZQgMXDuswgA14vavIp4mfpiqcWSr0qM6q4dFguTnnHddWqbpK5Mc3HsCN8XghlwawUUYApOOcxKNyg_9WqTofChJXxxD058_rL1HZkM5PDhUOh9YYQn1K", - "expires":1524508236322, - ssl: false - } - -### FeatureServer.error.authorize -Pass in an outgoing response object and this function will route and return a formattted authorization error. - - FeatureServer.error.authorize(res) - - { - "error": { - "code": 499, - "message": "Token Required", - "details": [] - } - } - -### FeatureServer.error.authenticate -Pass in an outgoing response object and this function will route and return a formatted authentication error. - - FeatureServer.error.authenticate(res) - - { - "error": { - "code": 400, - "message": "Unable to generate token.", - "details": ["Invalid username or password."] - } - } - - -[npm-image]: https://img.shields.io/npm/v/@koopjs/featureserver.svg?style=flat-square -[npm-url]: https://www.npmjs.com/package/@koopjs/featureserver - ### FeatureServer.queryRelatedRecords Pass in `geojson` and `options`, and the function will return a valid queryRelatedRecords object. Required attributes within `options` are `objectIds` and `relationshipId`. diff --git a/packages/featureserver/src/helpers/normalize-spatial-reference.js b/packages/featureserver/src/helpers/normalize-spatial-reference.js index 568cd7953..b2fc62cd1 100644 --- a/packages/featureserver/src/helpers/normalize-spatial-reference.js +++ b/packages/featureserver/src/helpers/normalize-spatial-reference.js @@ -88,6 +88,7 @@ function esriWktLookup (lookupValue) { const { wkid, latestWkid } = result; // Add the WKT to the local lookup so we don't need to scan the Esri lookups next time + // TODO: move to LRU cache wktLookup.set(wkid, { wkid, latestWkid }); return { latestWkid, wkid }; } diff --git a/packages/featureserver/src/rest-info-route-handler.js b/packages/featureserver/src/rest-info-route-handler.js index 7a5c61e3e..5c1905baa 100644 --- a/packages/featureserver/src/rest-info-route-handler.js +++ b/packages/featureserver/src/rest-info-route-handler.js @@ -9,7 +9,10 @@ function restInfo (data = {}, req) { return { currentVersion, fullVersion, - ...data + owningSystemUrl: data.owningSystemUrl, + authInfo: { + ...data.authInfo + } }; } diff --git a/packages/featureserver/src/rest-info-route-handler.spec.js b/packages/featureserver/src/rest-info-route-handler.spec.js index 447bfd693..ddddfc93f 100644 --- a/packages/featureserver/src/rest-info-route-handler.spec.js +++ b/packages/featureserver/src/rest-info-route-handler.spec.js @@ -1,4 +1,4 @@ -const should = require('should') // eslint-disable-line +const should = require('should'); // eslint-disable-line const restInfo = require('./rest-info-route-handler'); const CURRENT_VERSION = 11.1; const FULL_VERSION = '11.1.0'; @@ -7,34 +7,34 @@ describe('rest/info handler', () => { it('should return default info', () => { const req = { app: { - locals: {} - } + locals: {}, + }, }; const result = restInfo(undefined, req); result.should.deepEqual({ currentVersion: CURRENT_VERSION, - fullVersion: FULL_VERSION + fullVersion: FULL_VERSION, + authInfo: {}, + owningSystemUrl: undefined, }); }); it('should return default plus supplied info', () => { const data = { - hello: { - world: true - } + authInfo: { foo: 'bar' }, + owningSystemUrl: 'helloworld', }; const req = { app: { - locals: {} - } + locals: {}, + }, }; const result = restInfo(data, req); result.should.deepEqual({ currentVersion: CURRENT_VERSION, fullVersion: FULL_VERSION, - hello: { - world: true - } + authInfo: { foo: 'bar' }, + owningSystemUrl: 'helloworld', }); }); @@ -45,16 +45,23 @@ describe('rest/info handler', () => { config: { featureServer: { currentVersion: 10.81, - fullVersion: '10.8.1' - } - } - } - } + fullVersion: '10.8.1', + }, + }, + }, + }, }; - const result = restInfo({}, req); + const result = restInfo( + { authInfo: { foo: 'bar' }, owningSystemUrl: 'helloworld' }, + req, + ); result.should.deepEqual({ currentVersion: 10.81, - fullVersion: '10.8.1' + fullVersion: '10.8.1', + authInfo: { + foo: 'bar', + }, + owningSystemUrl: 'helloworld', }); }); }); diff --git a/packages/output-geoservices/README.md b/packages/output-geoservices/README.md index e55c94d3c..8beb7edbd 100644 --- a/packages/output-geoservices/README.md +++ b/packages/output-geoservices/README.md @@ -20,56 +20,67 @@ koop.register(provider) koop.server.listen(80) ``` -## Routes +## Options + +### `defaults (Object)` +The `defaults` options allows the setting of some FeatureServer metadata properties. The `defaults` option should be an object with some of the following properties: ```js -Geoservices.routes = [ - { - path: '$namespace/rest/info', - methods: ['get', 'post'], - handler: 'featureServerRestInfo' - }, - { - path: '$namespace/tokens/:method', - methods: ['get', 'post'], - handler: 'generateToken' - }, - { - path: '$namespace/tokens/', - methods: ['get', 'post'], - handler: 'generateToken' - }, - { - path: '$namespace/rest/services/$providerParams/FeatureServer/:layer/:method', - methods: ['get', 'post'], - handler: 'featureServer' - }, - { - path: '$namespace/rest/services/$providerParams/FeatureServer/layers', - methods: ['get', 'post'], - handler: 'featureServer' - }, - { - path: '$namespace/rest/services/$providerParams/FeatureServer/:layer', - methods: ['get', 'post'], - handler: 'featureServer' - }, - { - path: '$namespace/rest/services/$providerParams/FeatureServer', - methods: ['get', 'post'], - handler: 'featureServer' - }, - { - path: '$namespace/rest/services/$providerParams/FeatureServer*', - methods: ['get', 'post'], - handler: 'featureServer' - }, - { - path: '$namespace/rest/services/$providerParams/MapServer*', - methods: ['get', 'post'], - handler: 'featureServer' +{ + defaults: { + currentVersion, // number (11.2) + fullVersion, // string ('11.2.0') + maxRecordCount, // number (500) + server: { + serviceDescription, // string ('Default text for serviceDescription') + description, // string ('Default text for description') + copyrightText, // string ('Default text for copyright') + hasStaticData, // boolean (true) + spatialReference, // object (Esri spatial reference) + initialExtent, // object (Esri spatial envelope) + fullExtent, // object (Esri spatial envelope) + }, + layer: { + description, // string ('Default text for layer description') + copyrightText, // string ('Default text for layer copyright') + extent, // object (Esri spatial envelope) + }, } -] +} + +``` +Note that the `defaults` option only overrides FeatureServer's defaults. Providers that set metadata may override values set by the above `defaults` properties. + + +### `useHttpForTokenUrl (boolean)` +The `rest/info` route generates a property `tokenServicesUrl` with value for the URL to use when requesting a token. By default the protocol for this URL is `https`, but if you require it to be `http` set the value of this option to `true`. + +```js +{ + useHttpForTokenUrl: true +} +``` + +### `logger (Logger)` +You can leverage your own custom logger instance, but it must adhere to the Winston logger specification. + +```js +{ + logger // some custom Logger instance +} +``` + +## Routes + +```js +/rest/info +/rest/generateToken +/rest/services//FeatureServer/:layer/:method +/rest/services//FeatureServer/layers +/rest/services//FeatureServer/:layer +/rest/services//FeatureServer +/rest/services//FeatureServer* +/rest/services//MapServer* ``` [npm-img]: https://img.shields.io/npm/v/@koopjs/output-geoservices.svg?style=flat-square diff --git a/packages/output-geoservices/package.json b/packages/output-geoservices/package.json index 350bf9f21..341ba5950 100644 --- a/packages/output-geoservices/package.json +++ b/packages/output-geoservices/package.json @@ -37,6 +37,7 @@ }, "jest": { "coverageReporters": [ + "json-summary", "json", "text", "lcov" diff --git a/packages/output-geoservices/src/index.js b/packages/output-geoservices/src/index.js index d89008259..5675423fc 100644 --- a/packages/output-geoservices/src/index.js +++ b/packages/output-geoservices/src/index.js @@ -17,7 +17,8 @@ const tokenRequiredError = { error: { code: 499, message: 'Token Required', - details: [], + messageCode: 'GWM_0003', + details: ['Token Required'], }, }; @@ -39,8 +40,10 @@ const authorizationError = { }; class GeoServices { - #useHttp = false; + #useHttpForTokenUrl = false; + #authInfo; #pullData; + #logger; static type = 'output'; static version = require('../package.json').version; @@ -51,12 +54,7 @@ class GeoServices { handler: 'restInfoHandler', }, { - path: '$namespace/tokens/:method', - methods: ['get', 'post'], - handler: 'generateToken', - }, - { - path: '$namespace/tokens/', + path: '$namespace/rest/generateToken', methods: ['get', 'post'], handler: 'generateToken', }, @@ -95,30 +93,48 @@ class GeoServices { constructor(model, options = {}) { this.model = model; this.#pullData = promisify(this.model.pull).bind(this.model); - this.options = options; - this.logger = options.logger || logger; - this.authInfo = options.authInfo || {}; - this.#useHttp = - this.model.authenticationSpecification?.useHttp === true || - process.env.KOOP_AUTH_HTTP === 'true'; - FeatureServer.setLogger({ logger: this.logger }); + this.#logger = options.logger || logger; + this.#authInfo = options.authInfo || { + isTokenBasedSecurity: true, + }; + + this.#useHttpForTokenUrl = this.#getHttpSetting(options, model); + + FeatureServer.setLogger({ logger: this.#logger }); // Set overrides FeatureServer.setDefaults(options.defaults); } + #getHttpSetting(options, model) { + if (options.useHttpForTokenUrl || process.env.GEOSERVICES_HTTP === 'true') { + return ( + options.useHttpForTokenUrl || process.env.GEOSERVICES_HTTP === 'true' + ); + } + + if (typeof model.authenticationSpecification === 'function') { + return model.authenticationSpecification()?.useHttp === true; + } + + if (typeof process.env.KOOP_AUTH_HTTP !== 'undefined') { + this.#logger.warn( + 'Use of "KOOP_AUTH_HTTP" environment variable is deprecated. It will be removed in a future release. Use the "useHttpForTokenUrl" option or "GEOSERVICES_HTTP" environment variable.', + ); + return process.env.KOOP_AUTH_HTTP === 'true'; + } + + return false; + } + async generalHandler(req, res) { try { - if (this.#shouldAuthorize()) { - await this.model.authorize(req); - } - const data = await this.#pullData(req); return FeatureServer.route(req, res, data); } catch (error) { - this.logger.error(error); + this.#logger.error(error); - const token = this.#getToken(req); + const token = this.#extractTokenFromRequest(req); const { code, message, details = [] } = normalizeError(error); res.status(200); // ArcGIS standard is to wrap errors in 200 success @@ -145,11 +161,7 @@ class GeoServices { } } - #shouldAuthorize() { - return typeof this.model.authorize === 'function'; - } - - #getToken(req) { + #extractTokenFromRequest(req) { const { headers: { authorization }, query, @@ -164,37 +176,34 @@ class GeoServices { } restInfoHandler(req, res) { - const authInfo = { ...this.authInfo }; - - if (this.model.authenticationSpecification) { - authInfo.isTokenBasedSecurity = true; + const authInfo = { ...this.#authInfo }; + if (this.#authInfo.isTokenBasedSecurity) { authInfo.tokenServicesUrl = this.#buildTokensUrl( req.headers.host, req.baseUrl, ); } - FeatureServer.route(req, res, { authInfo }); + FeatureServer.route(req, res, { owningSystemUrl: this.#buildOwningSystemUrl(req.headers.host, + req.baseUrl), authInfo }); } #buildTokensUrl(host, baseUrl) { - const protocol = this.#useHttp ? 'http' : 'https'; - return `${protocol}://${host}${baseUrl}/${this.model.namespace}/tokens/`; + const protocol = this.#useHttpForTokenUrl ? 'http' : 'https'; + return `${protocol}://${host}${baseUrl}/${this.model.namespace}/rest/generateToken`; } - async generateToken(req, res) { - if (typeof this.model.authenticate !== 'function') { - return res - .status(500) - .json({ error: '"authenticate" not implemented for this provider' }); - } + #buildOwningSystemUrl(host, baseUrl) { + const protocol = this.#useHttpForTokenUrl ? 'http' : 'https'; + return `${protocol}://${host}${baseUrl}/${this.model.namespace}`; + } + async generateToken(req, res) { try { + //const decodedToken = await this.model.authorize(req); const tokenResponse = await this.model.authenticate(req); - res - .status(200) - .json({ ...tokenResponse, ssl: tokenResponse.ssl || false }); + res.status(200).json(tokenResponse); } catch (error) { const { code, message, details = [] } = normalizeError(error); diff --git a/packages/output-geoservices/src/index.spec.js b/packages/output-geoservices/src/index.spec.js index f8158f413..d069ad2b5 100644 --- a/packages/output-geoservices/src/index.spec.js +++ b/packages/output-geoservices/src/index.spec.js @@ -1,19 +1,19 @@ const OutputGeoServices = require('./index'); const FeatureServer = require('@koopjs/featureserver'); -// test.js jest.mock('@koopjs/featureserver', () => ({ setLogger: jest.fn(), route: jest.fn(), - setDefaults: jest.fn() + setDefaults: jest.fn(), })); const loggerMock = { silly: () => {}, - error: () => {} + error: () => {}, }; const modelMock = { + namespace: 'provider-name', pull: jest.fn((req, callback) => callback(null, 'someData')), }; @@ -51,12 +51,7 @@ describe('Output Geoservices', () => { handler: 'restInfoHandler', }, { - path: '$namespace/tokens/:method', - methods: ['get', 'post'], - handler: 'generateToken', - }, - { - path: '$namespace/tokens/', + path: '$namespace/rest/generateToken', methods: ['get', 'post'], handler: 'generateToken', }, @@ -92,24 +87,6 @@ describe('Output Geoservices', () => { }, ]); }); - - test('should include expected default properties', () => { - const output = new OutputGeoServices(modelMock); - expect(output.options).toEqual({}); - expect(output.logger).toBeDefined(); - expect(output.authInfo).toEqual({}); - expect(FeatureServer.setLogger.mock.calls.length).toBe(1); - }); - - test('should include properties with optional set values', () => { - const output = new OutputGeoServices(modelMock, { - logger: loggerMock, - authInfo: { food: 'baz' }, - }); - expect(Object.keys(output.logger)).toEqual(['silly', 'error']); - expect(output.authInfo).toEqual({ food: 'baz' }); - expect(FeatureServer.setLogger.mock.calls.length).toBe(1); - }); }); describe('generalHandler', () => { @@ -129,7 +106,6 @@ describe('Output Geoservices', () => { test('should authorize, then pull data and route', async () => { const modelMock = { pull: jest.fn((req, callback) => callback(null, 'someData')), - authorize: jest.fn(), }; const output = new OutputGeoServices(modelMock, { logger: loggerMock }); await output.generalHandler({ foo: 'bar' }, resMock); @@ -187,11 +163,10 @@ describe('Output Geoservices', () => { test('should handle required token error', async () => { const modelMock = { - pull: jest.fn((req, callback) => callback(null, 'someData')), - authorize: jest.fn(() => { + pull: jest.fn((req, callback) => { const err = new Error('no token'); err.code = 401; - throw err; + callback(err); }), }; const output = new OutputGeoServices(modelMock, { logger: loggerMock }); @@ -203,8 +178,9 @@ describe('Output Geoservices', () => { { error: { code: 499, - details: [], + details: ['Token Required'], message: 'Token Required', + messageCode: 'GWM_0003', }, }, ]); @@ -212,11 +188,10 @@ describe('Output Geoservices', () => { test('should handle invalid token error', async () => { const modelMock = { - pull: jest.fn((req, callback) => callback(null, 'someData')), - authorize: jest.fn(() => { + pull: jest.fn((req, callback) => { const err = new Error('invalid token'); err.code = 401; - throw err; + callback(err); }), }; const output = new OutputGeoServices(modelMock, { logger: loggerMock }); @@ -240,11 +215,10 @@ describe('Output Geoservices', () => { test('should handle invalid token error', async () => { const modelMock = { - pull: jest.fn((req, callback) => callback(null, 'someData')), - authorize: jest.fn(() => { + pull: jest.fn((req, callback) => { const err = new Error('Forbidden'); err.code = 403; - throw err; + callback(err); }), }; const output = new OutputGeoServices(modelMock, { logger: loggerMock }); @@ -291,51 +265,51 @@ describe('Output Geoservices', () => { }); describe('restInfoHandler', () => { - test('should return rest info', async () => { + test('should return rest info authInfo override', async () => { const output = new OutputGeoServices(modelMock, { authInfo: { food: 'baz' }, }); - await output.restInfoHandler({ foo: 'bar' }, resMock); + await output.restInfoHandler(reqMock, resMock); expect(FeatureServer.route.mock.calls.length).toBe(1); expect(FeatureServer.route.mock.calls[0]).toEqual([ - { foo: 'bar' }, + reqMock, resMock, - { authInfo: { food: 'baz' } }, + { + owningSystemUrl: 'https://some-host.com/api/v1/provider-name', + authInfo: { food: 'baz' }, + }, ]); }); - test('should return rest info with auth specification', async () => { + test('should return rest info with default authInfo, default https token url', async () => { const modelMock = { + namespace: 'provider-name', pull: jest.fn((req, callback) => callback(null, 'someData')), - authorize: jest.fn(), - authenticationSpecification: {}, }; - const output = new OutputGeoServices(modelMock, { - authInfo: { food: 'baz' }, - }); + const output = new OutputGeoServices(modelMock); await output.restInfoHandler(reqMock, resMock); expect(FeatureServer.route.mock.calls.length).toBe(1); expect(FeatureServer.route.mock.calls[0]).toEqual([ reqMock, resMock, { + owningSystemUrl: 'https://some-host.com/api/v1/provider-name', authInfo: { - food: 'baz', isTokenBasedSecurity: true, - tokenServicesUrl: 'https://some-host.com/api/v1/undefined/tokens/', + tokenServicesUrl: + 'https://some-host.com/api/v1/provider-name/rest/generateToken', }, }, ]); }); - test('should return rest info with auth specification, http', async () => { + test('should return rest info with default authInfo, http token url, set by option', async () => { const modelMock = { + namespace: 'provider-name', pull: jest.fn((req, callback) => callback(null, 'someData')), - authorize: jest.fn(), - authenticationSpecification: { useHttp: true }, }; const output = new OutputGeoServices(modelMock, { - authInfo: { food: 'baz' }, + useHttpForTokenUrl: true, }); await output.restInfoHandler(reqMock, resMock); expect(FeatureServer.route.mock.calls.length).toBe(1); @@ -343,10 +317,95 @@ describe('Output Geoservices', () => { reqMock, resMock, { + owningSystemUrl: 'http://some-host.com/api/v1/provider-name', authInfo: { - food: 'baz', isTokenBasedSecurity: true, - tokenServicesUrl: 'http://some-host.com/api/v1/undefined/tokens/', + tokenServicesUrl: + 'http://some-host.com/api/v1/provider-name/rest/generateToken', + }, + }, + ]); + }); + + test('should return rest info with default authInfo, http token url, set by GEOSERVICES_HTTP', async () => { + const modelMock = { + namespace: 'provider-name', + pull: jest.fn((req, callback) => callback(null, 'someData')), + }; + try { + process.env.GEOSERVICES_HTTP = 'true'; + const output = new OutputGeoServices(modelMock); + await output.restInfoHandler(reqMock, resMock); + expect(FeatureServer.route.mock.calls.length).toBe(1); + expect(FeatureServer.route.mock.calls[0]).toEqual([ + reqMock, + resMock, + { + owningSystemUrl: 'http://some-host.com/api/v1/provider-name', + authInfo: { + isTokenBasedSecurity: true, + tokenServicesUrl: + 'http://some-host.com/api/v1/provider-name/rest/generateToken', + }, + }, + ]); + } catch (error) { + expect(error).toBeUndefined(); + } finally { + delete process.env.GEOSERVICES_HTTP; + } + }); + + test('should return rest info with default authInfo, http token url, set by KOOP_AUTH_HTTP', async () => { + const modelMock = { + namespace: 'provider-name', + pull: jest.fn((req, callback) => callback(null, 'someData')), + }; + try { + process.env.KOOP_AUTH_HTTP = 'true'; + const output = new OutputGeoServices(modelMock); + await output.restInfoHandler(reqMock, resMock); + expect(FeatureServer.route.mock.calls.length).toBe(1); + expect(FeatureServer.route.mock.calls[0]).toEqual([ + reqMock, + resMock, + { + owningSystemUrl: 'http://some-host.com/api/v1/provider-name', + authInfo: { + isTokenBasedSecurity: true, + tokenServicesUrl: + 'http://some-host.com/api/v1/provider-name/rest/generateToken', + }, + }, + ]); + } catch (error) { + expect(error).toBeUndefined(); + } finally { + delete process.env.KOOP_AUTH_HTTP; + } + }); + + test('should return rest info with default authInfo, http token url, set by authenticationSpecification', async () => { + const modelMock = { + namespace: 'provider-name', + pull: jest.fn((req, callback) => callback(null, 'someData')), + authenticationSpecification: () => { + return { useHttp: true }; + }, + }; + + const output = new OutputGeoServices(modelMock); + await output.restInfoHandler(reqMock, resMock); + expect(FeatureServer.route.mock.calls.length).toBe(1); + expect(FeatureServer.route.mock.calls[0]).toEqual([ + reqMock, + resMock, + { + owningSystemUrl: 'http://some-host.com/api/v1/provider-name', + authInfo: { + isTokenBasedSecurity: true, + tokenServicesUrl: + 'http://some-host.com/api/v1/provider-name/rest/generateToken', }, }, ]); @@ -357,10 +416,9 @@ describe('Output Geoservices', () => { test('should generate token', async () => { const modelMock = { pull: jest.fn((req, callback) => callback(null, 'someData')), - authorize: jest.fn(), authenticate: jest.fn(() => { return { token: 'abc' }; - }) + }), }; const output = new OutputGeoServices(modelMock, { authInfo: { food: 'baz' }, @@ -369,38 +427,21 @@ describe('Output Geoservices', () => { expect(resMock.status.mock.calls.length).toBe(1); expect(resMock.status.mock.calls[0]).toEqual([200]); expect(resMock.json.mock.calls.length).toBe(1); - expect(resMock.json.mock.calls[0]).toEqual([{ - ssl: false, - token: 'abc' - }]); - }); - - test('should message that there is no authenticate method', async () => { - const modelMock = { - pull: jest.fn((req, callback) => callback(null, 'someData')), - authorize: jest.fn(), - }; - const output = new OutputGeoServices(modelMock, { - authInfo: { food: 'baz' }, - }); - await output.generateToken(reqMock, resMock); - expect(resMock.status.mock.calls.length).toBe(1); - expect(resMock.status.mock.calls[0]).toEqual([500]); - expect(resMock.json.mock.calls.length).toBe(1); - expect(resMock.json.mock.calls[0]).toEqual([{ - error: '"authenticate" not implemented for this provider' - }]); + expect(resMock.json.mock.calls[0]).toEqual([ + { + token: 'abc', + }, + ]); }); test('should fail to generate token due to 401', async () => { const modelMock = { pull: jest.fn((req, callback) => callback(null, 'someData')), - authorize: jest.fn(), authenticate: jest.fn(() => { const err = new Error('bad creds'); err.code = 401; throw err; - }) + }), }; const output = new OutputGeoServices(modelMock, { authInfo: { food: 'baz' }, @@ -409,28 +450,29 @@ describe('Output Geoservices', () => { expect(resMock.status.mock.calls.length).toBe(1); expect(resMock.status.mock.calls[0]).toEqual([200]); expect(resMock.json.mock.calls.length).toBe(1); - expect(resMock.json.mock.calls[0]).toEqual([{ - error: { - code: 400, - details: ['Invalid username or password.'], - message: 'Unable to generate token.' - } - }]); + expect(resMock.json.mock.calls[0]).toEqual([ + { + error: { + code: 400, + details: ['Invalid username or password.'], + message: 'Unable to generate token.', + }, + }, + ]); }); test('should fail to generate token due to credentials', async () => { const modelMock = { pull: jest.fn((req, callback) => callback(null, 'someData')), - authorize: jest.fn(), authenticate: jest.fn(() => { const err = { error: { code: 400, - message: 'Unable to generate token.' - } + message: 'Unable to generate token.', + }, }; throw err; - }) + }), }; const output = new OutputGeoServices(modelMock, { authInfo: { food: 'baz' }, @@ -439,24 +481,25 @@ describe('Output Geoservices', () => { expect(resMock.status.mock.calls.length).toBe(1); expect(resMock.status.mock.calls[0]).toEqual([200]); expect(resMock.json.mock.calls.length).toBe(1); - expect(resMock.json.mock.calls[0]).toEqual([{ - error: { - code: 400, - details: ['Invalid username or password.'], - message: 'Unable to generate token.' - } - }]); + expect(resMock.json.mock.calls[0]).toEqual([ + { + error: { + code: 400, + details: ['Invalid username or password.'], + message: 'Unable to generate token.', + }, + }, + ]); }); test('should fail to generate token due to 5xx', async () => { const modelMock = { pull: jest.fn((req, callback) => callback(null, 'someData')), - authorize: jest.fn(), authenticate: jest.fn(() => { const err = new Error('upstream'); err.code = 503; throw err; - }) + }), }; const output = new OutputGeoServices(modelMock, { authInfo: { food: 'baz' }, @@ -465,23 +508,24 @@ describe('Output Geoservices', () => { expect(resMock.status.mock.calls.length).toBe(1); expect(resMock.status.mock.calls[0]).toEqual([200]); expect(resMock.json.mock.calls.length).toBe(1); - expect(resMock.json.mock.calls[0]).toEqual([{ - error: { - code: 503, - details: [], - message: 'upstream' - } - }]); + expect(resMock.json.mock.calls[0]).toEqual([ + { + error: { + code: 503, + details: [], + message: 'upstream', + }, + }, + ]); }); test('should fail to generate token due to 500', async () => { const modelMock = { pull: jest.fn((req, callback) => callback(null, 'someData')), - authorize: jest.fn(), authenticate: jest.fn(() => { const err = new Error('upstream'); throw err; - }) + }), }; const output = new OutputGeoServices(modelMock, { authInfo: { food: 'baz' }, @@ -490,13 +534,15 @@ describe('Output Geoservices', () => { expect(resMock.status.mock.calls.length).toBe(1); expect(resMock.status.mock.calls[0]).toEqual([200]); expect(resMock.json.mock.calls.length).toBe(1); - expect(resMock.json.mock.calls[0]).toEqual([{ - error: { - code: 500, - details: [], - message: 'upstream' - } - }]); + expect(resMock.json.mock.calls[0]).toEqual([ + { + error: { + code: 500, + details: [], + message: 'upstream', + }, + }, + ]); }); }); }); diff --git a/test/geoservice-error-handling.spec.js b/test/geoservice-error-handling.spec.js index ce1dd9944..612083036 100644 --- a/test/geoservice-error-handling.spec.js +++ b/test/geoservice-error-handling.spec.js @@ -6,7 +6,7 @@ const mockLogger = { info: () => {}, silly: () => {}, warn: () => {}, - error: () => {} + error: () => {}, }; describe('geoservices error handling', () => { @@ -15,7 +15,8 @@ describe('geoservices error handling', () => { const koop = new Koop({ logLevel: 'error', logger: mockLogger }); koop.register(provider, { dataDir: './test/provider-data', - before: (req, callback) => { // eslint-disable-line + // eslint-disable-next-line + before: (req, callback) => { throw new Error('error in the provider'); }, }); @@ -42,7 +43,6 @@ describe('geoservices error handling', () => { let auth = require('@koopjs/auth-direct-file')( 'pass-in-your-secret', `${__dirname}/helpers/user-store.json`, - { useHttp: true }, ); koop.register(auth); koop.register(provider, { @@ -56,8 +56,9 @@ describe('geoservices error handling', () => { expect(response.body).toEqual({ error: { code: 499, - details: [], + details: ['Token Required'], message: 'Token Required', + messageCode: 'GWM_0003' }, }); } catch (error) { @@ -71,7 +72,6 @@ describe('geoservices error handling', () => { let auth = require('@koopjs/auth-direct-file')( 'pass-in-your-secret', `${__dirname}/helpers/user-store.json`, - { useHttp: true }, ); koop.register(auth); koop.register(provider, { @@ -102,14 +102,13 @@ describe('geoservices error handling', () => { let auth = require('@koopjs/auth-direct-file')( 'pass-in-your-secret', `${__dirname}/helpers/user-store.json`, - { useHttp: true }, ); koop.register(auth); koop.register(provider, { dataDir: './test/provider-data', }); try { - const response = await request(koop.server).get('/file-geojson/tokens'); + const response = await request(koop.server).get('/file-geojson/rest/generateToken'); expect(response.status).toBe(200); expect(response.body).toEqual({ error: {