Skip to content

An extension of Backbone to track models, manage sync methods and help building online, offline and hybrid solutions. Supports LocalStorage via Locally plugin.

License

Notifications You must be signed in to change notification settings

freak2geek/backbone-repository

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

92 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

backbone-repository

Backbone extension that implements purposes of the Repository pattern, which means the enhancement of the model management and synchronization, so as to provide strong features to build online, offline and hybrid web applications.

enter image description here

The library mainly supports the following features:

  • Model register: A Collection (C) keeps models instantiated.
  • Sync modes: Sync modes (m) are sync functions to different data sources that are managed by a Repository (R).
  • Sync state: A Model (M) keeps changes not synchronized for each sync mode defined.
  • Sync operations: A Model (M) runs operations (fetch, save, destroy, pull, push and check) on different sync modes.

Versions

  • backbone-repository - core library.
  • backbone-respository-locally - core library + localStorage support through the Locally plugin.

Table of Contents

Usage

Model register

The extension implements a model caching strategy. By this, a model register in the form of a collection will be responsible for keeping in-memory all instantiated models.

var User = Backbone.Model.extend();

// A static method that returns the collection of users
var userRegister = User.register();

Along with that, a factory method will manage such register and will instantiate or return a model ensuring that it is not instantiated twice according to its identifier attribute. Both model register and factory method will be accessed through a static method of the model.

// Creating instances through the factory method
var user = User.create({id: 1});
var duplicate = User.create({id: 1, name: "Nacho"});

// Ensures a model has been instantiated once.
user === duplicate; // true

// The user has been updated as well.
user.get("name") === "Nacho"; // true

Besides, collections will rely on the factory method to create the instances of the model class associated. So,

var Users = Backbone.Collection.extend({
	model: function(attrs, options) {
		return User.create(attrs, options);
	} 
});

Sync mode

A sync mode is responsible to provide logic to access to different data sources. There exists several predefined modes, but you are enable to configure your custom sync mode as well.

For using a mode, you must pass the mode as an option for every single sync operation, whether fetch, update, destroy, pull, push or check.

// E.g. fetching operation using the server mode
user.fetch({
	mode: "server"
});

Server mode

The server mode uses the Backbone's sync function to perform a remote call against a REST API. It is the mode executed by default when using the model manager methods. Both the success and error callbacks are allowed in this mode.

// E.g. saving operation using the server mode
user.save(null, {
	mode: "server"
});

Client mode

The client mode just perform local operations to the model. The success callback is only enabled in this mode.

// E.g. destroying operation using the client mode
user.destroy({
	mode: "client"
});

LocalStorage mode (requires Locally extension)

The LocalStorage mode runs the model operations against LocalStorage. Both the success and error callbacks are allowed in this mode. For working, it requires Locally plugin and the "backbone-repository-locally" version of the library.

For using this mode, you must specify an storeName whether in the model definition or by parameter when saving.

// Specify `storeName` in the model definition.
var User = Backbone.Model.extend({
	storeName: "User"
});

var user = User.create({id: 1});

// Specify `storeName` when saving.
user.save(null, {
	mode: "localStorage",
	storeName: "User"
});

You may also specify an storagePrefix for all your models when persisting in LocalStorage.

// Specify `storagePrefix`.
Backbone.Repository.storagePrefix = "MyPrefix"; 

Finally, the LocalStorage key will read as follows: storagePrefix:storeName:[id|localId].

Custom mode

The extension features a way to configure your custom sync modes.

var syncStrategy = function (method, model, options) {
    // sync function logic
};

// Registers a new sync mode.
Backbone.Repository.setMode({
    mySyncMethod: syncStrategy
});

It is also possible to establish which mode will be selected by default when none selected.

// Defaults a sync mode.
Backbone.Repository.setDefaultMode("mySyncMethod");

Sync state

The model prototype has been expanded to work up the Repository features intented.

Fetch state

The fetch state means whether the model has been fetched using a sync mode or not. It is possible to pass the sync mode against you wish to check as an option. By default, it will check against the server mode.

user.fetch({
    mode: "server",
	success: function (model, response, options) {
		user.isFetched({
			mode: "server"
        }); // true
	}
});

user.isFetched({
	mode: "server"
}); // false

Dirty attributes state

The dirty attributes is a hash that keeps the model attributes that have changed since its last sync for each existing sync mode. The hash will be progressively emptying as attributes are sucessfully synchronized against each sync mode.

For implementing this, the set method has been altered and configured to handle dirty changes.

// The 'set' method automatically stores dirtied attributes.
user.set({
	name: "Nacho"
}); 

Both dirtiedAttributes and hasDirtied methods are possible to pass the sync mode against you wish to check as an option. By default, it will check against the server mode.

user.hasDirtied({
	mode: "server"
}); // true

user.dirtiedAttributes({
	mode: "server"
}); // outputs {name: "Nacho"}

The dirty handler may be turned off by passing dirty option to set method.

// The 'set' method won't store dirtied attributes.
user.set({
	name: "Nacho"
}, {
	dirty: false
});

user.hasDirtied({
	mode: "server"
}); // false

Dirty destroy state

The dirty destroy state means whether the model has been destroyed locally or not.

user.destroy({
	mode: "client"
});

user.isDirtyDestroyed(); // true

Destroy state

The destroy state means whether the model has been destroyed using a sync mode or not. It is possible to pass the sync mode against you wish to check as an option. By default, it will check against the server mode.

user.destroy({
	mode: "server",
	success: function (model, response, options) {
		user.isDestroyed({
        	mode: "server"
        }); // true
	}
});

user.isDestroyed({
	mode: "server"
}); // false

Version attribute

The version state is an attribute configured for each model. It is useful to stamp versions on the models when working with serveral data sources, for example a REST server and LocalStorage.

var User = Backbone.Model.extend({
	versionAttribute: "version"
});

When calling some operation that uses set method, you are enabled to check if the version has been updated by passing an option. In case the model is outdated, it will force the fetched state to false. An event named outdated is also triggered when the version attribute is changed.

var user = User.create({
	version: 1
});

user.isFetched(); // true, lets suppose that the model is fetched

user.set({
	version: 2
}, {
	version: true // forces to check version status
});

user.isFetched(); // false, the version has changed.

Sync operations

Along with predefined sync operations such as fetch, save and destroy, the extension implements three useful methods: pull, push and check.

By default, all sync operations use the server mode, but it may use another one just by passing the mode as parameter.

Besides, all sync operations has been adapted to be used from Collection.

Pull method

The pull method performs a read request only if the model has not been fetched before.

var user = User.create({id: 1});

user.isFetched(); // false

// Performs a sync call on its first fetching, such as fetch.
user.pull({
	mode: "server",
    success: function (model, response, options) {
    	user.isFetched(); // true
    }
});

Once the model has been fetched, the pull method won't request it again. However, the success callback will be executed.

user.isFetched(); // true

// It won't request since it was fetched before.
user.pull({
	mode: "server"
});

Push method

The push method performs a request whether create, update or destroy methods according to the sync status.

// Create example
var user = User.create({
	name: "Nacho"
});

// A create request will be emitted.
user.push({
	mode: "server"
});
// Update example
user = User.create({
	id: 1,
	name: "Nacho"
});

// An update request will be emitted.
user.push({
	mode: "server"
});
// Destroy example
user.destroy({
	mode: "client"
});

// A destroy request will be emitted.
user.push({
	mode: "server"
});

Check method

The check method is devised to fetch only the model version attribute and check is up to date. This has the sense to be used with remote sync modes. The server mode is configured to accept the checkUrl, which represent the checking endpoint.

var User = Backbone.Model.extend({
  versionAttribute: "version",
  checkUrl: "A_CHECKING_ENDPOINT_FOR_MODEL"
});

var user = User.create({
	id: 1,
    version: 1
});

// Assuming that the user is already fetched
user.isFetched(); // true

user.check({
	mode: "server",
    success: function (model, response, options) {
    	// Lets assume a new version is available,        
        // then the user is about to be fetched again.
        user.isFetched(); // false
    }
});

This method meant to be useful using through a Collection since you will verify the models all at once.

var Users = Backbone.Collection.extend({
	checkUrl: "A_CHECKING_ENDPOINT_FOR_COLLECTION"
});

var user = User.create({id: 1, version: 1});
var user2 = User.create({id: 2, version: 1});

// Assuming that models are already fetched.
user.isFetched(); // true
user2.isFetched(); // true

var users = new Users([user, user2]);

users.check({
	mode: "server",
    success: function (collection, response, options) {
    	// Lets assume a new version is only available for the user one,        
        // then the user one is about to be fetched again.
        user.isFetched(); // false
		user2.isFetched(); // true
    }
});

Reference API

Backbone.Repository

modes Backbone.Repository.modes()

Returns an array with the name of the available modes.

getMode Backbone.Repository.getMode(name)

Returns the sync function of the provided mode.

Available parameters:

  • name {String} The mode name.

setMode Backbone.Repository.setMode(name, [fn])

Registers a new mode by provinding its name and sync function.

Available parameters:

  • name {String|Object} The mode name or an object containing the name and sync function.
  • fn {Function} The sync function

getDefaultMode Backbone.Repository.getDefaultMode()

Returns the default mode.

setDefaultMode Backbone.Repository.setDefaultMode(name)

Establish which mode will be used by default.

Available parameters:

  • name {String} The mode name.

removeMode Backbone.Repository.removeMode(name)

Removes the sync mode registered.

Available parameters:

  • name {String} The mode name.

reset Backbone.Repository.reset(name)

Cleans all the sync modes registered.

storagePrefix Backbone.Repository.storagePrefix

The prefix used in all storages. Requires Locally extension.

compressStorage Backbone.Repository.compressStorage

Wheter to use Locally compression by default or not. Requires Locally extension.

Backbone.Model

create Backbone.Model.create(attrs, [options])

Factory method that returns a model instance and ensures only one is gonna be created with same id.

Available parameters:

  • attrs {Object} The attributes for the new instance.
  • options {Object} The options for the new instance.

find Backbone.Model.find(attrs)

Returns a model by its id or cid from the local cache of the model.

Available parameters:

  • attrs {Object} An id or cid for looking up.

register Backbone.Model.register()

Returns the collection that represents the local cache of the model.

reset Backbone.Model.reset()

Resets the local cache of the model.


isFetched model.isFetched([options])

Returns true whether this model has been fetched through the sync mode or false otherwise.

Available parameters:

  • options.mode {String} [mode=defaultMode] The sync mode name for checking.

dirtied model.dirtied

Internal hash containing all attributes that have changed since the last sync for each sync mode.

dirtiedAttributes model.dirtiedAttributes([options])

Retrieve a copy of the attributes that have changed since the last sync.

Available parameters:

  • options.mode {String} [mode=defaultMode] The sync mode name.

hasDirtied model.hasDirtied([options])

Returns true in case the model changed since its last sync, false otherwise.

Available parameters:

  • options.mode {String} [mode=defaultMode] The sync mode name.

isDirtyDestroyed model.isDirtyDestroyed()

Returns true if this model has been destroyed locally, false otherwise.

clearDirtied model.clearDirtied()

Erases dirtied changes of the model, whether attribute change or model destroy.

set model.set(attrs, [options])

Alters set method to provide new options.

Available parameters:

  • options.dirty {Boolean} [dirty=true] Whether to handle dirtied changes or not.
  • options.version {Boolean} [version=true] Whether to handle version changes or not.
  • options.localStorage {Boolean} [localStorage= true] Forces to save the model in LocalStorage. Requires Locally extension.

isDestroyed model.isDestroyed([options])

Returns true if this model has been destroyed remotely, false otherwise.

Available parameters:

  • options.mode {String} [mode=defaultMode] The sync mode name.

pull model.pull([options])

Fetches the model using the sync mode selected if it has not been fetched before.

Available parameters:

  • options.mode {String} [mode=defaultMode] The sync mode name.

push model.push([options])

Pushes the changes performed to the model using the sync mode selected. It may emitt create, update or destroy operations.

Available parameters:

  • options.mode {String} [mode=defaultMode] The sync mode name.
  • options.patch {Boolean} Use patch request when updating.

checkUrl model.checkUrl

The checking endpoint in the server mode.

check model.check([options])

Fetches version attribute of the model and checks the uptodate status.

Available parameters:

  • options.mode {String} [mode=defaultMode] The sync mode name.
  • options.checkUrl {String} The checking endpoint in the server mode.

storeName model.storeName

The storage key for persisting the model. Requires Locally extension.

Backbone.Collection

save collection.save(attrs, [options])

Saves the whole collection using the sync mode. The server mode emitts one request per model.

Available parameters:

  • options.mode {String} [mode=defaultMode] The sync mode name.

destroy collection.destroy([options])

Destroys the whole collection using the sync mode. The server mode emitts one request per model.

Available parameters:

  • options.mode {String} [mode=defaultMode] The sync mode name.

pull collection.pull([options])

Pulls the whole collection using the sync mode. The server mode may emit urls that take the form of url+ subset of ids for fetching separated by comma.

Available parameters:

  • options.mode {String} [mode=defaultMode] The sync mode name.

push collection.push([options])

Pushes the changes of the whole collection using the sync mode. The server mode emitts one request per model.

Available parameters:

  • options.mode {String} [mode=defaultMode] The sync mode name.
  • options.patch {Boolean} Use patch request when updating.

check collection.check([options])

Checks the whole collection using the sync mode.

Available parameters:

  • options.mode {String} [mode=defaultMode] The sync mode name.
  • options.checkUrl {String} The checking endpoint in the server mode.

storeName collection.storeName

The storage key for persisting the collection. Requires Locally extension.

Building and Testing

First install locally all the required development dependencies.

npm install

Building

backbone-repository

grunt base

backbone-repository-locally

grunt baseLocally

Testing

grunt test

Release History

Read the CHANGELOG.md file distributed with the project.

License

Read the LICENSE file distributed with the project.

About

An extension of Backbone to track models, manage sync methods and help building online, offline and hybrid solutions. Supports LocalStorage via Locally plugin.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published