- Model Overview
- Creating a Model
- Working With Data Using a Model
- Working with JSON Graph Data using a Model
- The Model Cache
- Batching Outgoing Requests
- Path Optimization and the DataSource
Model Overview
Your application can use Data Sources to retrieve JSON Graph data from the network. However it is rarely ideal for your application’s views to interact directly with data sources for the following reasons:
- Application views typically navigate information hierarchically in JSON format, and Data Sources return data in JSON Graph format.
- Views need to be responsive to user input, but retrieving data from a Data Source may introduce considerable latency if the Data Source accesses the network.
- In response to user navigation (ex. scrolling through a list), views may need to repeatedly access small quantities of fine-grained data in rapid succession. Data Sources typically access the network, therefore fine-grained requests are often inefficient because of the overhead required to issue a request.
- Navigating information hierarchically rather than retrieving information using id’s can lead to inefficient back-end requests.
For these reasons, views retrieve their data from Model objects, which act as intermediaries between the view and the Data Source. Models abstract over Data Sources and provide several important services:
- Models convert JSON Graph information retrieved from the Data Source into JSON.
- Models reduce latency by caching data previously retrieved from the Data Source in an in-memory cache.
- Models achieve more efficient network access patterns by batching multiple concurrent requests for information from the view into batched requests to the Data Source.
- Models optimize your view’s outgoing requests to the Data Source using previously-cached JSON Graph references.
How the Model Works
Models use a DataSource to retrieve data from the JSON Graph. Falcor ships with HttpDataSource, an implementation of the DataSource interface which proxies requests to another DataSource running on an HTTP server (usually a falcor Router).
You can associate a DataSource with a Model by passing it to the Model constructor.
var model = new falcor.Model({ source: new falcor.HttpDataSource('/model.json') });
You can implement the DataSource interface to allow a Model to communicate with a remote JSON object over a different transport layer (ex. WebSocket). For more information see Data Sources.
If a Model does not have a DataSource, all Model operations will be performed on the Model’s local cache. When you initialize the Model, you can prime its cache with a JSON object.
Then you can transform and retrieve values by passing the Model Paths to values within its associated JSON object.
var falcor = require('falcor');
var model = new falcor.Model({
cache: {
todos: [
{
name: 'get milk from corner store',
done: false
},
{
name: 'withdraw money from ATM',
done: true
}
]
}
});
// This returns:
// "get milk from corner store"
var name = await model.getValue('todos[0].name');
model.getValue
(like most other getters and setters in falcor) returns a promise which can be resolved with .then()
or await
Using .then()
would look as follows:
model.getValue('todos[0].name').then(function(value) {
console.log(value);
});
It is common practice to begin working against mock data in a Model cache, and then replace it with a DataSource that retrieves data from the server later on.
var falcor = require('falcor');
var HttpDataSource = require('falcor-http-datasource');
var baseUrl = 'https://falcor-server-sample-nvutywuvbl.now.sh';
var model = new falcor.Model({
source: new HttpDataSource(baseUrl + '/model.json')
});
var name = await model.getValue('todos[0].name');
Note: You can check out the sample server source code for the data source used above here.
When data is retrieved from a DataSource, it is placed into the Model’s local cache. Subsequent requests for the same information will not trigger a request to the DataSource if the data has not been purged from the local cache.
// Does not trigger a request to the server because
// value is present already in local model cache
await model.getValue('todos[0].name');
There is one very important difference between working with a JSON object directly and working with that same JSON object through a Falcor Model: you can only retrieve value types from a Model.
“Why can’t I request Objects or Arrays from a Model?”
Falcor is optimized for displaying data catered to your views. Both Arrays and Objects can contain an unbounded amount of data. Requesting an Array or Object in its entirety is equivalent to your view requesting SELECT *
without a WHERE
clause in the SQL world. An Array that contains 5 items today, can grow to contain 10,000 items later on. This means that requests which are initially served quickly and fit the view’s requirements can become slower over time as more data is added to backend data stores.
Models force developers to be explicit about which value types they would like to retrieve in order to maximize the likelihood that server requests will have stable performance over time. Rather than allow you to retrieve an entire Object, Models force you to be explicit and retrieve only those values needed in a given scenario. Similarly, when displaying an Array of items, Models do not allow you to retrieve the entire Array upfront. Instead you must request the first visible page of an Array, and follow up with additional page requests as the user scrolls. This allows your client code to control performance boundaries, based on the amount of data actually used in the view, as opposed to being susceptible to unexpected increases in the total amount of data available.
In the following example we show one page worth of a list of TODOs, selecting the name
and done
properties of all the TODOs in the current page.
var falcor = require('falcor');
var HttpDataSource = require('falcor-http-datasource');
var baseUrl = 'https://falcor-server-sample-nvutywuvbl.now.sh';
var model = new falcor.Model({
source: new HttpDataSource(baseUrl + '/model.json')
});
//selecting just the props needed to display table row
var page = 0;
var from = page,
to = page + 5;
var response = await model.get('todos[' + from + '..' + to + ']["name", "done"]');
var html = '<ul>';
var todo;
for (var i = from; i < to; i++) {
todo = response.json.todos[i];
if (todo) html += '<li>' + todo.name + ' - ' + (todo.done ? '✓' : 'X') + '</li>';
}
html += '</ul>';
// // The code above prints the following html:
// <ul>
// <li>get milk from corner store - (✓)</li>
// <li>withdraw money from ATM - (X)</li>
// </ul>
If you are certain that an Object or Array will remain a constant size, you can indicate to a Model that they should always be retrieved in their entirety by using an Atom. For more information, see JSON Graph Atoms.
Creating a Model
A Model may be created by invoking the Model constructor. Model constructor can be passed an options object that supports the following keys:
- cache
- maxSize
- collectRatio
- source
- onChange
- comparator
- errorSelector
var modelOptions = {
/* options keys here */
};
var model = new falcor.Model(modelOptions);
The cache, maxSize, and collectRatio values
These optional values can be used to configure the Model Cache. For more information, see The Model Cache.
The source value
The optional source value in the Model constructor options object can be initialized to a DataSource. Models use DataSources to retrieve JSON information. For more information, see DataSources.
The onChange and comparator values
These optional values relate to change detection.
The errorSelector value
The optional errorSelector function can be used to transform errors that are returned from the DataSource before they are stored in the Model Cache.
In this example, we use an errorSelector function to add a relative expiration time of two minutes to every error received from the DataSource.
var model = new falcor.Model({
source: new falcor.HttpDataSource('/model.json'),
errorSelector: function(error) {
error.$expires = -1000 * 60 * 2;
}
});
Working With Data Using a Model
The Falcor Model is intended to be the “M” in your MVC. Rather than interact with JSON data directly, your application’s views interact with data indirectly through the Model object. The Model object provides a set of familiar JavaScript APIs for working with JSON data, including get, set, and call.
Here is an example of working with JSON data directly:
var model = {
todos: [
{
name: 'get milk from corner store',
done: false
},
{
name: 'withdraw money from ATM',
done: true
}
]
};
// This returns:
// "get milk from corner store"
var name = model.todos[0].name;
The main difference between working with JSON data directly and working with it indirectly through a Model object, is that the Falcor Model has an asynchronous API.
Here is an example of working with a JSON object indirectly using a Falcor Model:
var falcor = require('falcor');
var model = new falcor.Model({
cache: {
todos: [
{
name: 'get milk from corner store',
done: false
},
{
name: 'withdraw money from ATM',
done: true
}
]
}
});
// This eventually returns:
// "get milk from corner store"
var name = await model.getValue('todos[0].name');
Note that in the example above, the name of the TODO is returned by an asynchronous call to model.getValue
.
Calling model.getValue
returns a JavaScript Promise which is fulfilled by JavaScript’s await
keyword.
The main advantage of using an asynchronous API is that you can code against JSON data the same way regardless of whether the data is local or remote. This makes it very easy to begin coding your application against mocked data at first, and then work against server data later on without changing client code.
In the example above, we retrieve the name of the first TODO from a JSON Object. In the code sample below, the data has been moved to the cloud but the client code that retrieves the data remains the same:
var falcor = require('falcor');
var HttpDataSource = require('falcor-http-datasource');
var baseUrl = 'https://falcor-server-sample-nvutywuvbl.now.sh';
var model = new falcor.Model({
source: new HttpDataSource(baseUrl + '/model.json')
});
var name = await model.getValue('todos[0].name');
Retrieving Data from the Model
When retrieving data from the Model, developers are typically attempting to do one of the following:
- Retrieve a single value
- Retrieve JSON Graph data as JSON
- Transform JSON Graph data directly into view objects
1. Retrieve a Single Value
The Model’s getValue Method can be used to retrieve a single value from the DataSource. The getValue Method has the following signature:
class Model {
getValue(Path): ModelResponse
}
Here is an example of retrieving a single value from a Model using getValue:
var falcor = require('falcor');
var HttpDataSource = require('falcor-http-datasource');
var baseUrl = 'https://falcor-server-sample-nvutywuvbl.now.sh';
var model = new falcor.Model({
source: new HttpDataSource(baseUrl + '/model.json')
});
// prints "go to the ATM"
var name = await model.getValue(['todos', 0, 'name']);
2. Retrieving JSON Graph Data as JSON
The Model’s get Method can be used to retrieve data from the DataSource. The get Method has the following signature:
class Model {
get(...PathSet): ModelResponse
}
While Models retrieve information from DataSources in JSON Graph format, they emit information in JSON format. You can retrieve JSON data from a Model by using its get
method. In the following example, we will retrieve the names of the first two tasks in a TODOs list from a Model.
var falcor = require('falcor');
var HttpDataSource = require('falcor-http-datasource');
var baseUrl = 'https://falcor-server-sample-nvutywuvbl.now.sh';
var model = new falcor.Model({
source: new HttpDataSource(baseUrl + '/model.json')
});
var jsonGraph = await model.get(['todos', { from: 0, to: 1 }, 'name'], ['todos', 'length']);
The Model forwards the requested paths to the Data Source’s get method. The Data Source executes the abstract JSON Graph get operation on its associated JSON Graph object. The result is a subset of the Data Source’s JSON Graph object containing all of the values found at the requested paths, as well as any JSON Graph References encountered along the requested paths.
// DataSource Response
{
todos: {
"0": { $type: "ref", value: ["todosById", 44] },
"1": { $type: "ref", value: ["todosById", 54] },
"length": 10
},
todosById: {
"44": {
name: 'get milk from corner store'
},
"54": {
name: 'withdraw money from ATM'
}
}
};
The Model merges the response from the DataSource into its internal JSON Graph cache, and creates a JSON format copy of the response by replacing all JSON Graph Reference objects with real object references. The resulting JSON object is returned to the caller in an envelope, and printed to the console:
var falcor = require('falcor');
var model = new falcor.Model({
cache: {
todos: [
{ name: 'get milk from corner store', done: false },
{ name: 'withdraw money from ATM', done: true },
{ name: 'some other todo', done: false }
]
}
});
var result = await model.get(['todos', { from: 0, to: 1 }, 'name'], ['todos', 'length']);
// The following JSON envelope is eventually returned:
// {
// json: {
// todos: {
// "0": {
// name: 'get milk from corner store'
// },
// "1": {
// name: 'withdraw money from ATM'
// },
// "length": 3
// }
// }
// }
NOTE: await
used above may not be available in some browser environments.
You can just as well use model.get(..).then
to get the response (as shown in other examples):
model.get(['todos', { from: 0, to: 1 }, 'name'], ['todos', 'length']).then(function(response) {
console.log(JSON.stringify(response));
});
Setting Values Using the Model
In addition to retrieving values from a DataSource using a Model, you can also use a Model to set values into a DataSource. Set operations immediately write to the Model’s local cache, and then write to the DataSource. That means that changes are reflected on the client immediately. However, if an attempt to modify the data in the DataSource fails, the Model’s cache will eventually be updated with the post-set value of the DataSource. This is sometimes referred to as eventual consistency.
When setting information using the Model, developers are usually trying to do one of the following operations:
- Set a single value
- Set values using multiple PathValue objects
- Set values using a JSONEnvelope
1. Set a Single Value
The Model’s setValue Method can be used to set a single value in the DataSource. The setValue Method has the following signature:
class Model {
setValue(Path): ModelResponse
}
Here is an example of setting a single value using the setValue method:
var dataSource = new falcor.HttpDataSource('/model.json');
var model = new falcor.Model({
source: dataSource
});
// prints "true"
model.setValue(['todos', 0, 'done'], true).then(function(done) {
console.log(done);
});
2. Set Values Using Multiple PathValue Objects
In order to change multiple values using a single DataSource request, you can pass multiple PathValue objects to the Model’s set method.
class Model {
set(...PathValue): ModelResponse
}
A PathValue object is a combination of a path and a value. You can create a PathValue object using the pathValue factory function:
var pathValue = falcor.pathValue('todos[0].done', true);
// prints {path:["todos",0,"done"],value:true}
console.log(JSON.stringify(pathValue));
In the following example, we will set the done value of the first two tasks in a TODOs list to true
.
var dataSource = new falcor.HttpDataSource('/model.json');
var model = new falcor.Model({
source: dataSource
});
model
.set(falcor.pathValue(['todos', 0, 'done'], true), falcor.pathValue(['todos', 1, 'done'], true))
.then(function(response) {
console.log(JSON.stringify(response, null, 4));
});
The Model combines the PathValues into a single JSON Graph Envelope, and then forwards it to the DataSource’s set method. The DataSource executes the abstract JSON Graph set operation on its associated JSON Graph object. The result is a JSON Graph object containing the post set values of all of the paths that the Model attempted to set, as well as any JSON Graph References encountered along the requested paths.
// DataSource Response
{
todos: {
"0": { $type: "ref", value: ["todosById", 44] },
"1": { $type: "ref", value: ["todosById", 54] }
},
todosById: {
"44": {
done: true
},
"54": {
done: true
}
}
};
The Model merges the response from the DataSource into its internal JSON Graph cache, and creates a JSON format copy of the response by replacing all JSON Graph Reference objects with real object references. The resulting JSON object is returned to the caller in an envelope, and printed to the console:
model
.set(falcor.pathValue(['todos', 0, 'done'], true), falcor.pathValue(['todos', 1, 'done'], true))
.then(function(response) {
console.log(JSON.stringify(response));
});
// The following JSON envelope is eventually printed to the console:
// {
// json: {
// todos: {
// "0": {
// done: true
// },
// "1": {
// done: true
// }
// }
// }
// }
NOTE: In Node.js, falcor.pathValue
is available from
require('falcor-json-graph').pathValue
instead.
3. Set Values Using a JSONEnvelope
Occasionally, it can be less verbose to set multiple values with a JSON object rather than a series of PathValue objects. That is why set supports the following overload:
class Model {
set(JSONEnvelope): ModelResponse
}
A JSONEnvelope is just an object that contains a json
key with a JSON value. It is necessary to wrap JSON values within a JSONEnvelope so that the Model’s set method can differentiate between PathValue objects and JSON.
var taskCompletionValuesJSONEnvelope = {
json: {
todos: {
0: {
done: true
},
1: {
done: true
}
}
}
};
var dataSource = new falcor.HttpDataSource('/model.json');
var model = new falcor.Model({
source: dataSource
});
// The following code prints a JSONEnvelope to the console:
// {
// json: {
// todos: {
// "0": {
// done: true
// },
// "1": {
// done: true
// }
// }
// }
// }
model.set(taskCompletionValuesJSONEnvelope).then(function(response) {
console.log(JSON.stringify(response));
});
Calling Functions
Retrieving and setting values in a JSON Graph are both examples of idempotent operations. This means that you can repeat the same set or get operation multiple times without introducing any additional side effects. This is the reason the Model can safely serve subsequent requests for the same value from its cache instead of returning to the DataSource. Furthermore, the Model can optimistically update its local cache when a value is set because it can make an educated guess about the effect that the set operation will have on the DataSource’s JSON Graph object. Obviously, idempotent operations have nice properties. However, sometimes applications require a series of mutations to be applied transactionally. In these instances, set and get alone are not adequate. You need to call a function.
To allow for transactional operations, JSON Graph objects can contain functions just like JavaScript objects. Functions are considered JSON Graph objects, which means they cannot be retrieved from, nor set into, a DataSource. Rather, functions can only be invoked using the call method.
class Model {
call(callPath:Path, args:any[], refPaths?: PathSet[], extraPaths?:PathSet[]): ModelResponse
}
Unlike JavaScript functions, JSON Graph functions cannot return transient data. Instead JSON Graph functions may only return a subset of their this
object, typically after mutating it. The result of a JSON Graph function is always a JSONGraphEnvelope containing a subset of the function’s this
object.
The call method’s args
are explained below:
The callPath Argument
This argument is the path to the function within the DataSource’s JSON Graph object. Note that one invocation of call can only run a single function.
The args Argument
This is an array of arguments to be passed to the function being called.
The Optional refPaths Argument
Typically, returnValuePaths are used when the function creates a new object and returns a reference to that object. The refPaths
can be passed to the call method in order to allow fields to be retrieved from the newly-generated object without the need for a subsequent get operation.
In the event that any of the values returned from the function are JSON Graph References, the DataSource will append each of the refPaths
to each Reference path and evaluate the concatenated paths. The resulting values are added to the JSON Graph response by the DataSource.
The Optional extraPaths Arguments
A function is not obligated to return all of the changes that it makes to its this
object. On the contrary, functions typically return as little data as possible by default. Instead of forcing functions to return all of the changes they make to the JSON Graph object, DataSources allow callers to define exactly which values they would like to refresh after successful function execution. To this end, callers can provide refPaths
and extraPaths
to the DataSource’s call method along with the function path. After the DataSource runs the function, it retrieves the refPaths
and extraPaths
and adds them to the JSON Graph response.
After the refPaths
have been evaluated against any JSON Graph References returned by the function and added to the JSONGraphEnvelope Response, each PathSet in the extraPaths
array is evaluated on the function’s this
object. The resulting values are added to the JSON Graph Response returned by the DataSource’s call method.
How Call Works
When a JSON Graph function is called using a Model, the following steps are executed in order:
- The Model immediately forwards the call to its DataSource.
- The DataSource executes the abstract JSON Graph call operation on the function, receiving a JSONGraphEnvelope in response. The function’s response contains a subset of the DataSource’s JSON Graph object after the function’s successful completion. It also may optionally include an Array of “invalidated” PathSets which may have been changed by the function.
- If the function added any new objects to the JSON Graph, references to these objects are typically included in its response. The function caller can retrieve values from these newly-created objects by specifying
refPaths
alongside the other arguments to the call method (callPath
,args
, etc). IfrefPaths
have been specified, the DataSource attempts to retrieve each of these PathSets from the objects located at the references included in the function’s response. Once the DataSource has retrieved these values, it adds them to the JSON Graph subset in the function’s response. - Finally the DataSource attempts to retrieve each PathSet in the
extraPaths
argument. The resulting subset of the DataSource’s JSON Graph is added to the function’s response, and returned to the Model. - Upon receiving the response from the DataSource, The Model removes all of the “invalidated” paths from the cache, merges the data in the response into its cache, and returns a JSON version of the response to the caller.
Call By Example
Adding to the End of a List with a Call
In the example below, we add a new task to a TODOs list by invoking the add function on the TODOs list:
var dataSource = new falcor.HttpDataSource('/model.json');
var model = new falcor.Model({
source: dataSource
});
model.call(['todos', 'add'], ['pick up some eggs'], [['name'], ['done']]).then(
function(jsonEnvelope) {
console.log(JSON.stringify(jsonEnvelope));
},
function(error) {
console.error(error);
}
);
The Model forwards the call operation to the DataSource which implements the abstract JSON Graph call operation on its associated JSON Graph object:
// Inside Model
dataSource.call(
// callPath parsed into Path array by Model
['todos', 'add'],
// function arguments
['pick up some eggs'],
// refPaths parsed into PathSet arrays by Model
[['name'], ['done']]
);
Let’s say that the JSON Graph object managed by the DataSource looks like this:
{
todos: [
{ $type: "ref", value: ["todosById", 44] },
{ $type: "ref", value: ["todosById", 54] }
],
todosById: {
"44": {
name: 'get milk from corner store',
done: false
},
"54": {
name: 'withdraw money from ATM',
done: false
}
}
}});
After the DataSource executes the add function on the todos list, the JSON Graph object will look like this:
{
todos: [
{ $type: "ref", value: ["todosById", 44] },
{ $type: "ref", value: ["todosById", 54] },
{ $type: "ref", value: ["todosById", 93] }
],
todosById: {
"44": {
name: 'get milk from corner store',
done: false
},
"54": {
name: 'withdraw money from ATM',
done: false
},
"93": {
name: "pick up some eggs",
done: false
}
}
}});
After modifying the JSON Graph object, the function returns a JSONGraphEnvelope response to the DataSource.
{
paths: [
["todos", [2, "length"]]
],
jsonGraph: {
todos: {
2: { $type: "ref", value: ["todosById", 93] },
length: 3
}
}
}
Let’s see if we can try to understand why the function returns the response above.
Notice that the function’s JSONGraphEnvelope response contains a paths
array which contains paths to all of the values in the jsonGraph
key.
// partial JSONGraphEnvelope Response
{
paths: [
["todos", [2,"length"]]
],
jsonGraph: {
todos: {
2: { $type: "ref", value: ["todosById", 93] },
length: 3
}
},
// rest of response snipped...
}
Why is this necessary? Unlike get and set operations, there is no way for the Model to predict what values will be returned from a function call. By providing the Model with an array of paths to the values within the JSONGraph object, the function allows the Model to merge the response into its local cache without resorting to reflection.
Notice as well that the function does not include the entire contents of the todos
list in its response, nor does it include the newly-created task object in the todosById
map. As a rule, functions should return the minimum amount of data required to ensure the Model’s cache is consistent, as well as enable Models to retrieve data from any objects created by the function.
To allow the Model to retrieve data from the newly-created task object, the JSONGraphEnvelope response contains the reference to the newly-created task object in the todosById
map.
{
// beginning of response snipped...
jsonGraph: {
todos: {
2: { $type: "ref", value: ["todosById", 93] },
length: 3
}
},
// end of response snipped...
}
Instead of returning the task data in the response, including a reference to the task allows the caller to decide what values from the newly-created task should be retrieved by specifying the refPaths
argument. Recall that the following paths were passed as the refPaths
argument:
[['name'], ['done']];
Once the function has returned a response, the DataSource looks for any references inside of the JSONGraphEnvelope and attempts to retrieve the refPaths
from the reference path. The function’s response included a JSON Graph Reference { $type: "ref", value: ["todosById", 93] }
at the path ["todos", 2]
. Therefore the DataSource appends each of the paths in the refPaths
to the paths at which references are found in the response, yielding the following PathSets:
[['todos', 2, 'name'], ['todos', 2, 'done']];
The DataSource then attempts to retrieve these PathSets, and adds the values to the JSONGraphEnvelope returned by the function. This yields the following response:
{
paths: [
["todos", [2, "length"]]
],
jsonGraph: {
todos: {
2: { $type: "ref", value: ["todosById", 93] },
length: 3
},
todosById: {
"93": {
name: "pick up some eggs",
done: false
}
}
}
}
Notice that the response above now contains the name
and done
fields of the newly added task object. Using the refPaths
argument, we are able to retrieve values from object references returned from the function.
Now that the function has run successfully, and the values have been retrieved from the references in the response, the extraPaths
paths are retrieved. Recall that, for our implementation, we requested the following relative path from the this
object:
[['length']];
This DataSource implementation appends each one of the extraPaths
to the function path’s parent, yielding the following path:
[['todos', 'length']];
The DataSource then retrieves this path, and adds the resulting JSON Graph subset to the response, yielding the following JSONGraphEnvelope:
{
paths: [
["todos", 2]
],
jsonGraph: {
todos: {
2: { $type: "ref", value: ["todosById", 93] },
length: 3
},
todosById: {
"93": {
name: "pick up some eggs",
done: false
}
}
},
invalidated: [
["todos", "length"]
]
}
Notice that the value of the todos.length
is now in the JSON Graph object.
Dereferencing a Model (Version 1+)
When dealing with shared mutable data, it is important to avoid race conditions. For example, what if two different users are modifying objects in a list at the same time? How do we ensure that operations are applied to the right objects, even though objects may shift around in the list between being displayed to the user and being modified by the user?
Falcor provides the deref method to allow you to refer to objects by their identity rather than their location in the graph. When you dereference a Falcor Model against an object you previously retrieved from the Model, you create a new Model bound to that object. All future operations performed on this Model will be performed on the dereferenced object in the graph, whether it be get, set, or call operations.
To understand how the deref method is used, let’s take a look at an example. Imagine you are building an application that displays a list of video titles, and presents a Detail view when one of the titles is selected by the user. Let’s assume we are using the following JSON Graph representation of the list:
{
list: [
{ $type: "ref", value: ["titlesById", 53] },
{ $type: "ref", value: ["titlesById", 67] }
],
titlesById: {
53: {
name: "House of Cards",
rating: 5.0
},
67: {
name: "Daredevil",
rating: 5.0
}
}
}
Note that the graph contains a titlesById identity map which organizes each title by its unique ID.
To display the information, we can create a list view which accepts a Model and displays a list of summarized information about each title. When the user clicks on a title within the list, we will open the detail view and display additional information about the title. This begs the question of how the List view will send the selected title to the detail view.
When an item is selected, the List view could pass the Model to the Detail view, along with the path to the selected item.
new DetailView(model, ['list', selectedIndex]);
This is an anti-pattern because the path contains the index in the list where the title is located, and this item may shift in the list between the time the object is displayed and the user drills down and selects it.
Instead of passing paths around, the List view can pass a Model dereferenced to a specific object in the graph.
function item_selected(selectedIndex) {
// response was previous retrieved using model.get("list[0..1].name")
if (response.json.list && response.json.list[selectedIndex]) {
new DetailView(model.deref(json.list[selectedIndex]));
}
}
In this example, when a user clicks on a title, we dereference a Model against the title object we retrieved earlier. We pass the dereferenced Model to the Detail view which can use it to retrieve more information from the title.
function DetailView(titleModel) {
titleModel.getValue('rating').then(function(rating) {
// display rating
});
}
Operations performed on the dereferenced Model will always be applied to the same object, no matter where the object is moved within the graph. Furthermore code can interact with the object without any knowledge of its location within the graph.
Deref uses metadata about the references that get operation encounters when getting JSON responses. For example, retrieving list[0].name
will yield the following JSON response.
{
json: {
list: {
0: {
$__: ["titlesById", 53],
name: "House of Cards"
}
}
}
}
Note the $__path
added to the JSON response. When you dereference an object within a Falcor response, the new Model internally stores this path. Note that you should not use these metadata properties directly, as the specific property name may change over time and is an implementation detail. Rather you should use the public deref method instead.
Passing Dereferenced Models to Call
Dereferenced models can be passed as the argument to a function invoked with call. This approach allows you to pass an object to a function by identity, rather than a path to a volatile location within the graph.
var model = new falcor.Model(new HttpDataSource('/model.json'));
model.get('list[0]').then(response => {
var titleModel = model.deref(response.json.list[0]);
return model
.call('myList.push', [titleModel], [], ['length'])
.then(response => console.log(response.json.myList.length));
});
The code above works because JSON stringifying a dereferenced Model produces a JSON Graph reference that points to the object in the Graph.
var model = new falcor.Model(new HttpDataSource('/model.json'));
model.get('list[0]').then(response => {
var titleModel = model.deref(response.json.list[0]);
console.log(JSON.stringify(titleModel));
// prints { $type: "ref", value: ["titlesById", 53] }
});
Working with JSON Graph Data using a Model
In addition to being able to work with JSON documents, Models can also operate on JSON Graph documents. JSON Graph is a convention for modeling graph information in JSON. JSON Graph documents extend JSON with References. References can be used anywhere within a JSON object to refer to a value elsewhere within the same JSON object. This removes the need to duplicate objects when serializing a graph into a hierarchical JSON object.
Let’s say that we wanted to introduce a list of prerequisites for each TODO in a TODO list.
var json = {
todos: [
{
id: 2692,
name: 'get milk from corner store',
done: false,
prerequisites: [
{
name: 'withdraw money from ATM',
done: false
}
]
},
{
id: 4291,
name: 'withdraw money from ATM',
done: false
}
]
};
Notice that the TODO “withdraw money from the ATM” appears twice in the JSON object above. Let’s say we want to mark this task as done:
json.todos[1].done = true;
If we examine the JSON object after this change, we will notice that the change has not been propagated to all of the copies of the task. Because the same task also appears in the prerequisites array of task 2692, its done
value remains false.
console.log(JSON.stringify(json, null, 4));
/* Prints the following to the console:
{
todos: [
{
id: 2692,
name: 'get milk from corner store',
done: false,
prerequisites: [
{
id: 4291,
name: 'withdraw money from ATM',
done: false
}
]
},
{
id: 4291,
name: 'withdraw money from ATM',
done: true
}
]
};
*/
This highlights one of the hazards of representing your data as JSON: most application domains are Graphs and JSON models Trees.
When application servers send subsets of the graph across the network as JSON, they typically use the duplicate and identify strategy. If the same object appears more than once in the JSON response, the application server includes a unique ID within the object. The application client is expected to use the IDs to determine if the two copies of an object represent the same entity. This code must often be specialized for each new type of message that comes back from the server. Failing to de-dupe objects can lead to stale data being displayed to the user.
Falcor attempts to solve this problem by introducing JSON Graph. JSON Graph is a convention for modeling graph information in JSON. You can convert any JSON object into a JSON Graph in two steps:
- Move all objects to a unique location within the JSON object
- Replace all other occurrences of the object with a Reference to that object’s unique location
We can use the task ID to create a unique location in the JSON for each task. We start by adding a map of all Tasks that is organized by Task ID to the root of the document:
var json = {
todosById: {
'44': {
name: 'get milk from corner store',
done: false,
prerequisites: []
},
'54': {
name: 'withdraw money from ATM',
done: false
}
},
todos: [
{
id: 44,
name: 'get milk from corner store',
done: false,
prerequisites: [
{
id: 54,
name: 'withdraw money from ATM',
done: false
}
]
},
{
id: 54,
name: 'withdraw money from ATM',
done: true
}
]
};
Next we replace every other occurrence of each task with a Reference value. A Reference is a JSON object that contains a path to another location within an object. References can be constructed using the Model.ref factory function.
var $ref = falcor.Model.ref;
var json = {
todosById: {
'44': {
name: 'get milk from corner store',
done: false,
prerequisites: [$ref('todosById[54]')]
},
'54': {
name: 'withdraw money from ATM',
done: false
}
},
todos: [$ref('todosById[44]'), $ref('todosById[54]')]
};
Note that in the example above each TODO appears only once. If we use a Model to set a TODO to false we will observe that the new state will be reflected regardless of where in the JSON Graph we retrieve the TODO’s information.
var falcor = require('falcor');
var $ref = falcor.Model.ref;
var model = new falcor.Model({
cache: {
todos: [$ref('todosById[44]'), $ref('todosById[54]')],
todosById: {
'44': {
name: 'get milk from corner store',
done: false,
prerequisites: [$ref('todosById[54]')]
},
'54': {
name: 'withdraw money from ATM',
done: false // this is the value being set
}
}
}
});
console.log('before::', [
await model.getValue('todos[0].prerequisites[0].done'),
await model.getValue('todos[1].done')
]);
await model.setValue('todos[1].done', true);
console.log('after:', [await model.getValue('todos[0].prerequisites[0].done'), await model.getValue('todos[1].done')]);
// This outputs the following to the console:
// before: [false, false]
// after: [true, true]
Note that in the example operations above we use a path which extends beyond the reference object in the JSON Graph. However instead of short-circuiting and returning the reference, the Model follows the path in the reference and continues evaluating the remaining keys and the path at the location referred to by the path in the reference. In the next section we will explain how models evaluate paths against JSON and JSON Graph objects.
JSON Graph Path Evaluation
When evaluating paths against a JSON object, the Falcor model starts at the root of its associated JSON object and continues looking up keys until it arrives at a value type.
var falcor = require('falcor');
var $ref = falcor.Model.ref;
var model = new falcor.Model({
cache: {
todos: [$ref('todosById[44]'), $ref('todosById[54]')],
todosById: {
'44': {
name: 'get milk from corner store',
done: false,
prerequisites: [$ref('todosById[54]')],
customer: null
},
'54': {
name: 'withdraw money from ATM',
done: false
}
}
}
});
var name = await model.getValue('todosById[44].name');
// This returns:
// "get milk from corner store"
If a value type is encountered before the path is fully evaluated, the path evaluation process is short-circuited and the value discovered is returned.
var falcor = require('falcor');
var $ref = falcor.Model.ref;
var model = new falcor.Model({
cache: {
todos: [$ref('todosById[44]'), $ref('todosById[54]')],
todosById: {
'44': {
name: 'get milk from corner store',
done: false,
prerequisites: [$ref('todosById[54]')],
customer: null
},
'54': {
name: 'deliver pizza',
done: false,
prerequisites: [],
customer: {
name: 'Jim Donut',
address: '123 Seaside blvd. Pacifica, CA'
}
}
}
}
});
var name = await model.getValue('todosById[44].customer.name');
// This returns:
// null
The one exception to this rule is the case in which a Model encounters a Reference value type. When a Model encounters a reference while evaluating a path, it behaves differently than does if it encounters any other value type. If a Model encounters a reference before evaluating all of the keys in a path, the unevaluated keys are appended to the path within the reference and evaluation is resumed from root of the JSON object.
In the following piece of code, we attempt to retrieve the name of the first TODO:
var falcor = require('falcor');
var $ref = falcor.Model.ref;
var model = new falcor.Model({
cache: {
todos: [$ref('todosById[44]'), $ref('todosById[54]')],
todosById: {
'44': {
name: 'get milk from corner store',
done: false,
prerequisites: [$ref('todosById[54]')],
customer: null
},
'54': {
name: 'withdraw money from ATM',
done: false
}
}
}
});
var name = await model.getValue('todos[0].name');
// This returns:
// "get milk from corner store"
First the model evaluates the keys todo
and 0
and encounters a reference value. However instead of short-circuiting and returning the reference value, the Model resumes evaluation from the location in the JSON referred to in the reference path. This is accomplished by dynamically rewriting the path from todos[0].name
to todosById[44].name
and resuming evaluation from the root of the JSON object.
Note that references are only followed if there are more keys in the path that have not yet been evaluated. If we shorten the path to todos[0]
the model returns the reference path rather than the object it refers to.
var falcor = require('falcor');
var $ref = falcor.Model.ref;
var model = new falcor.Model({
cache: {
todos: [$ref('todosById[44]'), $ref('todosById[54]')],
todosById: {
'44': {
name: 'get milk from corner store',
done: false,
prerequisites: [$ref('todosById[54]')],
customer: null
},
'54': {
name: 'withdraw money from ATM',
done: false
}
}
}
});
var todo = await model.getValue('todos[0]');
// This returns:
// ["todosById", 44]
The process of rewriting a path when a reference is encountered is known as Path Optimization. For more information on how Path Optimization can improve the efficiency of server-side data retrieval, see Path Optimization.
JSON Graph Sentinels
In addition to References, JSON Graph introduces two more new value types: Atoms and Errors. These three special value types are all classified as Sentinels.
Sentinels are JSON objects that are treated by the Falcor Model as value types. References, Atoms, and Errors are all JSON objects with a $type
value of ref
, atom
, and error
respectively.
var $ref = falcor.Model.ref;
var model = new falcor.Model({
cache: {
todos: [
{
$type: 'ref',
value: ['todosById', 44]
},
{
$type: 'ref',
value: ['todosById', 54]
}
],
todosById: {
'44': {
name: 'get milk from corner store',
done: false
},
'45': {
$type: 'error',
value: 'todo #45 missing.'
},
'46': {
$type: 'atom',
value: [1, 2, 3]
}
}
}
});
Each Sentinel objects also contains a value
key with its actual value. One way to think about a Sentinel is a box around a value that indicates the type of the value within. Sentinels influence the way that Models interpret their values, allowing them to distinguish a path from a string or an regular object from an error for example.
Despite being JSON objects, all Sentinels are considered JSON Graph value types and therefore can be retrieved from a Model. However when a Sentinel is retrieved from a Model, the Model unboxes the value within the Sentinel and returns the value instead of the entire Sentinel object.
var falcor = require('falcor');
var model = new falcor.Model({
cache: {
todosById: {
'44': {
name: 'Get money from ATM.',
tags: {
$type: 'atom',
value: ['home', 'budget']
}
}
}
}
});
var tags = await model.getValue('todosById[44].tags');
// This returns:
// ["home", "budget"]
You can create a new Model which does not have this unboxing behavior by calling boxValues.
var falcor = require('falcor');
var $ref = falcor.Model.ref;
var model = new falcor.Model({
cache: {
todosById: {
'44': {
$type: 'atom',
value: [1, 2, 3, 4]
}
}
}
});
var todo = await model.boxValues().getValue('todosById[44]');
// This outputs the following to the console:
// {
// "$type": "atom",
// "value": [ 1, 2, 3, 4 ]
// }
As sentinels are value types, their contents cannot be changed. Like numbers and strings, they must be replaced entirely.
var falcor = require('falcor');
var model = new falcor.Model({
cache: {
todosById: {
'44': {
name: 'go to ATM',
tags: { $type: 'atom', value: ['money', 'store'] }
}
}
}
});
var tags = await model.setValue('todosById[44].tags', { $type: 'atom', value: ['money', 'store', 'debit card'] });
// This returns:
// ["money", "store", "debit card"]
Each Sentinel affects the way in which the Model interprets its value differently. References were explained in the previous section. In the next two sections, Atoms and Errors will be explained.
JSON Graph Atoms
JSON Graph allows metadata to be attached to values to control how they are handled by the Model. For example, metadata can be attached to values to control how long values stay in the Model cache and indicate whether one value is a more recent version of another value. For more information see Sentinel Metadata.
One issue is that JavaScript value types do not preserve any metadata attached to them when they are serialized as JSON:
var number = 4;
number.$expires = 5000;
number;
// number is:
// 4
Atoms “box” value types inside of a JSON object, allowing metadata to be attached to them.
var number = {
$type: 'atom',
value: 4
};
number.$expires = 5000;
number;
// number is:
// {
// "$type": "atom",
// "value": 4,
// "$expires": 5000
// }
The value of an Atom is always treated like a value type, meaning it is retrieved and set in its entirety. Mutating an Atom has no effect. Instead you must replace it entirely using the Model’s set operation.
In addition to making it possible to attach metadata to JSON values, Atoms can be used to get around the restriction against retrieving JSON Objects and Arrays from a Falcor Model.
Let’s say that we have an Array which we are certain will remain small, like a list of video subtitles for example. By boxing the subtitles Array in an Atom, we cause the Falcor model to treat it as a value and return it in its entirety.
var falcor = require('falcor');
var $ref = falcor.Model.ref;
var $atom = falcor.Model.atom;
var model = new falcor.Model({
cache: {
titlesById: {
'44': {
name: 'Die Hard',
subtitles: $atom(['en', 'fr'])
}
}
}
});
var subtitles = await model.getValue('titlesById[44].subtitles');
// This returns:
// ['en', 'fr']
Internally the Model boxes all retrieved values that have been successfully retrieved from the data source before storing these values in its local cache.
JSON Graph Errors
When a Model’s DataSource encounters an error while attempting to retrieve a value from a JSON object, an Error object is created and placed in the JSON instead of the value that was unable to be retrieved.
By default a Model delivers Errors differently than other values. If synchronous methods are used to retrieve the data from the Model the error is thrown. If the data is asynchronously being requested from the model as an Observable or a Promise, the error will be delivered in a special callback.
var falcor = require('falcor');
var model = new falcor.Model({
cache: {
titlesById: {
'44': {
$type: 'error',
value: 'failure to retrieve title.'
}
}
}
});
var result;
try {
result = await model.getValue('titlesById[44].name');
} catch (error) {
console.log('found error:');
result = error;
}
return result;
// This throws the error:
// [{
// path: ["titlesById", 44],
// value:"failure to retrieve title."
// }]
To learn more about the different ways to retrieve information from a Model, see Retrieving Data from a Model.
“What if I don’t want a Model to treat errors differently from other values?”
There are many reasons why you might want errors reported the same way as other values. For example you might retrieve several paths from a model in a single request, and resilient if one of them fails. Furthermore you might want to display errors in a template alongside successfully-retrieved values.
The treatErrorsAsValues
function creates a new Model which reports errors the same way as values.
var falcor = require('falcor');
var model = new falcor.Model({
cache: {
titlesById: {
'44': {
$type: 'error',
value: 'failure.'
}
}
}
});
var result;
try {
var response = await model.treatErrorsAsValues().get('titlesById[44].name');
} catch (error) {
console.log('error was thrown');
result = error;
} finally {
console.log('error treated like a value:');
result = response;
}
return result;
// This outputs the following and
// returns error object as a regular value:
//
// "error treated like a value:"
// {
// "json": {
// "titlesById": {
// "44": "failure."
// }
// }
// }
Note that using treatErrorsAsValues
will cause the model to deliver errors as values. However it will not provide you with a way to distinguish errors from values. If you would like to be able to receive errors alongside values, but retain the ability to distinguish between errors and values, you can chain treatErrorsAsValues
and boxValues
together. When a model is operating in boxValues
mode, it always returns the sentinels that box each value and indicate their type.
var falcor = require('falcor');
var model = new falcor.Model({
cache: {
titlesById: {
'44': {
$type: 'error',
value: 'failure.'
}
}
}
});
var response = await model
.treatErrorsAsValues()
.boxValues()
.getValue('titlesById[44]');
// This outputs the following to the regular console:
// {
// "$type": "error",
// "value": "failure."
// }
When you receive a Sentinel, you can check the $type
property of each sentinel to distinguish whether a value is an error (error
) or a successfully-retrieved value (atom
). For more information see Boxing and Unboxing.
Retrieving Data from a Model
Consider the following Model:
var falcor = require('falcor');
var $ref = falcor.Model.ref;
var model = new falcor.Model({
cache: {
todos: [$ref('todosById[79]'), $ref('todosById[99]')],
todosById: {
'99': {
name: 'deliver pizza',
done: false,
priority: 4,
customer: {
$type: 'atom',
value: {
name: 'Jim Hobart',
address: '123 pacifica ave., CA, US'
},
// this customer object expires in 30 minutes.
$expires: -30 * 60 * 1000
},
prerequisites: [$ref('todosById[79]')]
},
'79': {
$type: 'error',
value: 'error retrieving todo from database.'
}
}
}
});
var customer = await model.getValue('todos[1].customer');
// returns:
// {
// name: "Jim Hobart",
// address: "123 pacifica ave., CA, US"
// }
Note that the value
property of an Atom object is considered the value for that Atom’s path
Also note: Errors are cached in the Model just like any other value.
“Can I retrieve Arrays or Objects from a Model?”
No, instead of requesting entire Arrays or Objects, you must be explicit and request all of the individual values that you need. See JSON Graph Atoms for a way to include small, immutable collections as values in your JSON Graph.
In addition to the JavaScript path syntax, models can also process paths with ranges in indexers:
var falcor = require('falcor');
var $ref = falcor.Model.ref;
var model = new falcor.Model({
cache: {
todos: [$ref('todosById[44]'), $ref('todosById[54]')],
todosById: {
'44': {
name: 'get milk from corner store',
done: false,
prerequisites: [$ref('todosById[54]')]
},
'54': {
name: 'withdraw money from ATM',
done: false
}
}
}
});
var response = await model.get('todos[0..1].name');
// This returns:
// {
// "json": {
// "todos": {
// "0": {
// "name": "get milk from corner store"
// },
// "1": {
// "name": "withdraw money from ATM"
// }
// }
// }
// }
Models allow you to select as many paths as you want in a single network request.
model.get('todos[0..1].name', 'todos[0..1].done').then(console.log);
The paths in the previous example can be simplified to one path, because in addition to allowing ranges in indexers, Falcor models also allow multiple keys to be passed in a single indexer:
model.get('todos[0..1]["name", "done"]').then(console.log);
Boxing and Unboxing
Sentinel Metadata
Metadata can be attached to value types to control the way the Model handles them once they have been retrieved from the data source. Metadata can be added to any JSON object as a key that starts with the prefix $
. Note that any keys set on JSON value types (string, number, boolean, and null) will not persist when serialized to JSON.
Therefore in order to add metadata to JSON value types, the value types must be boxed in an Atom. For more information on Atoms see JSON Graph Atoms.
The Model supports a variety of different metadata fields, which can be attached to any JSON Graph Sentinel:
- $type
- $expires
- $timestamp
- $size
$type metadata
Every JSON Graph sentinel has to contain a $type key with one of the following three values:
atom
ref
error
For more information see JSON Graph Sentinels.
$expires metadata
The $expires metadata dictates how long a value should remain in the Model cache. The value of the $expires
key must a negative or positive integer. This value is overloaded to allow for two special values, as well as both positive and negative integers. Here are the meanings of these values:
Expire Immediately ($expires = 0)
An $expires value of 0 will cause the value to immediately be expired from the cache after being delivered.
Never Expire ($expires = 1)
An $expires value of 1 will prevent the Model from removing the value from the cache during its normal garbage collection process.
Expire at an Absolute Time ($expires > 0)
A positive integer indicates that the value should expire from the cache if the value < Date.now().
var falcor = require('falcor');
var model = new falcor.Model({
cache: {
todos: [
{
$type: 'atom',
$expires: new Date(2000, 0, 1).getTime(),
value: 'Fix Y2K bug'
}
]
}
});
var response = await model.getValue('todos[0]');
// returns:
// undefined
Expire at a Relative Time ($expires < 0)
If the $expires key value is a negative integer, the expiration time is relative to the time at which the value is written into the cache.
var falcor = require('falcor');
var model = new falcor.Model({
cache: {
todos: [
{
$type: 'atom',
$expires: -1000, // expires a second later
value: 'Deliver Pizza'
}
]
}
});
setTimeout(function() {
model.getValue('todos[0]').then(function(todo) {
console.log('value after 2 seconds:', todo);
});
}, 2000); // 2 seconds
var currentValue = await model.getValue('todos[0]');
// returns current value:
// "Deliver Pizza"
// then 2 seconds later, prints:
// "value after 2 seconds:" undefined
$timestamp metadata
Race conditions can often arise in eventually consistent systems. When multiple requests to the DataSource are in flight, there is no guarantee which request will be processed by the server first. Two sequential requests from the Model’s DataSource to the network may hit two different servers. If the application server that processes the first request is overloaded, the second request may make it to the backend data store before the first. If the second request mutates data and the backend data store is eventually consistent, changes may not be replicated by the time the first request makes it to the backend data store. If this occurs, the first request issued by the DataSource will return after the second, but will not reflect the mutation made by the second request. The result will be that the newer data in the Model cache will be overwritten by the stale data in the response from the first request.
The $timestamp metadata eliminates race conditions like this by allowing both the client and server to attach the time at which the value was last modified. If an older version of a value arrives from the network after a newer version, the Model Cache will ignore the older version of the value.
The $timestamp value is the number of milliseconds between midnight of January 1, 1970 and the date on which the value is last modified. In this example we make an attempt to overwrite the rating key with an older value. The Model will ignore our attempt to mutate the value and return the newer value instead.
var falcor = require('falcor');
var model = new falcor.Model({
cache: {
rating: {
$type: 'atom',
$timestamp: 500,
value: 3
}
}
});
// Attempt to set rating to older value will be ignored,
// and the newer value "3" will be returned;
var rating = await model.setValue('rating', {
$type: 'atom',
$timestamp: 200,
value: 5
});
$size metadata
The Model can be given a maximum cache size at creation time. When the maximum cache size is surpassed, the least-recently-used values in the cache are purged until there is enough space in the cache to add additional values. However, not all values will occupy the same amount of space in memory. For example, Atoms may contain larger values like JavaScript Objects and Arrays, and the Model will make no effort to approximate the size of these values. In order to more accurately represent the size of the value, it is possible to set a custom size using the $size metadata.
var model = new falcor.Model({
cache: {
titlesById: {
629: {
name: 'House of Cards',
subtitles: {
$type: 'atom',
value: ['en', 'fr', 'ge'],
$size: 10
}
}
}
},
maxSize: 500
});
For more information on the Model Cache, see Model Cache.
Error Handling
Cache Control
The Model Cache
Every Model contains a JSON Graph cache. By default the cache is empty, but it can be initialized to a value when the Model is constructed:
var model = new falcor.Model({
cache: {
todos: [
{
name: 'get milk from corner store',
done: false
},
{
name: 'withdraw money from ATM',
done: true
}
]
}
});
When values are requested from a Model, the Model attempts to retrieve the values from its cache using the JSON Graph get algorithm.
If a value is not found in the cache, it is requested from the data source.
Many applications run on devices with limited memory. As the Model Cache accumulates more and more data over time, there is a risk that the device may run out of memory and the application may crash. To minimize this risk, the Model Cache can be assigned a maximum size. When the size of the Model Cache outgrows the maximum size, the Model schedules a collection. During the collection phase, the Model removes values from the cache until the cache size is equal to a customizable ratio of the maximum size. The maximum cache size and collect ratio can be set when the model is constructed.
var model = new falcor.Model({ maxSize: 15000, collectRatio: 0.75 });
To minimize the risk of purging items from the cache that the application is currently using, the Model collection phase takes the following steps:
- The Model removes all of the values from the cache which have expired.
- The Model removes items from the cache in order of least recently used until the size of the Model Cache is equal to the collect ratio multiplied by the maximum size.
In the unlikely event that a value which the application is currently using is removed from the cache, the value will be reloaded from the data source when requested again.
Batching Outgoing Requests
By default, each request to the Model will map to a single request to the DataSource. Usually, the DataSource forwards the requests to a remote server on the network. Given the high cost of network requests, it can be efficient to batch multiple requests to the Model into a single request to the DataSource. This can be accomplished using the Model’s batch method:
var log = console.log.bind(console);
var httpDataSource = new falcor.HttpDataSource('/model.json');
var model = new falcor.Model({ source: httpDataSource });
var batchModel = model.batch();
batchModel.getValue('todos[0].name').then(log);
batchModel.getValue('todos[1].name').then(log);
batchModel.getValue('todos[2].name').then(log);
// The previous three model requests result in the following single request to the DataSource:
// httpDataSource.get([["todos", { from: 0, to: 2 }, "name"]]);
Path Optimization and the DataSource
When you request a path from a Model, the Model first attempts to retrieve the data from an in-memory cache. If the model fails to find the data in its cache, the Model requests the data from its DataSource.
Typically a path that has been optimized can be retrieved more efficiently by the Model’s DataSource because it requires fewer steps through the graph to retrieve the data.
When attempting to retrieve paths from the cache, Models optimize paths whenever they encounter references. This means that even if a Model is not able to find the requested data in its local cache, it may be able to request a more optimized path from the data source.