Browse Source

Merge branch 'release/0.9.0'

feature/nested-cache-fetch-fix 0.9.0
Bryan Donovan 11 years ago
parent
commit
44c37110cd
  1. 4
      History.md
  2. 28
      lib/caching.js
  3. 2
      package.json
  4. 65
      test/caching.unit.js
  5. 5
      test/support.js

4
History.md

@ -1,3 +1,7 @@
- 0.9.0 2014-08-19
Fixing issue #8 - parallel requests to a wrapped function were calling the
function multiple times. (Thanks alex-whitney).
- 0.8.0 2014-07-07 - 0.8.0 2014-07-07
Adding setex() (Thanks evanlucas) Adding setex() (Thanks evanlucas)

28
lib/caching.js

@ -18,6 +18,8 @@ var caching = function (args) {
// do we handle a cache error the same as a cache miss? // do we handle a cache error the same as a cache miss?
self.ignoreCacheErrors = args.ignoreCacheErrors || false; self.ignoreCacheErrors = args.ignoreCacheErrors || false;
self.queues = {};
/** /**
* Wraps a function in cache. I.e., the first time the function is run, * Wraps a function in cache. I.e., the first time the function is run,
* its results are stored in cache so subsequent calls retrieve from cache * its results are stored in cache so subsequent calls retrieve from cache
@ -34,21 +36,33 @@ var caching = function (args) {
*/ */
self.wrap = function (key, work, cb) { self.wrap = function (key, work, cb) {
self.store.get(key, function (err, result) { self.store.get(key, function (err, result) {
if (err && (!self.ignoreCacheErrors)) { return cb(err); } if (err && (!self.ignoreCacheErrors)) {
if (result) { cb(err);
return cb(null, result); } else if (result) {
} cb.apply(null, result);
} else if (self.queues[key]) {
self.queues[key].push(cb);
} else {
self.queues[key] = [cb];
work(function () { work(function () {
var work_args = Array.prototype.slice.call(arguments, 0); var work_args = Array.prototype.slice.call(arguments, 0);
if (work_args[0]) { // assume first arg is an error if (work_args[0]) { // assume first arg is an error
return cb(work_args[0]); return cb(work_args[0]);
} }
self.store.set(key, work_args[1], function (err) { self.store.set(key, work_args, function (err) {
if (err && (!self.ignoreCacheErrors)) { return cb(err); } if (err && (!self.ignoreCacheErrors)) {
cb.apply(null, work_args); return cb(err);
}
self.queues[key].forEach(function (done) {
done.apply(null, work_args);
});
delete self.queues[key];
}); });
}); });
}
}); });
}; };

2
package.json

@ -1,6 +1,6 @@
{ {
"name": "cache-manager", "name": "cache-manager",
"version": "0.8.0", "version": "0.9.0",
"description": "Cache module for Node.js", "description": "Cache module for Node.js",
"main": "index.js", "main": "index.js",
"scripts": { "scripts": {

65
test/caching.unit.js

@ -235,10 +235,15 @@ describe("caching", function () {
}); });
}); });
it("retrieves data from memory when available", function (done) { context("when result is already cached", function () {
cache.wrap(key, function (cb) { function get_cached_widget(name, cb) {
methods.get_widget(name, cb); cache.wrap(key, function (cache_cb) {
}, function (err, widget) { methods.get_widget(name, cache_cb);
}, cb);
}
beforeEach(function (done) {
get_cached_widget(name, function (err, widget) {
check_err(err); check_err(err);
assert.ok(widget); assert.ok(widget);
@ -247,6 +252,17 @@ describe("caching", function () {
assert.ok(result); assert.ok(result);
sinon.spy(memory_store_stub, 'get'); sinon.spy(memory_store_stub, 'get');
done();
});
});
});
afterEach(function () {
memory_store_stub.get.restore();
});
it("retrieves data from cache", function (done) {
var func_called = false; var func_called = false;
cache.wrap(key, function (cb) { cache.wrap(key, function (cb) {
@ -259,12 +275,10 @@ describe("caching", function () {
assert.deepEqual(widget, {name: name}); assert.deepEqual(widget, {name: name});
assert.ok(memory_store_stub.get.calledWith(key)); assert.ok(memory_store_stub.get.calledWith(key));
assert.ok(!func_called); assert.ok(!func_called);
memory_store_stub.get.restore();
done(); done();
}); });
}); });
}); });
});
it("expires cached result after ttl seconds", function (done) { it("expires cached result after ttl seconds", function (done) {
cache.wrap(key, function (cb) { cache.wrap(key, function (cb) {
@ -393,6 +407,45 @@ describe("caching", function () {
}); });
}); });
}); });
describe("when called multiple times in parallel with same key", function () {
var construct;
beforeEach(function () {
cache = caching({
store: 'memory',
max: 50,
ttl: 5 * 60
});
construct = sinon.spy(function (val, cb) {
var timeout = support.random.number(100);
setTimeout(function () {
cb(null, 'value');
}, timeout);
});
});
it("calls the wrapped function once", function (done) {
var values = [];
for (var i = 0; i < 2; i++) {
values.push(i);
}
async.each(values, function (val, async_cb) {
cache.wrap('key', function (cb) {
construct(val, cb);
}, function (err, result) {
assert.equal(result, 'value');
async_cb(err);
});
}, function (err) {
check_err(err);
assert.equal(construct.callCount, 1);
done();
});
});
});
}); });
describe("instantiating with no store passed in", function () { describe("instantiating with no store passed in", function () {

5
test/support.js

@ -13,6 +13,11 @@ var support = {
random_str += chars.substring(rnum, rnum + 1); random_str += chars.substring(rnum, rnum + 1);
} }
return random_str; return random_str;
},
number: function (max) {
max = max || 1000;
return Math.floor((Math.random() * max));
} }
}, },

Loading…
Cancel
Save