-
Notifications
You must be signed in to change notification settings - Fork 483
Home
LokiJS is a document oriented javascript database, somewhat similar to MongoDB. This is only an overview, for full documentation refer to lokijs.org.
It supports indexing, querying and filtering of data collections.
LokiJS also supports more advanced features such as mapReduce, transactions and lets you implement custom remote synchronization to save data to a server (or a local file on mobile).
Persistence to disk is already implemented for CommonJS environment such as nodejs, on a mobile you just have to request the filesystem and pass lokijs's serialize()
as content.
In a browser or mobile environment you only need to include build/lokijs.min.js
Data is stored in as javascript objects, serialized to JSON for disk persistence.
You can specify indexes to speed searches on certain object properties. The 'id' field is automatically indexed and uses binary search algorithm to avoid slower for(;;) loops
You can declare custom view functions to return result sets based on complex logic.
You can declare map and reduce functions to retrieve aggregate data from your database.
If you're working in a node.js environment, run npm install lokijs
and make sure to call var loki = require('lokijs')
Creating the db:
var db = new loki('Example');
Create a collection, specifying name, Type, index fields and whether the collection is transactional:
var users = db.addCollection('users', { indices: ['email'] });
Note that indexes and transactional flag are optional parameters. Transactions in LokiJS simply allow you to run operations and automatically restore the database to the stage it was before the beginning of the transaction if an error occurred.
Add a bunch of users in the database:
var odin = users.insert( { name : 'odin', email: '[email protected]', age: 38 } );
var thor = users.insert( { name : 'thor', email : '[email protected]', age: 25 } );
var stan = users.insert( { name : 'stan', email : '[email protected]', age: 29 } );
var oliver = users.insert( { name : 'oliver', email : '[email protected]', age: 31 } );
var hector = users.insert( { name : 'hector', email : '[email protected]', age: 15} );
var achilles = users.insert( { name : 'achilles', email : '[email protected]', age: 31 } );
Operate an update:
stan.name = 'Stan Laurel';
// update object (this really only syncs the index)
users.update(stan);
DynamicViews (recommended approach):
var dv = users.addDynamicView('a_complex_view');
dv.applyWhere(function aCustomFilter(obj){
return obj.name.length < 5 && obj.age > 30;
});
//view the data
dv.data();
// apply some changes
users.insert({ name: 'ratatosk', email: '[email protected]', age: 10320 });
// behold the dynamicview updating itself by inspecting the data
console.log(dv.data());
Create views:
function ageView(obj){
return obj.age > 30;
}
// a little more complicated, users with names longer than 3 characters and age over 30
function aCustomFilter(obj){
return obj.name.length < 5 && obj.age > 30;
}
// test the filters
var result = users.where(ageView);
var anotherResult = users.where(aCustomFilter);
// you can also store the view, without running it but for later usage
users.storeView('ageView', ageView);
// then
users.view('ageView');
Map Reduce (live example on lokijs.org ):
function getDuration( obj ){
return obj.complete ? null : obj.duration;
}
function getAverage( array ){
var cumulator = 0;
var i = array.length >>> 0;
var actual = 0;
while(i--){
if(array[i] != null){
cumulator += array[i];
actual++;
}
}
return ( cumulator / actual).toFixed(2);
}
var avgDuration = todos.mapReduce( getDuration, getAverage );
Querying via method chaining :
users.chain()
.find({'age':{'$gt': 25}})
.where(function(obj){ return obj.name.indexOf("in") != -1 })
.simplesort("age")
.offset(50)
.limit(10)
.data()