Merge pull request #38 from strongloop/refactor-ds
Remove inheritence from DataSource to ModelBuilder
This commit is contained in:
commit
13659ca19d
|
@ -791,11 +791,10 @@ var defineScope = require('./scope.js').defineScope;
|
|||
/**
|
||||
* Define scope
|
||||
*/
|
||||
DataAccessObject.scope = function (name, filter) {
|
||||
defineScope(this, this, name, filter);
|
||||
DataAccessObject.scope = function (name, filter, targetClass) {
|
||||
defineScope(this, targetClass || this, name, filter);
|
||||
};
|
||||
|
||||
|
||||
// jutil.mixin(DataAccessObject, validations.Validatable);
|
||||
jutil.mixin(DataAccessObject, Inclusion);
|
||||
jutil.mixin(DataAccessObject, Relation);
|
||||
|
|
|
@ -53,7 +53,7 @@ var slice = Array.prototype.slice;
|
|||
* });
|
||||
* ```
|
||||
*/
|
||||
function DataSource(name, settings) {
|
||||
function DataSource(name, settings, modelBuilder) {
|
||||
if (!(this instanceof DataSource)) {
|
||||
return new DataSource(name, settings);
|
||||
}
|
||||
|
@ -74,7 +74,9 @@ function DataSource(name, settings) {
|
|||
settings = utils.parseSettings(settings);
|
||||
}
|
||||
|
||||
ModelBuilder.call(this, name, settings);
|
||||
this.modelBuilder = modelBuilder || new ModelBuilder();
|
||||
this.models = this.modelBuilder.models;
|
||||
this.definitions = this.modelBuilder.definitions;
|
||||
|
||||
// operation metadata
|
||||
// Initialize it before calling setup as the connector might register operations
|
||||
|
@ -113,8 +115,7 @@ function DataSource(name, settings) {
|
|||
var fn = this.DataAccessObject.prototype[name];
|
||||
|
||||
if(typeof fn === 'function') {
|
||||
var returns = fn.returns;
|
||||
|
||||
|
||||
this.defineOperation(name, {
|
||||
prototype: true,
|
||||
accepts: fn.accepts,
|
||||
|
@ -128,17 +129,12 @@ function DataSource(name, settings) {
|
|||
}.bind(this));
|
||||
}
|
||||
|
||||
util.inherits(DataSource, ModelBuilder);
|
||||
util.inherits(DataSource, EventEmitter);
|
||||
|
||||
// allow child classes to supply a data access object
|
||||
DataSource.DataAccessObject = DataAccessObject;
|
||||
|
||||
// Copy over statics
|
||||
for (var m in ModelBuilder) {
|
||||
if (!DataSource.hasOwnProperty(m) && 'super_' !== m) {
|
||||
DataSource[m] = ModelBuilder[m];
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Set up the connector instance for backward compatibility with JugglingDB schema/adapter
|
||||
|
@ -312,6 +308,10 @@ function isModelClass(cls) {
|
|||
|
||||
DataSource.relationTypes = ['belongsTo', 'hasMany', 'hasAndBelongsToMany'];
|
||||
|
||||
|
||||
function isModelDataSourceAttached(model) {
|
||||
return model && (!model.settings.unresolved) && (model.dataSource instanceof DataSource);
|
||||
}
|
||||
/*!
|
||||
* Define relations for the model class from the relations object
|
||||
* @param modelClass
|
||||
|
@ -321,10 +321,10 @@ DataSource.prototype.defineRelations = function(modelClass, relations) {
|
|||
|
||||
// Create a function for the closure in the loop
|
||||
var createListener = function (name, relation, targetModel, throughModel) {
|
||||
if (targetModel && targetModel.settings.unresolved) {
|
||||
targetModel.once('defined', function (model) {
|
||||
if (!isModelDataSourceAttached(targetModel)) {
|
||||
targetModel.once('dataSourceAttached', function (model) {
|
||||
// Check if the through model doesn't exist or resolved
|
||||
if (!throughModel || !throughModel.settings.unresolved) {
|
||||
if (!throughModel || isModelDataSourceAttached(throughModel)) {
|
||||
// The target model is resolved
|
||||
var params = {
|
||||
foreignKey: relation.foreignKey,
|
||||
|
@ -337,11 +337,12 @@ DataSource.prototype.defineRelations = function(modelClass, relations) {
|
|||
modelClass[relation.type].call(modelClass, name, params);
|
||||
}
|
||||
});
|
||||
|
||||
}
|
||||
if (throughModel && throughModel.settings.unresolved) {
|
||||
if (throughModel && !isModelDataSourceAttached(throughModel)) {
|
||||
// Set up a listener to the through model
|
||||
throughModel.once('defined', function (model) {
|
||||
if (!targetModel.settings.unresolved) {
|
||||
throughModel.once('dataSourceAttached', function (model) {
|
||||
if (isModelDataSourceAttached(targetModel)) {
|
||||
// The target model is resolved
|
||||
var params = {
|
||||
foreignKey: relation.foreignKey,
|
||||
|
@ -360,20 +361,20 @@ DataSource.prototype.defineRelations = function(modelClass, relations) {
|
|||
for (var rn in relations) {
|
||||
var r = relations[rn];
|
||||
assert(DataSource.relationTypes.indexOf(r.type) !== -1, "Invalid relation type: " + r.type);
|
||||
var targetModel = isModelClass(r.model) ? r.model : this.models[r.model];
|
||||
var targetModel = isModelClass(r.model) ? r.model : this.getModel(r.model);
|
||||
if (!targetModel) {
|
||||
// The target model doesn't exist, let create a place holder for it
|
||||
targetModel = this.define(r.model, {}, {unresolved: true});
|
||||
}
|
||||
var throughModel = null;
|
||||
if (r.through) {
|
||||
throughModel = isModelClass(r.through) ? r.through : this.models[r.through];
|
||||
throughModel = isModelClass(r.through) ? r.through : this.getModel(r.through);
|
||||
if (!throughModel) {
|
||||
// The through model doesn't exist, let create a place holder for it
|
||||
throughModel = this.define(r.through, {}, {unresolved: true});
|
||||
}
|
||||
}
|
||||
if (targetModel.settings.unresolved || (throughModel && throughModel.settings.unresolved)) {
|
||||
if (!isModelDataSourceAttached(targetModel) || (throughModel && !isModelDataSourceAttached(throughModel))) {
|
||||
// Create a listener to defer the relation set up
|
||||
createListener(rn, r, targetModel, throughModel);
|
||||
} else {
|
||||
|
@ -475,12 +476,16 @@ DataSource.prototype.createModel = DataSource.prototype.define = function define
|
|||
}
|
||||
}
|
||||
|
||||
var modelClass = ModelBuilder.prototype.define.call(this, className, properties, settings);
|
||||
var modelClass = this.modelBuilder.define(className, properties, settings);
|
||||
modelClass.dataSource = this;
|
||||
|
||||
if(settings.unresolved) {
|
||||
return modelClass;
|
||||
}
|
||||
|
||||
this.setupDataAccess(modelClass, settings);
|
||||
modelClass.emit('dataSourceAttached', modelClass);
|
||||
|
||||
return modelClass;
|
||||
};
|
||||
|
||||
|
@ -526,6 +531,14 @@ DataSource.prototype.mixin = function (ModelCtor) {
|
|||
});
|
||||
};
|
||||
|
||||
DataSource.prototype.getModel = function(name) {
|
||||
return this.modelBuilder.getModel(name);
|
||||
};
|
||||
|
||||
DataSource.prototype.getModelDefinition = function(name) {
|
||||
return this.modelBuilder.getModelDefinition(name);
|
||||
};
|
||||
|
||||
/**
|
||||
* Attach an existing model to a data source.
|
||||
*
|
||||
|
@ -538,19 +551,22 @@ DataSource.prototype.attach = function (modelClass) {
|
|||
return;
|
||||
}
|
||||
var className = modelClass.modelName;
|
||||
var properties = modelClass.dataSource.definitions[className].properties;
|
||||
var settings = modelClass.dataSource.definitions[className].settings;
|
||||
var modelDef = modelClass.dataSource.getModelDefinition(className);
|
||||
var properties = modelDef.properties;
|
||||
var settings = modelDef.settings;
|
||||
|
||||
// redefine the dataSource
|
||||
modelClass.dataSource = this;
|
||||
// add to def
|
||||
var def = new ModelDefinition(this, className, properties, settings);
|
||||
var def = new ModelDefinition(this.modelBuilder, className, properties, settings);
|
||||
def.build();
|
||||
this.definitions[className] = def;
|
||||
this.models[className] = modelClass;
|
||||
this.modelBuilder.definitions[className] = def;
|
||||
this.modelBuilder.models[className] = modelClass;
|
||||
|
||||
this.setupDataAccess(modelClass, settings);
|
||||
|
||||
modelClass.emit('dataSourceAttached', modelClass);
|
||||
|
||||
return this;
|
||||
};
|
||||
|
||||
|
@ -558,13 +574,13 @@ DataSource.prototype.attach = function (modelClass) {
|
|||
* Define single property named `prop` on `model`
|
||||
*
|
||||
* @param {String} model - name of model
|
||||
* @param {String} prop - name of propery
|
||||
* @param {String} prop - name of property
|
||||
* @param {Object} params - property settings
|
||||
*/
|
||||
DataSource.prototype.defineProperty = function (model, prop, params) {
|
||||
ModelBuilder.prototype.defineProperty.call(this, model, prop, params);
|
||||
this.modelBuilder.defineProperty(model, prop, params);
|
||||
|
||||
var resolvedProp = this.definitions[model].properties[prop];
|
||||
var resolvedProp = this.getModelDefinition(model).properties[prop];
|
||||
if (this.connector.defineProperty) {
|
||||
this.connector.defineProperty(model, prop, resolvedProp);
|
||||
}
|
||||
|
@ -1225,7 +1241,7 @@ DataSource.prototype.discoverAndBuildModels = function (modelName, options, cb)
|
|||
schemaList.push(schema);
|
||||
}
|
||||
|
||||
var models = self.buildModels(schemaList);
|
||||
var models = self.modelBuilder.buildModels(schemaList);
|
||||
cb && cb(err, models);
|
||||
});
|
||||
};
|
||||
|
@ -1252,7 +1268,7 @@ DataSource.prototype.discoverAndBuildModelsSync = function (modelName, options)
|
|||
schemaList.push(schema);
|
||||
}
|
||||
|
||||
var models = this.buildModels(schemaList);
|
||||
var models = this.modelBuilder.buildModels(schemaList);
|
||||
return models;
|
||||
};
|
||||
|
||||
|
@ -1305,7 +1321,7 @@ DataSource.prototype.freeze = function freeze() {
|
|||
* @param {String} modelName The model name
|
||||
*/
|
||||
DataSource.prototype.tableName = function (modelName) {
|
||||
return this.definitions[modelName].tableName(this.connector.name);
|
||||
return this.getModelDefinition(modelName).tableName(this.connector.name);
|
||||
};
|
||||
|
||||
/**
|
||||
|
@ -1315,7 +1331,7 @@ DataSource.prototype.tableName = function (modelName) {
|
|||
* @returns {String} columnName
|
||||
*/
|
||||
DataSource.prototype.columnName = function (modelName, propertyName) {
|
||||
return this.definitions[modelName].columnName(this.connector.name, propertyName);
|
||||
return this.getModelDefinition(modelName).columnName(this.connector.name, propertyName);
|
||||
};
|
||||
|
||||
/**
|
||||
|
@ -1325,7 +1341,7 @@ DataSource.prototype.columnName = function (modelName, propertyName) {
|
|||
* @returns {Object} column metadata
|
||||
*/
|
||||
DataSource.prototype.columnMetadata = function (modelName, propertyName) {
|
||||
return this.definitions[modelName].columnMetadata(this.connector.name, propertyName);
|
||||
return this.getModelDefinition(modelName).columnMetadata(this.connector.name, propertyName);
|
||||
};
|
||||
|
||||
/**
|
||||
|
@ -1334,7 +1350,7 @@ DataSource.prototype.columnMetadata = function (modelName, propertyName) {
|
|||
* @returns {String[]} column names
|
||||
*/
|
||||
DataSource.prototype.columnNames = function (modelName) {
|
||||
return this.definitions[modelName].columnNames(this.connector.name);
|
||||
return this.getModelDefinition(modelName).columnNames(this.connector.name);
|
||||
};
|
||||
|
||||
/**
|
||||
|
@ -1343,7 +1359,7 @@ DataSource.prototype.columnNames = function (modelName) {
|
|||
* @returns {String} columnName for ID
|
||||
*/
|
||||
DataSource.prototype.idColumnName = function(modelName) {
|
||||
return this.definitions[modelName].idColumnName(this.connector.name);
|
||||
return this.getModelDefinition(modelName).idColumnName(this.connector.name);
|
||||
};
|
||||
|
||||
/**
|
||||
|
@ -1352,10 +1368,10 @@ DataSource.prototype.idColumnName = function(modelName) {
|
|||
* @returns {String} property name for ID
|
||||
*/
|
||||
DataSource.prototype.idName = function(modelName) {
|
||||
if(!this.definitions[modelName].idName) {
|
||||
console.log(this.definitions[modelName]);
|
||||
if(!this.getModelDefinition(modelName).idName) {
|
||||
console.error('No id name', this.getModelDefinition(modelName));
|
||||
}
|
||||
return this.definitions[modelName].idName();
|
||||
return this.getModelDefinition(modelName).idName();
|
||||
};
|
||||
|
||||
/**
|
||||
|
@ -1364,7 +1380,7 @@ DataSource.prototype.idName = function(modelName) {
|
|||
* @returns {String[]} property names for IDs
|
||||
*/
|
||||
DataSource.prototype.idNames = function (modelName) {
|
||||
return this.definitions[modelName].idNames();
|
||||
return this.getModelDefinition(modelName).idNames();
|
||||
};
|
||||
|
||||
|
||||
|
@ -1376,7 +1392,7 @@ DataSource.prototype.idNames = function (modelName) {
|
|||
*/
|
||||
DataSource.prototype.defineForeignKey = function defineForeignKey(className, key, foreignClassName) {
|
||||
// quit if key already defined
|
||||
if (this.definitions[className].rawProperties[key]) return;
|
||||
if (this.getModelDefinition(className).rawProperties[key]) return;
|
||||
|
||||
if (this.connector.defineForeignKey) {
|
||||
var cb = function (err, keyType) {
|
||||
|
@ -1428,7 +1444,7 @@ DataSource.prototype.disconnect = function disconnect(cb) {
|
|||
DataSource.prototype.copyModel = function copyModel(Master) {
|
||||
var dataSource = this;
|
||||
var className = Master.modelName;
|
||||
var md = Master.dataSource.definitions[className];
|
||||
var md = Master.dataSource.getModelDefinition(className);
|
||||
var Slave = function SlaveModel() {
|
||||
Master.apply(this, [].slice.call(arguments));
|
||||
};
|
||||
|
@ -1442,11 +1458,11 @@ DataSource.prototype.copyModel = function copyModel(Master) {
|
|||
hiddenProperty(Slave, 'modelName', className);
|
||||
hiddenProperty(Slave, 'relations', Master.relations);
|
||||
|
||||
if (!(className in dataSource.models)) {
|
||||
if (!(className in dataSource.modelBuilder.models)) {
|
||||
|
||||
// store class in model pool
|
||||
dataSource.models[className] = Slave;
|
||||
dataSource.definitions[className] = new ModelDefinition(dataSource, md.name, md.properties, md.settings);
|
||||
dataSource.modelBuilder.models[className] = Slave;
|
||||
dataSource.modelBuilder.definitions[className] = new ModelDefinition(dataSource.modelBuilder, md.name, md.properties, md.settings);
|
||||
|
||||
if ((!dataSource.isTransaction) && dataSource.connector && dataSource.connector.define) {
|
||||
dataSource.connector.define({
|
||||
|
@ -1483,11 +1499,12 @@ DataSource.prototype.transaction = function() {
|
|||
transaction.connector = dataSource.connector.transaction();
|
||||
|
||||
// create blank models pool
|
||||
transaction.models = {};
|
||||
transaction.definitions = {};
|
||||
transaction.modelBuilder = new ModelBuilder();
|
||||
transaction.models = transaction.modelBuilder.models;
|
||||
transaction.definitions = transaction.modelBuilder.definitions;
|
||||
|
||||
for (var i in dataSource.models) {
|
||||
dataSource.copyModel.call(transaction, dataSource.models[i]);
|
||||
for (var i in dataSource.modelBuilder.models) {
|
||||
dataSource.copyModel.call(transaction, dataSource.modelBuilder.models[i]);
|
||||
}
|
||||
|
||||
transaction.exec = function(cb) {
|
||||
|
|
|
@ -55,6 +55,14 @@ function isModelClass(cls) {
|
|||
return cls.prototype instanceof DefaultModelBaseClass;
|
||||
}
|
||||
|
||||
ModelBuilder.prototype.getModel = function(name) {
|
||||
return this.models[name];
|
||||
};
|
||||
|
||||
ModelBuilder.prototype.getModelDefinition = function(name) {
|
||||
return this.definitions[name];
|
||||
};
|
||||
|
||||
/**
|
||||
* Define a model class
|
||||
*
|
||||
|
@ -146,7 +154,7 @@ ModelBuilder.prototype.define = function defineClass(className, properties, sett
|
|||
var events = new EventEmitter();
|
||||
for (var f in EventEmitter.prototype) {
|
||||
if (typeof EventEmitter.prototype[f] === 'function') {
|
||||
ModelClass[f] = events[f].bind(events);
|
||||
ModelClass[f] = EventEmitter.prototype[f].bind(events);
|
||||
}
|
||||
}
|
||||
util.inherits(ModelClass, ModelBaseClass);
|
||||
|
@ -170,9 +178,10 @@ ModelBuilder.prototype.define = function defineClass(className, properties, sett
|
|||
|
||||
// inherit ModelBaseClass static methods
|
||||
for (var i in ModelBaseClass) {
|
||||
if(i !== '_mixins') {
|
||||
ModelClass[i] = ModelBaseClass[i];
|
||||
}
|
||||
// We need to skip properties that are already in the subclass, for example, the event emitter methods
|
||||
if(i !== '_mixins' && !(i in ModelClass)) {
|
||||
ModelClass[i] = ModelBaseClass[i];
|
||||
}
|
||||
}
|
||||
|
||||
ModelClass.getter = {};
|
||||
|
|
|
@ -34,9 +34,9 @@ Relation.hasMany = function hasMany(anotherClass, params) {
|
|||
anotherClass = params.model;
|
||||
} else {
|
||||
var anotherClassName = i8n.singularize(anotherClass).toLowerCase();
|
||||
for(var name in this.dataSource.models) {
|
||||
for(var name in this.dataSource.modelBuilder.models) {
|
||||
if (name.toLowerCase() === anotherClassName) {
|
||||
anotherClass = this.dataSource.models[name];
|
||||
anotherClass = this.dataSource.modelBuilder.models[name];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -187,9 +187,9 @@ Relation.belongsTo = function (anotherClass, params) {
|
|||
anotherClass = params.model;
|
||||
} else {
|
||||
var anotherClassName = anotherClass.toLowerCase();
|
||||
for(var name in this.dataSource.models) {
|
||||
for(var name in this.dataSource.modelBuilder.models) {
|
||||
if (name.toLowerCase() === anotherClassName) {
|
||||
anotherClass = this.dataSource.models[name];
|
||||
anotherClass = this.dataSource.modelBuilder.models[name];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -271,7 +271,7 @@ Relation.belongsTo = function (anotherClass, params) {
|
|||
*/
|
||||
Relation.hasAndBelongsToMany = function hasAndBelongsToMany(anotherClass, params) {
|
||||
params = params || {};
|
||||
var models = this.dataSource.models;
|
||||
var models = this.dataSource.modelBuilder.models;
|
||||
|
||||
if ('string' === typeof anotherClass) {
|
||||
params.as = anotherClass;
|
||||
|
|
|
@ -2,13 +2,14 @@
|
|||
var should = require('./init.js');
|
||||
|
||||
var Schema = require('../').Schema;
|
||||
var ModelBuilder = require('../').ModelBuilder;
|
||||
|
||||
describe('JSON property', function() {
|
||||
var dataSource, Model;
|
||||
|
||||
it('should be defined', function() {
|
||||
dataSource = getSchema();
|
||||
Model = dataSource.define('Model', {propertyName: Schema.JSON});
|
||||
Model = dataSource.define('Model', {propertyName: ModelBuilder.JSON});
|
||||
var m = new Model;
|
||||
(new Boolean('propertyName' in m)).should.eql(true);
|
||||
should.not.exist(m.propertyName);
|
||||
|
|
|
@ -200,7 +200,7 @@ describe('DataSource define model', function () {
|
|||
// define models
|
||||
var Post = ds.define('Post', {
|
||||
title: { type: String, length: 255 },
|
||||
content: { type: DataSource.Text },
|
||||
content: { type: ModelBuilder.Text },
|
||||
date: { type: Date, default: function () {
|
||||
return new Date();
|
||||
} },
|
||||
|
@ -211,7 +211,7 @@ describe('DataSource define model', function () {
|
|||
// simpler way to describe model
|
||||
var User = ds.define('User', {
|
||||
name: String,
|
||||
bio: DataSource.Text,
|
||||
bio: ModelBuilder.Text,
|
||||
approved: Boolean,
|
||||
joinedAt: Date,
|
||||
age: Number
|
||||
|
@ -592,7 +592,7 @@ describe('Load models from json', function () {
|
|||
// Read the dataSource JSON file
|
||||
var schemas = JSON.parse(fs.readFileSync(schemaFile));
|
||||
|
||||
return dataSource.buildModels(schemas);
|
||||
return dataSource.modelBuilder.buildModels(schemas);
|
||||
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue