Merge branch 'master' into fix/examples

* master:
  Add feature to hide disabled remote methods after explorer is initialized
  More fixes of indentation in index.js
  Fix broken indentation
  Fix linting errors
  Auto-update by eslint --fix
  Add eslint infrastructure
  2.4.0
  Add `swaggerUI` option to enable/disable UI serving
  2.3.0
  2.2.0
  remove references to ubuntu font
  Update swaggerObject when a new model was added

# Conflicts:
#	example/hidden.js
#	example/simple.js
This commit is contained in:
Alexander Ryzhikov 2016-05-01 17:39:59 +03:00
commit 5385d3aaa2
14 changed files with 193 additions and 94 deletions

0
.eslintignore Normal file
View File

10
.eslintrc Normal file
View File

@ -0,0 +1,10 @@
{
"extends": "loopback",
"rules": {
"max-len": ["error", 90, 4, {
"ignoreComments": true,
"ignoreUrls": true,
"ignorePattern": "^\\s*var\\s.+=\\s*(require\\s*\\()|(/)"
}]
}
}

View File

@ -1,13 +0,0 @@
{
"node": true,
"camelcase" : true,
"eqnull" : true,
"indent": 2,
"undef": true,
"quotmark": "single",
"maxlen": 80,
"trailing": true,
"newcap": true,
"nonew": true,
"undef": false
}

View File

@ -1,3 +1,24 @@
2016-03-08, Version 2.4.0
=========================
* Add `swaggerUI` option to enable/disable UI serving (Raymond Feng)
2016-02-02, Version 2.3.0
=========================
2016-01-13, Version 2.2.0
=========================
* remove references to ubuntu font (Anthony Ettinger)
* Update swaggerObject when a new model was added (Pradeep Kumar Tippa)
* Refer to licenses with a link (Sam Roberts)
2015-10-01, Version 2.1.1 2015-10-01, Version 2.1.1
========================= =========================

View File

@ -7,13 +7,13 @@ var User = loopback.Model.extend('user', {
username: 'string', username: 'string',
email: 'string', email: 'string',
sensitiveInternalProperty: 'string', sensitiveInternalProperty: 'string',
}, {hidden: ['sensitiveInternalProperty']}); }, { hidden: ['sensitiveInternalProperty'] });
User.attachTo(loopback.memory()); User.attachTo(loopback.memory());
app.model(User); app.model(User);
var apiPath = '/api'; var apiPath = '/api';
explorer(app, {basePath: apiPath}); explorer(app, { basePath: apiPath });
app.use(apiPath, loopback.rest()); app.use(apiPath, loopback.rest());
console.log('Explorer mounted at localhost:' + port + '/explorer'); console.log('Explorer mounted at localhost:' + port + '/explorer');

View File

@ -4,15 +4,15 @@ var explorer = require('../');
var port = 3000; var port = 3000;
var Product = loopback.PersistedModel.extend('product', { var Product = loopback.PersistedModel.extend('product', {
foo: {type: 'string', required: true}, foo: { type: 'string', required: true },
bar: 'string', bar: 'string',
aNum: {type: 'number', min: 1, max: 10, required: true, default: 5} aNum: { type: 'number', min: 1, max: 10, required: true, default: 5 },
}); });
Product.attachTo(loopback.memory()); Product.attachTo(loopback.memory());
app.model(Product); app.model(Product);
var apiPath = '/api'; var apiPath = '/api';
explorer(app, {basePath: apiPath}); explorer(app, { basePath: apiPath });
app.use(apiPath, loopback.rest()); app.use(apiPath, loopback.rest());
console.log('Explorer mounted at http://localhost:' + port + '/explorer'); console.log('Explorer mounted at http://localhost:' + port + '/explorer');

View File

@ -1,4 +1,4 @@
'use strict'; 'use strict';
/*! /*!
* Adds dynamically-updated docs as /explorer * Adds dynamically-updated docs as /explorer
*/ */
@ -30,7 +30,7 @@ function explorer(loopbackApplication, options) {
function routes(loopbackApplication, options) { function routes(loopbackApplication, options) {
var loopback = loopbackApplication.loopback; var loopback = loopbackApplication.loopback;
var loopbackMajor = loopback && loopback.version && var loopbackMajor = loopback && loopback.version &&
loopback.version.split('.')[0] || 1; loopback.version.split('.')[0] || 1;
if (loopbackMajor < 2) { if (loopbackMajor < 2) {
throw new Error('loopback-component-explorer requires loopback 2.0 or newer'); throw new Error('loopback-component-explorer requires loopback 2.0 or newer');
@ -38,7 +38,8 @@ function routes(loopbackApplication, options) {
options = _defaults({}, options, { options = _defaults({}, options, {
resourcePath: 'swagger.json', resourcePath: 'swagger.json',
apiInfo: loopbackApplication.get('apiInfo') || {} apiInfo: loopbackApplication.get('apiInfo') || {},
swaggerUI: true,
}); });
var router = new loopback.Router(); var router = new loopback.Router();
@ -56,30 +57,32 @@ function routes(loopbackApplication, options) {
source = req.originalUrl.replace(/\/config.json(\?.*)?$/, ''); source = req.originalUrl.replace(/\/config.json(\?.*)?$/, '');
} }
res.send({ res.send({
url: urlJoin(source, '/' + options.resourcePath) url: urlJoin(source, '/' + options.resourcePath),
}); });
}); });
// Allow specifying a static file roots for swagger files. Any files in if (options.swaggerUI) {
// these folders will override those in the swagger-ui distribution. // Allow specifying a static file roots for swagger files. Any files in
// In this way one could e.g. make changes to index.html without having // these folders will override those in the swagger-ui distribution.
// to worry about constantly pulling in JS updates. // In this way one could e.g. make changes to index.html without having
if (options.uiDirs) { // to worry about constantly pulling in JS updates.
if (typeof options.uiDirs === 'string') { if (options.uiDirs) {
router.use(loopback.static(options.uiDirs)); if (typeof options.uiDirs === 'string') {
} else if (Array.isArray(options.uiDirs)) { router.use(loopback.static(options.uiDirs));
options.uiDirs.forEach(function(dir) { } else if (Array.isArray(options.uiDirs)) {
router.use(loopback.static(dir)); 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));
} }
// 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; return router;
} }
@ -95,6 +98,18 @@ function routes(loopbackApplication, options) {
function mountSwagger(loopbackApplication, swaggerApp, opts) { function mountSwagger(loopbackApplication, swaggerApp, opts) {
var swaggerObject = createSwaggerObject(loopbackApplication, 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'; var resourcePath = opts && opts.resourcePath || 'swagger.json';
if (resourcePath[0] !== '/') resourcePath = '/' + resourcePath; if (resourcePath[0] !== '/') resourcePath = '/' + resourcePath;
@ -108,7 +123,7 @@ function mountSwagger(loopbackApplication, swaggerApp, opts) {
function setupCors(swaggerApp, remotes) { function setupCors(swaggerApp, remotes) {
var corsOptions = remotes.options && remotes.options.cors || var corsOptions = remotes.options && remotes.options.cors ||
{ origin: true, credentials: true }; { origin: true, credentials: true };
// TODO(bajtos) Skip CORS when remotes.options.cors === false // TODO(bajtos) Skip CORS when remotes.options.cors === false
swaggerApp.use(cors(corsOptions)); swaggerApp.use(cors(corsOptions));

View File

@ -1,10 +1,12 @@
{ {
"name": "loopback-component-explorer", "name": "loopback-component-explorer",
"version": "2.1.1", "version": "2.4.0",
"description": "Browse and test your LoopBack app's APIs", "description": "Browse and test your LoopBack app's APIs",
"main": "index.js", "main": "index.js",
"scripts": { "scripts": {
"test": "mocha" "lint": "eslint .",
"test": "mocha",
"posttest": "npm run lint"
}, },
"repository": { "repository": {
"type": "git", "type": "git",
@ -21,10 +23,12 @@
"url": "https://github.com/strongloop/loopback-component-explorer/issues" "url": "https://github.com/strongloop/loopback-component-explorer/issues"
}, },
"devDependencies": { "devDependencies": {
"chai": "^3.2.0",
"eslint": "^2.8.0",
"eslint-config-loopback": "^2.0.0",
"loopback": "^2.19.1", "loopback": "^2.19.1",
"mocha": "^2.2.5", "mocha": "^2.2.5",
"supertest": "^1.0.1", "supertest": "^1.0.1"
"chai": "^3.2.0"
}, },
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {

View File

@ -1,37 +1,5 @@
/* Styles used for loopback explorer customizations */ /* Styles used for loopback explorer customizations */
/* Custom font */
@font-face {
font-family: 'Ubuntu';
font-style: normal;
font-weight: 300;
src: local('Ubuntu Light'),
local('Ubuntu-Light'),
url(../fonts/_aijTyevf54tkVDLy-dlnLO3LdcAZYWl9Si6vvxL-qU.woff)
format('woff');
}
@font-face {
font-family: 'Ubuntu';
font-style: normal;
font-weight: 500;
src: local('Ubuntu Medium'),
local('Ubuntu-Medium'),
url(../fonts/OsJ2DjdpjqFRVUSto6IffLO3LdcAZYWl9Si6vvxL-qU.woff)
format('woff');
}
@font-face {
font-family: 'Ubuntu';
font-style: normal;
font-weight: 700;
src: local('Ubuntu Bold'),
local('Ubuntu-Bold'),
url(../fonts/0ihfXUL2emPh0ROJezvraLO3LdcAZYWl9Si6vvxL-qU.woff)
format('woff');
}
.swagger-section .swagger-ui-wrap, .swagger-section .swagger-ui-wrap,
.swagger-section .swagger-ui-wrap b, .swagger-section .swagger-ui-wrap b,
.swagger-section .swagger-ui-wrap strong, .swagger-section .swagger-ui-wrap strong,
@ -43,13 +11,13 @@
.swagger-section .swagger-ui-wrap p#colophon, .swagger-section .swagger-ui-wrap p#colophon,
.swagger-section .swagger-ui-wrap .markdown ul .swagger-section .swagger-ui-wrap .markdown ul
.model-signature { .model-signature {
font-family: "Ubuntu", sans-serif !important; font-family: 'helvetica neue', helvetica, arial, sans-serif !important;
} }
/* layout spacing and global colors */ /* layout spacing and global colors */
body { body {
padding-top: 60px; padding-top: 60px;
font-family: "Ubuntu", sans-serif; font-family: 'helvetica neue', helvetica, arial, sans-serif;
} }
body #header { body #header {

View File

@ -4,7 +4,7 @@
/*global SwaggerUi, log, ApiKeyAuthorization, hljs, window, $ */ /*global SwaggerUi, log, ApiKeyAuthorization, hljs, window, $ */
$(function() { $(function() {
// Pre load translate... // Pre load translate...
if(window.SwaggerTranslator) { if (window.SwaggerTranslator) {
window.SwaggerTranslator.translate(); window.SwaggerTranslator.translate();
} }
@ -17,6 +17,7 @@ $(function() {
var accessToken; var accessToken;
function loadSwaggerUi(config) { function loadSwaggerUi(config) {
var methodOrder = ['get', 'head', 'options', 'put', 'post', 'delete']; var methodOrder = ['get', 'head', 'options', 'put', 'post', 'delete'];
/* eslint-disable camelcase */
window.swaggerUi = new SwaggerUi({ window.swaggerUi = new SwaggerUi({
validatorUrl: null, validatorUrl: null,
url: config.url || '/swagger/resources', url: config.url || '/swagger/resources',
@ -28,7 +29,7 @@ $(function() {
log(swaggerApi); log(swaggerApi);
log(swaggerUi); log(swaggerUi);
if(window.SwaggerTranslator) { if (window.SwaggerTranslator) {
window.SwaggerTranslator.translate(); window.SwaggerTranslator.translate();
} }
@ -56,8 +57,9 @@ $(function() {
return pathCompare !== 0 ? return pathCompare !== 0 ?
pathCompare : pathCompare :
methodOrder.indexOf(a.method) - methodOrder.indexOf(b.method); methodOrder.indexOf(a.method) - methodOrder.indexOf(b.method);
} },
}); });
/* eslint-disable camelcase */
$('#explore').click(setAccessToken); $('#explore').click(setAccessToken);
$('#api_selector').submit(setAccessToken); $('#api_selector').submit(setAccessToken);
@ -71,9 +73,10 @@ $(function() {
e.preventDefault(); e.preventDefault();
var key = $('#input_accessToken')[0].value; var key = $('#input_accessToken')[0].value;
log('key: ' + key); log('key: ' + key);
if(key && key.trim() !== '') { if (key && key.trim() !== '') {
log('added accessToken ' + key); log('added accessToken ' + key);
var apiKeyAuth = new SwaggerClient.ApiKeyAuthorization('access_token', key, 'query'); var apiKeyAuth =
new SwaggerClient.ApiKeyAuthorization('access_token', key, 'query');
window.swaggerUi.api.clientAuthorizations.add('key', apiKeyAuth); window.swaggerUi.api.clientAuthorizations.add('key', apiKeyAuth);
accessToken = key; accessToken = key;
$('.accessTokenDisplay').text('Token Set.').addClass('set'); $('.accessTokenDisplay').text('Token Set.').addClass('set');
@ -83,10 +86,9 @@ $(function() {
if (window.localStorage) { if (window.localStorage) {
window.localStorage.setItem(lsKey, key); window.localStorage.setItem(lsKey, key);
} }
} } else {
// If submitted with an empty token, remove the current token. Can be // If submitted with an empty token, remove the current token. Can be
// useful to intentionally remove authorization. // useful to intentionally remove authorization.
else {
log('removed accessToken.'); log('removed accessToken.');
$('.accessTokenDisplay').text('Token Not Set.').removeClass('set'); $('.accessTokenDisplay').text('Token Not Set.').removeClass('set');
$('.accessTokenDisplay').removeAttr('data-tooltip'); $('.accessTokenDisplay').removeAttr('data-tooltip');
@ -116,5 +118,3 @@ $(function() {
} }
} }
}); });

View File

@ -8,7 +8,6 @@ var urlJoin = require('../lib/url-join');
var os = require('os'); var os = require('os');
describe('explorer', function() { describe('explorer', function() {
describe('with default config', function() { describe('with default config', function() {
beforeEach(givenLoopBackAppWithExplorer()); beforeEach(givenLoopBackAppWithExplorer());
@ -117,7 +116,7 @@ describe('explorer', function() {
beforeEach(function setupExplorerWithUiDirs() { beforeEach(function setupExplorerWithUiDirs() {
app = loopback(); app = loopback();
explorer(app, { explorer(app, {
uiDirs: [path.resolve(__dirname, 'fixtures', 'dummy-swagger-ui')] uiDirs: [path.resolve(__dirname, 'fixtures', 'dummy-swagger-ui')],
}); });
}); });
@ -138,6 +137,39 @@ describe('explorer', function() {
}); });
}); });
describe('with swaggerUI option', function() {
var app;
beforeEach(function setupExplorerWithoutUI() {
app = loopback();
explorer(app, {
swaggerUI: false,
});
});
it('overrides swagger-ui files', function(done) {
request(app).get('/explorer/swagger-ui.js')
.expect(404, done);
});
it('should serve config.json', function(done) {
request(app)
.get('/explorer/config.json')
.expect(200)
.end(function(err, res) {
if (err) return done(err);
expect(res.body).to
.have.property('url', '/explorer/swagger.json');
done();
});
});
it('should serve swagger.json', function(done) {
request(app)
.get('/explorer/swagger.json')
.expect(200, done);
});
});
describe('explorer.routes API', function() { describe('explorer.routes API', function() {
var app; var app;
beforeEach(function() { beforeEach(function() {
@ -167,7 +199,7 @@ describe('explorer', function() {
it('should allow `uiDirs` to be defined as an Array', function(done) { it('should allow `uiDirs` to be defined as an Array', function(done) {
explorer(app, { explorer(app, {
uiDirs: [path.resolve(__dirname, 'fixtures', 'dummy-swagger-ui')] uiDirs: [path.resolve(__dirname, 'fixtures', 'dummy-swagger-ui')],
}); });
request(app).get('/explorer/') request(app).get('/explorer/')
@ -179,7 +211,7 @@ describe('explorer', function() {
it('should allow `uiDirs` to be defined as an String', function(done) { it('should allow `uiDirs` to be defined as an String', function(done) {
explorer(app, { explorer(app, {
uiDirs: path.resolve(__dirname, 'fixtures', 'dummy-swagger-ui') uiDirs: path.resolve(__dirname, 'fixtures', 'dummy-swagger-ui'),
}); });
request(app).get('/explorer/') request(app).get('/explorer/')
@ -205,7 +237,7 @@ describe('explorer', function() {
it('can be disabled by configuration', function(done) { it('can be disabled by configuration', function(done) {
var app = loopback(); var app = loopback();
app.set('remoting', { cors: { origin: false } }); app.set('remoting', { cors: { origin: false }});
configureRestApiAndExplorer(app, '/explorer'); configureRestApiAndExplorer(app, '/explorer');
request(app) request(app)
@ -220,6 +252,68 @@ describe('explorer', function() {
}); });
}); });
it('updates swagger object when a new model is added', function(done) {
var app = loopback();
configureRestApiAndExplorer(app, '/explorer');
// Ensure the swagger object was built
request(app)
.get('/explorer/swagger.json')
.expect(200)
.end(function(err) {
if (err) return done(err);
// Create a new model
var Model = loopback.PersistedModel.extend('Customer');
Model.attachTo(loopback.memory());
app.model(Model);
// Request swagger.json again
request(app)
.get('/explorer/swagger.json')
.expect(200)
.end(function(err, res) {
if (err) return done(err);
var modelNames = Object.keys(res.body.definitions);
expect(modelNames).to.contain('Customer');
var paths = Object.keys(res.body.paths);
expect(paths).to.have.contain('/Customers');
done();
});
});
});
it('updates swagger object when a remote method is disabled', function(done) {
var app = loopback();
configureRestApiAndExplorer(app, '/explorer');
// Ensure the swagger object was built
request(app)
.get('/explorer/swagger.json')
.expect(200)
.end(function(err, res) {
if (err) return done(err);
// Check the method that will be disabled
var paths = Object.keys(res.body.paths);
expect(paths).to.contain('/products/findOne');
var Product = app.models.Product;
Product.disableRemoteMethod('findOne', true);
// Request swagger.json again
request(app)
.get('/explorer/swagger.json')
.expect(200)
.end(function(err, res) {
if (err) return done(err);
var paths = Object.keys(res.body.paths);
expect(paths).to.not.contain('/products/findOne');
done();
});
});
});
function givenLoopBackAppWithExplorer(explorerBase) { function givenLoopBackAppWithExplorer(explorerBase) {
return function(done) { return function(done) {
var app = this.app = loopback(); var app = this.app = loopback();