Memoization framework for NodeJS. There are several memoization modules already for Node.JS that are very good. The difference is that this module is built with plugins for cache and ease of use in mind.
This is not a complex ES6-compliant memoization library but it will work for simple caches. In particular, this comes with a very basic memory cache but REDIS, MongoDB and S3 plugins are in the works. In addition, it is possible to do multi-layered caching this way.
This module is experimental and the kinks have yet to be worked out.
npm install memojs
- A Redis plugin is available -- memojs-redis
- A MongoDB plugin is available -- memojs-mongodb
- A simple in memory plugin is included by default
- A per-domain cache is also available but not included by default
There are two seperate ways to invoke this library. The simpler approach is by Function prototype and the other approach is explicit initialization.
More explicit documentation will be added soon.
var memojs = require('memojs');
function someSyncFunction(arg1, arg2) {
return arg1 + arg2;
}
function someFunction(arg1, arg2, done) {
return done(null, arg1 + arg2);
}
// note that someFunction.memoized is declared via prototype automatically.
someFunction.memoized(5, 3, function (err, val) {
// now if you call someFunction.memoized(5, 3, cb) it will use the cache.
});
// you can configure the cache and other details as follows
someFunction.memoized.configure({store: yourStoreImplementation()});
// you can also get a one-off implementation that uses a different store as follows
var someFunctionButDifferentCache = someFunction.memoize({store: anotherStore()});
// you can get contents from the cache directly via the cached property.
// the following call will not actually ever call someFunction
someFunction.memoized.cached(5, 3, callback);
// you can readThrough values via readThrough. This does not read from the cache
// but will end up saving to the cache though.
someFunction.memoized.readThrough(5, 3, callback);
// you can also write through to the cache. This does not call someFunction but allows updating
// the caches directly.
someFunction.memoized.writeThrough(42)(5, 3, callback);
// if you want to delete an item from the cache, writeThrough 'undefined'
someFunction.memoized.writeThrough(undefined)(5, 3, callback);
// if you api was sync instead of callback, you can use the .sync variants but note that
// all the responses are still async as the cache underneath is expected to be async.
someSyncFunction.memoized.sync(5, 3, callback);
// By default memoized uses the name of the function and JSON.stringify of the arguments to determine
// the cache key. If you have two functions with the same name but destined for different caches, you
// can configure different names like so:
someSyncFunction.memoized.configure({name: 'someSyncFunction-alternate'});
// if you want to normalize the arguments for the cache, you can provide your own method that does that.
someSyncFunction.memoized.configure({getCacheProperties: function (info) { info.key = some_func_of(info.args); }});
}
var memojs = require('memojs');
function someSyncFunction(arg1, arg2) {
return arg1 + arg2;
}
function someFunction(arg1, arg2, done) {
return done(null, arg1 + arg2);
}
var memoized;
// note that someFunction.memoized is declared via prototype automatically.
memoized = memojs(someFunction);
memoized(5, 3, function (err, val) {
// now if you call someFunction.memoized(5, 3, cb) it will use the cache.
});
// you can configure the cache and other details as follows
memoized.configure({store: yourStoreImplementation()});
// you can also get a one-off implementation that uses a different store as follows
var someFunctionButDifferentCache = memojs(someFunction, {store: anotherStore()});
// you can get contents from the cache directly via the cached property.
// the following call will not actually ever call someFunction
memoized.cached(5, 3, callback);
// you can readThrough values via readThrough. This does not read from the cache
// but will end up saving to the cache though.
memoized.readThrough(5, 3, callback);
// you can also write through to the cache. This does not call someFunction but allows updating
// the caches directly.
memoized.writeThrough(42)(5, 3, callback);
// if you want to delete an item from the cache, writeThrough 'undefined'
memoized.writeThrough(undefined)(5, 3, callback);
// if you api was sync instead of callback, you can use the .sync variants but note that
// all the responses are still async as the cache underneath is expected to be async.
var memoizedSync = memojs(memoizedSync);
memoizedSync(5, 3, callback);
// By default memoized uses the name of the function and JSON.stringify of the arguments to determine
// the cache key. If you have two functions with the same name but destined for different caches, you
// can configure different names like so:
memoized.configure({name: 'someSyncFunction-alternate'});
// if you want to normalize the arguments for the cache, you can provide your own method that does that.
memoized.configure({getCacheProperties: function (info) { info.key = some_func_of(info.args); }});
}
The module comes with a built in in-memory cache. While this is interesting for short-lived applications, the memory cache needs to be pruned for long lived applications. To do that, you need to explicitly set the store as follows:
var memojs = require('memojs');
var store = require('memojs/memory.js').create();
// configure memojs to use this store.
memojs.configure({store: store});
// whenever you want to prune the keys, do:
// be warned that this could take a while.
store.prune();
Sometimes, the cache is only needed on a per-domain basis. For example, if you are using a new node domain per HTTP request, or you are using one of the excellent expressjs domain middlewares (connect-domain or express-domain-middleware), you may want to provide a cache that is only used on a request and thrown away. For example, you read the user object and any database calls to read it again need to hit the cache.
In that case, you can use the memoryDomain store which stores the cache only on the current active domain.
var memojs = require('memojs');
var store = require('memojs/memoryDomain').create();
// configure memojs to use this store
memojs.configure({store: store});
// any future calls to memojs will use the local store as needed.
Sometimes you want to chain stores in a write-through fashion. For example, use a per-domain store backed by a REDIS store.
All the stores accept a previous store to pass the request through in case of a cache miss.
var redisStore = require('memojs-redis').create(null, {redisArgs: [port, host, {auth_pass: pwd}]});
var memoryStore = require('memojs/memory').create(redisStore);
// now use this
memojs.configure({store: memoryStore});