Recollect is an abstraction layer over IndexedDB, providing a rich Promise-based interface to interact with and query client-side databases. It supports all major browsers, and support for some older browsers can be attained through use of a polyfill.
- Example
- Getting Started
- API
- Recollect
- ObjectStore
.find(query)
->Promise
.findOne(query)
->Promise
.findByIndex(fieldName, value, query)
->Promise
.findOneByIndex(fieldName, value, query)
->Promise
.insertOne(newObject)
->Promise
.insertMany(newObjects)
->Promise
.update(options)
->Promise
.replace(keyPath, newObject)
->Promise
.delete(key)
->Promise
.drop()
->Promise
- Queries
- Errors
Here's a quick look at what you can do.
var oz = new Recollect("landOfOz");
oz.initialize()
.then(() => oz.createObjectStore({ osName: "characters" }))
.then(() => oz.characters.insertMany([{
name: "Scarecrow",
hasPets: false,
age: 24
}, {
name: "Dorothy",
hasPets: true,
age: 17
}, {
name: "Witch of the West",
hasPets: true,
age: 849
}]))
.then(ids => console.log(`Created new character entries with unique IDs: ${ids}.`))
.then(() => oz.characters.findOne({
hasPets: true,
age: { $gt: 15, $lt: 30 }
}))
.then(character => console.log(`${character.name} matches your search.`))
.then(() => oz.characters.drop())
.then(() => oz.drop());
You can try it for yourself, here.
If you use NPM for your frontend projects, it can be included in your project by installing (npm install --save recollect
) and requiring var Recollect = require("recollect");
. If you want to use the ES6 sources directly, do var Recollect = require("recollect/src");
To run off a CDN, the UMD module is available at the following URL: https://cdn.rawgit.com/divmain/recollect/v1.0.0/dist/recollect.min.js
Make sure the URL you use points to the version that you want.
Creates a new Recollect instance. Provides interface for interacting with specified database. Should be initialize
d before use.
Initializes the Recollect instance and prepares it for use. For any object stores previously created on the specified database, a new instance of ObjectStore will be created and attached to the Recollect instance.
The returned promise resolves to the initialized Recollect instance. Additionally, any pre-existing object stores are initialized and attached to the Recollect instance.
var recollect = new Recollect("someDb") // someDb has two object stores: obstoreA, obstoreB
recollect.initialize().then(function (db) {
db.obstoreA
// ObjectStore instance
db.obstoreB
// ObjectStore instance
});
Creates and configures new datastore. Options:
osName
- Name of object store.autoIncrement
- Indicates whether key values should automatically be created as objects are added. (optional, defaults to true)keyPath
- Indicates the path to the key for each object stored. (optional, defaults to_id
)indexes
- An object indicating which fields to index.[fieldName]
- Key path of field to index.[fieldName].unique
- If true, no two entries should share value in this field.[fieldName].multiEntry
- If true, values in fieldName should be an array of values that can be independently queried.
The returned promise resolves to the new ObjectStore instance. Additionally, this ObjectStore instance is attached to the parent Recollect instance.
recollect.createObjectStore({
osName: "newDatabase"
}).then(function (objStore) {
console.log(objStore === recollect.newDatabase);
// true
});
Destroys the database.
The returned promise resolves to undefined
on success.
Takes a query, and finds all matching objects.
The returned promise resolves to an array of matching objects in the object store. If no matches are found, it resolves to an empty array.
recollect.animals.find({
name: /er$/,
age: { $lt: 5 }
}).then(function (matches) {
console.log(matches);
// [{
// _id: 148,
// name: "tiger",
// age: 4
// }, {
// _id: 177,
// name: "panther",
// age: 3
// }]
});
Takes a query and finds a single matching object.
The returned promise resolves to the first matching object found in the object store. If no matches are found, it resolves to undefined
.
recollect.animals.findOne({
name: /er$/,
age: { $lt: 5 }
}).then(function (match) {
console.log(match);
// {
// _id: 148,
// name: "tiger",
// age: 4
// }
});
Takes the name of an indexed field, an expected value for that field, and an optional query representing additional constraints, and finds all matching objects. This method is faster than .find
, as the possible results are first filtered by the indexed field/value before applying the provided query.
The returned promise resolves to an array of matching objects in the object store. If no matches are found, it resolves to an empty array.
recollect.sportsTeams.findByIndex("sport", "football", {
hasWonChampionship: true
}).then(function (teams) {
console.log(teams);
// [{
// _id: 7261,
// name: "New England Patriots",
// sport: "football",
// hasWonChampionship: true
// }, {
// _id: 7199,
// name: "Seattle Seahawks",
// sport: "football",
// hasWonChampionship: true
// }, {
// // ...
// }]
});
Takes the name of an indexed field, an expected value for that field, and an optional query representing additional constraints, and finds one matching object.
The returned promise resolves to the first matching object found in the object store. If no matches are found, it resolves to undefined
.
recollect.sportsTeams.findOneByIndex("sport", "football", {
hasWonChampionship: true
}).then(function (team) {
console.log(team);
// {
// _id: 7261,
// name: "New England Patriots",
// sport: "football",
// hasWonChampionship: true
// }
});
Takes an object and inserts it into the object store. If autoIncrement
was set to true
for the object store, an error will be thrown if a value is present in the key field. If autoIncrement
was set to false
, an error will be thrown if the value is not present.
The returned promise resolves to the key of the inserted object.
recollect.sportsTeams.insertOne({
name: "Formidable Labs",
sport: "software-development",
hasWonChampionship: true
}).then(function (key) {
console.log(key);
// 42
});
Takes an array of objects and inserts them into the object store. The same constraints on key values and autoIncrement
are the same as for .insertOne
.
The returned promise resolves to an array of keys for the inserted objects.
recollect.sportsTeams.insertMany([{
name: "Microsoft",
sport: "software-development",
hasWonChampionship: true
}, {
name: "Google",
sport: "software-development",
hasWonChampionship: true
}]).then(function (keys) {
console.log(keys);
// [ 455, 456 ]
});
Given a query, finds all matching objects and overwrites any properties in each of those objects with the provided new properties.
query
- a Recollect query.newProperties
- properties to merge into objects matched byquery
.
The returned promise resolves to undefined
.
Given a unique key
, replaces the identified object with the provided newObject
The returned promise resolves to undefined
.
Given a unique key, finds object with key and removes it from the object store.
The returned promise resolves to undefined
.
Removes the object store from the database.
The returned promise resolves to undefined
.
Query-literals are fundamental to extracting data from your databases. They're used by several methods, including the find-class methods and update.
A query-literal is an object constructed of key-paths and a set of conditions related to the desired data at that key-path. They look like the following:
{
"thing": "something",
"key.val": { $lte: 5.5 }
}
Key-paths are period-delimited, meaning that you can query against deep values of objects in your database. The query above translates to the following english phrase:
Find an object that has a property
thing
whose value equals"something"
. This object should also have a propertykey
, whose value is an object with propertyval
. The value ofval
should be less than or equal to5.5
It would match the following object, among others:
{
thing: "something",
key: {
"val": 4
}
}
There are several query operators, documented below.
If you do not use operators, and instead pass in specific values, the following rules apply:
- if a key-path's value is a regular expression, it is interpreted as shorthand for the $regex operator;
- if a key-path's value is a javascript primitive, it is interpreted as shorthand for the $eq operator;
- if a key-path's value is an object that includes non-operators (
$gt
,$lt
, etc.), it is interpreted as shorthand for the $eq-operator operator.
The following is an example of the above three conditions:
{
val1: /someRegexMatch/,
val2: true,
val3: { $gt: 5, nonOperator: true }
}
If you have an object with a property that contains an actual .
character, you can escape it like so:
{
"shallow\\.key": "someValue"
}
This condition is true if the value found at the key-path indicated is greater than the provided value.
Given:
{
"keypath": { $gt: 5 }
}
True:
{
keypath: 8
}
False:
{
keypath: 4
}
All comparisons are done Lexicographically.
This condition is true if the value found at the key-path indicated is less than the provided value.
Given:
{
"keypath": { $lt: 5 }
}
True:
{
keypath: 4
}
False:
{
keypath: 8
}
All comparisons are done lexicographically.
This condition is true if the value found at the key-path indicated is greater than or equal to the provided value.
Given:
{
"keypath": { $gte: 5 }
}
True:
{
keypath: 5
}
False:
{
keypath: 4.9
}
All comparisons are done Lexicographically.
This condition is true if the value found at the key-path indicated is less than or equal to the provided value.
Given:
{
"keypath": { $lte: 5 }
}
True:
{
keypath: 5
}
False:
{
keypath: 5.1
}
All comparisons are done Lexicographically.
This condition is true if the value found at the key-path indicated is not equal to the provided value.
Given:
{
"keypath": { $neq: 5 }
}
True:
{
keypath: 5
}
False:
{
keypath: "hello"
}
This condition is true if the value found at the key-path indicated is a string that contains the provided value.
Given:
{
"keypath": { $contains: "name" }
}
True:
{
keypath: "Hello, my name is George."
}
False:
{
keypath: "Hello, I am Jerry."
}
This condition is true if the value found at the key-path indicated is a string that matches the provided regular expression.
Given:
{
"keypath": { $regex: /ion$/ }
}
True:
{
keypath: "diction"
}
False:
{
keypath: "dictionary"
}
The $fn
operator provides a mechanism for queries that are not supported by the other operators. The $fn
operator expects a function that returns a truey or falsey value. This function should take a single argument - this argument will be the object currently being tested for a match.
Given:
{
"keypath": {
$fn: function (obj) {
if (obj.thing) {
return _.isArray(obj.thing.deepThing);
}
return false;
}
}
}
True:
{
keypath: {
thing: {
deepThing: ["easy as", 1, 2, 3]
}
}
}
False:
{
keypath: "I have no thing property."
}
False:
{
keypath: {
thing: {
deepThing: "I am not an array."
}
}
}
If the function throws an error, this will be interpreted as a falsey response. However, it is still considered best practice to protect against unnecessary and expensive exceptions, such as non-existent deep values in an object.
In many situations, the $eq
operator will provide little benefit. The query { name: "Bob" }
behaves identically to { name: { $eq: "Bob" } }
However, it will be very useful in certain circumstances. For example, what if you are searching for an object that has a regular expression as a property? If we were to query like so, { a: /er$/ }
, it wouldn't find an object whose a
property equals /er$/
. It would find an object whose property matches the pattern.
$eq
provides an alternative: { a: { $eq: /er$/ } }
.
Similarly, consider a situation where you are searching for an object with sub-property $gte
. If you were testing for equality, instead of { a: { $gte: 1 } }
, you could construct a query like so: { a: { $eq: { $gte: 1 } } }
.
Note that, at present, there is no mechanism to search for an object that has a property name matching one of the provided query operators.
For each object that Recollect stores, metadata related to that object is also stored. You can access this metadata with the special keypath namespace $meta
. Standard operators can be used to query this metadata alongside any other conditions.
Note: Because the $meta
keypath is reserved by Recollect, you cannot use it to perform queries for objects that contain property $meta
.
This is an integer representing the number of milliseconds since the epoch when the object was created. It is set once, when the object is originally added to the object store.
It has no relation to corresponding backend records, should they exist, or the life-cycle of those records.
Example:
{
// find any records created on or before midnight on January 1, 1985
"$meta.created": {
$lte: 473414400000
}
}
This is an integer representing the number of milliseconds since the epoch when the object was last updated. It is updated whenever a change is made. If no change has ever been made, its value will be null
.
Example:
{
// find any records modified after 16:45, February 5, 2014
"$meta.modified": {
$gt: new Date(2014, 1, 5, 16, 45).getTime()
}
}
The following error prototypes are available as properties on Recollect.Errors
. They are thrown (or provided as the reject
-ed value) in the indicated circumstances.
Error name | Description |
---|---|
ObjectNotFoundError | Thrown on replace if object with keyPath is not found. |
IndexedDbNotFound | Thrown if IndexedDB is unsupported in browser. |
ConnectionError | Thrown upon failure to open database. |
CursorError | Thrown upon unexpected condition while iterating over records. |
InvalidArgumentError | Thrown if required options not provided or invalid. |
TransactionError | Thrown upon unexpected condition while performing transaction. |
InitializationError | Thrown if initialization happens improperly or more than once. |
Oftentimes, additional information about the problem will be available on the error object as err.message
.