Make sure scoped methods are remoted

See https://github.com/strongloop/loopback/issues/454. It's regression in 2.x.
This commit is contained in:
Raymond Feng 2014-08-08 15:55:47 -07:00
parent cec7deb4a5
commit 1c61cedee7
3 changed files with 42 additions and 21 deletions

View File

@ -185,10 +185,9 @@ Model.setup = function () {
// resolve relation functions // resolve relation functions
sharedClass.resolve(function resolver(define) { sharedClass.resolve(function resolver(define) {
var relations = ModelCtor.relations;
if (!relations) { var relations = ModelCtor.relations || {};
return;
}
// get the relations // get the relations
for (var relationName in relations) { for (var relationName in relations) {
var relation = relations[relationName]; var relation = relations[relationName];
@ -201,11 +200,14 @@ Model.setup = function () {
relation.type === 'embedsMany' || relation.type === 'embedsMany' ||
relation.type === 'referencesMany') { relation.type === 'referencesMany') {
ModelCtor.hasManyRemoting(relationName, relation, define); ModelCtor.hasManyRemoting(relationName, relation, define);
ModelCtor.scopeRemoting(relationName, relation, define);
} else {
ModelCtor.scopeRemoting(relationName, relation, define);
} }
} }
// handle scopes
var scopes = ModelCtor.scopes || {};
for (var scopeName in scopes) {
ModelCtor.scopeRemoting(scopeName, scopes[scopeName], define);
}
}); });
return ModelCtor; return ModelCtor;
@ -474,30 +476,32 @@ Model.hasManyRemoting = function (relationName, relation, define) {
} }
}; };
Model.scopeRemoting = function(relationName, relation, define) { Model.scopeRemoting = function(scopeName, scope, define) {
var pathName = (relation.options.http && relation.options.http.path) || relationName; var pathName = (scope.options && scope.options.http && scope.options.http.path)
var toModelName = relation.modelTo.modelName; || scopeName;
var isStatic = scope.isStatic;
var toModelName = scope.modelTo.modelName;
define('__get__' + relationName, { define('__get__' + scopeName, {
isStatic: false, isStatic: isStatic,
http: {verb: 'get', path: '/' + pathName}, http: {verb: 'get', path: '/' + pathName},
accepts: {arg: 'filter', type: 'object'}, accepts: {arg: 'filter', type: 'object'},
description: 'Queries ' + relationName + ' of ' + this.modelName + '.', description: 'Queries ' + scopeName + ' of ' + this.modelName + '.',
returns: {arg: relationName, type: [toModelName], root: true} returns: {arg: scopeName, type: [toModelName], root: true}
}); });
define('__create__' + relationName, { define('__create__' + scopeName, {
isStatic: false, isStatic: isStatic,
http: {verb: 'post', path: '/' + pathName}, http: {verb: 'post', path: '/' + pathName},
accepts: {arg: 'data', type: toModelName, http: {source: 'body'}}, accepts: {arg: 'data', type: toModelName, http: {source: 'body'}},
description: 'Creates a new instance in ' + relationName + ' of this model.', description: 'Creates a new instance in ' + scopeName + ' of this model.',
returns: {arg: 'data', type: toModelName, root: true} returns: {arg: 'data', type: toModelName, root: true}
}); });
define('__delete__' + relationName, { define('__delete__' + scopeName, {
isStatic: false, isStatic: isStatic,
http: {verb: 'delete', path: '/' + pathName}, http: {verb: 'delete', path: '/' + pathName},
description: 'Deletes all ' + relationName + ' of this model.' description: 'Deletes all ' + scopeName + ' of this model.'
}); });
}; };

View File

@ -45,6 +45,13 @@
"public": true, "public": true,
"dataSource": "db", "dataSource": "db",
"options": { "options": {
"scopes": {
"superStores": {
"where": {
"size": "super"
}
}
},
"relations": { "relations": {
"widgets": { "widgets": {
"model": "widget", "model": "widget",

View File

@ -25,6 +25,16 @@ describe('relations - integration', function () {
this.app.models.widget.destroyAll(done); this.app.models.widget.destroyAll(done);
}); });
describe('/store/superStores', function() {
it('should invoke scoped methods remotely', function(done) {
this.get('/api/stores/superStores')
.expect(200, function(err, res) {
expect(res.body).to.be.array;
done();
});
});
});
describe('/store/:id/widgets', function () { describe('/store/:id/widgets', function () {
beforeEach(function() { beforeEach(function() {
this.url = '/api/stores/' + this.store.id + '/widgets'; this.url = '/api/stores/' + this.store.id + '/widgets';