214 lines
7.9 KiB
JavaScript
214 lines
7.9 KiB
JavaScript
// Copyright IBM Corp. 2013,2016. All Rights Reserved.
|
|
// Node module: loopback-component-explorer
|
|
// This file is licensed under the MIT License.
|
|
// License text available at https://opensource.org/licenses/MIT
|
|
|
|
'use strict';
|
|
|
|
var SG = require('strong-globalize');
|
|
SG.SetRootDir(__dirname);
|
|
var g = SG();
|
|
|
|
/*!
|
|
* Adds dynamically-updated docs as /explorer
|
|
*/
|
|
var deprecated = require('depd')('loopback-explorer');
|
|
var url = require('url');
|
|
var path = require('path');
|
|
var urlJoin = require('./lib/url-join');
|
|
var _defaults = require('lodash').defaults;
|
|
var _ = require('lodash');
|
|
var cors = require('cors');
|
|
var createSwaggerObject = require('loopback-swagger').generateSwaggerSpec;
|
|
var SWAGGER_UI_ROOT = require('swagger-ui/index').dist;
|
|
var STATIC_ROOT = path.join(__dirname, 'public');
|
|
|
|
module.exports = explorer;
|
|
explorer.routes = routes;
|
|
|
|
/**
|
|
* Example usage:
|
|
*
|
|
* var explorer = require('loopback-component-explorer');
|
|
* explorer(app, options);
|
|
*/
|
|
|
|
function explorer(loopbackApplication, options) {
|
|
options = _defaults({}, options, { mountPath: '/explorer' });
|
|
loopbackApplication.use(options.mountPath, routes(loopbackApplication, options));
|
|
loopbackApplication.set('loopback-component-explorer', options);
|
|
}
|
|
|
|
function routes(loopbackApplication, options) {
|
|
var loopback = loopbackApplication.loopback;
|
|
var loopbackMajor = loopback && loopback.version &&
|
|
loopback.version.split('.')[0] || 1;
|
|
|
|
if (loopbackMajor < 2) {
|
|
throw new Error(g.f('{{loopback-component-explorer}} requires ' +
|
|
'{{loopback}} 2.0 or newer'));
|
|
}
|
|
|
|
options = _defaults({}, options, {
|
|
resourcePath: 'swagger.json',
|
|
apiInfo: loopbackApplication.get('apiInfo') || {},
|
|
swaggerUI: true,
|
|
});
|
|
|
|
var router = new loopback.Router();
|
|
|
|
mountSwagger(loopbackApplication, router, options);
|
|
|
|
// config.json is loaded by swagger-ui. The server should respond
|
|
// with the relative URI of the resource doc.
|
|
router.get('/config.json', function(req, res) {
|
|
// Get the path we're mounted at. It's best to get this from the referer
|
|
// in case we're proxied at a deep path.
|
|
var source = url.parse(req.headers.referer || '').pathname;
|
|
// If no referer is available, use the incoming url.
|
|
if (!source) {
|
|
source = req.originalUrl.replace(/\/config.json(\?.*)?$/, '');
|
|
}
|
|
res.send({
|
|
url: urlJoin(source, '/' + options.resourcePath),
|
|
});
|
|
});
|
|
|
|
if (options.swaggerUI) {
|
|
// Allow specifying a static file roots for swagger files. Any files in
|
|
// these folders will override those in the swagger-ui distribution.
|
|
// In this way one could e.g. make changes to index.html without having
|
|
// to worry about constantly pulling in JS updates.
|
|
if (options.uiDirs) {
|
|
if (typeof options.uiDirs === 'string') {
|
|
router.use(loopback.static(options.uiDirs));
|
|
} else if (Array.isArray(options.uiDirs)) {
|
|
options.uiDirs.forEach(function(dir) {
|
|
router.use(loopback.static(dir));
|
|
});
|
|
}
|
|
}
|
|
|
|
// File in node_modules are overridden by a few customizations
|
|
router.use(loopback.static(STATIC_ROOT));
|
|
|
|
// Swagger UI distribution
|
|
router.use(loopback.static(SWAGGER_UI_ROOT));
|
|
}
|
|
|
|
return router;
|
|
}
|
|
|
|
/**
|
|
* Setup Swagger documentation on the given express app.
|
|
*
|
|
* @param {Application} loopbackApplication The loopback application to
|
|
* document.
|
|
* @param {Application} swaggerApp Swagger application used for hosting
|
|
* swagger documentation.
|
|
* @param {Object} opts Options.
|
|
*/
|
|
function mountSwagger(loopbackApplication, swaggerApp, opts) {
|
|
var swaggerObject = createSwaggerObject(loopbackApplication, opts);
|
|
|
|
// listening to modelRemoted event for updating the swaggerObject
|
|
// with the newly created model to appear in the Swagger UI.
|
|
loopbackApplication.on('modelRemoted', function() {
|
|
swaggerObject = createSwaggerObject(loopbackApplication, opts);
|
|
});
|
|
|
|
// listening to remoteMethodDisabled event for updating the swaggerObject
|
|
// when a remote method is disabled to hide that method in the Swagger UI.
|
|
loopbackApplication.on('remoteMethodDisabled', function() {
|
|
swaggerObject = createSwaggerObject(loopbackApplication, opts);
|
|
});
|
|
|
|
var resourcePath = opts && opts.resourcePath || 'swagger.json';
|
|
if (resourcePath[0] !== '/') resourcePath = '/' + resourcePath;
|
|
|
|
var remotes = loopbackApplication.remotes();
|
|
setupCors(swaggerApp, remotes);
|
|
|
|
swaggerApp.get(resourcePath, function sendSwaggerObject(req, res) {
|
|
var tenantId = require('loopback-context').getCurrentContext().get('tenantId');
|
|
var swaggerFilterPath;
|
|
var verbs = ['head', 'get', 'post', 'put', 'patch', 'delete'];
|
|
if (tenantId) {
|
|
swaggerFilterPath = '/custom/' + tenantId;
|
|
if (req.query && req.query.model) {
|
|
const modelName = tenantId + '-' + req.query.model;
|
|
const modelDef = req.app.models[modelName];
|
|
if (modelDef && modelDef.definition && modelDef.definition.settings && modelDef.definition.settings.slug) {
|
|
swaggerFilterPath += '-' + modelDef.definition.settings.slug;
|
|
} else swaggerFilterPath += '-' + req.query.model;
|
|
}
|
|
|
|
if (swaggerObject && swaggerObject.paths) {
|
|
var filteredSwaggerObject = JSON.parse(JSON.stringify(swaggerObject));
|
|
filteredSwaggerObject.paths = _.reduce(_.pickBy(swaggerObject.paths, function(val, key) {
|
|
return key.indexOf(swaggerFilterPath) === 0;
|
|
}), function(result, val, key) {
|
|
var newKey = key.replace(tenantId + '-', '');
|
|
result[newKey] = val;
|
|
_.each(verbs, function(verb) {
|
|
if (result[newKey][verb] && result[newKey][verb].tags) {
|
|
result[newKey][verb].tags = _.map(result[newKey][verb].tags, function(tag){
|
|
return tag.replace(tenantId + '-', '');
|
|
});
|
|
}
|
|
if (verb === 'get' && result[newKey].get) {
|
|
if (result[newKey].get.parameters) {
|
|
if (result[newKey].get.parameters[0].name === 'filter') {
|
|
result[newKey].get.parameters[0].default = JSON.stringify({limit: 10});
|
|
}
|
|
} else {
|
|
result[newKey].get.parameters = [];
|
|
}
|
|
var nonElasticOperationIds = ['.prototype.__get__model', '.prototype.__get__user', '.prototype.__get__tenant'];
|
|
var nonElasticOperationId = nonElasticOperationIds.filter(opId => _.includes(result[newKey].get.operationId, opId))[0];
|
|
if (!nonElasticOperationId && _.endsWith(newKey, '/change-stream') && _.isArray(result[newKey].get.parameters) && _.findIndex(result[newKey].get.parameters, { name: 'getFrom', enum: ['elastic'] }) === -1) {
|
|
result[newKey].get.parameters.push({
|
|
name: 'getFrom',
|
|
in: 'query',
|
|
required: false,
|
|
type: 'string',
|
|
enum: ['elastic'],
|
|
description: 'Gets the resource from the indexed elastic store rather than the default'
|
|
});
|
|
}
|
|
}
|
|
});
|
|
return result;
|
|
}, {});
|
|
const includeDefaultDefinitions = ['ObjectID', 'x-any'];
|
|
filteredSwaggerObject.definitions = _.pickBy(swaggerObject.definitions, function(val, key) {
|
|
return key.startsWith(tenantId) || includeDefaultDefinitions.indexOf(key) !== -1;
|
|
});
|
|
res.status(200).send(filteredSwaggerObject);
|
|
} else {
|
|
res.status(200).send(swaggerObject);
|
|
}
|
|
} else {
|
|
res.status(401).send({});
|
|
}
|
|
});
|
|
}
|
|
|
|
function setupCors(swaggerApp, remotes) {
|
|
var corsOptions = remotes.options && remotes.options.cors;
|
|
if (corsOptions === false)
|
|
return;
|
|
|
|
deprecated(g.f(
|
|
'The built-in CORS middleware provided by loopback-component-explorer ' +
|
|
'was deprecated. See %s for more details.',
|
|
'https://docs.strongloop.com/display/public/LB/Security+considerations'
|
|
));
|
|
|
|
if (corsOptions === undefined) {
|
|
corsOptions = { origin: true, credentials: true };
|
|
}
|
|
|
|
swaggerApp.use(cors(corsOptions));
|
|
}
|