I have a question about loading and caching remote objects with Ember. I\'m developing an Ember app that uses server-side storage through a REST API. Some of the fetched data is
Just to "bump" this thread up a little, because it was one of the top results when I researched solutions for ember local cache of restful api, etc.:
Dan Gebhardt seems to do a bloody good job with Orbit.js and its integration into Ember: https://github.com/orbitjs/ember-orbit
Orbit is a standalone library for coordinating access to data sources and keeping their contents synchronized.
Orbit provides a foundation for building advanced features in client-side applications such as offline operation, maintenance and synchronization of local caches, undo/redo stacks and ad hoc editing contexts.
Orbit.js features:
Support any number of different data sources in an application and provide access to them through common interfaces.
Allow for the fulfilment of requests by different sources, including the ability to specify priority and fallback plans.
Allow records to simultaneously exist in different states across sources.
Coordinate transformations across sources. Handle merges automatically where possible but allow for complete custom control.
Allow for blocking and non-blocking transformations.
Allow for synchronous and asynchronous requests.
Support transactions and undo/redo by tracking inverses of operations.
Work with plain JavaScript objects.
And don't miss his great speech and slides about Orbit:
Introduction to Orbit.js
(UPDATE: I added some more descriptive information from the Orbit pages, as my posting got downvoted for "just" referencing external resources and not containing the actual solution in itself. But Orbit seems to me like the solution, and the only way to "include" this here is via links.)
Here's a way to do it. A mixin for your adapters, with a method localStoreRecord
you can use to cache the record, lastly an initializer to preload the store.
Local Storage is simply a key:value store for stringified objects, so we can store all of our application data under a single key.
Note: this is using es6 modules
// app/mixins/local-storage.js
import Ember from 'ember';
export default Ember.Mixin.create({
appName: 'myApp',
// how many records per model to store locally, can be improved.
// needed to prevent going over localStorage's 5mb limit
localStorageLimit: 5,
localStoreRecord: function(record) {
var data = JSON.parse(localStorage.getItem(this.appName));
data = data || {};
data[this.modelName] = data[this.modelName] || [];
var isNew = data[this.modelName].every(function(rec) {
rec.id !== record.id;
});
if (isNew) {
data[this.modelName].push(record);
if (data[this.modelName].length > this.localStorageLimit) {
data[this.modelName].shift();
}
localStorage.setItem(this.appName, JSON.stringify(data));
}
}
});
// app/adapters/skateboard.js
import DS from 'ember-data';
import Ember from 'ember';
import LocalStorageMixin from '../mixins/local-storage';
export default DS.RESTAdapter.extend(LocalStorageMixin, {
modelName: 'skateboard',
find: function(store, type, id) {
var self = this;
var url = [type,id].join('/');
return new Ember.RSVP.Promise(function(resolve, reject) {
Ember.$.ajax({
url: 'api/' + url,
type: 'GET'
}).done(function (response) {
// cache the response in localStorage
self.localStoreRecord(response);
resolve({ type: response });
}).fail(function(jqHXR, responseStatus) {
reject(new Error(type +
' request failed with status=' + reponseStatus);
});
});
},
updateRecord: function(store, type, record) {
var data = this.serialize(record, { includeId: true });
var id = record.get('id');
var url = [type, id].join('/');
return new Ember.RSVP.Promise(function(resolve, reject) {
Ember.$.ajax({
type: 'PUT',
url: 'api/' + url,
dataType: 'json',
data: data
}).then(function(data) {
// cache the response in localStorage
self.localStoreRecord(response);
resolve({ type: response });
}).fail(function(jqXHR, responseData) {
reject(new Error(type +
' request failed with status=' + reponseStatus);
});
});
}
});
// app/initializers/local-storage.js
export var initialize = function(container/*, application*/) {
var appName = 'myApp';
var store = container.lookup('store:main');
var data = JSON.parse(localStorage.getItem(appName));
console.log('localStorage:',data);
if (!data) {
return;
}
var keys = Object.keys(data);
if (keys.length) {
keys.forEach(function(key) {
console.log(key,data[key][0]);
store.createRecord(key, data[key][0]);
});
}
};
export default {
name: 'local-storage',
after: 'store',
initialize: initialize
};
There is an implementation of a local storage adapter that you might find useful. Have a look at https://github.com/rpflorence/ember-localstorage-adapter