Merge pull request #564 from strongloop/feature/allow-filter-on-findById

Extend findById to accept an optional filter object
This commit is contained in:
Raymond Feng 2015-04-16 11:03:03 -07:00
commit fdd8ee4ca6
2 changed files with 139 additions and 8 deletions

View File

@ -685,26 +685,44 @@ DataAccessObject.exists = function exists(id, options, cb) {
* ``` * ```
* *
* @param {*} id Primary key value * @param {*} id Primary key value
* @param {Object} [filter] The filter that contains `include` or `fields`.
* Other settings such as `where`, `order`, `limit`, or `offset` will be
* ignored.
* @param {Object} [options] Options * @param {Object} [options] Options
* @param {Function} cb Callback called with (err, instance) * @param {Function} cb Callback called with (err, instance)
*/ */
DataAccessObject.findById = function find(id, options, cb) { DataAccessObject.findById = function find(id, filter, options, cb) {
var connectionPromise = stillConnecting(this.getDataSource(), this, arguments); var connectionPromise = stillConnecting(this.getDataSource(), this, arguments);
if (connectionPromise) { if (connectionPromise) {
return connectionPromise; return connectionPromise;
} }
assert(arguments.length >= 1, 'The id argument is required'); assert(arguments.length >= 1, 'The id argument is required');
if (cb === undefined) {
if (typeof options === 'function') { if (options === undefined && cb === undefined) {
if (typeof filter === 'function') {
// findById(id, cb) // findById(id, cb)
cb = filter;
filter = {};
}
} else if (cb === undefined) {
if (typeof options === 'function') {
// findById(id, query, cb)
cb = options; cb = options;
options = {}; options = {};
if (typeof filter === 'object' && !(filter.include || filter.fields)) {
// If filter doesn't have include or fields, assuming it's options
options = filter;
filter = {};
}
} }
} }
cb = cb || utils.createPromiseCallback(); cb = cb || utils.createPromiseCallback();
options = options || {}; options = options || {};
filter = filter || {};
assert(typeof filter === 'object', 'The filter argument must be an object');
assert(typeof options === 'object', 'The options argument must be an object'); assert(typeof options === 'object', 'The options argument must be an object');
assert(typeof cb === 'function', 'The cb argument must be a function'); assert(typeof cb === 'function', 'The cb argument must be a function');
@ -713,7 +731,14 @@ DataAccessObject.findById = function find(id, options, cb) {
cb(new Error('Model::findById requires the id argument')); cb(new Error('Model::findById requires the id argument'));
}); });
} else { } else {
this.findOne(byIdQuery(this, id), options, cb); var query = byIdQuery(this, id);
if (filter.include) {
query.include = filter.include;
}
if (filter.fields) {
query.fields = filter.fields;
}
this.findOne(query, options, cb);
} }
return cb.promise; return cb.promise;
}; };

View File

@ -1,7 +1,7 @@
// This test written in mocha+should.js // This test written in mocha+should.js
var should = require('./init.js'); var should = require('./init.js');
var async = require('async'); var async = require('async');
var db, User, options; var db, User, options, filter;
describe('crud-with-options', function () { describe('crud-with-options', function () {
@ -17,6 +17,7 @@ describe('crud-with-options', function () {
vip: {type: Boolean} vip: {type: Boolean}
}); });
options = {}; options = {};
filter = {fields: ['name', 'id']};
db.automigrate(done); db.automigrate(done);
@ -36,23 +37,48 @@ describe('crud-with-options', function () {
}); });
}); });
it('should allow findById(id, filter, cb)', function (done) {
User.findById(1, filter, function (err, u) {
should.not.exist(u);
should.not.exist(err);
done();
});
});
it('should allow findById(id)', function () { it('should allow findById(id)', function () {
User.findById(1); User.findById(1);
}); });
it('should allow findById(id, filter)', function () {
User.findById(1, filter);
});
it('should allow findById(id, options)', function () { it('should allow findById(id, options)', function () {
User.findById(1, options); User.findById(1, options);
}); });
it('should throw when invalid options are provided for findById', it('should allow findById(id, filter, options)', function () {
User.findById(1, filter, options);
});
it('should throw when invalid filter are provided for findById',
function(done) { function(done) {
(function() { (function() {
User.findById(1, '123', function(err, u) { User.findById(1, '123', function(err, u) {
}); });
}).should.throw('The options argument must be an object'); }).should.throw('The filter argument must be an object');
done(); done();
}); });
it('should throw when invalid options are provided for findById',
function(done) {
(function() {
User.findById(1, filter, '123', function(err, u) {
});
}).should.throw('The options argument must be an object');
done();
});
it('should report an invalid id via callback for findById', it('should report an invalid id via callback for findById',
function(done) { function(done) {
User.findById(undefined, {}, function(err, u) { User.findById(undefined, {}, function(err, u) {
@ -62,20 +88,100 @@ describe('crud-with-options', function () {
}); });
}); });
it('should allow findById(id, filter, cb) for a matching id',
function(done) {
User.create({name: 'x', email: 'x@y.com'}, function(err, u) {
should.not.exist(err);
should.exist(u.id);
User.findById(u.id, filter, function(err, u) {
should.exist(u);
should.not.exist(err);
u.should.be.an.instanceOf(User);
u.should.have.property('name', 'x');
u.should.have.property('email', undefined);
done();
});
});
});
it('should allow findById(id, options, cb) for a matching id', it('should allow findById(id, options, cb) for a matching id',
function(done) { function(done) {
User.create(function(err, u) { User.create({name: 'y', email: 'y@y.com'}, function(err, u) {
should.not.exist(err); should.not.exist(err);
should.exist(u.id); should.exist(u.id);
User.findById(u.id, options, function(err, u) { User.findById(u.id, options, function(err, u) {
should.exist(u); should.exist(u);
should.not.exist(err); should.not.exist(err);
u.should.be.an.instanceOf(User); u.should.be.an.instanceOf(User);
u.should.have.property('name', 'y');
u.should.have.property('email', 'y@y.com');
done(); done();
}); });
}); });
}); });
it('should allow findById(id, filter, options, cb) for a matching id',
function(done) {
User.create({name: 'z', email: 'z@y.com'}, function(err, u) {
should.not.exist(err);
should.exist(u.id);
User.findById(u.id, filter, options, function(err, u) {
should.exist(u);
should.not.exist(err);
u.should.be.an.instanceOf(User);
u.should.have.property('name', 'z');
u.should.have.property('email', undefined);
done();
});
});
});
it('should allow promise-style findById',
function(done) {
User.create({name: 'w', email: 'w@y.com'}).then(function(u) {
should.exist(u.id);
return User.findById(u.id).then(function(u) {
should.exist(u);
u.should.be.an.instanceOf(User);
u.should.have.property('name', 'w');
u.should.have.property('email', 'w@y.com');
return u;
});
}).then(function(u) {
should.exist(u);
should.exist(u.id);
return User.findById(u.id, filter).then(function(u) {
should.exist(u);
u.should.be.an.instanceOf(User);
u.should.have.property('name', 'w');
u.should.have.property('email', undefined);
return u;
});
}).then(function(u) {
should.exist(u);
should.exist(u.id);
return User.findById(u.id, options).then(function(u) {
should.exist(u);
u.should.be.an.instanceOf(User);
u.should.have.property('name', 'w');
u.should.have.property('email', 'w@y.com');
return u;
});
}).then(function(u) {
should.exist(u);
should.exist(u.id);
return User.findById(u.id, filter, options).then(function(u) {
should.exist(u);
u.should.be.an.instanceOf(User);
u.should.have.property('name', 'w');
u.should.have.property('email', undefined);
done();
});
}).catch(function(err) {
done(err);
});
});
}); });
describe('findByIds', function () { describe('findByIds', function () {