[api] More work on Redis store

master
indexzero 2011-03-31 04:11:11 -04:00
parent 67b5c8ab2a
commit 4c8e7a5b7f
2 changed files with 165 additions and 4 deletions

View File

@ -6,11 +6,13 @@
*/
var async = require('async'),
eyes = require('eyes'),
redis = require('redis'),
nconf = require('nconf'),
Memory = require('./Memory').Memory;
var Redis = exports.Redis = function (options) {
options = options || {}
this.namespace = options.namespace || 'nconf';
this.host = options.host || 'localhost';
this.port = options.port || 6379;
@ -19,7 +21,33 @@ var Redis = exports.Redis = function (options) {
};
Redis.prototype.get = function (key, callback) {
var self = this,
result = {},
fullKey = nconf.utils.key(this.namespace, key);
this.redis.smembers(nconf.utils.key(fullKey, 'keys'), function (err, keys) {
function addValue (source, next) {
self.get(nconf.utils.key(key, source), function (err, value) {
if (err) {
return next(err);
}
result[source] = value;
next();
});
}
if (keys && keys.length > 0) {
async.forEach(keys, addValue, function (err) {
return err ? callback(err) : callback(null, result);
})
}
else {
self.redis.get(fullKey, function (err, value) {
return err ? callback(err) : callback(null, JSON.parse(value));
});
}
});
};
Redis.prototype.set = function (key, value, callback) {
@ -29,20 +57,56 @@ Redis.prototype.set = function (key, value, callback) {
function addKey (partial, next) {
var index = path.indexOf(partial),
base = [self.namespace].concat(path.slice(0, index)),
parent = nconf.utils.key(base.concat('keys'));
self.redis.rpush(parent, partial, next);
parent = nconf.utils.key(base.concat(['keys']));
self.redis.sadd(parent, partial, next);
};
async.forEach(path, addKey, function (err) {
if (err) {
return callback(err);
}
var fullKey = nconf.utils.key(self.namespace, key);
if (!Array.isArray(value) && typeof value === 'object') {
self._setObject(fullKey, value, callback);
}
else {
value = JSON.stringify(value);
self.redis.set(fullKey, value, callback);
}
});
};
Redis.prototype.clear = function (key, callback) {
};
Redis.prototype._setObject = function (key, value, callback) {
var self = this,
keys = Object.keys(value);
function addValue (child, next) {
self.redis.sadd(nconf.utils.key(key, 'keys'), child, function (err) {
if (err) {
return next(err);
}
var fullKey = nconf.utils.key(key, child),
childValue = value[child];
if (!Array.isArray(childValue) && typeof childValue === 'object') {
self._setObject(fullKey, childValue, next);
}
else {
childValue = JSON.stringify(childValue);
self.redis.set(fullKey, childValue, next);
}
});
}
async.forEach(keys, addValue, function (err) {
return err ? callback(err) : callback();
});
};

97
test/stores/redis-test.js Normal file
View File

@ -0,0 +1,97 @@
/*
* redis-test.js: Tests for the redis nconf storage engine.
*
* (C) 2011, Charlie Robbins
*
*/
require.paths.unshift(require('path').join(__dirname, '..', '..', 'lib'));
var vows = require('vows'),
assert = require('assert'),
nconf = require('nconf');
var data = {
literal: 'bazz',
arr: ['one', 2, true, { value: 'foo' }],
obj: {
host: 'localhost',
port: 5984,
array: ['one', 2, true, { foo: 'bar' }],
auth: {
username: 'admin',
password: 'password'
}
}
}
vows.describe('nconf/stores/redis').addBatch({
"When using the nconf redis store": {
topic: new nconf.stores.Redis(),
"the set() method": {
"with a literal": {
topic: function (store) {
store.set('foo:literal', 'bazz', this.callback)
},
"should respond without an error": function (err, ok) {
assert.isNull(err);
}
},
"with an Array": {
topic: function (store) {
store.set('foo:array', data.arr, this.callback)
},
"should respond without an": function (err, ok) {
assert.isNull(err);
}
},
"with an Object": {
topic: function (store) {
store.set('foo:object', data.obj, this.callback)
},
"should respond without an error": function (err, ok) {
assert.isNull(err);
}
}
}
}
}).addBatch({
"When using the nconf redis store": {
topic: new nconf.stores.Redis(),
"the get() method": {
"with a literal value": {
topic: function (store) {
store.get('foo:literal', this.callback);
},
"should respond with the correct value": function (err, value) {
assert.equal(value, data.literal);
}
},
"with an Array value": {
topic: function (store) {
store.get('foo:array', this.callback);
},
"should respond with the correct value": function (err, value) {
assert.deepEqual(value, data.arr);
}
},
"with an Object value": {
topic: function (store) {
store.get('foo:object', this.callback);
},
"should respond with the correct value": function (err, value) {
assert.deepEqual(value, data.obj);
}
}
}
}
}).addBatch({
/*,
"the clear() method": {
"should respond with the true": function (store) {
assert.equal(store.get('foo:bar:bazz'), 'buzz');
assert.isTrue(store.clear('foo:bar:bazz'));
assert.isTrue(typeof store.get('foo:bar:bazz') === 'undefined');
}
}*/
}).export(module);