You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

709 lines
16 KiB
JavaScript

11 years ago
(function(module) {
'use strict';
var mongoClient = require('mongodb').MongoClient,
11 years ago
winston = require('winston'),
async = require('async'),
11 years ago
nconf = require('nconf'),
11 years ago
express = require('express'),
11 years ago
mongoStore = require('connect-mongo')(express),
mongoHost = nconf.get('mongo:host'),
db;
11 years ago
module.init = function(callback) {
mongoClient.connect('mongodb://'+ mongoHost + ':' + nconf.get('mongo:port') + '/' + nconf.get('mongo:database'), function(err, _db) {
db = _db;
11 years ago
if(err) {
winston.error("NodeBB could not connect to your Mongo database. Mongo returned the following error: " + err.message);
process.exit();
}
11 years ago
module.sessionStore = new mongoStore({
db: db
});
11 years ago
11 years ago
// TODO : what is the db user name??
/*if(nconf.get('mongo:password')) {
db.authenticate(dbUser, nconf.get('mongo:password'), function (err) {
});
}*/
db.createCollection('objects', function(err, collection) {
if(err) {
winston.error("Error creating collection " + err.message);
return;
}
if(collection) {
collection.ensureIndex({_key :1, setName:1}, {background:true}, function(err, name){
if(err) {
winston.error("Error creating index " + err.message);
}
});
}
});
11 years ago
db.createCollection('search', function(err, collection) {
if(err) {
winston.error("Error creating collection " + err.message);
return;
}
if(collection) {
collection.ensureIndex({content:'text'}, {background:true}, function(err, name){
if(err) {
winston.error("Error creating index " + err.message);
}
});
}
})
callback(err);
});
11 years ago
}
11 years ago
11 years ago
//
// Exported functions
//
module.searchIndex = function(key, content, id) {
11 years ago
var data = {
id:id,
key:key,
content:content
};
db.collection('search').update({id:id, key:key}, {$set:data}, {upsert:true, w: 1}, function(err, result) {
if(err) {
winston.error('Error indexing ' + err.message);
}
});
}
module.search = function(key, term, callback) {
11 years ago
db.command({text:"search" , search: term, filter: {key:key} }, function(err, result) {
if(err) {
return callback(err);
}
if(!result) {
return callback(null, []);
}
if(result.results && result.results.length) {
var data = result.results.map(function(item) {
return item.obj.id;
});
callback(null, data);
} else {
callback(null, []);
}
});
}
module.searchRemove = function(key, id) {
11 years ago
db.collection('search').remove({id:id, key:key}, function(err, result) {
11 years ago
if(err) {
winston.error('Error removing search ' + err.message);
}
});
}
module.flushdb = function(callback) {
db.dropDatabase(function(err, result) {
if(err){
winston.error(error);
if(callback) {
return callback(err);
}
}
if(callback) {
callback(null);
}
});
}
11 years ago
module.getFileName = function(callback) {
11 years ago
throw new Error('not-implemented');
}
module.info = function(callback) {
db.stats({scale:1024}, function(err, stats) {
// TODO : if this it not deleted the templates break,
// it is a nested object inside stats
delete stats.dataFileVersion;
stats.avgObjSize = (stats.avgObjSize / 1024).toFixed(2);
11 years ago
stats.raw = JSON.stringify(stats, null, 4);
stats.mongo = true;
//remove this when andrew adds in undefined checking to templates
stats.redis = false;
callback(err, stats);
});
11 years ago
}
// key
module.exists = function(key, callback) {
db.collection('objects').findOne({$or:[{_key:key}, {setName:key}]}, function(err, item) {
11 years ago
callback(err, item !== undefined && item !== null);
});
11 years ago
}
module.delete = function(key, callback) {
11 years ago
db.collection('objects').remove({_key:key}, function(err, result) {
if(err) {
if(callback) {
return callback(err);
} else {
return winston.error(err.message);
}
11 years ago
}
11 years ago
if(result === 0) {
db.collection('objects').remove({setName:key}, function(err, result) {
if(callback) {
callback(err, result);
}
11 years ago
});
} else {
if(callback) {
callback(null, result);
}
11 years ago
}
});
11 years ago
}
module.get = function(key, callback) {
11 years ago
module.getObjectField(key, 'value', callback);
11 years ago
}
module.set = function(key, value, callback) {
11 years ago
var data = {value:value};
module.setObject(key, data, callback);
11 years ago
}
module.keys = function(key, callback) {
db.collection('objects').find( { _key: { $regex: key /*, $options: 'i'*/ } }, function(err, result) {
callback(err, result);
});
}
11 years ago
//hashes
function removeHiddenFields(item) {
if(item) {
if(item._id) {
delete item._id;
}
if(item._key) {
delete item._key;
}
if(item.setName) {
delete item.setName;
}
}
return item;
}
11 years ago
module.setObject = function(key, data, callback) {
11 years ago
data['_key'] = key;
11 years ago
db.collection('objects').update({_key:key}, {$set:data}, {upsert:true, w: 1}, function(err, result) {
if(callback) {
callback(err, result);
}
11 years ago
});
11 years ago
}
11 years ago
module.setObjectField = function(key, field, value, callback) {
11 years ago
var data = {};
// if there is a '.' in the field name it inserts subdocument in mongo, replace '.'s with \uff0E
field = field.replace(/\./g, '\uff0E');
11 years ago
data[field] = value;
db.collection('objects').update({_key:key}, {$set:data}, {upsert:true, w: 1}, function(err, result) {
if(callback) {
callback(err, result);
}
11 years ago
});
11 years ago
}
module.getObject = function(key, callback) {
db.collection('objects').findOne({_key:key}, function(err, item) {
removeHiddenFields(item);
11 years ago
callback(err, item);
});
11 years ago
}
module.getObjects = function(keys, callback) {
db.collection('objects').find({_key:{$in:keys}}, {_id:0}).toArray(function(err, data) {
if(err) {
return callback(err);
}
var returnData = [],
resultIndex = 0;
function findData(key) {
if(!data) {
return null;
}
for(var i=0; i<data.length; ++i) {
if(data[i]._key === key) {
var item = data.splice(i, 1);
if(item && item.length) {
return item[0];
} else {
return null;
}
}
}
return null;
}
for(var i=0; i<keys.length; ++i) {
returnData.push(findData(keys[i]));
}
callback(err, returnData);
});
}
11 years ago
module.getObjectField = function(key, field, callback) {
module.getObjectFields(key, [field], function(err, data) {
if(err) {
return callback(err);
}
callback(null, data[field]);
11 years ago
});
11 years ago
}
module.getObjectFields = function(key, fields, callback) {
var _fields = {};
for(var i=0; i<fields.length; ++i) {
_fields[fields[i].replace(/\./g, '\uff0E')] = 1;
}
db.collection('objects').findOne({_key:key}, _fields, function(err, item) {
if(err) {
return callback(err);
}
if(item === null) {
item = {};
}
for(var i=0; i<fields.length; ++i) {
if(item[fields[i]] === null || item[fields[i]] === undefined) {
item[fields[i]] = null;
}
}
removeHiddenFields(item);
callback(err, item);
});
11 years ago
}
module.getObjectValues = function(key, callback) {
module.getObject(key, function(err, data) {
if(err) {
return callback(err);
}
var values = [];
for(var key in data) {
values.push(data[key]);
}
callback(null, values);
});
11 years ago
}
module.isObjectField = function(key, field, callback) {
11 years ago
var data = {};
field = field.replace(/\./g, '\uff0E');
11 years ago
data[field] = '';
db.collection('objects').findOne({_key:key}, {fields:data}, function(err, item) {
if(err) {
return callback(err);
}
callback(err, !!item && item[field] !== undefined && item[field] !== null);
});
11 years ago
}
module.deleteObjectField = function(key, field, callback) {
var data = {};
field = field.replace(/\./g, '\uff0E');
data[field] = "";
db.collection('objects').update({_key:key}, {$unset : data}, function(err, result) {
if(callback) {
callback(err, result);
}
});
11 years ago
}
module.incrObjectField = function(key, field, callback) {
module.incrObjectFieldBy(key, field, 1, callback);
11 years ago
}
module.decrObjectField = function(key, field, callback) {
module.incrObjectFieldBy(key, field, -1, callback);
11 years ago
}
11 years ago
module.incrObjectFieldBy = function(key, field, value, callback) {
var data = {};
field = field.replace(/\./g, '\uff0E');
data[field] = value;
11 years ago
db.collection('objects').update({_key:key}, {$inc : data}, {upsert:true}, function(err, result) {
module.getObjectField(key, field, function(err, value) {
if(callback) {
callback(err, value);
}
});
});
11 years ago
}
11 years ago
// sets
11 years ago
11 years ago
module.setAdd = function(key, value, callback) {
if(value !== null && value !== undefined) {
value = value.toString();
}
db.collection('objects').update({_key:key}, {$addToSet: { members: value }}, {upsert:true, w: 1}, function(err, result) {
if(callback) {
callback(err, result);
}
});
11 years ago
}
module.setRemove = function(key, value, callback) {
if(value !== null && value !== undefined) {
value = value.toString();
}
db.collection('objects').update({_key:key, members: value}, {$pull : {members: value}}, function(err, result) {
if(callback) {
callback(err, result);
}
11 years ago
});
11 years ago
}
module.isSetMember = function(key, value, callback) {
if(value !== null && value !== undefined) {
value = value.toString();
}
db.collection('objects').findOne({_key:key, members: value}, function(err, item) {
11 years ago
callback(err, item !== null && item !== undefined);
});
11 years ago
}
module.isMemberOfSets = function(sets, value, callback) {
function iterator(set, next) {
module.isSetMember(set, value, function(err, result) {
if(err) {
return next(err);
}
next(null, result?1:0);
});
}
async.map(sets, iterator, function(err, result) {
callback(err, result);
});
11 years ago
}
module.getSetMembers = function(key, callback) {
db.collection('objects').findOne({_key:key}, {members:1}, function(err, data) {
11 years ago
if(err) {
return callback(err);
}
if(!data) {
callback(null, []);
} else {
callback(null, data.members);
11 years ago
}
11 years ago
});
}
module.setCount = function(key, callback) {
db.collection('objects').findOne({_key:key}, function(err, data) {
11 years ago
if(err) {
return callback(err);
}
if(!data) {
11 years ago
return callback(null, 0);
}
callback(null, data.members.length);
11 years ago
});
11 years ago
}
module.setRemoveRandom = function(key, callback) {
db.collection('objects').findOne({_key:key}, function(err, data) {
11 years ago
if(err) {
if(callback) {
return callback(err);
} else {
return winston.error(err.message);
}
11 years ago
}
if(!data) {
if(callback) {
callback(null, 0);
}
11 years ago
} else {
var randomIndex = Math.floor(Math.random() * data.members.length);
var value = data.members[randomIndex];
module.setRemove(data._key, value, function(err, result) {
if(callback) {
callback(err, value);
}
11 years ago
});
}
});
11 years ago
}
11 years ago
// sorted sets
module.sortedSetAdd = function(key, score, value, callback) {
var data = {
score:score,
value:value
};
data.setName = key;
11 years ago
module.setObject(key + ':' + value, data, callback);
11 years ago
}
module.sortedSetRemove = function(key, value, callback) {
db.collection('objects').remove({setName:key, value:value}, function(err, result) {
if(callback) {
callback(err, result);
}
});
11 years ago
}
function getSortedSetRange(key, start, stop, sort, callback) {
db.collection('objects').find({setName:key}, {fields:{value:1}})
.limit(stop - start + 1)
.skip(start)
.sort({score: sort})
.toArray(function(err, data) {
if(err) {
return callback(err);
}
// maybe this can be done with mongo?
data = data.map(function(item) {
return item.value;
});
callback(err, data);
});
}
11 years ago
module.getSortedSetRange = function(key, start, stop, callback) {
getSortedSetRange(key, start, stop, 1, callback);
11 years ago
}
module.getSortedSetRevRange = function(key, start, stop, callback) {
getSortedSetRange(key, start, stop, -1, callback);
11 years ago
}
module.getSortedSetRevRangeByScore = function(args, callback) {
//var args = ['topics:recent', '+inf', timestamp - since, 'LIMIT', start, end - start + 1];
var key = args[0],
max = (args[1] === '+inf')?Number.MAX_VALUE:args[1],
min = args[2],
start = args[4],
stop = args[5];
db.collection('objects').find({setName:key, score: {$gt:min, $lt:max}}, {fields:{value:1}})
.limit(stop - start + 1)
.skip(start)
.sort({score: -1})
.toArray(function(err, data) {
if(err) {
return callback(err);
}
// maybe this can be done with mongo?
data = data.map(function(item) {
return item.value;
});
callback(err, data);
});
11 years ago
}
module.sortedSetCount = function(key, min, max, callback) {
11 years ago
db.collection('objects').count({setName:key, score: {$gt:min, $lt:max}}, function(err, count) {
if(err) {
return callback(err);
}
if(!count) {
return callback(null, 0);
}
callback(null,count);
});
11 years ago
}
module.sortedSetRank = function(key, value, callback) {
module.getSortedSetRange(key, 0, -1, function(err, result) {
if(err) {
return callback(err);
}
var rank = result.indexOf(value);
if(rank === -1) {
return callback(null, null);
}
callback(null, rank);
});
}
module.sortedSetScore = function(key, value, callback) {
db.collection('objects').findOne({setName:key, value: value}, {fields:{score:1}}, function(err, result) {
if(err) {
return callback(err);
}
if(result) {
return callback(null, result.score);
}
callback(err, null);
});
}
module.sortedSetsScore = function(keys, value, callback) {
db.collection('objects').find({setName:{$in:keys}, value: value}).toArray(function(err, result) {
if(err) {
return callback(err);
}
var returnData = [],
resultIndex = 0;
for(var i=0; i<keys.length; ++i) {
if(result && resultIndex < result.length && keys[i] === result[resultIndex].setName) {
returnData.push(result[resultIndex].score);
++resultIndex;
} else {
returnData.push(null);
}
}
callback(null, returnData);
});
}
11 years ago
// lists
module.listPrepend = function(key, value, callback) {
module.isObjectField(key, 'array', function(err, exists) {
if(err) {
if(callback) {
return callback(err);
} else {
return winston.error(err.message);
}
}
if(exists) {
db.collection('objects').update({_key:key}, {'$set': {'array.-1': value}}, {upsert:true, w:1 }, function(err, result) {
if(callback) {
callback(err, result);
}
});
} else {
module.listAppend(key, value, callback);
}
})
11 years ago
}
module.listAppend = function(key, value, callback) {
db.collection('objects').update({ _key: key }, { $push: { array: value } }, {upsert:true, w:1}, function(err, result) {
if(callback) {
callback(err, result);
}
});
11 years ago
}
module.listRemoveLast = function(key, callback) {
module.getListRange(key, -1, 0, function(err, value) {
if(err) {
return callback(err);
}
db.collection('objects').update({_key: key }, { $pop: { array: 1 } }, function(err, result) {
if(err) {
if(callback) {
return callback(err);
}
return;
}
if(value && value.length) {
if(callback) {
callback(err, value[0]);
}
} else {
if(callback) {
callback(err, null);
}
}
});
});
}
11 years ago
module.getListRange = function(key, start, stop, callback) {
if(stop === -1) {
// mongo doesnt allow -1 as the count argument in slice
// pass in a large value to retrieve the whole array
stop = Math.pow(2, 31) - 2;
}
db.collection('objects').findOne({_key:key}, { array: { $slice: [start, stop - start + 1] }}, function(err, data) {
if(err) {
return callback(err);
}
if(data && data.array) {
callback(null, data.array);
} else {
callback(null, []);
}
});
11 years ago
}
11 years ago
}(exports));