var crypto = require('crypto') var MongoDB = require('mongodb').Db; var Server = require('mongodb').Server; var moment = require('moment'); var dbPort = 27017; var dbHost = 'localhost'; var dbName = 'node-login'; /* establish the database connection */ var db = new MongoDB(dbName, new Server(dbHost, dbPort, {auto_reconnect: true}), {w: 1}); db.open(function(e, d){ if (e) { console.log(e); } else{ console.log('connected to database :: ' + dbName); } }); var accounts = db.collection('accounts'); /* login validation methods */ exports.autoLogin = function(user, pass, callback) { accounts.findOne({user:user}, function(e, o) { if (o){ o.pass == pass ? callback(o) : callback(null); } else{ callback(null); } }); } exports.manualLogin = function(user, pass, callback) { accounts.findOne({user:user}, function(e, o) { if (o == null){ callback('user-not-found'); } else{ validatePassword(pass, o.pass, function(err, res) { if (res){ callback(null, o); } else{ callback('invalid-password'); } }); } }); } /* record insertion, update & deletion methods */ exports.addNewAccount = function(newData, callback) { accounts.findOne({user:newData.user}, function(e, o) { if (o){ callback('username-taken'); } else{ accounts.findOne({email:newData.email}, function(e, o) { if (o){ callback('email-taken'); } else{ saltAndHash(newData.pass, function(hash){ newData.pass = hash; // append date stamp when record was created // newData.date = moment().format('MMMM Do YYYY, h:mm:ss a'); accounts.insert(newData, {safe: true}, callback); }); } }); } }); } exports.updateAccount = function(newData, callback) { accounts.findOne({user:newData.user}, function(e, o){ o.name = newData.name; o.email = newData.email; o.country = newData.country; if (newData.pass == ''){ accounts.save(o, {safe: true}, callback); } else{ saltAndHash(newData.pass, function(hash){ o.pass = hash; accounts.save(o, {safe: true}, callback); }); } }); } exports.updatePassword = function(email, newPass, callback) { accounts.findOne({email:email}, function(e, o){ if (e){ callback(e, null); } else{ saltAndHash(newPass, function(hash){ o.pass = hash; accounts.save(o, {safe: true}, callback); }); } }); } /* account lookup methods */ exports.deleteAccount = function(id, callback) { accounts.remove({_id: getObjectId(id)}, callback); } exports.getAccountByEmail = function(email, callback) { accounts.findOne({email:email}, function(e, o){ callback(o); }); } exports.validateResetLink = function(email, passHash, callback) { accounts.find({ $and: [{email:email, pass:passHash}] }, function(e, o){ callback(o ? 'ok' : null); }); } exports.getAllRecords = function(callback) { accounts.find().toArray( function(e, res) { if (e) callback(e) else callback(null, res) }); }; exports.delAllRecords = function(callback) { accounts.remove({}, callback); // reset accounts collection for testing // } /* private encryption & validation methods */ var generateSalt = function() { var set = '0123456789abcdefghijklmnopqurstuvwxyzABCDEFGHIJKLMNOPQURSTUVWXYZ'; var salt = ''; for (var i = 0; i < 10; i++) { var p = Math.floor(Math.random() * set.length); salt += set[p]; } return salt; } var md5 = function(str) { return crypto.createHash('md5').update(str).digest('hex'); } var saltAndHash = function(pass, callback) { var salt = generateSalt(); callback(salt + md5(pass + salt)); } var validatePassword = function(plainPass, hashedPass, callback) { var salt = hashedPass.substr(0, 10); var validHash = salt + md5(plainPass + salt); callback(null, hashedPass === validHash); } /* auxiliary methods */ var getObjectId = function(id) { return accounts.db.bson_serializer.ObjectID.createFromHexString(id) } var findById = function(id, callback) { accounts.findOne({_id: getObjectId(id)}, function(e, res) { if (e) callback(e) else callback(null, res) }); }; var findByMultipleFields = function(a, callback) { // this takes an array of name/val pairs to search against {fieldName : 'value'} // accounts.find( { $or : a } ).toArray( function(e, results) { if (e) callback(e) else callback(null, results) }); }