'use strict'; var assert = require('assert'); var async = require('async'); var nconf = require('nconf'); var request = require('request'); var db = require('./mocks/databasemock'); var User = require('../src/user'); var Topics = require('../src/topics'); var Categories = require('../src/categories'); var Meta = require('../src/meta'); var Password = require('../src/password'); var groups = require('../src/groups'); var helpers = require('./helpers'); describe('User', function () { var userData; var testUid; var testCid; before(function (done) { groups.resetCache(); Categories.create({ name: 'Test Category', description: 'A test', order: 1 }, function (err, categoryObj) { if (err) { return done(err); } testCid = categoryObj.cid; done(); }); }); beforeEach(function () { userData = { username: 'John Smith', fullname: 'John Smith McNamara', password: 'swordfish', email: 'john@example.com', callback: undefined }; }); describe('.create(), when created', function () { it('should be created properly', function (done) { User.create({username: userData.username, password: userData.password, email: userData.email}, function (error,userId) { assert.equal(error, null, 'was created with error'); assert.ok(userId); testUid = userId; done(); }); }); it('should have a valid email, if using an email', function (done) { User.create({username: userData.username, password: userData.password, email: 'fakeMail'},function (err) { assert(err); assert.equal(err.message, '[[error:invalid-email]]'); done(); }); }); }); describe('.isModerator()', function () { it('should return false', function (done) { User.isModerator(testUid, testCid, function (err, isModerator) { assert.equal(err, null); assert.equal(isModerator, false); done(); }); }); it('should return two false results', function (done) { User.isModerator([testUid, testUid], testCid, function (err, isModerator) { assert.equal(err, null); assert.equal(isModerator[0], false); assert.equal(isModerator[1], false); done(); }); }); it('should return two false results', function (done) { User.isModerator(testUid, [testCid, testCid], function (err, isModerator) { assert.equal(err, null); assert.equal(isModerator[0], false); assert.equal(isModerator[1], false); done(); }); }); }); describe('.isReadyToPost()', function () { it('should error when a user makes two posts in quick succession', function (done) { Meta.config = Meta.config || {}; Meta.config.postDelay = '10'; async.series([ async.apply(Topics.post, { uid: testUid, title: 'Topic 1', content: 'lorem ipsum', cid: testCid }), async.apply(Topics.post, { uid: testUid, title: 'Topic 2', content: 'lorem ipsum', cid: testCid }) ], function (err) { assert(err); done(); }); }); it('should allow a post if the last post time is > 10 seconds', function (done) { User.setUserField(testUid, 'lastposttime', +new Date() - (11 * 1000), function () { Topics.post({ uid: testUid, title: 'Topic 3', content: 'lorem ipsum', cid: testCid }, function (err) { assert.ifError(err); done(); }); }); }); it('should error when a new user posts if the last post time is 10 < 30 seconds', function (done) { Meta.config.newbiePostDelay = 30; Meta.config.newbiePostDelayThreshold = 3; User.setUserField(testUid, 'lastposttime', +new Date() - (20 * 1000), function () { Topics.post({ uid: testUid, title: 'Topic 4', content: 'lorem ipsum', cid: testCid }, function (err) { assert(err); done(); }); }); }); it('should not error if a non-newbie user posts if the last post time is 10 < 30 seconds', function (done) { User.setUserFields(testUid, { lastposttime: +new Date() - (20 * 1000), reputation: 10 }, function () { Topics.post({ uid: testUid, title: 'Topic 5', content: 'lorem ipsum', cid: testCid }, function (err) { assert.ifError(err); done(); }); }); }); }); describe('.search()', function () { var socketUser = require('../src/socket.io/user'); it('should return an object containing an array of matching users', function (done) { User.search({query: 'john'}, function (err, searchData) { assert.ifError(err); assert.equal(Array.isArray(searchData.users) && searchData.users.length > 0, true); assert.equal(searchData.users[0].username, 'John Smith'); done(); }); }); it('should search user', function (done) { socketUser.search({uid: testUid}, {query: 'john'}, function (err, searchData) { assert.ifError(err); assert.equal(searchData.users[0].username, 'John Smith'); done(); }); }); it('should error for guest', function (done) { Meta.config.allowGuestUserSearching = 0; socketUser.search({uid: 0}, {query: 'john'}, function (err) { assert.equal(err.message, '[[error:not-logged-in]]'); Meta.config.allowGuestUserSearching = 1; done(); }); }); it('should error with invalid data', function (done) { socketUser.search({uid: testUid}, null, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); }); describe('.delete()', function () { var uid; before(function (done) { User.create({username: 'usertodelete', password: '123456', email: 'delete@me.com'}, function (err, newUid) { assert.ifError(err); uid = newUid; done(); }); }); it('should delete a user account', function (done) { User.delete(1, uid, function (err) { assert.ifError(err); User.existsBySlug('usertodelete', function (err, exists) { assert.ifError(err); assert.equal(exists, false); done(); }); }); }); }); describe('passwordReset', function () { var uid, code; before(function (done) { User.create({username: 'resetuser', password: '123456', email: 'reset@me.com'}, function (err, newUid) { assert.ifError(err); uid = newUid; done(); }); }); it('.generate() should generate a new reset code', function (done) { User.reset.generate(uid, function (err, _code) { assert.ifError(err); assert(_code); code = _code; done(); }); }); it('.validate() should ensure that this new code is valid', function (done) { User.reset.validate(code, function (err, valid) { assert.ifError(err); assert.strictEqual(valid, true); done(); }); }); it('.validate() should correctly identify an invalid code', function (done) { User.reset.validate(code + 'abcdef', function (err, valid) { assert.ifError(err); assert.strictEqual(valid, false); done(); }); }); it('.send() should create a new reset code and reset password', function (done) { User.reset.send('reset@me.com', function (err, code) { if (err) { console.log(err); } done(); }); }); it('.commit() should update the user\'s password', function (done) { User.reset.commit(code, 'newpassword', function (err) { assert.ifError(err); db.getObjectField('user:' + uid, 'password', function (err, newPassword) { assert.ifError(err); Password.compare('newpassword', newPassword, function (err, match) { assert.ifError(err); assert(match); done(); }); }); }); }); }); describe('hash methods', function () { it('should return uid from email', function (done) { User.getUidByEmail('john@example.com', function (err, uid) { assert.ifError(err); assert.equal(parseInt(uid, 10), parseInt(testUid, 10)); done(); }); }); it('should return uid from username', function (done) { User.getUidByUsername('John Smith', function (err, uid) { assert.ifError(err); assert.equal(parseInt(uid, 10), parseInt(testUid, 10)); done(); }); }); it('should return uid from userslug', function (done) { User.getUidByUserslug('john-smith', function (err, uid) { assert.ifError(err); assert.equal(parseInt(uid, 10), parseInt(testUid, 10)); done(); }); }); }); describe('not logged in', function () { var jar; var io; before(function (done) { helpers.initSocketIO(function (err, _jar, _io) { assert.ifError(err); jar = _jar; io = _io; done(); }); }); it('should return error if not logged in', function (done) { io.emit('user.updateProfile', {}, function (err) { assert.equal(err.message, '[[error:invalid-uid]]'); done(); }); }); }); describe('profile methods', function () { var uid; var jar; var io; before(function (done) { User.create({username: 'updateprofile', email: 'update@me.com', password: '123456'}, function (err, newUid) { assert.ifError(err); uid = newUid; helpers.loginUser('updateprofile', '123456', function (err, _jar, _io) { assert.ifError(err); jar = _jar; io = _io; done(); }); }); }); it('should return error if data is invalid', function (done) { io.emit('user.updateProfile', null, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should return error if data is missing uid', function (done) { io.emit('user.updateProfile', {username: 'bip', email: 'bop'}, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should update a user\'s profile', function (done) { var data = { uid: uid, username: 'updatedUserName', email: 'updatedEmail@me.com', fullname: 'updatedFullname', website: 'http://nodebb.org', location: 'izmir', groupTitle: 'testGroup', birthday: '01/01/1980', signature: 'nodebb is good' }; io.emit('user.updateProfile', data, function (err, result) { assert.ifError(err); assert.equal(result.username, 'updatedUserName'); assert.equal(result.userslug, 'updatedusername'); assert.equal(result.email, 'updatedEmail@me.com'); db.getObject('user:' + uid, function (err, userData) { assert.ifError(err); Object.keys(data).forEach(function (key) { assert.equal(data[key], userData[key]); }); done(); }); }); }); it('should change a user\'s password', function (done) { this.timeout(20000); io.emit('user.changePassword', {uid: uid, newPassword: '654321', currentPassword: '123456'}, function (err) { assert.ifError(err); User.isPasswordCorrect(uid, '654321', function (err, correct) { assert.ifError(err); assert(correct); done(); }); }); }); it('should change username', function (done) { io.emit('user.changeUsernameEmail', {uid: uid, username: 'updatedAgain', password: '654321'}, function (err) { assert.ifError(err); db.getObjectField('user:' + uid, 'username', function (err, username) { assert.ifError(err); assert.equal(username, 'updatedAgain'); done(); }); }); }); it('should change email', function (done) { io.emit('user.changeUsernameEmail', {uid: uid, email: 'updatedAgain@me.com', password: '654321'}, function (err) { assert.ifError(err); db.getObjectField('user:' + uid, 'email', function (err, email) { assert.ifError(err); assert.equal(email, 'updatedAgain@me.com'); done(); }); }); }); it('should update cover image', function (done) { var imageData = ''; var position = '50.0301% 19.2464%'; io.emit('user.updateCover', {uid: uid, imageData: imageData, position: position}, function (err, result) { assert.ifError(err); assert(result.url); db.getObjectFields('user:' + uid, ['cover:url', 'cover:position'], function (err, data) { assert.ifError(err); assert.equal(data['cover:url'], result.url); assert.equal(data['cover:position'], position); done(); }); }); }); it('should remove cover image', function (done) { io.emit('user.removeCover', {uid: uid}, function (err) { assert.ifError(err); db.getObjectField('user:' + uid, 'cover:url', function (err, url) { assert.ifError(err); assert.equal(url, null); done(); }); }); }); it('should set user status', function (done) { io.emit('user.setStatus', 'away', function (err, data) { assert.ifError(err); assert.equal(data.uid, uid); assert.equal(data.status, 'away'); done(); }); }); it('should fail for invalid status', function (done) { io.emit('user.setStatus', '12345', function (err) { assert.equal(err.message, '[[error:invalid-user-status]]'); done(); }); }); it('should get user status', function (done) { io.emit('user.checkStatus', uid, function (err, status) { assert.ifError(err); assert.equal(status, 'away'); done(); }); }); it('should change user picture', function (done) { io.emit('user.changePicture', {type: 'default', uid: uid}, function (err) { assert.ifError(err); User.getUserField(uid, 'picture', function (err, picture) { assert.ifError(err); assert.equal(picture, ''); done(); }); }); }); it('should upload profile picture', function (done) { var path = require('path'); var picture = { path: path.join(nconf.get('base_dir'), 'public', 'logo.png'), size: 7189, name: 'logo.png' }; User.uploadPicture(uid, picture, function (err, uploadedPicture) { assert.ifError(err); assert.equal(uploadedPicture.url, '/uploads/profile/' + uid + '-profileimg.png'); assert.equal(uploadedPicture.path, path.join(nconf.get('base_dir'), 'public', 'uploads', 'profile', uid + '-profileimg.png')); done(); }); }); it('should get profile pictures', function (done) { io.emit('user.getProfilePictures', {uid: uid}, function (err, data) { assert.ifError(err); assert(data); assert(Array.isArray(data)); assert.equal(data[0].type, 'uploaded'); assert.equal(data[0].text, '[[user:uploaded_picture]]'); done(); }); }); it('should remove uploaded picture', function (done) { io.emit('user.removeUploadedPicture', {uid: uid}, function (err) { assert.ifError(err); User.getUserField(uid, 'uploadedpicture', function (err, uploadedpicture) { assert.ifError(err); assert.equal(uploadedpicture, ''); done(); }); }); }); it('should load profile page', function (done) { request(nconf.get('url') + '/api/user/updatedagain', {jar: jar, json: true}, function (err, res, body) { assert.ifError(err); assert.equal(res.statusCode, 200); assert(body); done(); }); }); it('should load settings page', function (done) { request(nconf.get('url') + '/api/user/updatedagain/settings', {jar: jar, json: true}, function (err, res, body) { assert.ifError(err); assert.equal(res.statusCode, 200); assert(body.settings); assert(body.languages); assert(body.homePageRoutes); done(); }); }); it('should load edit page', function (done) { request(nconf.get('url') + '/api/user/updatedagain/edit', {jar: jar, json: true}, function (err, res, body) { assert.ifError(err); assert.equal(res.statusCode, 200); assert(body); done(); }); }); it('should load edit/email page', function (done) { request(nconf.get('url') + '/api/user/updatedagain/edit/email', {jar: jar, json: true}, function (err, res, body) { assert.ifError(err); assert.equal(res.statusCode, 200); assert(body); done(); }); }); it('should load user\'s groups page', function (done) { groups.create({ name: 'Test', description: 'Foobar!' }, function (err) { assert.ifError(err); groups.join('Test', uid, function (err) { assert.ifError(err); request(nconf.get('url') + '/api/user/updatedagain/groups', {jar: jar, json: true}, function (err, res, body) { assert.ifError(err); assert.equal(res.statusCode, 200); assert(Array.isArray(body.groups)); assert.equal(body.groups[0].name, 'Test'); done(); }); }); }); }); }); describe('.getModerationHistory', function () { it('should return the correct ban reason', function (done) { async.series([ function (next) { User.ban(testUid, 0, '', function (err) { assert.ifError(err); next(err); }); }, function (next) { User.getModerationHistory(testUid, function (err, data) { assert.ifError(err); assert.equal(data.bans.length, 1, 'one ban'); assert.equal(data.bans[0].reason, '[[user:info.banned-no-reason]]', 'no ban reason'); next(err); }); } ], function (err) { assert.ifError(err); User.unban(testUid, function (err) { assert.ifError(err); done(); }); }); }); }); describe('digests', function () { var uid; before(function (done) { User.create({username: 'digestuser', email: 'test@example.com'}, function (err, _uid) { assert.ifError(err); uid = _uid; done(); }); }); it('should send digests', function (done) { User.updateDigestSetting(uid, 'day', function (err) { assert.ifError(err); User.digest.execute('day', function (err) { assert.ifError(err); done(); }); }); }); }); describe('socket methods', function () { var socketUser = require('../src/socket.io/user'); it('should fail with invalid data', function (done) { socketUser.exists({uid: testUid}, null, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should return true if user/group exists', function (done) { socketUser.exists({uid: testUid}, {username: 'registered-users'}, function (err, exists) { assert.ifError(err); assert(exists); done(); }); }); it('should return true if user/group exists', function (done) { socketUser.exists({uid: testUid}, {username: 'John Smith'}, function (err, exists) { assert.ifError(err); assert(exists); done(); }); }); it('should return false if user/group does not exists', function (done) { socketUser.exists({uid: testUid}, {username: 'doesnot exist'}, function (err, exists) { assert.ifError(err); assert(!exists); done(); }); }); it('should delete user', function (done) { User.create({username: 'tobedeleted'}, function (err, _uid) { assert.ifError(err); socketUser.deleteAccount({uid: _uid}, {}, function (err) { assert.ifError(err); socketUser.exists({uid: testUid}, {username: 'doesnot exist'}, function (err, exists) { assert.ifError(err); assert(!exists); done(); }); }); }); }); }); describe('approval queue', function () { var socketAdmin = require('../src/socket.io/admin'); var oldRegistrationType; var adminUid; before(function (done) { oldRegistrationType = Meta.config.registrationType; Meta.config.registrationType = 'admin-approval'; User.create({username: 'admin', password: '123456'}, function (err, uid) { assert.ifError(err); adminUid = uid; groups.join('administrators', uid, done); }); }); after(function (done) { Meta.config.registrationType = oldRegistrationType; done(); }); it('should add user to approval queue', function (done) { helpers.registerUser({ username: 'rejectme', password: '123456', email: 'reject@me.com' }, function (err) { assert.ifError(err); helpers.loginUser('admin', '123456', function (err, jar) { assert.ifError(err); request(nconf.get('url') + '/api/admin/manage/registration', {jar: jar, json: true}, function (err, res, body) { assert.ifError(err); assert.equal(body.users[0].username, 'rejectme'); assert.equal(body.users[0].email, 'reject@me.com'); done(); }); }); }); }); it('should reject user registration', function (done) { socketAdmin.user.rejectRegistration({uid: adminUid}, {username: 'rejectme'}, function (err) { assert.ifError(err); User.getRegistrationQueue(0, -1, function (err, users) { assert.ifError(err); assert.equal(users.length, 0); done(); }); }); }); it('should accept user registration', function (done) { helpers.registerUser({ username: 'acceptme', password: '123456', email: 'accept@me.com' }, function (err) { assert.ifError(err); socketAdmin.user.acceptRegistration({uid: adminUid}, {username: 'acceptme'}, function (err, uid) { assert.ifError(err); User.exists(uid, function (err, exists) { assert.ifError(err); assert(exists); User.getRegistrationQueue(0, -1, function (err, users) { assert.ifError(err); assert.equal(users.length, 0); done(); }); }); }); }); }); }); after(function (done) { db.emptydb(done); }); });