'use strict'; const assert = require('assert'); const async = require('async'); const fs = require('fs'); const path = require('path'); const nconf = require('nconf'); const validator = require('validator'); const request = require('request'); const requestAsync = require('request-promise-native'); const jwt = require('jsonwebtoken'); const db = require('./mocks/databasemock'); const User = require('../src/user'); const Topics = require('../src/topics'); const Categories = require('../src/categories'); const Posts = require('../src/posts'); const groups = require('../src/groups'); const messaging = require('../src/messaging'); const helpers = require('./helpers'); const meta = require('../src/meta'); const file = require('../src/file'); const socketUser = require('../src/socket.io/user'); const apiUser = require('../src/api/users'); const utils = require('../src/utils'); const privileges = require('../src/privileges'); describe('User', () => { let userData; let testUid; let testCid; const plugins = require('../src/plugins'); async function dummyEmailerHook(data) { // pretend to handle sending emails } before((done) => { // Attach an emailer hook so related requests do not error plugins.hooks.register('emailer-test', { hook: 'filter:email.send', method: dummyEmailerHook, }); Categories.create({ name: 'Test Category', description: 'A test', order: 1, }, (err, categoryObj) => { if (err) { return done(err); } testCid = categoryObj.cid; done(); }); }); after(() => { plugins.hooks.unregister('emailer-test', 'filter:email.send'); }); beforeEach(() => { userData = { username: 'John Smith', fullname: 'John Smith McNamara', password: 'swordfish', email: 'john@example.com', callback: undefined, }; }); const goodImage = ''; describe('.create(), when created', () => { it('should be created properly', async () => { testUid = await User.create({ username: userData.username, password: userData.password }); assert.ok(testUid); await User.setUserField(testUid, 'email', userData.email); await User.email.confirmByUid(testUid); }); it('should be created properly', async () => { const email = '

test

@gmail.com'; const uid = await User.create({ username: 'weirdemail', email: email }); const data = await User.getUserData(uid); const validationPending = await User.email.isValidationPending(uid, email); assert.strictEqual(validationPending, true); assert.equal(data.email, ''); assert.strictEqual(data.profileviews, 0); assert.strictEqual(data.reputation, 0); assert.strictEqual(data.postcount, 0); assert.strictEqual(data.topiccount, 0); assert.strictEqual(data.lastposttime, 0); assert.strictEqual(data.banned, false); }); it('should have a valid email, if using an email', (done) => { User.create({ username: userData.username, password: userData.password, email: 'fakeMail' }, (err) => { assert(err); assert.equal(err.message, '[[error:invalid-email]]'); done(); }); }); it('should error with invalid password', (done) => { User.create({ username: 'test', password: '1' }, (err) => { assert.equal(err.message, '[[reset_password:password_too_short]]'); done(); }); }); it('should error with invalid password', (done) => { User.create({ username: 'test', password: {} }, (err) => { assert.equal(err.message, '[[error:invalid-password]]'); done(); }); }); it('should error with a too long password', (done) => { let toolong = ''; for (let i = 0; i < 5000; i++) { toolong += 'a'; } User.create({ username: 'test', password: toolong }, (err) => { assert.equal(err.message, '[[error:password-too-long]]'); done(); }); }); it('should error if username is already taken or rename user', async () => { let err; async function tryCreate(data) { try { return await User.create(data); } catch (_err) { err = _err; } } const [uid1, uid2] = await Promise.all([ tryCreate({ username: 'dupe1' }), tryCreate({ username: 'dupe1' }), ]); if (err) { assert.strictEqual(err.message, '[[error:username-taken]]'); } else { const userData = await User.getUsersFields([uid1, uid2], ['username']); const userNames = userData.map(u => u.username); // make sure only 1 dupe1 is created assert.equal(userNames.filter(username => username === 'dupe1').length, 1); assert.equal(userNames.filter(username => username === 'dupe1 0').length, 1); } }); it('should error if email is already taken', async () => { let err; async function tryCreate(data) { try { return await User.create(data); } catch (_err) { err = _err; } } await Promise.all([ tryCreate({ username: 'notdupe1', email: 'dupe@dupe.com' }), tryCreate({ username: 'notdupe2', email: 'dupe@dupe.com' }), ]); assert.strictEqual(err.message, '[[error:email-taken]]'); }); }); describe('.uniqueUsername()', () => { it('should deal with collisions', (done) => { const users = []; for (let i = 0; i < 10; i += 1) { users.push({ username: 'Jane Doe', email: `jane.doe${i}@example.com`, }); } async.series([ function (next) { async.eachSeries(users, (user, next) => { User.create(user, next); }, next); }, function (next) { User.uniqueUsername({ username: 'Jane Doe', userslug: 'jane-doe', }, (err, username) => { assert.ifError(err); assert.strictEqual(username, 'Jane Doe 9'); next(); }); }, ], done); }); }); describe('.isModerator()', () => { it('should return false', (done) => { User.isModerator(testUid, testCid, (err, isModerator) => { assert.equal(err, null); assert.equal(isModerator, false); done(); }); }); it('should return two false results', (done) => { User.isModerator([testUid, testUid], testCid, (err, isModerator) => { assert.equal(err, null); assert.equal(isModerator[0], false); assert.equal(isModerator[1], false); done(); }); }); it('should return two false results', (done) => { User.isModerator(testUid, [testCid, testCid], (err, isModerator) => { assert.equal(err, null); assert.equal(isModerator[0], false); assert.equal(isModerator[1], false); done(); }); }); }); describe('.getModeratorUids()', () => { before((done) => { groups.join('cid:1:privileges:moderate', 1, done); }); it('should retrieve all users with moderator bit in category privilege', (done) => { User.getModeratorUids((err, uids) => { assert.ifError(err); assert.strictEqual(1, uids.length); assert.strictEqual(1, parseInt(uids[0], 10)); done(); }); }); after((done) => { groups.leave('cid:1:privileges:moderate', 1, done); }); }); describe('.getModeratorUids()', () => { before((done) => { async.series([ async.apply(groups.create, { name: 'testGroup' }), async.apply(groups.join, 'cid:1:privileges:groups:moderate', 'testGroup'), async.apply(groups.join, 'testGroup', 1), ], done); }); it('should retrieve all users with moderator bit in category privilege', (done) => { User.getModeratorUids((err, uids) => { assert.ifError(err); assert.strictEqual(1, uids.length); assert.strictEqual(1, parseInt(uids[0], 10)); done(); }); }); after((done) => { async.series([ async.apply(groups.leave, 'cid:1:privileges:groups:moderate', 'testGroup'), async.apply(groups.destroy, 'testGroup'), ], done); }); }); describe('.isReadyToPost()', () => { it('should error when a user makes two posts in quick succession', (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, }), ], (err) => { assert(err); done(); }); }); it('should allow a post if the last post time is > 10 seconds', (done) => { User.setUserField(testUid, 'lastposttime', +new Date() - (11 * 1000), () => { Topics.post({ uid: testUid, title: 'Topic 3', content: 'lorem ipsum', cid: testCid, }, (err) => { assert.ifError(err); done(); }); }); }); it('should error when a new user posts if the last post time is 10 < 30 seconds', (done) => { meta.config.newbiePostDelay = 30; meta.config.newbiePostDelayThreshold = 3; User.setUserField(testUid, 'lastposttime', +new Date() - (20 * 1000), () => { Topics.post({ uid: testUid, title: 'Topic 4', content: 'lorem ipsum', cid: testCid, }, (err) => { assert(err); done(); }); }); }); it('should not error if a non-newbie user posts if the last post time is 10 < 30 seconds', (done) => { User.setUserFields(testUid, { lastposttime: +new Date() - (20 * 1000), reputation: 10, }, () => { Topics.post({ uid: testUid, title: 'Topic 5', content: 'lorem ipsum', cid: testCid, }, (err) => { assert.ifError(err); done(); }); }); }); it('should only post 1 topic out of 10', async () => { await User.create({ username: 'flooder', password: '123456' }); const { jar } = await helpers.loginUser('flooder', '123456'); const titles = new Array(10).fill('topic title'); const res = await Promise.allSettled(titles.map(async (title) => { const { body } = await helpers.request('post', '/api/v3/topics', { form: { cid: testCid, title: title, content: 'the content', }, jar: jar, json: true, }); return body.status; })); const failed = res.filter(res => res.value.code === 'bad-request'); const success = res.filter(res => res.value.code === 'ok'); assert.strictEqual(failed.length, 9); assert.strictEqual(success.length, 1); }); }); describe('.search()', () => { let adminUid; let uid; before(async () => { adminUid = await User.create({ username: 'noteadmin' }); await groups.join('administrators', adminUid); }); it('should return an object containing an array of matching users', (done) => { User.search({ query: 'john' }, (err, searchData) => { assert.ifError(err); uid = searchData.users[0].uid; assert.equal(Array.isArray(searchData.users) && searchData.users.length > 0, true); assert.equal(searchData.users[0].username, 'John Smith'); done(); }); }); it('should search user', async () => { const searchData = await apiUser.search({ uid: testUid }, { query: 'john' }); assert.equal(searchData.users[0].username, 'John Smith'); }); it('should error for guest', async () => { try { await apiUser.search({ uid: 0 }, { query: 'john' }); assert(false); } catch (err) { assert.equal(err.message, '[[error:no-privileges]]'); } }); it('should error with invalid data', async () => { try { await apiUser.search({ uid: testUid }, null); assert(false); } catch (err) { assert.equal(err.message, '[[error:invalid-data]]'); } }); it('should error for unprivileged user', async () => { try { await apiUser.search({ uid: testUid }, { searchBy: 'ip', query: '123' }); assert(false); } catch (err) { assert.equal(err.message, '[[error:no-privileges]]'); } }); it('should error for unprivileged user', async () => { try { await apiUser.search({ uid: testUid }, { filters: ['banned'], query: '123' }); assert(false); } catch (err) { assert.equal(err.message, '[[error:no-privileges]]'); } }); it('should error for unprivileged user', async () => { try { await apiUser.search({ uid: testUid }, { filters: ['flagged'], query: '123' }); assert(false); } catch (err) { assert.equal(err.message, '[[error:no-privileges]]'); } }); it('should search users by ip', async () => { const uid = await User.create({ username: 'ipsearch' }); await db.sortedSetAdd('ip:1.1.1.1:uid', [1, 1], [testUid, uid]); const data = await apiUser.search({ uid: adminUid }, { query: '1.1.1.1', searchBy: 'ip' }); assert(Array.isArray(data.users)); assert.equal(data.users.length, 2); }); it('should search users by uid', async () => { const data = await apiUser.search({ uid: testUid }, { query: uid, searchBy: 'uid' }); assert(Array.isArray(data.users)); assert.equal(data.users[0].uid, uid); }); it('should search users by fullname', async () => { const uid = await User.create({ username: 'fullnamesearch1', fullname: 'Mr. Fullname' }); const data = await apiUser.search({ uid: adminUid }, { query: 'mr', searchBy: 'fullname' }); assert(Array.isArray(data.users)); assert.equal(data.users.length, 1); assert.equal(uid, data.users[0].uid); }); it('should search users by fullname', async () => { const uid = await User.create({ username: 'fullnamesearch2', fullname: 'Baris:Usakli' }); const data = await apiUser.search({ uid: adminUid }, { query: 'baris:', searchBy: 'fullname' }); assert(Array.isArray(data.users)); assert.equal(data.users.length, 1); assert.equal(uid, data.users[0].uid); }); it('should return empty array if query is empty', async () => { const data = await apiUser.search({ uid: testUid }, { query: '' }); assert.equal(data.users.length, 0); }); it('should filter users', async () => { const uid = await User.create({ username: 'ipsearch_filter' }); await User.bans.ban(uid, 0, ''); await User.setUserFields(uid, { flags: 10 }); const data = await apiUser.search({ uid: adminUid }, { query: 'ipsearch', filters: ['online', 'banned', 'flagged'], }); assert.equal(data.users[0].username, 'ipsearch_filter'); }); it('should sort results by username', (done) => { async.waterfall([ function (next) { User.create({ username: 'brian' }, next); }, function (uid, next) { User.create({ username: 'baris' }, next); }, function (uid, next) { User.create({ username: 'bzari' }, next); }, function (uid, next) { User.search({ uid: testUid, query: 'b', sortBy: 'username', paginate: false, }, next); }, ], (err, data) => { assert.ifError(err); assert.equal(data.users[0].username, 'baris'); assert.equal(data.users[1].username, 'brian'); assert.equal(data.users[2].username, 'bzari'); done(); }); }); }); describe('.delete()', () => { let uid; before((done) => { User.create({ username: 'usertodelete', password: '123456', email: 'delete@me.com' }, (err, newUid) => { assert.ifError(err); uid = newUid; done(); }); }); it('should delete a user account', (done) => { User.delete(1, uid, (err) => { assert.ifError(err); User.existsBySlug('usertodelete', (err, exists) => { assert.ifError(err); assert.equal(exists, false); done(); }); }); }); it('should not re-add user to users:postcount if post is purged after user account deletion', async () => { const uid = await User.create({ username: 'olduserwithposts' }); assert(await db.isSortedSetMember('users:postcount', uid)); const result = await Topics.post({ uid: uid, title: 'old user topic', content: 'old user topic post content', cid: testCid, }); assert.equal(await db.sortedSetScore('users:postcount', uid), 1); await User.deleteAccount(uid); assert(!await db.isSortedSetMember('users:postcount', uid)); await Posts.purge(result.postData.pid, 1); assert(!await db.isSortedSetMember('users:postcount', uid)); }); it('should not re-add user to users:reputation if post is upvoted after user account deletion', async () => { const uid = await User.create({ username: 'olduserwithpostsupvote' }); assert(await db.isSortedSetMember('users:reputation', uid)); const result = await Topics.post({ uid: uid, title: 'old user topic', content: 'old user topic post content', cid: testCid, }); assert.equal(await db.sortedSetScore('users:reputation', uid), 0); await User.deleteAccount(uid); assert(!await db.isSortedSetMember('users:reputation', uid)); await Posts.upvote(result.postData.pid, 1); assert(!await db.isSortedSetMember('users:reputation', uid)); }); it('should delete user even if they started a chat', async () => { const socketModules = require('../src/socket.io/modules'); const uid1 = await User.create({ username: 'chatuserdelete1' }); const uid2 = await User.create({ username: 'chatuserdelete2' }); const roomId = await messaging.newRoom(uid1, { uids: [uid2] }); await messaging.addMessage({ uid: uid1, content: 'hello', roomId, }); await messaging.leaveRoom([uid2], roomId); await User.delete(1, uid1); assert.strictEqual(await User.exists(uid1), false); }); }); describe('hash methods', () => { it('should return uid from email', (done) => { User.getUidByEmail('john@example.com', (err, uid) => { assert.ifError(err); assert.equal(parseInt(uid, 10), parseInt(testUid, 10)); done(); }); }); it('should return uid from username', (done) => { User.getUidByUsername('John Smith', (err, uid) => { assert.ifError(err); assert.equal(parseInt(uid, 10), parseInt(testUid, 10)); done(); }); }); it('should return uid from userslug', (done) => { User.getUidByUserslug('john-smith', (err, uid) => { assert.ifError(err); assert.equal(parseInt(uid, 10), parseInt(testUid, 10)); done(); }); }); it('should get user data even if one uid is NaN', (done) => { User.getUsersData([NaN, testUid], (err, data) => { assert.ifError(err); assert(data[0]); assert.equal(data[0].username, '[[global:guest]]'); assert(data[1]); assert.equal(data[1].username, userData.username); done(); }); }); it('should not return private user data', (done) => { User.setUserFields(testUid, { fb_token: '123123123', another_secret: 'abcde', postcount: '123', }, (err) => { assert.ifError(err); User.getUserData(testUid, (err, userData) => { assert.ifError(err); assert(!userData.hasOwnProperty('fb_token')); assert(!userData.hasOwnProperty('another_secret')); assert(!userData.hasOwnProperty('password')); assert(!userData.hasOwnProperty('rss_token')); assert.strictEqual(userData.postcount, 123); assert.strictEqual(userData.uid, testUid); done(); }); }); }); it('should not return password even if explicitly requested', (done) => { User.getUserFields(testUid, ['password'], (err, payload) => { assert.ifError(err); assert(!payload.hasOwnProperty('password')); done(); }); }); it('should not modify the fields array passed in', async () => { const fields = ['username', 'email']; await User.getUserFields(testUid, fields); assert.deepStrictEqual(fields, ['username', 'email']); }); it('should return an icon text and valid background if username and picture is explicitly requested', async () => { const payload = await User.getUserFields(testUid, ['username', 'picture']); const validBackgrounds = await User.getIconBackgrounds(testUid); assert.strictEqual(payload['icon:text'], userData.username.slice(0, 1).toUpperCase()); assert(payload['icon:bgColor']); assert(validBackgrounds.includes(payload['icon:bgColor'])); }); it('should return a valid background, even if an invalid background colour is set', async () => { await User.setUserField(testUid, 'icon:bgColor', 'teal'); const payload = await User.getUserFields(testUid, ['username', 'picture']); const validBackgrounds = await User.getIconBackgrounds(testUid); assert(payload['icon:bgColor']); assert(validBackgrounds.includes(payload['icon:bgColor'])); }); it('should return private data if field is whitelisted', (done) => { function filterMethod(data, callback) { data.whitelist.push('another_secret'); callback(null, data); } plugins.hooks.register('test-plugin', { hook: 'filter:user.whitelistFields', method: filterMethod }); User.getUserData(testUid, (err, userData) => { assert.ifError(err); assert(!userData.hasOwnProperty('fb_token')); assert.equal(userData.another_secret, 'abcde'); plugins.hooks.unregister('test-plugin', 'filter:user.whitelistFields', filterMethod); done(); }); }); it('should return 0 as uid if username is falsy', (done) => { User.getUidByUsername('', (err, uid) => { assert.ifError(err); assert.strictEqual(uid, 0); done(); }); }); it('should get username by userslug', (done) => { User.getUsernameByUserslug('john-smith', (err, username) => { assert.ifError(err); assert.strictEqual('John Smith', username); done(); }); }); it('should get uids by emails', (done) => { User.getUidsByEmails(['john@example.com'], (err, uids) => { assert.ifError(err); assert.equal(uids[0], testUid); done(); }); }); it('should not get groupTitle for guests', (done) => { User.getUserData(0, (err, userData) => { assert.ifError(err); assert.strictEqual(userData.groupTitle, ''); assert.deepStrictEqual(userData.groupTitleArray, []); done(); }); }); it('should load guest data', (done) => { User.getUsersData([1, 0], (err, data) => { assert.ifError(err); assert.strictEqual(data[1].username, '[[global:guest]]'); assert.strictEqual(data[1].userslug, ''); assert.strictEqual(data[1].uid, 0); done(); }); }); }); describe('profile methods', () => { let uid; let jar; let csrf_token; before(async () => { const newUid = await User.create({ username: 'updateprofile', email: 'update@me.com', password: '123456' }); uid = newUid; await User.setUserField(uid, 'email', 'update@me.com'); await User.email.confirmByUid(uid); ({ jar, csrf_token } = await helpers.loginUser('updateprofile', '123456')); }); it('should return error if not logged in', async () => { try { await apiUser.update({ uid: 0 }, { uid: 1 }); assert(false); } catch (err) { assert.equal(err.message, '[[error:invalid-uid]]'); } }); it('should return error if data is invalid', async () => { try { await apiUser.update({ uid: uid }, null); assert(false); } catch (err) { assert.equal(err.message, '[[error:invalid-data]]'); } }); it('should return error if data is missing uid', async () => { try { await apiUser.update({ uid: uid }, { username: 'bip', email: 'bop' }); assert(false); } catch (err) { assert.equal(err.message, '[[error:invalid-data]]'); } }); describe('.updateProfile()', () => { let uid; it('should update a user\'s profile', async () => { uid = await User.create({ username: 'justforupdate', email: 'just@for.updated', password: '123456' }); await User.setUserField(uid, 'email', 'just@for.updated'); await User.email.confirmByUid(uid); const 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', password: '123456', }; const result = await apiUser.update({ uid: uid }, { ...data, password: '123456', invalid: 'field' }); assert.equal(result.username, 'updatedUserName'); assert.equal(result.userslug, 'updatedusername'); assert.equal(result.location, 'izmir'); const userData = await db.getObject(`user:${uid}`); Object.keys(data).forEach((key) => { if (key === 'email') { assert.strictEqual(userData.email, 'just@for.updated'); // email remains the same until confirmed } else if (key !== 'password') { assert.equal(data[key], userData[key]); } else { assert(userData[key].startsWith('$2a$')); } }); // updateProfile only saves valid fields assert.strictEqual(userData.invalid, undefined); }); it('should also generate an email confirmation code for the changed email', async () => { const confirmSent = await User.email.isValidationPending(uid, 'updatedemail@me.com'); assert.strictEqual(confirmSent, true); }); }); it('should change a user\'s password', async () => { const uid = await User.create({ username: 'changepassword', password: '123456' }); await apiUser.changePassword({ uid: uid }, { uid: uid, newPassword: '654321', currentPassword: '123456' }); const correct = await User.isPasswordCorrect(uid, '654321', '127.0.0.1'); assert(correct); }); it('should not let user change another user\'s password', async () => { const regularUserUid = await User.create({ username: 'regularuserpwdchange', password: 'regularuser1234' }); const uid = await User.create({ username: 'changeadminpwd1', password: '123456' }); try { await apiUser.changePassword({ uid: uid }, { uid: regularUserUid, newPassword: '654321', currentPassword: '123456' }); assert(false); } catch (err) { assert.equal(err.message, '[[user:change_password_error_privileges]]'); } }); it('should not let user change admin\'s password', async () => { const adminUid = await User.create({ username: 'adminpwdchange', password: 'admin1234' }); await groups.join('administrators', adminUid); const uid = await User.create({ username: 'changeadminpwd2', password: '123456' }); try { await apiUser.changePassword({ uid: uid }, { uid: adminUid, newPassword: '654321', currentPassword: '123456' }); assert(false); } catch (err) { assert.equal(err.message, '[[user:change_password_error_privileges]]'); } }); it('should let admin change another users password', async () => { const adminUid = await User.create({ username: 'adminpwdchange2', password: 'admin1234' }); await groups.join('administrators', adminUid); const uid = await User.create({ username: 'forgotmypassword', password: '123456' }); await apiUser.changePassword({ uid: adminUid }, { uid: uid, newPassword: '654321' }); const correct = await User.isPasswordCorrect(uid, '654321', '127.0.0.1'); assert(correct); }); it('should not let admin change their password if current password is incorrect', async () => { const adminUid = await User.create({ username: 'adminforgotpwd', password: 'admin1234' }); await groups.join('administrators', adminUid); try { await apiUser.changePassword({ uid: adminUid }, { uid: adminUid, newPassword: '654321', currentPassword: 'wrongpwd' }); assert(false); } catch (err) { assert.equal(err.message, '[[user:change_password_error_wrong_current]]'); } }); it('should change username', async () => { await apiUser.update({ uid: uid }, { uid: uid, username: 'updatedAgain', password: '123456' }); const username = await db.getObjectField(`user:${uid}`, 'username'); assert.equal(username, 'updatedAgain'); }); it('should not let setting an empty username', async () => { await apiUser.update({ uid: uid }, { uid: uid, username: '', password: '123456' }); const username = await db.getObjectField(`user:${uid}`, 'username'); assert.strictEqual(username, 'updatedAgain'); }); it('should let updating profile if current username is above max length and it is not being changed', async () => { const maxLength = meta.config.maximumUsernameLength + 1; const longName = new Array(maxLength).fill('a').join(''); const uid = await User.create({ username: longName }); await apiUser.update({ uid: uid }, { uid: uid, username: longName, email: 'verylong@name.com' }); const userData = await db.getObject(`user:${uid}`); const awaitingValidation = await User.email.isValidationPending(uid, 'verylong@name.com'); assert.strictEqual(userData.username, longName); assert.strictEqual(awaitingValidation, true); }); it('should not update a user\'s username if it did not change', async () => { await apiUser.update({ uid: uid }, { uid: uid, username: 'updatedAgain', password: '123456' }); const data = await db.getSortedSetRevRange(`user:${uid}:usernames`, 0, -1); assert.equal(data.length, 2); assert(data[0].startsWith('updatedAgain')); }); it('should not update a user\'s username if a password is not supplied', async () => { try { await apiUser.update({ uid: uid }, { uid: uid, username: 'updatedAgain', password: '' }); assert(false); } catch (err) { assert.strictEqual(err.message, '[[error:invalid-password]]'); } }); it('should properly change username and clean up old sorted sets', async () => { const uid = await User.create({ username: 'DennyO', password: '123456' }); let usernames = await db.getSortedSetRevRangeWithScores('username:uid', 0, -1); usernames = usernames.filter(d => d.score === uid); assert.deepStrictEqual(usernames, [{ value: 'DennyO', score: uid }]); await apiUser.update({ uid: uid }, { uid: uid, username: 'DennyO\'s', password: '123456' }); usernames = await db.getSortedSetRevRangeWithScores('username:uid', 0, -1); usernames = usernames.filter(d => d.score === uid); assert.deepStrictEqual(usernames, [{ value: 'DennyO\'s', score: uid }]); await apiUser.update({ uid: uid }, { uid: uid, username: 'Denny O', password: '123456' }); usernames = await db.getSortedSetRevRangeWithScores('username:uid', 0, -1); usernames = usernames.filter(d => d.score === uid); assert.deepStrictEqual(usernames, [{ value: 'Denny O', score: uid }]); }); it('should send validation email', async () => { const uid = await User.create({ username: 'pooremailupdate', email: 'poor@update.me', password: '123456' }); await User.email.expireValidation(uid); await apiUser.update({ uid: uid }, { uid: uid, email: 'updatedAgain@me.com', password: '123456' }); assert.strictEqual(await User.email.isValidationPending(uid, 'updatedAgain@me.com'.toLowerCase()), true); }); it('should update cover image', (done) => { const position = '50.0301% 19.2464%'; socketUser.updateCover({ uid: uid }, { uid: uid, imageData: goodImage, position: position }, (err, result) => { assert.ifError(err); assert(result.url); db.getObjectFields(`user:${uid}`, ['cover:url', 'cover:position'], (err, data) => { assert.ifError(err); assert.equal(data['cover:url'], result.url); assert.equal(data['cover:position'], position); done(); }); }); }); it('should remove cover image', async () => { const coverPath = await User.getLocalCoverPath(uid); await socketUser.removeCover({ uid: uid }, { uid: uid }); const coverUrlNow = await db.getObjectField(`user:${uid}`, 'cover:url'); assert.strictEqual(coverUrlNow, null); assert.strictEqual(fs.existsSync(coverPath), false); }); it('should set user status', (done) => { socketUser.setStatus({ uid: uid }, 'away', (err, data) => { assert.ifError(err); assert.equal(data.uid, uid); assert.equal(data.status, 'away'); done(); }); }); it('should fail for invalid status', (done) => { socketUser.setStatus({ uid: uid }, '12345', (err) => { assert.equal(err.message, '[[error:invalid-user-status]]'); done(); }); }); it('should get user status', (done) => { socketUser.checkStatus({ uid: uid }, uid, (err, status) => { assert.ifError(err); assert.equal(status, 'away'); done(); }); }); it('should change user picture', async () => { await apiUser.changePicture({ uid: uid }, { type: 'default', uid: uid }); const picture = await User.getUserField(uid, 'picture'); assert.equal(picture, ''); }); it('should let you set an external image', async () => { const token = await helpers.getCsrfToken(jar); const body = await requestAsync(`${nconf.get('url')}/api/v3/users/${uid}/picture`, { jar, method: 'put', json: true, headers: { 'x-csrf-token': token, }, body: { type: 'external', url: 'https://example.org/picture.jpg', }, }); assert(body && body.status && body.response); assert.strictEqual(body.status.code, 'ok'); const picture = await User.getUserField(uid, 'picture'); assert.strictEqual(picture, validator.escape('https://example.org/picture.jpg')); }); it('should fail to change user picture with invalid data', async () => { try { await apiUser.changePicture({ uid: uid }, null); assert(false); } catch (err) { assert.equal(err.message, '[[error:invalid-data]]'); } }); it('should fail to change user picture with invalid uid', async () => { try { await apiUser.changePicture({ uid: 0 }, { uid: 1 }); assert(false); } catch (err) { assert.equal(err.message, '[[error:no-privileges]]'); } }); it('should set user picture to uploaded', async () => { await User.setUserField(uid, 'uploadedpicture', '/test'); await apiUser.changePicture({ uid: uid }, { type: 'uploaded', uid: uid }); const picture = await User.getUserField(uid, 'picture'); assert.equal(picture, `${nconf.get('relative_path')}/test`); }); it('should return error if profile image uploads disabled', (done) => { meta.config.allowProfileImageUploads = 0; const picture = { path: path.join(nconf.get('base_dir'), 'test/files/test_copy.png'), size: 7189, name: 'test.png', type: 'image/png', }; User.uploadCroppedPicture({ callerUid: uid, uid: uid, file: picture, }, (err) => { assert.equal(err.message, '[[error:profile-image-uploads-disabled]]'); meta.config.allowProfileImageUploads = 1; done(); }); }); it('should return error if profile image has no mime type', (done) => { User.uploadCroppedPicture({ callerUid: uid, uid: uid, imageData: '', }, (err) => { assert.equal(err.message, '[[error:invalid-image]]'); done(); }); }); describe('user.uploadCroppedPicture', () => { const badImage = 'data:audio/mp3;base64,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'; it('should upload cropped profile picture', async () => { const result = await socketUser.uploadCroppedPicture({ uid: uid }, { uid: uid, imageData: goodImage }); assert(result.url); const data = await db.getObjectFields(`user:${uid}`, ['uploadedpicture', 'picture']); assert.strictEqual(result.url, data.uploadedpicture); assert.strictEqual(result.url, data.picture); }); it('should upload cropped profile picture in chunks', async () => { const socketUploads = require('../src/socket.io/uploads'); const socketData = { uid, method: 'user.uploadCroppedPicture', size: goodImage.length, progress: 0, }; const chunkSize = 1000; let result; do { const chunk = goodImage.slice(socketData.progress, socketData.progress + chunkSize); socketData.progress += chunk.length; // eslint-disable-next-line result = await socketUploads.upload({ uid: uid }, { chunk: chunk, params: socketData, }); } while (socketData.progress < socketData.size); assert(result.url); const data = await db.getObjectFields(`user:${uid}`, ['uploadedpicture', 'picture']); assert.strictEqual(result.url, data.uploadedpicture); assert.strictEqual(result.url, data.picture); }); it('should error if both file and imageData are missing', (done) => { User.uploadCroppedPicture({}, (err) => { assert.equal('[[error:invalid-data]]', err.message); done(); }); }); it('should error if file size is too big', (done) => { const temp = meta.config.maximumProfileImageSize; meta.config.maximumProfileImageSize = 1; User.uploadCroppedPicture({ callerUid: uid, uid: 1, imageData: goodImage, }, (err) => { assert.equal('[[error:file-too-big, 1]]', err.message); // Restore old value meta.config.maximumProfileImageSize = temp; done(); }); }); it('should not allow image data with bad MIME type to be passed in', (done) => { User.uploadCroppedPicture({ callerUid: uid, uid: 1, imageData: badImage, }, (err) => { assert.equal('[[error:invalid-image]]', err.message); done(); }); }); it('should get profile pictures', (done) => { socketUser.getProfilePictures({ uid: uid }, { uid: uid }, (err, data) => { assert.ifError(err); assert(data); assert(Array.isArray(data)); assert.equal(data[0].type, 'default'); assert.equal(data[0].username, '[[user:default_picture]]'); assert.equal(data[1].type, 'uploaded'); assert.equal(data[1].username, '[[user:uploaded_picture]]'); done(); }); }); it('should get default profile avatar', (done) => { assert.strictEqual(User.getDefaultAvatar(), ''); meta.config.defaultAvatar = 'https://path/to/default/avatar'; assert.strictEqual(User.getDefaultAvatar(), meta.config.defaultAvatar); meta.config.defaultAvatar = '/path/to/default/avatar'; assert.strictEqual(User.getDefaultAvatar(), nconf.get('relative_path') + meta.config.defaultAvatar); meta.config.defaultAvatar = ''; done(); }); it('should fail to get profile pictures with invalid data', (done) => { socketUser.getProfilePictures({ uid: uid }, null, (err) => { assert.equal(err.message, '[[error:invalid-data]]'); socketUser.getProfilePictures({ uid: uid }, { uid: null }, (err) => { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); }); it('should remove uploaded picture', async () => { const avatarPath = await User.getLocalAvatarPath(uid); assert.notStrictEqual(avatarPath, false); await socketUser.removeUploadedPicture({ uid: uid }, { uid: uid }); const uploadedPicture = await User.getUserField(uid, 'uploadedpicture'); assert.strictEqual(uploadedPicture, ''); assert.strictEqual(fs.existsSync(avatarPath), false); }); it('should fail to remove uploaded picture with invalid-data', (done) => { socketUser.removeUploadedPicture({ uid: uid }, null, (err) => { assert.equal(err.message, '[[error:invalid-data]]'); socketUser.removeUploadedPicture({ uid: uid }, { }, (err) => { assert.equal(err.message, '[[error:invalid-data]]'); socketUser.removeUploadedPicture({ uid: null }, { }, (err) => { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); }); }); }); it('should load profile page', (done) => { request(`${nconf.get('url')}/api/user/updatedagain`, { jar: jar, json: true }, (err, res, body) => { assert.ifError(err); assert.equal(res.statusCode, 200); assert(body); done(); }); }); it('should load settings page', (done) => { request(`${nconf.get('url')}/api/user/updatedagain/settings`, { jar: jar, json: true }, (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', (done) => { request(`${nconf.get('url')}/api/user/updatedagain/edit`, { jar: jar, json: true }, (err, res, body) => { assert.ifError(err); assert.equal(res.statusCode, 200); assert(body); done(); }); }); it('should load edit/email page', async () => { const res = await requestAsync(`${nconf.get('url')}/api/user/updatedagain/edit/email`, { jar: jar, json: true, resolveWithFullResponse: true }); assert.strictEqual(res.statusCode, 200); assert(res.body); // Accessing this page will mark the user's account as needing an updated email, below code undo's. await requestAsync({ uri: `${nconf.get('url')}/register/abort`, jar, method: 'POST', simple: false, headers: { 'x-csrf-token': csrf_token, }, }); }); it('should load user\'s groups page', async () => { await groups.create({ name: 'Test', description: 'Foobar!', }); await groups.join('Test', uid); const body = await requestAsync(`${nconf.get('url')}/api/user/updatedagain/groups`, { jar: jar, json: true }); assert(Array.isArray(body.groups)); assert.equal(body.groups[0].name, 'Test'); }); }); describe('user info', () => { let testUserUid; let verifiedTestUserUid; before(async () => { // Might be the first user thus a verified one if this test part is ran alone verifiedTestUserUid = await User.create({ username: 'bannedUser', password: '123456', email: 'banneduser@example.com' }); await User.setUserField(verifiedTestUserUid, 'email:confirmed', 1); testUserUid = await User.create({ username: 'bannedUser2', password: '123456', email: 'banneduser2@example.com' }); }); it('should return error if there is no ban reason', (done) => { User.getLatestBanInfo(123, (err) => { assert.equal(err.message, 'no-ban-info'); done(); }); }); it('should get history from set', async () => { const now = Date.now(); await db.sortedSetAdd(`user:${testUserUid}:usernames`, now, `derp:${now}`); const data = await User.getHistory(`user:${testUserUid}:usernames`); assert.equal(data[0].value, 'derp'); assert.equal(data[0].timestamp, now); }); it('should return the correct ban reason', (done) => { async.series([ function (next) { User.bans.ban(testUserUid, 0, '', (err) => { assert.ifError(err); next(err); }); }, function (next) { User.getModerationHistory(testUserUid, (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); }); }, ], (err) => { assert.ifError(err); User.bans.unban(testUserUid, (err) => { assert.ifError(err); done(); }); }); }); it('should ban user permanently', (done) => { User.bans.ban(testUserUid, (err) => { assert.ifError(err); User.bans.isBanned(testUserUid, (err, isBanned) => { assert.ifError(err); assert.equal(isBanned, true); User.bans.unban(testUserUid, done); }); }); }); it('should ban user temporarily', (done) => { User.bans.ban(testUserUid, Date.now() + 2000, (err) => { assert.ifError(err); User.bans.isBanned(testUserUid, (err, isBanned) => { assert.ifError(err); assert.equal(isBanned, true); setTimeout(() => { User.bans.isBanned(testUserUid, (err, isBanned) => { assert.ifError(err); assert.equal(isBanned, false); User.bans.unban(testUserUid, done); }); }, 3000); }); }); }); it('should error if until is NaN', (done) => { User.bans.ban(testUserUid, 'asd', (err) => { assert.equal(err.message, '[[error:ban-expiry-missing]]'); done(); }); }); it('should be member of "banned-users" system group only after a ban', async () => { await User.bans.ban(testUserUid); const systemGroups = groups.systemGroups.filter(group => group !== groups.BANNED_USERS); const isMember = await groups.isMember(testUserUid, groups.BANNED_USERS); const isMemberOfAny = await groups.isMemberOfAny(testUserUid, systemGroups); assert.strictEqual(isMember, true); assert.strictEqual(isMemberOfAny, false); }); it('should restore system group memberships after an unban (for an unverified user)', async () => { await User.bans.unban(testUserUid); const isMemberOfGroups = await groups.isMemberOfGroups(testUserUid, groups.systemGroups); const membership = new Map(groups.systemGroups.map((item, index) => [item, isMemberOfGroups[index]])); assert.strictEqual(membership.get('registered-users'), true); assert.strictEqual(membership.get('verified-users'), false); assert.strictEqual(membership.get('unverified-users'), true); assert.strictEqual(membership.get(groups.BANNED_USERS), false); // administrators cannot be banned assert.strictEqual(membership.get('administrators'), false); // This will not restored assert.strictEqual(membership.get('Global Moderators'), false); }); it('should restore system group memberships after an unban (for a verified user)', async () => { await User.bans.ban(verifiedTestUserUid); await User.bans.unban(verifiedTestUserUid); const isMemberOfGroups = await groups.isMemberOfGroups(verifiedTestUserUid, groups.systemGroups); const membership = new Map(groups.systemGroups.map((item, index) => [item, isMemberOfGroups[index]])); assert.strictEqual(membership.get('verified-users'), true); assert.strictEqual(membership.get('unverified-users'), false); }); it('should be able to post in category for banned users', async () => { const { cid } = await Categories.create({ name: 'Test Category', description: 'A test', order: 1, }); const testUid = await User.create({ username: userData.username }); await User.bans.ban(testUid); let _err; try { await Topics.post({ title: 'banned topic', content: 'tttttttttttt', cid: cid, uid: testUid }); } catch (err) { _err = err; } assert.strictEqual(_err && _err.message, '[[error:no-privileges]]'); await Promise.all([ privileges.categories.give(['groups:topics:create', 'groups:topics:reply'], cid, 'banned-users'), privileges.categories.rescind(['groups:topics:create', 'groups:topics:reply'], cid, 'registered-users'), ]); const result = await Topics.post({ title: 'banned topic', content: 'tttttttttttt', cid: cid, uid: testUid }); assert(result); assert.strictEqual(result.topicData.title, 'banned topic'); }); }); describe('Digest.getSubscribers', () => { const uidIndex = {}; before((done) => { const testUsers = ['daysub', 'offsub', 'nullsub', 'weeksub']; async.each(testUsers, (username, next) => { async.waterfall([ async.apply(User.create, { username: username, email: `${username}@example.com` }), function (uid, next) { if (username === 'nullsub') { return setImmediate(next); } uidIndex[username] = uid; const sub = username.slice(0, -3); async.parallel([ async.apply(User.updateDigestSetting, uid, sub), async.apply(User.setSetting, uid, 'dailyDigestFreq', sub), ], next); }, ], next); }, done); }); it('should accurately build digest list given ACP default "null" (not set)', (done) => { User.digest.getSubscribers('day', (err, subs) => { assert.ifError(err); assert.strictEqual(subs.length, 1); done(); }); }); it('should accurately build digest list given ACP default "day"', (done) => { async.series([ async.apply(meta.configs.set, 'dailyDigestFreq', 'day'), function (next) { User.digest.getSubscribers('day', (err, subs) => { assert.ifError(err); assert.strictEqual(subs.includes(uidIndex.daysub.toString()), true); // daysub does get emailed assert.strictEqual(subs.includes(uidIndex.weeksub.toString()), false); // weeksub does not get emailed assert.strictEqual(subs.includes(uidIndex.offsub.toString()), false); // offsub doesn't get emailed next(); }); }, ], done); }); it('should accurately build digest list given ACP default "week"', (done) => { async.series([ async.apply(meta.configs.set, 'dailyDigestFreq', 'week'), function (next) { User.digest.getSubscribers('week', (err, subs) => { assert.ifError(err); assert.strictEqual(subs.includes(uidIndex.weeksub.toString()), true); // weeksub gets emailed assert.strictEqual(subs.includes(uidIndex.daysub.toString()), false); // daysub gets emailed assert.strictEqual(subs.includes(uidIndex.offsub.toString()), false); // offsub does not get emailed next(); }); }, ], done); }); it('should accurately build digest list given ACP default "off"', (done) => { async.series([ async.apply(meta.configs.set, 'dailyDigestFreq', 'off'), function (next) { User.digest.getSubscribers('day', (err, subs) => { assert.ifError(err); assert.strictEqual(subs.length, 1); next(); }); }, ], done); }); }); describe('digests', () => { let uid; before((done) => { async.waterfall([ function (next) { User.create({ username: 'digestuser', email: 'test@example.com' }, next); }, function (_uid, next) { uid = _uid; User.updateDigestSetting(uid, 'day', next); }, function (next) { User.setSetting(uid, 'dailyDigestFreq', 'day', next); }, function (next) { User.setSetting(uid, 'notificationType_test', 'notificationemail', next); }, ], done); }); it('should send digests', async () => { const oldValue = meta.config.includeUnverifiedEmails; meta.config.includeUnverifiedEmails = true; const uid = await User.create({ username: 'digest' }); await User.setUserField(uid, 'email', 'email@test.com'); await User.email.confirmByUid(uid); await User.digest.execute({ interval: 'day', subscribers: [uid], }); meta.config.includeUnverifiedEmails = oldValue; }); it('should return 0', async () => { const sent = await User.digest.send({ subscribers: [] }); assert.strictEqual(sent, 0); }); it('should get users with single uid', async () => { const res = await User.digest.getUsersInterval(1); assert.strictEqual(res, false); }); it('should not send digests', async () => { const oldValue = meta.config.disableEmailSubsriptions; meta.config.disableEmailSubsriptions = 1; const res = await User.digest.execute({}); assert.strictEqual(res, false); meta.config.disableEmailSubsriptions = oldValue; }); it('should not send digests', async () => { await User.digest.execute({ interval: 'month' }); }); it('should get delivery times', async () => { const data = await User.digest.getDeliveryTimes(0, -1); const users = data.users.filter(u => u.username === 'digestuser'); assert.strictEqual(users[0].setting, 'day'); }); describe('unsubscribe via POST', () => { it('should unsubscribe from digest if one-click unsubscribe is POSTed', (done) => { const token = jwt.sign({ template: 'digest', uid: uid, }, nconf.get('secret')); request({ method: 'post', url: `${nconf.get('url')}/email/unsubscribe/${token}`, }, (err, res) => { assert.ifError(err); assert.strictEqual(res.statusCode, 200); db.getObjectField(`user:${uid}:settings`, 'dailyDigestFreq', (err, value) => { assert.ifError(err); assert.strictEqual(value, 'off'); done(); }); }); }); it('should unsubscribe from notifications if one-click unsubscribe is POSTed', (done) => { const token = jwt.sign({ template: 'notification', type: 'test', uid: uid, }, nconf.get('secret')); request({ method: 'post', url: `${nconf.get('url')}/email/unsubscribe/${token}`, }, (err, res) => { assert.ifError(err); assert.strictEqual(res.statusCode, 200); db.getObjectField(`user:${uid}:settings`, 'notificationType_test', (err, value) => { assert.ifError(err); assert.strictEqual(value, 'notification'); done(); }); }); }); it('should return errors on missing template in token', (done) => { const token = jwt.sign({ uid: uid, }, nconf.get('secret')); request({ method: 'post', url: `${nconf.get('url')}/email/unsubscribe/${token}`, }, (err, res) => { assert.ifError(err); assert.strictEqual(res.statusCode, 404); done(); }); }); it('should return errors on wrong template in token', (done) => { const token = jwt.sign({ template: 'user', uid: uid, }, nconf.get('secret')); request({ method: 'post', url: `${nconf.get('url')}/email/unsubscribe/${token}`, }, (err, res) => { assert.ifError(err); assert.strictEqual(res.statusCode, 404); done(); }); }); it('should return errors on missing token', (done) => { request({ method: 'post', url: `${nconf.get('url')}/email/unsubscribe/`, }, (err, res) => { assert.ifError(err); assert.strictEqual(res.statusCode, 404); done(); }); }); it('should return errors on token signed with wrong secret (verify-failure)', (done) => { const token = jwt.sign({ template: 'notification', type: 'test', uid: uid, }, `${nconf.get('secret')}aababacaba`); request({ method: 'post', url: `${nconf.get('url')}/email/unsubscribe/${token}`, }, (err, res) => { assert.ifError(err); assert.strictEqual(res.statusCode, 403); done(); }); }); }); }); describe('socket methods', () => { const socketUser = require('../src/socket.io/user'); let delUid; it('should fail with invalid data', (done) => { meta.userOrGroupExists(null, (err) => { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should return true if user/group exists', (done) => { meta.userOrGroupExists('registered-users', (err, exists) => { assert.ifError(err); assert(exists); done(); }); }); it('should return true if user/group exists', (done) => { meta.userOrGroupExists('John Smith', (err, exists) => { assert.ifError(err); assert(exists); done(); }); }); it('should return false if user/group does not exists', (done) => { meta.userOrGroupExists('doesnot exist', (err, exists) => { assert.ifError(err); assert(!exists); done(); }); }); it('should delete user', async () => { delUid = await User.create({ username: 'willbedeleted' }); // Upload some avatars and covers before deleting meta.config['profile:keepAllUserImages'] = 1; let result = await socketUser.uploadCroppedPicture({ uid: delUid }, { uid: delUid, imageData: goodImage }); assert(result.url); result = await socketUser.uploadCroppedPicture({ uid: delUid }, { uid: delUid, imageData: goodImage }); assert(result.url); const position = '50.0301% 19.2464%'; result = await socketUser.updateCover({ uid: delUid }, { uid: delUid, imageData: goodImage, position: position }); assert(result.url); result = await socketUser.updateCover({ uid: delUid }, { uid: delUid, imageData: goodImage, position: position }); assert(result.url); meta.config['profile:keepAllUserImages'] = 0; await apiUser.deleteAccount({ uid: delUid }, { uid: delUid }); const exists = await meta.userOrGroupExists('willbedeleted'); assert(!exists); }); it('should clean profile images after account deletion', () => { const allProfileFiles = fs.readdirSync(path.join(nconf.get('upload_path'), 'profile')); const deletedUserImages = allProfileFiles.filter( f => f.startsWith(`${delUid}-profilecover`) || f.startsWith(`${delUid}-profileavatar`) ); assert.strictEqual(deletedUserImages.length, 0); }); it('should fail to delete user with wrong password', async () => { const uid = await User.create({ username: 'willbedeletedpwd', password: '123456' }); try { await apiUser.deleteAccount({ uid: uid }, { uid: uid, password: '654321' }); assert(false); } catch (err) { assert.strictEqual(err.message, '[[error:invalid-password]]'); } }); it('should delete user with correct password', async () => { const uid = await User.create({ username: 'willbedeletedcorrectpwd', password: '123456' }); await apiUser.deleteAccount({ uid: uid }, { uid: uid, password: '123456' }); const exists = await User.exists(uid); assert(!exists); }); it('should fail to delete user if account deletion is not allowed', async () => { const oldValue = meta.config.allowAccountDelete; meta.config.allowAccountDelete = 0; const uid = await User.create({ username: 'tobedeleted' }); try { await apiUser.deleteAccount({ uid: uid }, { uid: uid }); assert(false); } catch (err) { assert.strictEqual(err.message, '[[error:account-deletion-disabled]]'); } meta.config.allowAccountDelete = oldValue; }); it('should send reset email', (done) => { socketUser.reset.send({ uid: 0 }, 'john@example.com', (err) => { assert.ifError(err); done(); }); }); it('should return invalid-data error', (done) => { socketUser.reset.send({ uid: 0 }, null, (err) => { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should not error', (done) => { socketUser.reset.send({ uid: 0 }, 'doestnot@exist.com', (err) => { assert.ifError(err); done(); }); }); it('should commit reset', (done) => { db.getObject('reset:uid', (err, data) => { assert.ifError(err); const code = Object.keys(data).find(code => parseInt(data[code], 10) === parseInt(testUid, 10)); socketUser.reset.commit({ uid: 0 }, { code: code, password: 'pwdchange' }, (err) => { assert.ifError(err); done(); }); }); }); it('should save user settings', async () => { const data = { uid: testUid, settings: { bootswatchSkin: 'default', homePageRoute: 'none', homePageCustom: '', openOutgoingLinksInNewTab: 0, scrollToMyPost: 1, userLang: 'en-GB', usePagination: 1, topicsPerPage: '10', postsPerPage: '5', showemail: 1, showfullname: 1, restrictChat: 0, followTopicsOnCreate: 1, followTopicsOnReply: 1, }, }; await apiUser.updateSettings({ uid: testUid }, data); const userSettings = await User.getSettings(testUid); assert.strictEqual(userSettings.usePagination, true); }); it('should properly escape homePageRoute', async () => { const data = { uid: testUid, settings: { bootswatchSkin: 'default', homePageRoute: 'category/6/testing-ground', homePageCustom: '', openOutgoingLinksInNewTab: 0, scrollToMyPost: 1, userLang: 'en-GB', usePagination: 1, topicsPerPage: '10', postsPerPage: '5', showemail: 1, showfullname: 1, restrictChat: 0, followTopicsOnCreate: 1, followTopicsOnReply: 1, }, }; await apiUser.updateSettings({ uid: testUid }, data); const userSettings = await User.getSettings(testUid); assert.strictEqual(userSettings.homePageRoute, 'category/6/testing-ground'); }); it('should error if language is invalid', async () => { const data = { uid: testUid, settings: { userLang: '', topicsPerPage: '10', postsPerPage: '5', }, }; try { await apiUser.updateSettings({ uid: testUid }, data); assert(false); } catch (err) { assert.equal(err.message, '[[error:invalid-language]]'); } }); it('should set moderation note', (done) => { let adminUid; async.waterfall([ function (next) { User.create({ username: 'noteadmin' }, next); }, function (_adminUid, next) { adminUid = _adminUid; groups.join('administrators', adminUid, next); }, function (next) { socketUser.setModerationNote({ uid: adminUid }, { uid: testUid, note: 'this is a test user' }, next); }, function (next) { setTimeout(next, 50); }, function (next) { socketUser.setModerationNote({ uid: adminUid }, { uid: testUid, note: ' { assert.ifError(err); assert.equal(notes[0].note, '<svg/onload=alert(document.location);//'); assert.equal(notes[0].uid, adminUid); assert.equal(notes[1].note, 'this is a test user'); assert(notes[0].timestamp); done(); }); }); it('should get unread count 0 for guest', async () => { const count = await socketUser.getUnreadCount({ uid: 0 }); assert.strictEqual(count, 0); }); it('should get unread count for user', async () => { const count = await socketUser.getUnreadCount({ uid: testUid }); assert.strictEqual(count, 4); }); it('should get unread chat count 0 for guest', async () => { const count = await socketUser.getUnreadChatCount({ uid: 0 }); assert.strictEqual(count, 0); }); it('should get unread chat count for user', async () => { const count = await socketUser.getUnreadChatCount({ uid: testUid }); assert.strictEqual(count, 0); }); it('should get unread counts 0 for guest', async () => { const counts = await socketUser.getUnreadCounts({ uid: 0 }); assert.deepStrictEqual(counts, {}); }); it('should get unread counts for user', async () => { const counts = await socketUser.getUnreadCounts({ uid: testUid }); assert.deepStrictEqual(counts, { unreadChatCount: 0, unreadCounts: { '': 4, new: 4, unreplied: 4, watched: 0, }, unreadNewTopicCount: 4, unreadNotificationCount: 0, unreadTopicCount: 4, unreadUnrepliedTopicCount: 4, unreadWatchedTopicCount: 0, }); }); it('should get user data by uid', async () => { const userData = await socketUser.getUserByUID({ uid: testUid }, testUid); assert.strictEqual(userData.uid, testUid); }); it('should get user data by username', async () => { const userData = await socketUser.getUserByUsername({ uid: testUid }, 'John Smith'); assert.strictEqual(userData.uid, testUid); }); it('should get user data by email', async () => { const userData = await socketUser.getUserByEmail({ uid: testUid }, 'john@example.com'); assert.strictEqual(userData.uid, testUid); }); it('should check/consent gdpr status', async () => { const consent = await socketUser.gdpr.check({ uid: testUid }, { uid: testUid }); assert(!consent); await socketUser.gdpr.consent({ uid: testUid }); const consentAfter = await socketUser.gdpr.check({ uid: testUid }, { uid: testUid }); assert(consentAfter); }); }); describe('approval queue', () => { let oldRegistrationApprovalType; let adminUid; before((done) => { oldRegistrationApprovalType = meta.config.registrationApprovalType; meta.config.registrationApprovalType = 'admin-approval'; User.create({ username: 'admin', password: '123456' }, (err, uid) => { assert.ifError(err); adminUid = uid; groups.join('administrators', uid, done); }); }); after((done) => { meta.config.registrationApprovalType = oldRegistrationApprovalType; done(); }); it('should add user to approval queue', async () => { await helpers.registerUser({ username: 'rejectme', password: '123456', 'password-confirm': '123456', email: '