Skip to content

A backbone.js plugin that provides one-to-one, one-to-many and many-to-many relations between backbone models

License

Notifications You must be signed in to change notification settings

jj-studio/Backbone-JJRelational

Repository files navigation

__________                __   ___.                         
\______   \_____    ____ |  | _\_ |__   ____   ____   ____  
 |    |  _/\__  \ _/ ___\|  |/ /| __ \ /  _ \ /    \_/ __ \ 
 |    |   \ / __ \\  \___|    < | \_\ (  <_> )   |  \  ___/ 
 |______  /(____  /\___  >__|_ \|___  /\____/|___|  /\___  >
        \/      \/     \/     \/    \/            \/     \/ 
     ____.    ____.__________       .__          __  .__                     .__   
    |    |   |    |\______   \ ____ |  | _____ _/  |_|__| ____   ____ _____  |  |  
    |    |   |    | |       _// __ \|  | \__  \\   __\  |/  _ \ /    \\__  \ |  |  
/\__|    /\__|    | |    |   \  ___/|  |__/ __ \|  | |  (  <_> )   |  \/ __ \|  |__
\________\________| |____|_  /\___  >____(____  /__| |__|\____/|___|  (____  /____/
                           \/     \/          \/                    \/     \/      

Version 0.2.12

Backbone JJRelational is a plugin that provides one-to-one, one-to-many, many-to-one and many-to-many bi-directional relations between Backbone models.

Backbone JJRelational is inspired by Paul Uithol's Backbone-relational, but supports many-to-many relations out of the box.

Backbone JJRelational has been tested with Backbone 1.0.0 and Underscore 1.5.0

Table Of Contents

## Installation

Backbone JJRelational depends - who would have thought - on Backbone.JS, Underscore.JS and jQuery.
Simply include backbone.JJRelational.js right after jQuery, Underscore and Backbone.

<script type="text/javascript" src="jquery.js"></script>
<script type="text/javascript" src="underscore.js"></script>
<script type="text/javascript" src="backbone.js"></script>
<script type="text/javascript" src="backbone.JJRelational.js"></script>
## How to use When defining your models, simply extend from `Backbone.JJRelationalModel` instead of the regular `Backbone.Model` and define a property named `storeIdentifier` and a property named `relations`, which takes an array of objects containing your relational options. Each defined relational options object must at least contain `type`, `relatedModel`, `key` and `reverseKey`. For each specified relation, you must define the reverse relation on the other side. __This is very important__, if you want JJRelational to work properly. Let's take a closer look at this.

A string and mandatory. As each model registers itself in Backbone.JJStore upon creation, you have to define an identifier for a store, where instances of the model are put.
Example:

Book = Backbone.JJRelationalModel.extend({
	storeIdentifier: 'Book'
});

Relational options

You can use the following options when defining relations.

type

mandatory
A string that defines the type of the relation (from the model's point of view!). Possible values are
has_one: for one-to-one or many-to-one relations
has_many: for one-to-many relations
many_many: for many-to-many relations

relatedModel

mandatory
Same as in Paul Uithol's Backbone-relational, this is a string which can be resolved to an object type on the global scope, or a reference to a Backbone.JJRelationalModel type. This - of course - defines which kind of model fills this relation.

Since 0.2.10 you also have the possibility to register a scope which will be searched by using Backbone.JJStore.addModelScope, for example:

Literature = {}
Literature.Author = Backbone.JJRelationalModel.extend({ ... });
Backbone.JJStore.addModelScope(Literature);

Thus, specifying a relation with relatedModel: 'Author' will look for the Author model type not only on the global scope, but also in the Literature-object.

key

mandatory
A string naming the attribute under which the relation is stored.
has_one relations are stored as a single Backbone.JJRelationalModel, or null when empty.
has_many relations are stored as a Backbone.Collection or an extension of it - provided collectionType has been set and the collection type has been registered with Backbone.JJRelational.registerCollectionTypes.

reverseKey

mandatory
A string naming the attribute of the related model's reverse relation. This is the same as key from the related model's point of view.
For every relation you specify on a model, you always have to specify a reverse relation on the related model in a fitting pattern. Remember: Each key is the same as reverseKey on the related side.
Confused? Take a look at the Setup example

collectionType

A string referencing a collection type which has been registered with Backbone.JJRelational.registerCollectionTypes(). Explanation: If the relational attribute of a has_many or many_many relation should use one of your extensions of Backbone.Collection, you have to register the collection(s) with a key of your choice by using Backbone.JJRelational.registerCollectionTypes(), and then use that key within the relation's collectionType option.
Example:
javascript BooksColl = Backbone.Collection.extend({ model: Book });

Backbone.JJRelational.registerCollectionTypes({ 'Books': BooksColl // … // … })

…and then within your relation options `collectionType: 'Books'`.

#### includeInJSON
This is an array of strings defining which attributes should be included when the model is saved.
Leaving `includeInJSON` empty means that merely the related model's `id` is included.

#### polymorphic
If you have a polymorphic collection, set this option to true on the side that `has_many`.
Note that this will completely disable the type checking when adding a model to the collection, so make sure you use it responsibly.
In the following example, both CerealBoxes and MilkJugs will appear inside of the Backbone.JJStore.Models.Product collection,
because they extend `Product` and do not override its `storeIdentifier` key.
Example:
```javascript```
var Store = Backbone.JJRelationalModel.extend({
  relations: [{
    type: "has_many",
    key: "products",
    relatedModel: "Product",
    collectionType: "ProductsCollection",
    reverseKey: "store",
    polymorphic: true
  }]
});

var Product = Backbone.JJRelationalModel.extend({
  storeIdentifier: "Product",
  relations: [{
    type: "has_one",
    key: "store",
    relatedModel: "Store",
    reverseKey: "products",
  }]
});
var CerealBox = Product.extend({ ... });
var MilkJug   = Product.extend({ ... });

var ProductFactory = (function() {
  return {
    buildModel: function(attrs, options) {
      if (attrs.type === "cereal")    return new CerealBox(attrs, options);
      else if (attrs.type === "milk") return new MilkCarton(attrs, options);
      else                            return new Product(attrs, options);
    }
  };
)();

var ProductsCollection = Backbone.Collection.extend({
  storeIdentifier: "ProductsCollection",
  model: ProductFactory.buildModel
});
## Setup example Here is an example of how the relations between different models could be defined:
/*
** Author
*/

// each author can have many books, and many publishers
Author = Backbone.JJRelationalModel.extend({
	storeIdentifier: 'Author',
	relations: [
		{
			type: 'has_many',
			relatedModel: 'Book',
			key: 'books',
			reverseKey: 'author',
			collectionType: 'Books',
			includeInJSON: ['id', 'title']
		},
		{
			type: 'many_many',
			relatedModel: 'Publisher',
			key: 'publishers',
			reverseKey: 'publishedAuthors',
			collectionType: 'Publishers'
		}
	]
});

AuthorsColl = Backbone.Collection.extend({
	model: Author
});

/*
** Book
*/

// each book has one author
Book = Backbone.JJRelationalModel.extend({
	storeIdentifier: 'Book',
	relations: [{
		type: 'has_one',
		relatedModel: 'Author',
		key: 'author',
		reverseKey: 'books',
		includeInJSON: ['id', 'firstname', 'surname']
	}]
});

BooksColl = Backbone.Collection.extend({
	model: Book
});

/*
** Publisher
*/

// each publisher has many authors
Publisher = Backbone.JJRelationalModel.extend({
	storeIdentifier: 'Publisher',
	relations: [{
		type: 'many_many',
		relatedModel: 'Author',
		key: 'publishedAuthors',
		reverseKey: 'publishers',
		collectionType: 'Authors'
	}]
});

PublishersColl = Backbone.Collection.extend({
	model: Publisher
});

// register collection types
Backbone.JJRelational.registerCollectionTypes({
	'Authors': AuthorsColl,
	'Books': BooksColl,
	'Publishers': PublishersColl
});

Once a new instance of a relational model has been created, a relationsInstalled event is emitted on the model.

## Getting and setting data

As stated before, in a has_one relation, the model stores a single Backbone.JJRelationalModel. You get and set it in the usual Backbone fashion.

var whichBook = function (book) {
		var a = book.get('author');
		console.log('"' + book.get('title') + '" by ' + a.get('firstname') + ' ' + a.get('surname'));
	},
	twist = new Book({
		title: 'Oliver Twist',
		author: {
			firstname: 'Charles',
			surname: 'Dickens'
		}
	});
	
whichBook(twist); // <-- logs out '"Oliver Twist" by Charles Dickens

In has_many and many_many relations, the model stores an instance of Backbone.Collection (or an extension of it, if registered). Thus you should use collection methods on it.
Tying to the lines of code above:

var carroll = new Author({ name: 'Lewis Carroll' }),
	penguin = new Publisher({
		name: 'Penguin Classics',
		publishedAuthors: [caroll, dickens]
	}),
	randomHouse = new Publisher({ name: 'Random House' });
	
randomHouse.get('publishedAuthors').add([carroll, dickens]);

carroll.get('publishers').each(function (publisher) {
	console.log(publisher.get('name')); // <-- logs out 'Penguin Classics' and 'Random House'
});

carroll.get('publishers').remove(randomHouse);

randomHouse.get('publishedAuthors').each(function (author) {
	console.log(author.get('firstname')); // <-- logs out 'Charles'
});

You also have the possibility to merely store ids within the relational attributes. These are automatically replaced with appropriate models, as soon as a model with one of these ids is created. In a has_one relation, the id is directly stored under the key-attribute.
In has_many and many_many relations, the relational collections have an own id-queue they juggle around with. For clarification, check out the following lines of code:

var theRoad = new Book({ title: 'The Road', author: 1 }),	cormac = new Author({ id: 1, firstname: 'Cormac', surname: 'McCarthy' });
whichBook(theRoad); // <-- logs out '"The Road" by Cormac McCarthy'

When the author model is created, it registers itself in the store and the book model gets notified of it - so it can replace the id with the real model. Take a look at these two lines, which basically have the same effect:

var cormac = new Author({ id: 1, firstname: 'Cormac', surname: 'McCarthy'}),
	theRoad = new Book({ title: 'The Road', author: 1 });
	

Although the author is created before the book, giving author an id (on the book model) triggers a lookup of the author. Same same for has_many and many_many relations:

var george = new Author({firstname: 'George', surname: 'Martin', books: [1]}),
	one = new Book({ id: 1, title: 'A Game Of Thrones' }),
	two = new Book({ id: 2, title: 'A Clash Of Kings' }),
	three = new Book({ id: 3, title: 'A Storm Of Swords' }),
	
	books = george.get('books');
	
	books.add([2, 3, {title: 'A Feast For Crows'}]).each(function (book) {
		whichBook(book);
	});
	
	// logs out all four books

NOTE: There is one thing that you should never do, and that is directly setting a collection on a relational attribute. For example:

var author = new Author({books: new BooksColl([{title: 'a book'}])}); // don't do that!

This will throw an error. Better use:

var author = new Author({books: [{title: 'a book'}]}); // works fine!
## Sync - saving and fetching data

Backbone.JJRelational handles everything for you automatically. Nevertheless, this section should explain some of the concepts and possibilities you have when fetching from/persisting to the server.

When calling save on a model, the includeInJSON property you defined for the relation is used to generate the JSON which gets persisted to the server. Going further, it is checked for you if a related model is new: If yes, the related model gets saved too!

When you call save on a Backbone.JJRelationalModel, it will walk through that model's relations. If the model has_one of something that has_many of it (like a Stores -> Products relationship), then the parent model (Store) will be saved first! Otherwise, the related model will be saved first.

Assume that a Store has_many Products, and Product has_one Store. You call store.save() ... So the Store will save itself first. After this completes, the Product's save will be called.

The Deferred object returned by save won't be resolved until all of the model's dependent models have finished saving too.

It should be noted that these related models will only be saved before/after if the model isNew(). If these are existing models, then you should save them each yourself individually.

You should also be aware that save will inevitably call set() on the response returned from the server, and so the usual backbone set options apply. Please read the backbone docs on how to use the add, remove, and merge options. In some cases, you'll probably want to pass these to save, which will in turn pass them through to set. Consider the above case where we've called store.save(). If the server responded with this json for a store {"name": "storename", "products": []}, this would end up wiping out all the products for the store when we set this response. Calling save(attrs, {remove: false}) would fix this.

Saving (many-to-many example)

// We pretend our relational setup is the same as in the setup example.
// So Publishers include publishedAuthors.id, Authors include books.id and books.title

var publisher = new Publisher({ name: "Faber & Faber", authors: [{ firstname: "Martin", surname: "McDonagh", books: [{ title: "The Pillowman" }] }] });
publisher.save();

Okay, now what's going to happen? Publisher relies on the author's id, author relies on the book's id. So at first a save request is fired for "The Pillowman":

{
	"title": "The Pillowman"
}

Provided the book gets persisted to a database, for example, and gets an id, "Martin McDonagh"'s request is fired:

{
	"firstname": "Martin"
	"surname": "McDonagh"
	"books": [{
		"id": 1,
		"title: "The Pillowman"
	}]
}

And now, at last, the publisher will be saved:

{
	"name": "Faber & Faber"
	"authors": [1]
}

Fetching

When fetching data from the server, Backbone.JJRelational automatically creates the needed models in a relation. Imagine we would call fetch on a collection of publishers and it would return:

[
{
	"id": 1,
	"name": "Faber & Faber",
	"authors": [{
		"id": 1,
		"firstname": "Martin",
		"surname": "McDonagh"
	}]
}
]

The author model "Martin McDonagh" will be created automatically. Now let's pretend, fetching the publishers would result in merely an ID array of its authors:

[
{
	"id": 1,
	"name": "Faber & Faber",
	"authors": [1,2]
},
{
	"id": 2,
	"name": "Random House",
	"authors": [3,4]
}
]

In this case, JJRelational doesn't create new author models, but stores the IDs in a queue within the relational collection. (Backbone.Collection._relational.idQueue) If faber is our first publisher model, we can call

faber.get('authors').fetchByIdQueue();

This will fetch the authors with IDs 1 and 2 and add them to the collection. (this works for one-to-one and one-to-many relations as well, of course) Or, if we want to fetch all related authors of the whole collection, we can call: (pretending pubColl is our collection of publishers):

pubColl.fetchByIdQueueOfModels('authors');

This will fetch the authors with IDs 1, 2, 3 & 4 and add them to their related publishers appropriately.

## Working with the store - prevent duplicate data

Backbone.JJStore acts as the big data store in your application which every newly created model registers itself at with its id and storeIdentifier. Each time a new model is created, all other models (which could be interested in a relationship to this new model) are informed of its creation. It's the same the other way round: Each time a mere ID is added to a relation, the store checks if there's already a model with the same id/storeIdentifier combination. If yes, the model from the store is used rather than the raw ID. Naturally, all this can work only if there are no duplicates present. This is what the configuration Backbone.JJRelational.Config.work_with_store (defaults to true) ensures. In JJRelational, when you create a new model with an existing storeIdentifier/id combination, the existing model with updated attributes will be returned. Let's clarify this with an example.

var a = new Author({ id: 1, firstname: "Jane", surname: "Doe" });
console.log("a: cid is %s and name is %s %s", a.cid, a.get('firstname'), a.get('surname'));

var b = new Author({ id: 1, surname: "Austen" });
console.log("b: cid is %s and name is %s %s", b.cid, b.get('firstname'), b.get('surname'));

In a regular Backbone application, this would output:

>  a: cid is c1 and name is Jane Doe
>  b: cid is c2 and name is undefined Austen

In JJRelational when work_with_store is set to true, the same logs would output:

> a: cid is c1 and name is Jane Doe
> b: cid is c1 and name is Jane Austen

Smash. The store realizes there's already an author with the same ID, so it merely updates the existing one.

You can turn this behaviour off by setting Backbone.JJRelational.Config.work_with_store to false, however you should keep in mind that the synchronization of relations only works properly in a duplicate-free environment.

## The Devil in the details

The concept behind JJRelational is actually dirt-simple. On the creation of a model, it registers itself in Backbone.JJStore, and other models that could be interested in it are notified of the creation. Because relations are defined from both sides, it's easy to keep everything in sync. Basically it's just juggling with models and attributes.

The following Backbone methods were overridden, all other methods (like Collection.reset, Collection.set etc.) merely perform some relational actions before actually calling the core method. See Reference for more information on the overrides:

  • Backbone.Model.set
  • Backbone.Model.save
  • Backbone.Model._validate
  • Backbone.Model.toJSON
  • Backbone.Collection.fetch

## Reference

Config

( Object ) Global configuration object. Possible values are:

  • url_id_appendix: used for fetchByIdQueue- and fetchByIdQueueOfModels-calls. Default value is '?ids=' , the needed ids are then added comma separated.
  • work_with_store: Default is true. This option indicates whether you want Backbone.JJStore to prevent duplication of models. See Working with the store for more information.

registerCollectionTypes (collTypes<Object>)

Registers one or many collection types, in order to build a correct collection instance for many-relations.

Backbone.JJRelationalModel

set (key<String|Object>, val<mixed|Object>, options<Object>)

This is pretty much the most important override. Of course it does the same as Backbone core, although it filters relational values and handles them accordingly.
Example:

author.set({ publishers: [2, 3, publisherObj, { name: 'new publisher' }] });

save (key<String|Object>, value<mixed|Object>, options<Object>)

Backbone's original save-method wrapped in relational stuff. The concept is: When saving a model, it is checked whether it has any relations containing a new model. If yes, the new model is saved first. When all new models have been saved, only then is the calling model saved.
Relational collections are saved as an array of models + idQueue.
Concerning relations, the includeInJSON property is used for serializing to JSON. See Sync - saving and fetching data for detailed information.

_validate (attrs<Object>, options<Object>)

The difference to Backbone core's _validate method is that this one flattens relational collections down to its model array. We've found that this is more convenient for more general validation functions.

toJSON (options<Object>)

If it's for saving ( options.isSave == true ) then the includeInJSON option of the relation is used. That can go down as many levels as required. If isSave is false, the related models are serialized regularly with their relations represented only by ids, however.

toJSONWithRelIDs ()

Returns a JSON of the model with all relations represented only by ids.

fetchByIdQueue (relation<String|Object>, options<Object>)

Fetches missing models of a relation, if their ids are known. See Sync - saving and fetching data for detailed information.

Backbone.Collection

fetch (options)

Of course the same as Backbone core's fetch, with the only exception that already existing models in the store are always smartly updated.

fetchByIdQueue (options)

If any IDs are stored in the collection's idQueue, this method will fetch the missing models.

fetchByIdQueueOfModels (relation<String|Object>, options<Object>)

Sums up fetchByIdQueue-calls on the same relation in a whole collection by collection the idQueues of each model and firing a single request. The fetched models are automatically added to their appropriate relations.

getIDArray

Returns an array of the collection's models' IDs + any IDs stored within the collection's idQueue (if present)


## Running the tests

If you want to run the tests supplied, open up your terminal, change into the tests directory, run

$ npm update

once and then start the server with

$ node server.js

In your browser, navigate to http://localhost:3000/tests.html. That's it.


## License

MIT

About

A backbone.js plugin that provides one-to-one, one-to-many and many-to-many relations between backbone models

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 4

  •  
  •  
  •  
  •