2014-07-10 16:32:58 +00:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
var modelHelper = require('../lib/model-helper');
|
2015-07-02 18:03:27 +00:00
|
|
|
var _defaults = require('lodash').defaults;
|
2014-08-05 06:22:41 +00:00
|
|
|
var loopback = require('loopback');
|
2014-07-10 16:32:58 +00:00
|
|
|
var expect = require('chai').expect;
|
|
|
|
|
|
|
|
describe('model-helper', function() {
|
|
|
|
describe('properly converts LDL definitions to swagger types', function() {
|
|
|
|
it('converts constructor types', function() {
|
2014-07-24 20:35:02 +00:00
|
|
|
var def = buildSwaggerModels({
|
2014-07-10 16:32:58 +00:00
|
|
|
str: String, // 'string'
|
|
|
|
num: Number, // {type: 'number', format: 'double'}
|
|
|
|
date: Date, // {type: 'string', format: 'date'}
|
|
|
|
bool: Boolean, // 'boolean'
|
|
|
|
buf: Buffer // {type: 'string', format: 'byte'}
|
|
|
|
});
|
|
|
|
var props = def.properties;
|
2014-07-11 13:55:19 +00:00
|
|
|
expect(props.str).to.eql({ type: 'string' });
|
|
|
|
expect(props.num).to.eql({ type: 'number', format: 'double' });
|
|
|
|
expect(props.date).eql({ type: 'string', format: 'date' });
|
|
|
|
expect(props.bool).to.eql({ type: 'boolean' });
|
|
|
|
expect(props.buf).to.eql({ type: 'string', format: 'byte' });
|
2014-07-10 16:32:58 +00:00
|
|
|
});
|
|
|
|
it('converts string types', function() {
|
2014-07-24 20:35:02 +00:00
|
|
|
var def = buildSwaggerModels({
|
2014-07-10 16:32:58 +00:00
|
|
|
str: 'string', // 'string'
|
|
|
|
num: 'number', // {type: 'number', format: 'double'}
|
|
|
|
date: 'date', // {type: 'string', format: 'date'}
|
|
|
|
bool: 'boolean', // 'boolean'
|
|
|
|
buf: 'buffer' // {type: 'string', format: 'byte'}
|
|
|
|
});
|
|
|
|
var props = def.properties;
|
2014-07-11 13:55:19 +00:00
|
|
|
expect(props.str).to.eql({ type: 'string' });
|
|
|
|
expect(props.num).to.eql({ type: 'number', format: 'double' });
|
|
|
|
expect(props.date).eql({ type: 'string', format: 'date' });
|
|
|
|
expect(props.bool).to.eql({ type: 'boolean' });
|
|
|
|
expect(props.buf).to.eql({ type: 'string', format: 'byte' });
|
2014-07-10 16:32:58 +00:00
|
|
|
});
|
|
|
|
describe('array definitions', function() {
|
|
|
|
// There are three types we want to checK:
|
|
|
|
// [String]
|
|
|
|
// ["string"],
|
|
|
|
// [{type: String, ...}]
|
|
|
|
it('converts [Constructor] type', function() {
|
2014-07-24 20:35:02 +00:00
|
|
|
var def = buildSwaggerModels({
|
2014-07-10 16:32:58 +00:00
|
|
|
array: [String]
|
|
|
|
});
|
|
|
|
var props = def.properties;
|
2014-07-11 13:55:19 +00:00
|
|
|
expect(props.array).to.eql({ type: 'array', items: {
|
|
|
|
type: 'string'
|
|
|
|
}});
|
2014-07-10 16:32:58 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('converts ["string"] type', function() {
|
2014-07-24 20:35:02 +00:00
|
|
|
var def = buildSwaggerModels({
|
2014-07-10 16:32:58 +00:00
|
|
|
array: ['string']
|
|
|
|
});
|
|
|
|
var props = def.properties;
|
2014-07-11 13:55:19 +00:00
|
|
|
expect(props.array).to.eql({ type: 'array', items: {
|
|
|
|
type: 'string'
|
|
|
|
}});
|
2014-07-10 16:32:58 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('converts [{type: "string", length: 64}] type', function() {
|
2014-07-24 20:35:02 +00:00
|
|
|
var def = buildSwaggerModels({
|
2014-07-10 16:32:58 +00:00
|
|
|
array: [{type: 'string', length: 64}]
|
|
|
|
});
|
|
|
|
var props = def.properties;
|
2014-07-11 13:55:19 +00:00
|
|
|
expect(props.array).to.eql({ type: 'array', items: {
|
|
|
|
type: 'string',
|
|
|
|
length: 64
|
|
|
|
}});
|
2014-07-10 16:32:58 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('converts [{type: "date"}] type (with `format`)', function() {
|
2014-07-24 20:35:02 +00:00
|
|
|
var def = buildSwaggerModels({
|
2014-07-10 16:32:58 +00:00
|
|
|
array: [{type: 'date'}]
|
|
|
|
});
|
|
|
|
var props = def.properties;
|
2014-07-11 13:55:19 +00:00
|
|
|
expect(props.array).to.eql({ type: 'array', items: {
|
|
|
|
type: 'string', format: 'date'
|
|
|
|
}});
|
2014-07-10 16:32:58 +00:00
|
|
|
});
|
2014-07-22 19:34:42 +00:00
|
|
|
|
|
|
|
it('converts [] type', function() {
|
2014-07-24 20:35:02 +00:00
|
|
|
var def = buildSwaggerModels({
|
2014-07-22 19:34:42 +00:00
|
|
|
array: []
|
|
|
|
});
|
|
|
|
var prop = def.properties.array;
|
2014-07-31 22:52:52 +00:00
|
|
|
expect(prop).to.eql({
|
|
|
|
type: 'array',
|
|
|
|
items: { type: 'any' }
|
|
|
|
});
|
2014-07-22 19:34:42 +00:00
|
|
|
});
|
|
|
|
|
2014-07-22 19:52:25 +00:00
|
|
|
it('converts [undefined] type', function() {
|
2014-07-24 20:35:02 +00:00
|
|
|
var def = buildSwaggerModels({
|
2014-07-22 19:52:25 +00:00
|
|
|
// This value is somehow provided by loopback-boot called from
|
|
|
|
// loopback-workspace.
|
|
|
|
array: [undefined]
|
|
|
|
});
|
|
|
|
var prop = def.properties.array;
|
2014-07-31 22:52:52 +00:00
|
|
|
expect(prop).to.eql({ type: 'array', items: { type: 'any' } });
|
2014-07-22 19:52:25 +00:00
|
|
|
});
|
|
|
|
|
2014-07-22 19:34:42 +00:00
|
|
|
it('converts "array" type', function() {
|
2014-07-24 20:35:02 +00:00
|
|
|
var def = buildSwaggerModels({
|
2014-07-22 19:34:42 +00:00
|
|
|
array: 'array'
|
|
|
|
});
|
|
|
|
var prop = def.properties.array;
|
2014-07-31 22:52:52 +00:00
|
|
|
expect(prop).to.eql({ type: 'array', items: { type: 'any' } });
|
2014-07-22 19:34:42 +00:00
|
|
|
});
|
2014-08-05 06:22:41 +00:00
|
|
|
|
2015-04-07 00:00:32 +00:00
|
|
|
it('converts Model type to $ref', function() {
|
2014-08-05 06:22:41 +00:00
|
|
|
var Address = loopback.createModel('Address', {street: String});
|
|
|
|
var def = buildSwaggerModels({
|
|
|
|
str: String,
|
|
|
|
address: Address
|
|
|
|
});
|
|
|
|
var prop = def.properties.address;
|
2015-04-07 02:57:57 +00:00
|
|
|
expect(prop).to.eql({ $ref: 'Address' });
|
2014-08-05 06:22:41 +00:00
|
|
|
});
|
|
|
|
|
2014-07-10 16:32:58 +00:00
|
|
|
});
|
2015-07-02 18:03:27 +00:00
|
|
|
|
|
|
|
it('converts model property field `doc`', function() {
|
|
|
|
var def = buildSwaggerModels({
|
|
|
|
name: { type: String, doc: 'a-description' }
|
|
|
|
});
|
|
|
|
var nameProp = def.properties.name;
|
|
|
|
expect(nameProp).to.have.property('description', 'a-description');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('converts model property field `description`', function() {
|
|
|
|
var def = buildSwaggerModels({
|
|
|
|
name: { type: String, description: 'a-description' }
|
|
|
|
});
|
|
|
|
var nameProp = def.properties.name;
|
|
|
|
expect(nameProp).to.have.property('description', 'a-description');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('converts model field `description`', function() {
|
|
|
|
var def = buildSwaggerModels({}, { description: 'a-description' });
|
|
|
|
expect(def).to.have.property('description', 'a-description');
|
|
|
|
});
|
2014-07-10 16:32:58 +00:00
|
|
|
});
|
2014-08-28 22:20:03 +00:00
|
|
|
|
2014-07-24 20:35:02 +00:00
|
|
|
describe('related models', function() {
|
2014-08-28 22:20:03 +00:00
|
|
|
it('should include related models', function () {
|
2014-07-24 20:35:02 +00:00
|
|
|
var defs = buildSwaggerModelsWithRelations({
|
|
|
|
str: String // 'string'
|
|
|
|
});
|
|
|
|
expect(defs).has.property('testModel');
|
|
|
|
expect(defs).has.property('relatedModel');
|
|
|
|
});
|
2014-08-28 22:20:03 +00:00
|
|
|
|
|
|
|
it('should include nesting models', function() {
|
|
|
|
var Model2 = loopback.createModel('Model2', {street: String});
|
|
|
|
var Model1 = loopback.createModel('Model1', {
|
|
|
|
str: String, // 'string'
|
|
|
|
address: Model2
|
2015-07-02 18:03:27 +00:00
|
|
|
}, { models: { Model2: Model2 } });
|
2014-08-28 22:20:03 +00:00
|
|
|
var defs = modelHelper.generateModelDefinition(Model1, {});
|
|
|
|
expect(defs).has.property('Model1');
|
|
|
|
expect(defs).has.property('Model2');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should include used models', function() {
|
|
|
|
var Model4 = loopback.createModel('Model4', {street: String});
|
|
|
|
var Model3 = loopback.createModel('Model3', {
|
2015-07-02 18:03:27 +00:00
|
|
|
str: String // 'string'
|
2014-08-28 22:20:03 +00:00
|
|
|
}, {models: {model4: 'Model4'}});
|
|
|
|
var defs = modelHelper.generateModelDefinition(Model3, {});
|
|
|
|
expect(defs).has.property('Model3');
|
|
|
|
expect(defs).has.property('Model4');
|
|
|
|
});
|
2014-08-29 08:06:05 +00:00
|
|
|
|
|
|
|
it('should include nesting models in array', function() {
|
|
|
|
var Model6 = loopback.createModel('Model6', {street: String});
|
|
|
|
var Model5 = loopback.createModel('Model5', {
|
|
|
|
str: String, // 'string'
|
|
|
|
addresses: [Model6]
|
2015-07-02 18:03:27 +00:00
|
|
|
}, { models: { Model6: Model6 } });
|
2014-08-29 08:06:05 +00:00
|
|
|
var defs = modelHelper.generateModelDefinition(Model5, {});
|
|
|
|
expect(defs).has.property('Model5');
|
|
|
|
expect(defs).has.property('Model6');
|
|
|
|
});
|
2014-09-19 16:35:11 +00:00
|
|
|
|
|
|
|
// https://github.com/strongloop/loopback-explorer/issues/49
|
|
|
|
it('should work if Array class is extended and no related models are found',
|
|
|
|
function() {
|
|
|
|
var Model7 = loopback.createModel('Model7', {street: String});
|
|
|
|
Array.prototype.customFunc = function() {
|
|
|
|
};
|
|
|
|
var defs = modelHelper.generateModelDefinition(Model7, {});
|
|
|
|
expect(defs).has.property('Model7');
|
|
|
|
expect(Object.keys(defs)).has.property('length', 1);
|
|
|
|
});
|
|
|
|
|
2015-07-02 18:03:27 +00:00
|
|
|
// https://github.com/strongloop/loopback-explorer/issues/71
|
|
|
|
it('should skip unknown types', function() {
|
|
|
|
var Model8 = loopback.createModel('Model8', {
|
|
|
|
patient: {
|
|
|
|
model: 'physician',
|
|
|
|
type: 'hasMany',
|
|
|
|
through: 'appointment'
|
|
|
|
}
|
|
|
|
});
|
|
|
|
var defs = modelHelper.generateModelDefinition(Model8, {});
|
|
|
|
expect(Object.keys(defs)).to.not.contain('hasMany');
|
|
|
|
});
|
2014-07-24 20:35:02 +00:00
|
|
|
});
|
2014-08-28 22:20:03 +00:00
|
|
|
|
2014-07-26 17:12:05 +00:00
|
|
|
describe('hidden properties', function() {
|
|
|
|
it('should hide properties marked as "hidden"', function() {
|
|
|
|
var aClass = createModelCtor({
|
|
|
|
visibleProperty: 'string',
|
|
|
|
hiddenProperty: 'string'
|
|
|
|
});
|
|
|
|
aClass.ctor.definition.settings = {
|
|
|
|
hidden: ['hiddenProperty']
|
|
|
|
};
|
|
|
|
var def = modelHelper.generateModelDefinition(aClass.ctor, {}).testModel;
|
|
|
|
expect(def.properties).to.not.have.property('hiddenProperty');
|
|
|
|
expect(def.properties).to.have.property('visibleProperty');
|
|
|
|
});
|
|
|
|
});
|
2015-04-07 22:53:56 +00:00
|
|
|
|
2015-07-02 18:03:27 +00:00
|
|
|
describe('#generateModelDefinition', function() {
|
|
|
|
it('should convert top level array description to string', function () {
|
2015-04-07 22:53:56 +00:00
|
|
|
var model = {};
|
|
|
|
model.definition = {
|
|
|
|
name: 'test',
|
2015-07-02 18:03:27 +00:00
|
|
|
description: ['1', '2', '3'],
|
2015-04-07 22:53:56 +00:00
|
|
|
properties: {}
|
|
|
|
};
|
|
|
|
var models = {};
|
|
|
|
modelHelper.generateModelDefinition(model, models);
|
2015-07-02 18:03:27 +00:00
|
|
|
expect(models.test.description).to.equal("1\n2\n3");
|
2015-04-07 22:53:56 +00:00
|
|
|
});
|
|
|
|
|
2015-07-02 18:03:27 +00:00
|
|
|
it('should convert property level array description to string', function () {
|
2015-04-07 22:53:56 +00:00
|
|
|
var model = {};
|
|
|
|
model.definition = {
|
|
|
|
name: 'test',
|
|
|
|
properties: {
|
|
|
|
prop1: {
|
|
|
|
type: 'string',
|
2015-07-02 18:03:27 +00:00
|
|
|
description: ['1', '2', '3']
|
2015-04-07 22:53:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
var models = {};
|
|
|
|
modelHelper.generateModelDefinition(model, models);
|
2015-07-02 18:03:27 +00:00
|
|
|
expect(models.test.properties.prop1.description).to.equal("1\n2\n3");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('getPropType', function() {
|
|
|
|
it('converts anonymous object types', function() {
|
|
|
|
var type = modelHelper.getPropType({ name: 'string', value: 'string' });
|
|
|
|
expect(type).to.eql('object');
|
2015-04-07 22:53:56 +00:00
|
|
|
});
|
|
|
|
});
|
2014-07-10 16:32:58 +00:00
|
|
|
});
|
|
|
|
|
2014-07-11 13:55:19 +00:00
|
|
|
// Simulates the format of a remoting class.
|
2015-07-02 18:03:27 +00:00
|
|
|
function buildSwaggerModels(modelProperties, modelOptions) {
|
|
|
|
var aClass = createModelCtor(modelProperties, modelOptions);
|
2014-07-26 17:12:05 +00:00
|
|
|
return modelHelper.generateModelDefinition(aClass.ctor, {}).testModel;
|
|
|
|
}
|
|
|
|
|
2015-07-02 18:03:27 +00:00
|
|
|
function createModelCtor(properties, modelOptions) {
|
|
|
|
Object.keys(properties).forEach(function(name) {
|
|
|
|
var type = properties[name];
|
|
|
|
if (typeof type !== 'object' || Array.isArray(type))
|
|
|
|
properties[name] = { type: type };
|
2014-07-24 20:35:02 +00:00
|
|
|
});
|
2015-07-02 18:03:27 +00:00
|
|
|
|
|
|
|
var definition = {
|
|
|
|
name: 'testModel',
|
|
|
|
properties: properties
|
|
|
|
};
|
|
|
|
_defaults(definition, modelOptions);
|
|
|
|
|
2014-07-24 20:35:02 +00:00
|
|
|
var aClass = {
|
|
|
|
ctor: {
|
2015-07-02 18:03:27 +00:00
|
|
|
definition: definition
|
2014-07-24 20:35:02 +00:00
|
|
|
}
|
|
|
|
};
|
2014-07-26 17:12:05 +00:00
|
|
|
return aClass;
|
2014-07-24 20:35:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function buildSwaggerModelsWithRelations(model) {
|
2014-07-10 16:32:58 +00:00
|
|
|
Object.keys(model).forEach(function(name) {
|
|
|
|
model[name] = {type: model[name]};
|
|
|
|
});
|
2014-07-24 20:35:02 +00:00
|
|
|
// Mock up the related model
|
|
|
|
var relatedModel = {
|
|
|
|
definition: {
|
|
|
|
name: 'relatedModel',
|
|
|
|
properties: {
|
|
|
|
fk: String
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2014-07-10 16:32:58 +00:00
|
|
|
var aClass = {
|
|
|
|
ctor: {
|
|
|
|
definition: {
|
|
|
|
name: 'testModel',
|
|
|
|
properties: model
|
2014-07-24 20:35:02 +00:00
|
|
|
},
|
|
|
|
// Mock up relations
|
|
|
|
relations: {
|
|
|
|
other: {
|
|
|
|
modelTo: relatedModel
|
|
|
|
}
|
2014-07-10 16:32:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2014-07-24 20:35:02 +00:00
|
|
|
return modelHelper.generateModelDefinition(aClass.ctor, {});
|
2014-07-10 16:32:58 +00:00
|
|
|
}
|
2014-07-24 20:35:02 +00:00
|
|
|
|