Relations and models for Meteor collections.
Allows you to:
- Transform your Mongo docs into models with attributes and methods.
- Define relationships between collections.
- Traverse relationships and retrieve related models.
- Use other packages to handle validation (meteor-simple-schema) hooks (meteor-collection-hooks) and relational pub/sub (cottz:publish-relations).
Collections defined with Graviton automatically convert retrieved objects into models. You specify the type(s) when define the collection. Passing {transform: null}
to find()
etc. will bypass model transformation. The raw document is stored in model.attributes. Use built-in transformation methods like set, push, pop to make changes to your model locally. Call model.save()
to persist changes to the database. All methods work on both server and client.
In a Meteor app directory run:
$ meteor add emmerge:graviton
The following are added to all your meteor collections:
all()
- an alias for find().fetch()
.
build()
- returns a new local Gravition.Model
based on your collection definition. Does not save to db. The instance can be saved using Model.save()
.
create()
- calls build() to generate a Model instance AND inserts it into the db.
###Graviton.define(collectionName, options)
Use to define your collections. Returns a Mongo.Collection
instantiated with a transform function based on the options passed.
Param | Type | Description |
---|---|---|
collectionName | String |
The collection name to define. |
[options] | Object |
Options. |
[options.persist] | boolean |
Passing false will define a local collection. defaults to true |
[options.modelCls] | Gravtion.Model constructor or Object |
A model constructor generated by calling Graviton.Model.extend. Used to transform for objects in the collection. For polymorphism, pass an object that maps _type to model constructor like {key1: ModelClassA, key2: ModelClassB} . collection.build({_type: 'key2'}) would return an instance of ModelClassB . |
[options.defaultType] | String |
Use when supplying a object for modelCls. Specify the type to use when object being transformed does not contain _type. |
[options.typeAttribute] | String |
Optional key to the type attribute if you want to use something other than the default _type . |
[options.timestamps] | boolean |
If true and you have the collection-hooks package installed, createdAt and updatedAt timestamps will be added to your models. |
[options.<relationships>] | Object |
DEPRECATED (define relations on the modelCl) - Use to define relationships with other collections. See Relations section below. |
###Graviton.getProperty(key)
Use to access deeply-nested attributes without worry of throwing errors. Use period-delimited strings as keys. For example:
Gravition.getPropery({}, 'some.deep.nested.prop')
would simply return undefined
instead of throwing an error because obj.some
is undefined
. This method works best with keys that are meant to be stored in mongo since periods are not allowed in them.
###Graviton.setProperty(thing, [value])
Set deeply-nested attributes. Example:
var obj = {};
Graviton.setProperty(obj, 'some.deeply.nested.prop', 'hello');
>> {some: {deeply: {nested: {prop: 'hello'}}}}
You can pass an object instead of [key, value] to set several at once.
###Graviton.isModel(obj)
Use to check if an object is a model.
Singular relation types:
- belongsTo
- hasOne
- embeds
Multiple relation types:
- belongsToMany
- hasMany
- embedsMany
- hasAndBelongsToMany
Each of these is defined as a configuration object with the following keys/value pairs:
Key | Type | Description |
---|---|---|
[collection] | Mongo.Collection |
collection this relation refereneces |
[collectionName] or [klass] or [relationName] | String |
the name of another collection defined in Graviton |
[field] | String |
the key name on this document |
[foreignKey] | String |
used to specify the key name on the foreign document (if not _id ) |
Pass these as keys in the options Object to Model.extend
or Graviton.define
to declare relationships with other collections. Example:
CarModel = Graviton.Model.extend({
belongsTo: {
owner: {
klass: 'people',
foreignKey: 'ownerId'
}
},
belongsToMany: {
drivers: {
klass: 'drivers',
field: 'driverIds'
}
},
hasOne: {
// note that manufacturer should really be a belongsTo relationship since manufacturer shouldn't have a single carId
manufacturer: {
klass: 'manufacturers',
foreignKey: 'carId'
}
},
hasMany: {
wheels: {
klass: 'wheels',
foreignKey: 'carId'
}
},
embeds: {
plate: {
klass: 'plates'
}
},
embedsMany: {
windows: {
klass: 'windows'
}
}
},{});
Car = Graviton.define("cars", {
modelCls: CarModel
});
Would make the following possible:
var car = Car.findOne();
car.owner(); // returns a model
car.owner(person); // sets the ownerId of person
car.wheels.find(); // returns a cursor. same as Wheel.find({carId: car._id})
car.wheels.add({}); // insert a document into the wheels collection with carId = car._id
car.drivers.find(); // returns a cursor. same as Driver.find({_id: {$in: car.get('driverIds')}})
car.manufacturer();
car.plate();
// embedded models don't have the same finder capability since they aren't kept in minimongo
car.windows.all(); // returns all models
car.windows.at(2); // only builds one model
You can have relations to the Meteor.users collection (added by Meteor built in package accounts-base
). You can choose to make relations directly to the collection like this:
hasMany: {
users: {
collection: Meteor.users,
foreignKey: 'foreignId'
}
}
You can also define or register 'users' in Graviton. Graviton.define()
treats 'users' as a special collection and backs it with Meteor.users.
Graviton.define('users', options);
or
Graviton.registerCollection(Meteor.users);
Then the following configuration is valid:
hasMany: {
users: {
collectionName: 'users',
foreignKey: 'foreignId'
}
}
Note: Unlike other Graviton defined collections the special 'user' collection defined in Graviton will not tranform results into models by default. This is also true for user objects returned from relations. This is done for maximum compatability with Meteor itself and other external pacakges. The following are examples of how to get a user Model.
GravitonUsers = Graviton.define('users', options);
GravitonUsers.build(Meteor.user());
// or from a relation
GravitonUsers.build(Chats.findOne().user());
Graviton transforms collection objects into Models for you. This allows them to carry useful metadata and functions with the data. The vanilla Graviton.Model allows for basic functionality. Defining an extension of the Graviton.Model allows you to specify details of the collection's relationships or other custom functionality.
###Graviton.Model.extend(options, extensionPrototype)
Use to create a model constructor.
Param | Type | Description |
---|---|---|
[options] | Object |
options |
[options.defaults] | Object |
an object containing default key:value pairs for the collection. These key:values will be added to all model instances where there is not already a stored value with the same key. Functions should not be placed here as stored records cannot have functions as values. |
[options.initialize] | Function |
a function which will be run on initialization. |
[options.<relationships>] | Object |
define how this model relates to other collections. belongsTo , belongsToMany , hasOne , hasMany , etc. |
[extensionPrototype] | Object |
This object contains the extension prototype. This is the place to add functions to the model. Values could also be placed here if they relate to this specific model. These do not behave as attributes - any values placed here will not be stored. |
##Model manipulation
##Model database interactions
Graviscope - a graviton version of the Microscope app built in the book Discover Meteor
PostModel = Graviton.Model.extend({
hasMany: {
comments: {
collection: 'comments',
foreignKey: 'postId'
}
}
},{};
Posts = Graviton.define('posts', {
modelCls: PostModel
});
post = Posts.findOne();
comment = Comments.build();
comment.set({author:'Steve', body:'Check out the Posts.hasMany.comments relationship.'})
post.comments.add(comment);
console.log('Newly created comment id', comment._id);
Posts = Graviton.define('posts', {
hasMany: {
comments: {
collection: 'comments',
foreignKey: 'postId'
}
}
});
Comments = Graviton.define('comments', {
belongsTo: {
post: {
collection: 'posts',
field: 'postId'
}
}
})
post = Posts.findOne();
comment = Comments.build();
comment.set({author:'Steve', body:'Check out the Posts.hasMany.comments relationship.'})
post.comments.add(comment);
comment.post()
// return Posts object.
console.log('It will return same _id :',post._id,comment.post()._id)