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.
2441 lines
90 KiB
JavaScript
2441 lines
90 KiB
JavaScript
'use strict';
|
|
|
|
var assert = require('assert');
|
|
var async = require('async');
|
|
var path = require('path');
|
|
var nconf = require('nconf');
|
|
var request = require('request');
|
|
var jwt = require('jsonwebtoken');
|
|
|
|
var db = require('./mocks/databasemock');
|
|
var User = require('../src/user');
|
|
var Topics = require('../src/topics');
|
|
var Categories = require('../src/categories');
|
|
var Posts = require('../src/posts');
|
|
var Password = require('../src/password');
|
|
var groups = require('../src/groups');
|
|
var helpers = require('./helpers');
|
|
var meta = require('../src/meta');
|
|
var plugins = require('../src/plugins');
|
|
var socketUser = require('../src/socket.io/user');
|
|
|
|
describe('User', function () {
|
|
var userData;
|
|
var testUid;
|
|
var testCid;
|
|
|
|
before(function (done) {
|
|
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: '[email protected]',
|
|
callback: undefined,
|
|
};
|
|
});
|
|
|
|
|
|
describe('.create(), when created', function () {
|
|
it('should be created properly', async function () {
|
|
testUid = await User.create({ username: userData.username, password: userData.password, email: userData.email });
|
|
assert.ok(testUid);
|
|
});
|
|
|
|
it('should be created properly', async function () {
|
|
const uid = await User.create({ username: 'weirdemail', email: '<h1>test</h1>@gmail.com' });
|
|
const data = await User.getUserData(uid);
|
|
assert.equal(data.email, '<h1>test</h1>@gmail.com');
|
|
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, 0);
|
|
});
|
|
|
|
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();
|
|
});
|
|
});
|
|
|
|
it('should error with invalid password', function (done) {
|
|
User.create({ username: 'test', password: '1' }, function (err) {
|
|
assert.equal(err.message, '[[reset_password:password_too_short]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error with invalid password', function (done) {
|
|
User.create({ username: 'test', password: {} }, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-password]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error with a too long password', function (done) {
|
|
var toolong = '';
|
|
for (var i = 0; i < 5000; i++) {
|
|
toolong += 'a';
|
|
}
|
|
User.create({ username: 'test', password: toolong }, function (err) {
|
|
assert.equal(err.message, '[[error:password-too-long]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error if username is already taken or rename user', async function () {
|
|
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 function () {
|
|
let err;
|
|
async function tryCreate(data) {
|
|
try {
|
|
return await User.create(data);
|
|
} catch (_err) {
|
|
err = _err;
|
|
}
|
|
}
|
|
|
|
await Promise.all([
|
|
tryCreate({ username: 'notdupe1', email: '[email protected]' }),
|
|
tryCreate({ username: 'notdupe2', email: '[email protected]' }),
|
|
]);
|
|
assert.strictEqual(err.message, '[[error:email-taken]]');
|
|
});
|
|
});
|
|
|
|
describe('.uniqueUsername()', function () {
|
|
it('should deal with collisions', function (done) {
|
|
var users = [];
|
|
for (var i = 0; i < 10; i += 1) {
|
|
users.push({
|
|
username: 'Jane Doe',
|
|
email: 'jane.doe' + i + '@example.com',
|
|
});
|
|
}
|
|
|
|
async.series([
|
|
function (next) {
|
|
async.eachSeries(users, function (user, next) {
|
|
User.create(user, next);
|
|
}, next);
|
|
},
|
|
function (next) {
|
|
User.uniqueUsername({
|
|
username: 'Jane Doe',
|
|
userslug: 'jane-doe',
|
|
}, function (err, username) {
|
|
assert.ifError(err);
|
|
|
|
assert.strictEqual(username, 'Jane Doe 9');
|
|
next();
|
|
});
|
|
},
|
|
], 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('.getModeratorUids()', function () {
|
|
before(function (done) {
|
|
groups.join('cid:1:privileges:moderate', 1, done);
|
|
});
|
|
|
|
it('should retrieve all users with moderator bit in category privilege', function (done) {
|
|
User.getModeratorUids(function (err, uids) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(1, uids.length);
|
|
assert.strictEqual(1, parseInt(uids[0], 10));
|
|
done();
|
|
});
|
|
});
|
|
|
|
after(function (done) {
|
|
groups.leave('cid:1:privileges:moderate', 1, done);
|
|
});
|
|
});
|
|
|
|
describe('.getModeratorUids()', function () {
|
|
before(function (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', function (done) {
|
|
User.getModeratorUids(function (err, uids) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(1, uids.length);
|
|
assert.strictEqual(1, parseInt(uids[0], 10));
|
|
done();
|
|
});
|
|
});
|
|
|
|
after(function (done) {
|
|
async.series([
|
|
async.apply(groups.leave, 'cid:1:privileges:groups:moderate', 'testGroup'),
|
|
async.apply(groups.destroy, 'testGroup'),
|
|
], 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 () {
|
|
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', function (done) {
|
|
User.search({ query: 'john' }, function (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', 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) {
|
|
socketUser.search({ uid: 0 }, { query: 'john' }, function (err) {
|
|
assert.equal(err.message, '[[error:no-privileges]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error with invalid data', function (done) {
|
|
socketUser.search({ uid: testUid }, null, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error for unprivileged user', function (done) {
|
|
socketUser.search({ uid: testUid }, { searchBy: 'ip', query: '123' }, function (err) {
|
|
assert.equal(err.message, '[[error:no-privileges]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error for unprivileged user', function (done) {
|
|
socketUser.search({ uid: testUid }, { filters: ['banned'], query: '123' }, function (err) {
|
|
assert.equal(err.message, '[[error:no-privileges]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error for unprivileged user', function (done) {
|
|
socketUser.search({ uid: testUid }, { filters: ['flagged'], query: '123' }, function (err) {
|
|
assert.equal(err.message, '[[error:no-privileges]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should search users by ip', async function () {
|
|
const uid = await User.create({ username: 'ipsearch' });
|
|
await db.sortedSetAdd('ip:1.1.1.1:uid', [1, 1], [testUid, uid]);
|
|
const data = await socketUser.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', function (done) {
|
|
socketUser.search({ uid: testUid }, { query: uid, searchBy: 'uid' }, function (err, data) {
|
|
assert.ifError(err);
|
|
assert(Array.isArray(data.users));
|
|
assert.equal(data.users[0].uid, uid);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should search users by fullname', async function () {
|
|
const uid = await User.create({ username: 'fullnamesearch1', fullname: 'Mr. Fullname' });
|
|
const data = await socketUser.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 function () {
|
|
const uid = await User.create({ username: 'fullnamesearch2', fullname: 'Baris:Usakli' });
|
|
const data = await socketUser.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', function (done) {
|
|
socketUser.search({ uid: testUid }, { query: '' }, function (err, data) {
|
|
assert.ifError(err);
|
|
assert.equal(data.users.length, 0);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should filter users', function (done) {
|
|
User.create({ username: 'ipsearch_filter' }, function (err, uid) {
|
|
assert.ifError(err);
|
|
User.setUserFields(uid, { banned: 1, flags: 10 }, function (err) {
|
|
assert.ifError(err);
|
|
socketUser.search({ uid: adminUid }, {
|
|
query: 'ipsearch',
|
|
filters: ['online', 'banned', 'flagged'],
|
|
}, function (err, data) {
|
|
assert.ifError(err);
|
|
assert.equal(data.users[0].username, 'ipsearch_filter');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should sort results by username', function (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);
|
|
},
|
|
], function (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()', function () {
|
|
var uid;
|
|
before(function (done) {
|
|
User.create({ username: 'usertodelete', password: '123456', email: '[email protected]' }, 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();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should not re-add user to users:postcount if post is deleted after user deletion', async function () {
|
|
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 deletion', async function () {
|
|
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));
|
|
});
|
|
});
|
|
|
|
describe('passwordReset', function () {
|
|
var uid;
|
|
var code;
|
|
before(function (done) {
|
|
User.create({ username: 'resetuser', password: '123456', email: '[email protected]' }, 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('[email protected]', function (err) {
|
|
if (err) {
|
|
console.log(err);
|
|
}
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('.commit() should update the user\'s password and confirm their email', function (done) {
|
|
User.reset.commit(code, 'newpassword', function (err) {
|
|
assert.ifError(err);
|
|
|
|
async.parallel({
|
|
userData: function (next) {
|
|
User.getUserData(uid, next);
|
|
},
|
|
password: function (next) {
|
|
db.getObjectField('user:' + uid, 'password', next);
|
|
},
|
|
}, function (err, results) {
|
|
assert.ifError(err);
|
|
Password.compare('newpassword', results.password, function (err, match) {
|
|
assert.ifError(err);
|
|
assert(match);
|
|
assert.strictEqual(results.userData['email:confirmed'], 1);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it('.commit() should invalidate old codes', function (done) {
|
|
var code1;
|
|
var code2;
|
|
var uid;
|
|
async.waterfall([
|
|
function (next) {
|
|
User.create({ username: 'doublereseter', email: '[email protected]', password: '123456' }, next);
|
|
},
|
|
function (_uid, next) {
|
|
uid = _uid;
|
|
User.reset.generate(uid, next);
|
|
},
|
|
function (code, next) {
|
|
code1 = code;
|
|
User.reset.generate(uid, next);
|
|
},
|
|
function (code, next) {
|
|
code2 = code;
|
|
User.reset.validate(code1, next);
|
|
},
|
|
function (isValid, next) {
|
|
assert(isValid);
|
|
User.reset.commit(code2, 'newPwd123', next);
|
|
},
|
|
function (next) {
|
|
User.reset.validate(code1, next);
|
|
},
|
|
function (isValid, next) {
|
|
assert(!isValid);
|
|
next();
|
|
},
|
|
], done);
|
|
});
|
|
});
|
|
|
|
describe('hash methods', function () {
|
|
it('should return uid from email', function (done) {
|
|
User.getUidByEmail('[email protected]', 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();
|
|
});
|
|
});
|
|
|
|
it('should get user data even if one uid is NaN', function (done) {
|
|
User.getUsersData([NaN, testUid], function (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', function (done) {
|
|
User.setUserFields(testUid, {
|
|
fb_token: '123123123',
|
|
another_secret: 'abcde',
|
|
postcount: '123',
|
|
}, function (err) {
|
|
assert.ifError(err);
|
|
User.getUserData(testUid, function (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', function (done) {
|
|
User.getUserFields(testUid, ['password'], function (err, payload) {
|
|
assert.ifError(err);
|
|
assert(!payload.hasOwnProperty('password'));
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should return private data if field is whitelisted', function (done) {
|
|
function filterMethod(data, callback) {
|
|
data.whitelist.push('another_secret');
|
|
callback(null, data);
|
|
}
|
|
|
|
plugins.registerHook('test-plugin', { hook: 'filter:user.whitelistFields', method: filterMethod });
|
|
User.getUserData(testUid, function (err, userData) {
|
|
assert.ifError(err);
|
|
assert(!userData.hasOwnProperty('fb_token'));
|
|
assert.equal(userData.another_secret, 'abcde');
|
|
plugins.unregisterHook('test-plugin', 'filter:user.whitelistFields', filterMethod);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should return 0 as uid if username is falsy', function (done) {
|
|
User.getUidByUsername('', function (err, uid) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(uid, 0);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should get username by userslug', function (done) {
|
|
User.getUsernameByUserslug('john-smith', function (err, username) {
|
|
assert.ifError(err);
|
|
assert.strictEqual('John Smith', username);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should get uids by emails', function (done) {
|
|
User.getUidsByEmails(['[email protected]'], function (err, uids) {
|
|
assert.ifError(err);
|
|
assert.equal(uids[0], testUid);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should not get groupTitle for guests', function (done) {
|
|
User.getUserData(0, function (err, userData) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(userData.groupTitle, '');
|
|
assert.deepStrictEqual(userData.groupTitleArray, []);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should load guest data', function (done) {
|
|
User.getUsersData([1, 0], function (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('not logged in', function () {
|
|
it('should return error if not logged in', function (done) {
|
|
socketUser.updateProfile({ uid: 0 }, { uid: 1 }, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-uid]]');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('profile methods', function () {
|
|
var uid;
|
|
var jar;
|
|
|
|
before(function (done) {
|
|
User.create({ username: 'updateprofile', email: '[email protected]', password: '123456' }, function (err, newUid) {
|
|
assert.ifError(err);
|
|
uid = newUid;
|
|
helpers.loginUser('updateprofile', '123456', function (err, _jar) {
|
|
assert.ifError(err);
|
|
jar = _jar;
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should return error if data is invalid', function (done) {
|
|
socketUser.updateProfile({ uid: uid }, null, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should return error if data is missing uid', function (done) {
|
|
socketUser.updateProfile({ uid: uid }, { username: 'bip', email: 'bop' }, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should update a user\'s profile', function (done) {
|
|
User.create({ username: 'justforupdate', email: '[email protected]', password: '123456' }, function (err, uid) {
|
|
assert.ifError(err);
|
|
var data = {
|
|
uid: uid,
|
|
username: 'updatedUserName',
|
|
email: '[email protected]',
|
|
fullname: 'updatedFullname',
|
|
website: 'http://nodebb.org',
|
|
location: 'izmir',
|
|
groupTitle: 'testGroup',
|
|
birthday: '01/01/1980',
|
|
signature: 'nodebb is good',
|
|
password: '123456',
|
|
};
|
|
socketUser.updateProfile({ uid: uid }, { ...data, password: '123456' }, function (err, result) {
|
|
assert.ifError(err);
|
|
|
|
assert.equal(result.username, 'updatedUserName');
|
|
assert.equal(result.userslug, 'updatedusername');
|
|
assert.equal(result.email, '[email protected]');
|
|
|
|
db.getObject('user:' + uid, function (err, userData) {
|
|
assert.ifError(err);
|
|
Object.keys(data).forEach(function (key) {
|
|
if (key !== 'password') {
|
|
assert.equal(data[key], userData[key]);
|
|
} else {
|
|
assert(userData[key].startsWith('$2a$'));
|
|
}
|
|
});
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should change a user\'s password', function (done) {
|
|
User.create({ username: 'changepassword', password: '123456' }, function (err, uid) {
|
|
assert.ifError(err);
|
|
socketUser.changePassword({ uid: uid }, { uid: uid, newPassword: '654321', currentPassword: '123456' }, function (err) {
|
|
assert.ifError(err);
|
|
User.isPasswordCorrect(uid, '654321', '127.0.0.1', function (err, correct) {
|
|
assert.ifError(err);
|
|
assert(correct);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should not let user change another user\'s password', async function () {
|
|
const regularUserUid = await User.create({ username: 'regularuserpwdchange', password: 'regularuser1234' });
|
|
const uid = await User.create({ username: 'changeadminpwd1', password: '123456' });
|
|
let err;
|
|
try {
|
|
await socketUser.changePassword({ uid: uid }, { uid: regularUserUid, newPassword: '654321', currentPassword: '123456' });
|
|
} catch (_err) {
|
|
err = _err;
|
|
}
|
|
assert.equal(err.message, '[[user:change_password_error_privileges]]');
|
|
});
|
|
|
|
it('should not let user change admin\'s password', async function () {
|
|
const adminUid = await User.create({ username: 'adminpwdchange', password: 'admin1234' });
|
|
await groups.join('administrators', adminUid);
|
|
const uid = await User.create({ username: 'changeadminpwd2', password: '123456' });
|
|
|
|
let err;
|
|
try {
|
|
await socketUser.changePassword({ uid: uid }, { uid: adminUid, newPassword: '654321', currentPassword: '123456' });
|
|
} catch (_err) {
|
|
err = _err;
|
|
}
|
|
assert.equal(err.message, '[[user:change_password_error_privileges]]');
|
|
});
|
|
|
|
it('should let admin change another users password', async function () {
|
|
const adminUid = await User.create({ username: 'adminpwdchange2', password: 'admin1234' });
|
|
await groups.join('administrators', adminUid);
|
|
const uid = await User.create({ username: 'forgotmypassword', password: '123456' });
|
|
|
|
await socketUser.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 function () {
|
|
const adminUid = await User.create({ username: 'adminforgotpwd', password: 'admin1234' });
|
|
await groups.join('administrators', adminUid);
|
|
|
|
let err;
|
|
try {
|
|
await socketUser.changePassword({ uid: adminUid }, { uid: adminUid, newPassword: '654321', currentPassword: 'wrongpwd' });
|
|
} catch (_err) {
|
|
err = _err;
|
|
}
|
|
assert.equal(err.message, '[[user:change_password_error_wrong_current]]');
|
|
});
|
|
|
|
it('should change username', function (done) {
|
|
socketUser.changeUsernameEmail({ uid: uid }, { uid: uid, username: 'updatedAgain', password: '123456' }, function (err) {
|
|
assert.ifError(err);
|
|
db.getObjectField('user:' + uid, 'username', function (err, username) {
|
|
assert.ifError(err);
|
|
assert.equal(username, 'updatedAgain');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should not let setting an empty username', async function () {
|
|
await socketUser.changeUsernameEmail({ 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 function () {
|
|
const maxLength = meta.config.maximumUsernameLength + 1;
|
|
const longName = new Array(maxLength).fill('a').join('');
|
|
const uid = await User.create({ username: longName });
|
|
await socketUser.changeUsernameEmail({ uid: uid }, { uid: uid, username: longName, email: '[email protected]' });
|
|
const userData = await db.getObject('user:' + uid);
|
|
assert.strictEqual(userData.username, longName);
|
|
assert.strictEqual(userData.email, '[email protected]');
|
|
});
|
|
|
|
it('should not update a user\'s username if it did not change', function (done) {
|
|
socketUser.changeUsernameEmail({ uid: uid }, { uid: uid, username: 'updatedAgain', password: '123456' }, function (err) {
|
|
assert.ifError(err);
|
|
db.getSortedSetRevRange('user:' + uid + ':usernames', 0, -1, function (err, data) {
|
|
assert.ifError(err);
|
|
assert.equal(data.length, 2);
|
|
assert(data[0].startsWith('updatedAgain'));
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should not update a user\'s username if a password is not supplied', async () => {
|
|
let _err;
|
|
try {
|
|
await socketUser.updateProfile({ uid: uid }, { uid: uid, username: 'updatedAgain', password: '' });
|
|
} catch (err) {
|
|
_err = err;
|
|
}
|
|
|
|
assert(_err);
|
|
assert.strictEqual(_err.message, '[[error:invalid-password]]');
|
|
});
|
|
|
|
it('should change email', function (done) {
|
|
User.create({ username: 'pooremailupdate', email: '[email protected]', password: '123456' }, function (err, uid) {
|
|
assert.ifError(err);
|
|
socketUser.changeUsernameEmail({ uid: uid }, { uid: uid, email: '[email protected]', password: '123456' }, function (err) {
|
|
assert.ifError(err);
|
|
db.getObjectField('user:' + uid, 'email', function (err, email) {
|
|
assert.ifError(err);
|
|
assert.equal(email, '[email protected]');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should error if email is identical', async function () {
|
|
await User.create({
|
|
username: 'trimtest1',
|
|
email: '[email protected]',
|
|
});
|
|
const uid2 = await User.create({
|
|
username: 'trimtest2',
|
|
email: '[email protected]',
|
|
});
|
|
let err;
|
|
try {
|
|
await socketUser.changeUsernameEmail({ uid: uid2 }, {
|
|
uid: uid2,
|
|
email: ' [email protected]',
|
|
});
|
|
} catch (_err) {
|
|
err = _err;
|
|
}
|
|
assert.strictEqual(err.message, '[[error:email-taken]]');
|
|
});
|
|
|
|
it('should update cover image', function (done) {
|
|
var imageData = '';
|
|
var position = '50.0301% 19.2464%';
|
|
socketUser.updateCover({ uid: uid }, { 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 upload cropped profile picture', function (done) {
|
|
var imageData = '';
|
|
socketUser.uploadCroppedPicture({ uid: uid }, { uid: uid, imageData: imageData }, function (err, result) {
|
|
assert.ifError(err);
|
|
assert(result.url);
|
|
db.getObjectFields('user:' + uid, ['uploadedpicture', 'picture'], function (err, data) {
|
|
assert.ifError(err);
|
|
assert.equal(result.url, data.uploadedpicture);
|
|
assert.equal(result.url, data.picture);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should remove cover image', function (done) {
|
|
socketUser.removeCover({ uid: uid }, { 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) {
|
|
socketUser.setStatus({ uid: uid }, '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) {
|
|
socketUser.setStatus({ uid: uid }, '12345', function (err) {
|
|
assert.equal(err.message, '[[error:invalid-user-status]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should get user status', function (done) {
|
|
socketUser.checkStatus({ uid: uid }, uid, function (err, status) {
|
|
assert.ifError(err);
|
|
assert.equal(status, 'away');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should change user picture', function (done) {
|
|
socketUser.changePicture({ uid: uid }, { 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 fail to change user picture with invalid data', function (done) {
|
|
socketUser.changePicture({ uid: uid }, null, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should fail to change user picture with invalid uid', function (done) {
|
|
socketUser.changePicture({ uid: 0 }, null, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-uid]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should set user picture to uploaded', function (done) {
|
|
User.setUserField(uid, 'uploadedpicture', '/test', function (err) {
|
|
assert.ifError(err);
|
|
socketUser.changePicture({ uid: uid }, { type: 'uploaded', uid: uid }, function (err) {
|
|
assert.ifError(err);
|
|
User.getUserField(uid, 'picture', function (err, picture) {
|
|
assert.ifError(err);
|
|
assert.equal(picture, nconf.get('relative_path') + '/test');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should return error if profile image uploads disabled', function (done) {
|
|
meta.config.allowProfileImageUploads = 0;
|
|
var picture = {
|
|
path: path.join(nconf.get('base_dir'), 'test/files/test_copy.png'),
|
|
size: 7189,
|
|
name: 'test.png',
|
|
type: 'image/png',
|
|
};
|
|
User.uploadCroppedPicture({
|
|
uid: uid,
|
|
file: picture,
|
|
}, function (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', function (done) {
|
|
User.uploadCroppedPicture({
|
|
uid: uid,
|
|
imageData: '',
|
|
}, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-image]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
describe('user.uploadCroppedPicture', function () {
|
|
var goodImage = '';
|
|
var badImage = 'data:audio/mp3;base64,R0lGODlhPQBEAPeoAJosM//AwO/AwHVYZ/z595kzAP/s7P+goOXMv8+fhw/v739/f+8PD98fH/8mJl+fn/9ZWb8/PzWlwv///6wWGbImAPgTEMImIN9gUFCEm/gDALULDN8PAD6atYdCTX9gUNKlj8wZAKUsAOzZz+UMAOsJAP/Z2ccMDA8PD/95eX5NWvsJCOVNQPtfX/8zM8+QePLl38MGBr8JCP+zs9myn/8GBqwpAP/GxgwJCPny78lzYLgjAJ8vAP9fX/+MjMUcAN8zM/9wcM8ZGcATEL+QePdZWf/29uc/P9cmJu9MTDImIN+/r7+/vz8/P8VNQGNugV8AAF9fX8swMNgTAFlDOICAgPNSUnNWSMQ5MBAQEJE3QPIGAM9AQMqGcG9vb6MhJsEdGM8vLx8fH98AANIWAMuQeL8fABkTEPPQ0OM5OSYdGFl5jo+Pj/+pqcsTE78wMFNGQLYmID4dGPvd3UBAQJmTkP+8vH9QUK+vr8ZWSHpzcJMmILdwcLOGcHRQUHxwcK9PT9DQ0O/v70w5MLypoG8wKOuwsP/g4P/Q0IcwKEswKMl8aJ9fX2xjdOtGRs/Pz+Dg4GImIP8gIH0sKEAwKKmTiKZ8aB/f39Wsl+LFt8dgUE9PT5x5aHBwcP+AgP+WltdgYMyZfyywz78AAAAAAAD///8AAP9mZv///wAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAEAAKgALAAAAAA9AEQAAAj/AFEJHEiwoMGDCBMqXMiwocAbBww4nEhxoYkUpzJGrMixogkfGUNqlNixJEIDB0SqHGmyJSojM1bKZOmyop0gM3Oe2liTISKMOoPy7GnwY9CjIYcSRYm0aVKSLmE6nfq05QycVLPuhDrxBlCtYJUqNAq2bNWEBj6ZXRuyxZyDRtqwnXvkhACDV+euTeJm1Ki7A73qNWtFiF+/gA95Gly2CJLDhwEHMOUAAuOpLYDEgBxZ4GRTlC1fDnpkM+fOqD6DDj1aZpITp0dtGCDhr+fVuCu3zlg49ijaokTZTo27uG7Gjn2P+hI8+PDPERoUB318bWbfAJ5sUNFcuGRTYUqV/3ogfXp1rWlMc6awJjiAAd2fm4ogXjz56aypOoIde4OE5u/F9x199dlXnnGiHZWEYbGpsAEA3QXYnHwEFliKAgswgJ8LPeiUXGwedCAKABACCN+EA1pYIIYaFlcDhytd51sGAJbo3onOpajiihlO92KHGaUXGwWjUBChjSPiWJuOO/LYIm4v1tXfE6J4gCSJEZ7YgRYUNrkji9P55sF/ogxw5ZkSqIDaZBV6aSGYq/lGZplndkckZ98xoICbTcIJGQAZcNmdmUc210hs35nCyJ58fgmIKX5RQGOZowxaZwYA+JaoKQwswGijBV4C6SiTUmpphMspJx9unX4KaimjDv9aaXOEBteBqmuuxgEHoLX6Kqx+yXqqBANsgCtit4FWQAEkrNbpq7HSOmtwag5w57GrmlJBASEU18ADjUYb3ADTinIttsgSB1oJFfA63bduimuqKB1keqwUhoCSK374wbujvOSu4QG6UvxBRydcpKsav++Ca6G8A6Pr1x2kVMyHwsVxUALDq/krnrhPSOzXG1lUTIoffqGR7Goi2MAxbv6O2kEG56I7CSlRsEFKFVyovDJoIRTg7sugNRDGqCJzJgcKE0ywc0ELm6KBCCJo8DIPFeCWNGcyqNFE06ToAfV0HBRgxsvLThHn1oddQMrXj5DyAQgjEHSAJMWZwS3HPxT/QMbabI/iBCliMLEJKX2EEkomBAUCxRi42VDADxyTYDVogV+wSChqmKxEKCDAYFDFj4OmwbY7bDGdBhtrnTQYOigeChUmc1K3QTnAUfEgGFgAWt88hKA6aCRIXhxnQ1yg3BCayK44EWdkUQcBByEQChFXfCB776aQsG0BIlQgQgE8qO26X1h8cEUep8ngRBnOy74E9QgRgEAC8SvOfQkh7FDBDmS43PmGoIiKUUEGkMEC/PJHgxw0xH74yx/3XnaYRJgMB8obxQW6kL9QYEJ0FIFgByfIL7/IQAlvQwEpnAC7DtLNJCKUoO/w45c44GwCXiAFB/OXAATQryUxdN4LfFiwgjCNYg+kYMIEFkCKDs6PKAIJouyGWMS1FSKJOMRB/BoIxYJIUXFUxNwoIkEKPAgCBZSQHQ1A2EWDfDEUVLyADj5AChSIQW6gu10bE/JG2VnCZGfo4R4d0sdQoBAHhPjhIB94v/wRoRKQWGRHgrhGSQJxCS+0pCZbEhAAOw==';
|
|
it('should error if both file and imageData are missing', function (done) {
|
|
User.uploadCroppedPicture({}, function (err) {
|
|
assert.equal('[[error:invalid-data]]', err.message);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error if file size is too big', function (done) {
|
|
var temp = meta.config.maximumProfileImageSize;
|
|
meta.config.maximumProfileImageSize = 1;
|
|
User.uploadCroppedPicture({
|
|
uid: 1,
|
|
imageData: goodImage,
|
|
}, function (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', function (done) {
|
|
User.uploadCroppedPicture({
|
|
uid: 1,
|
|
imageData: badImage,
|
|
}, function (err) {
|
|
assert.equal('[[error:invalid-image]]', err.message);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should get profile pictures', function (done) {
|
|
socketUser.getProfilePictures({ uid: uid }, { 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 get default profile avatar', function (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';
|
|
nconf.set('relative_path', '/community');
|
|
assert.strictEqual(User.getDefaultAvatar(), '/community' + meta.config.defaultAvatar);
|
|
meta.config.defaultAvatar = '';
|
|
nconf.set('relative_path', '');
|
|
done();
|
|
});
|
|
|
|
it('should fail to get profile pictures with invalid data', function (done) {
|
|
socketUser.getProfilePictures({ uid: uid }, null, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
socketUser.getProfilePictures({ uid: uid }, { uid: null }, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should remove uploaded picture', function (done) {
|
|
socketUser.removeUploadedPicture({ uid: uid }, { uid: uid }, function (err) {
|
|
assert.ifError(err);
|
|
User.getUserField(uid, 'uploadedpicture', function (err, uploadedpicture) {
|
|
assert.ifError(err);
|
|
assert.equal(uploadedpicture, '');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should fail to remove uploaded picture with invalid-data', function (done) {
|
|
socketUser.removeUploadedPicture({ uid: uid }, null, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
socketUser.removeUploadedPicture({ uid: uid }, { }, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
socketUser.removeUploadedPicture({ uid: null }, { }, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
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('user info', function () {
|
|
it('should return error if there is no ban reason', function (done) {
|
|
User.getLatestBanInfo(123, function (err) {
|
|
assert.equal(err.message, 'no-ban-info');
|
|
done();
|
|
});
|
|
});
|
|
|
|
|
|
it('should get history from set', async function () {
|
|
const now = Date.now();
|
|
await db.sortedSetAdd('user:' + testUid + ':usernames', now, 'derp:' + now);
|
|
const data = await User.getHistory('user:' + testUid + ':usernames');
|
|
assert.equal(data[0].value, 'derp');
|
|
assert.equal(data[0].timestamp, now);
|
|
});
|
|
|
|
it('should return the correct ban reason', function (done) {
|
|
async.series([
|
|
function (next) {
|
|
User.bans.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.bans.unban(testUid, function (err) {
|
|
assert.ifError(err);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should ban user permanently', function (done) {
|
|
User.bans.ban(testUid, function (err) {
|
|
assert.ifError(err);
|
|
User.bans.isBanned(testUid, function (err, isBanned) {
|
|
assert.ifError(err);
|
|
assert.equal(isBanned, true);
|
|
User.bans.unban(testUid, done);
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should ban user temporarily', function (done) {
|
|
User.bans.ban(testUid, Date.now() + 2000, function (err) {
|
|
assert.ifError(err);
|
|
|
|
User.bans.isBanned(testUid, function (err, isBanned) {
|
|
assert.ifError(err);
|
|
assert.equal(isBanned, true);
|
|
setTimeout(function () {
|
|
User.bans.isBanned(testUid, function (err, isBanned) {
|
|
assert.ifError(err);
|
|
assert.equal(isBanned, false);
|
|
User.bans.unban(testUid, done);
|
|
});
|
|
}, 3000);
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should error if until is NaN', function (done) {
|
|
User.bans.ban(testUid, 'asd', function (err) {
|
|
assert.equal(err.message, '[[error:ban-expiry-missing]]');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('Digest.getSubscribers', function (done) {
|
|
var uidIndex = {};
|
|
|
|
before(function (done) {
|
|
var testUsers = ['daysub', 'offsub', 'nullsub', 'weeksub'];
|
|
async.each(testUsers, function (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;
|
|
|
|
var 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)', function (done) {
|
|
User.digest.getSubscribers('day', function (err, subs) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(subs.length, 1);
|
|
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should accurately build digest list given ACP default "day"', function (done) {
|
|
async.series([
|
|
async.apply(meta.configs.set, 'dailyDigestFreq', 'day'),
|
|
function (next) {
|
|
User.digest.getSubscribers('day', function (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"', function (done) {
|
|
async.series([
|
|
async.apply(meta.configs.set, 'dailyDigestFreq', 'week'),
|
|
function (next) {
|
|
User.digest.getSubscribers('week', function (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"', function (done) {
|
|
async.series([
|
|
async.apply(meta.configs.set, 'dailyDigestFreq', 'off'),
|
|
function (next) {
|
|
User.digest.getSubscribers('day', function (err, subs) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(subs.length, 1);
|
|
|
|
next();
|
|
});
|
|
},
|
|
], done);
|
|
});
|
|
});
|
|
|
|
describe('digests', function () {
|
|
var uid;
|
|
before(function (done) {
|
|
async.waterfall([
|
|
function (next) {
|
|
User.create({ username: 'digestuser', email: '[email protected]' }, 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', function (done) {
|
|
User.digest.execute({ interval: 'day' }, function (err) {
|
|
assert.ifError(err);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should not send digests', function (done) {
|
|
User.digest.execute({ interval: 'month' }, function (err) {
|
|
assert.ifError(err);
|
|
done();
|
|
});
|
|
});
|
|
|
|
describe('unsubscribe via POST', function () {
|
|
it('should unsubscribe from digest if one-click unsubscribe is POSTed', function (done) {
|
|
const token = jwt.sign({
|
|
template: 'digest',
|
|
uid: uid,
|
|
}, nconf.get('secret'));
|
|
|
|
request({
|
|
method: 'post',
|
|
url: nconf.get('url') + '/email/unsubscribe/' + token,
|
|
}, function (err, res) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(res.statusCode, 200);
|
|
|
|
db.getObjectField('user:' + uid + ':settings', 'dailyDigestFreq', function (err, value) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(value, 'off');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should unsubscribe from notifications if one-click unsubscribe is POSTed', function (done) {
|
|
const token = jwt.sign({
|
|
template: 'notification',
|
|
type: 'test',
|
|
uid: uid,
|
|
}, nconf.get('secret'));
|
|
|
|
request({
|
|
method: 'post',
|
|
url: nconf.get('url') + '/email/unsubscribe/' + token,
|
|
}, function (err, res) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(res.statusCode, 200);
|
|
|
|
db.getObjectField('user:' + uid + ':settings', 'notificationType_test', function (err, value) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(value, 'notification');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should return errors on missing template in token', function (done) {
|
|
const token = jwt.sign({
|
|
uid: uid,
|
|
}, nconf.get('secret'));
|
|
|
|
request({
|
|
method: 'post',
|
|
url: nconf.get('url') + '/email/unsubscribe/' + token,
|
|
}, function (err, res) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(res.statusCode, 404);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should return errors on wrong template in token', function (done) {
|
|
const token = jwt.sign({
|
|
template: 'user',
|
|
uid: uid,
|
|
}, nconf.get('secret'));
|
|
|
|
request({
|
|
method: 'post',
|
|
url: nconf.get('url') + '/email/unsubscribe/' + token,
|
|
}, function (err, res) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(res.statusCode, 404);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should return errors on missing token', function (done) {
|
|
request({
|
|
method: 'post',
|
|
url: nconf.get('url') + '/email/unsubscribe/',
|
|
}, function (err, res) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(res.statusCode, 404);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should return errors on token signed with wrong secret (verify-failure)', function (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,
|
|
}, function (err, res) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(res.statusCode, 403);
|
|
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();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should fail to delete user if account deletion is not allowed', async function () {
|
|
const oldValue = meta.config.allowAccountDeletion;
|
|
meta.config.allowAccountDeletion = 0;
|
|
const uid = await User.create({ username: 'tobedeleted' });
|
|
try {
|
|
await socketUser.deleteAccount({ uid: uid }, {});
|
|
} catch (err) {
|
|
assert.equal(err.message, '[[error:no-privileges]]');
|
|
}
|
|
meta.config.allowAccountDeletion = oldValue;
|
|
});
|
|
|
|
it('should fail if data is invalid', function (done) {
|
|
socketUser.emailExists({ uid: testUid }, null, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should return true if email exists', function (done) {
|
|
socketUser.emailExists({ uid: testUid }, { email: '[email protected]' }, function (err, exists) {
|
|
assert.ifError(err);
|
|
assert(exists);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should return false if email does not exist', function (done) {
|
|
socketUser.emailExists({ uid: testUid }, { email: '[email protected]' }, function (err, exists) {
|
|
assert.ifError(err);
|
|
assert(!exists);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error if requireEmailConfirmation is disabled', function (done) {
|
|
socketUser.emailConfirm({ uid: testUid }, {}, function (err) {
|
|
assert.equal(err.message, '[[error:email-confirmations-are-disabled]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should send email confirm', function (done) {
|
|
meta.config.requireEmailConfirmation = 1;
|
|
socketUser.emailConfirm({ uid: testUid }, {}, function (err) {
|
|
assert.ifError(err);
|
|
meta.config.requireEmailConfirmation = 0;
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should send reset email', function (done) {
|
|
socketUser.reset.send({ uid: 0 }, '[email protected]', function (err) {
|
|
assert.ifError(err);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should return invalid-data error', function (done) {
|
|
socketUser.reset.send({ uid: 0 }, null, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should not error', function (done) {
|
|
socketUser.reset.send({ uid: 0 }, '[email protected]', function (err) {
|
|
assert.ifError(err);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should commit reset', function (done) {
|
|
db.getObject('reset:uid', function (err, data) {
|
|
assert.ifError(err);
|
|
var code = Object.keys(data)[0];
|
|
socketUser.reset.commit({ uid: 0 }, { code: code, password: 'swordfish' }, function (err) {
|
|
assert.ifError(err);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should save user settings', function (done) {
|
|
var 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,
|
|
},
|
|
};
|
|
socketUser.saveSettings({ uid: testUid }, data, function (err) {
|
|
assert.ifError(err);
|
|
User.getSettings(testUid, function (err, data) {
|
|
assert.ifError(err);
|
|
assert.equal(data.usePagination, true);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should properly escape homePageRoute', function (done) {
|
|
var 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,
|
|
},
|
|
};
|
|
socketUser.saveSettings({ uid: testUid }, data, function (err) {
|
|
assert.ifError(err);
|
|
User.getSettings(testUid, function (err, data) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(data.homePageRoute, 'category/6/testing-ground');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
|
|
it('should error if language is invalid', function (done) {
|
|
var data = {
|
|
uid: testUid,
|
|
settings: {
|
|
userLang: '<invalid-string>',
|
|
topicsPerPage: '10',
|
|
postsPerPage: '5',
|
|
},
|
|
};
|
|
socketUser.saveSettings({ uid: testUid }, data, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-language]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should set moderation note', function (done) {
|
|
var 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: '<svg/onload=alert(document.location);//' }, next);
|
|
},
|
|
function (next) {
|
|
User.getModerationNotes(testUid, 0, -1, next);
|
|
},
|
|
], function (err, notes) {
|
|
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();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('approval queue', function () {
|
|
var oldRegistrationApprovalType;
|
|
var adminUid;
|
|
before(function (done) {
|
|
oldRegistrationApprovalType = meta.config.registrationApprovalType;
|
|
meta.config.registrationApprovalType = '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.registrationApprovalType = oldRegistrationApprovalType;
|
|
done();
|
|
});
|
|
|
|
it('should add user to approval queue', function (done) {
|
|
helpers.registerUser({
|
|
username: 'rejectme',
|
|
password: '123456',
|
|
'password-confirm': '123456',
|
|
email: '<script>alert("ok")<script>[email protected]',
|
|
gdpr_consent: true,
|
|
}, 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, '<script>alert("ok")<script>[email protected]');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should fail to add user to queue if username is taken', function (done) {
|
|
helpers.registerUser({
|
|
username: 'rejectme',
|
|
password: '123456',
|
|
'password-confirm': '123456',
|
|
email: '<script>alert("ok")<script>[email protected]',
|
|
gdpr_consent: true,
|
|
}, function (err, jar, res, body) {
|
|
assert.ifError(err);
|
|
assert.equal(body, '[[error:username-taken]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should fail to add user to queue if email is taken', function (done) {
|
|
helpers.registerUser({
|
|
username: 'rejectmenew',
|
|
password: '123456',
|
|
'password-confirm': '123456',
|
|
email: '<script>alert("ok")<script>[email protected]',
|
|
gdpr_consent: true,
|
|
}, function (err, jar, res, body) {
|
|
assert.ifError(err);
|
|
assert.equal(body, '[[error:email-taken]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should reject user registration', function (done) {
|
|
socketUser.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',
|
|
'password-confirm': '123456',
|
|
email: '[email protected]',
|
|
gdpr_consent: true,
|
|
}, function (err) {
|
|
assert.ifError(err);
|
|
socketUser.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();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should trim username and add user to registration queue', function (done) {
|
|
helpers.registerUser({
|
|
username: 'invalidname\r\n',
|
|
password: '123456',
|
|
'password-confirm': '123456',
|
|
email: '[email protected]',
|
|
gdpr_consent: true,
|
|
}, function (err) {
|
|
assert.ifError(err);
|
|
db.getSortedSetRange('registration:queue', 0, -1, function (err, data) {
|
|
assert.ifError(err);
|
|
assert.equal(data[0], 'invalidname');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('invites', function () {
|
|
var socketUser = require('../src/socket.io/user');
|
|
var inviterUid;
|
|
var adminUid;
|
|
|
|
before(function (done) {
|
|
async.parallel({
|
|
inviter: async.apply(User.create, { username: 'inviter', email: '[email protected]' }),
|
|
admin: async.apply(User.create, { username: 'adminInvite' }),
|
|
}, function (err, results) {
|
|
assert.ifError(err);
|
|
inviterUid = results.inviter;
|
|
adminUid = results.admin;
|
|
groups.join('administrators', adminUid, done);
|
|
});
|
|
});
|
|
|
|
it('should error with invalid data', function (done) {
|
|
socketUser.invite({ uid: inviterUid }, null, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should eror if forum is not invite only', function (done) {
|
|
socketUser.invite({ uid: inviterUid }, '[email protected]', function (err) {
|
|
assert.equal(err.message, '[[error:forum-not-invite-only]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error if user is not admin and type is admin-invite-only', function (done) {
|
|
meta.config.registrationType = 'admin-invite-only';
|
|
socketUser.invite({ uid: inviterUid }, '[email protected]', function (err) {
|
|
assert.equal(err.message, '[[error:no-privileges]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should send invitation email', function (done) {
|
|
meta.config.registrationType = 'invite-only';
|
|
socketUser.invite({ uid: inviterUid }, '[email protected]', function (err) {
|
|
assert.ifError(err);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error if ouf of invitations', function (done) {
|
|
meta.config.maximumInvites = 1;
|
|
socketUser.invite({ uid: inviterUid }, '[email protected]', function (err) {
|
|
assert.equal(err.message, '[[error:invite-maximum-met, ' + 1 + ', ' + 1 + ']]');
|
|
meta.config.maximumInvites = 5;
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error if email exists', function (done) {
|
|
socketUser.invite({ uid: inviterUid }, '[email protected]', function (err) {
|
|
assert.equal(err.message, '[[error:email-taken]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should send invitation email', function (done) {
|
|
socketUser.invite({ uid: inviterUid }, '[email protected]', function (err) {
|
|
assert.ifError(err);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should get user\'s invites', function (done) {
|
|
User.getInvites(inviterUid, function (err, data) {
|
|
assert.ifError(err);
|
|
assert.notEqual(data.indexOf('[email protected]'), -1);
|
|
assert.notEqual(data.indexOf('[email protected]'), -1);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should get all invites', function (done) {
|
|
User.getAllInvites(function (err, data) {
|
|
assert.ifError(err);
|
|
assert.equal(data[0].uid, inviterUid);
|
|
assert.notEqual(data[0].invitations.indexOf('[email protected]'), -1);
|
|
assert.notEqual(data[0].invitations.indexOf('[email protected]'), -1);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should fail to verify invitation with invalid data', function (done) {
|
|
User.verifyInvitation({ token: '', email: '' }, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should fail to verify invitation with invalid email', function (done) {
|
|
User.verifyInvitation({ token: 'test', email: '[email protected]' }, function (err) {
|
|
assert.equal(err.message, '[[error:invalid-token]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should verify installation with no errors', function (done) {
|
|
var email = '[email protected]';
|
|
db.get('invitation:email:' + email, function (err, token) {
|
|
assert.ifError(err);
|
|
User.verifyInvitation({ token: token, email: '[email protected]' }, function (err) {
|
|
assert.ifError(err);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should error with invalid username', function (done) {
|
|
User.deleteInvitation('doesnotexist', '[email protected]', function (err) {
|
|
assert.equal(err.message, '[[error:invalid-username]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should delete invitation', function (done) {
|
|
var socketUser = require('../src/socket.io/user');
|
|
socketUser.deleteInvitation({ uid: adminUid }, { invitedBy: 'inviter', email: '[email protected]' }, function (err) {
|
|
assert.ifError(err);
|
|
db.isSetMember('invitation:uid:' + inviterUid, '[email protected]', function (err, isMember) {
|
|
assert.ifError(err);
|
|
assert.equal(isMember, false);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should delete invitation key', function (done) {
|
|
User.deleteInvitationKey('[email protected]', function (err) {
|
|
assert.ifError(err);
|
|
db.isSetMember('invitation:uid:' + inviterUid, '[email protected]', function (err, isMember) {
|
|
assert.ifError(err);
|
|
assert.equal(isMember, false);
|
|
db.isSetMember('invitation:uids', inviterUid, function (err, isMember) {
|
|
assert.ifError(err);
|
|
assert.equal(isMember, false);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should escape email', function (done) {
|
|
socketUser.invite({ uid: inviterUid }, '<script>alert("ok");</script>', function (err) {
|
|
assert.ifError(err);
|
|
User.getInvites(inviterUid, function (err, data) {
|
|
assert.ifError(err);
|
|
assert.equal(data[0], '<script>alert("ok");</script>');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('email confirm', function () {
|
|
it('should error with invalid code', function (done) {
|
|
User.email.confirm('asdasda', function (err) {
|
|
assert.equal(err.message, '[[error:invalid-data]]');
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should confirm email of user', async function () {
|
|
const email = '[email protected]';
|
|
const uid = await User.create({
|
|
username: 'confirme',
|
|
email: email,
|
|
});
|
|
|
|
const code = await User.email.sendValidationEmail(uid, email);
|
|
const unverified = await groups.isMember(uid, 'unverified-users');
|
|
assert.strictEqual(unverified, true);
|
|
await User.email.confirm(code);
|
|
const [confirmed, isVerified] = await Promise.all([
|
|
db.getObjectField('user:' + uid, 'email:confirmed'),
|
|
groups.isMember(uid, 'verified-users', uid),
|
|
]);
|
|
assert.strictEqual(parseInt(confirmed, 10), 1);
|
|
assert.strictEqual(isVerified, true);
|
|
});
|
|
});
|
|
|
|
describe('user jobs', function () {
|
|
it('should start user jobs', function (done) {
|
|
User.startJobs();
|
|
done();
|
|
});
|
|
|
|
it('should stop user jobs', function (done) {
|
|
User.stopJobs();
|
|
done();
|
|
});
|
|
|
|
it('should send digest', function (done) {
|
|
db.sortedSetAdd('digest:day:uids', [Date.now(), Date.now()], [1, 2], function (err) {
|
|
assert.ifError(err);
|
|
User.digest.execute({ interval: 'day' }, function (err) {
|
|
assert.ifError(err);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('hideEmail/hideFullname', function () {
|
|
var uid;
|
|
after(function (done) {
|
|
meta.config.hideEmail = 0;
|
|
meta.config.hideFullname = 0;
|
|
done();
|
|
});
|
|
|
|
it('should hide email and fullname', function (done) {
|
|
meta.config.hideEmail = 1;
|
|
meta.config.hideFullname = 1;
|
|
|
|
User.create({
|
|
username: 'hiddenemail',
|
|
email: '[email protected]',
|
|
fullname: 'baris soner usakli',
|
|
}, function (err, _uid) {
|
|
uid = _uid;
|
|
assert.ifError(err);
|
|
request(nconf.get('url') + '/api/user/hiddenemail', { json: true }, function (err, res, body) {
|
|
assert.ifError(err);
|
|
assert.equal(body.fullname, '');
|
|
assert.equal(body.email, '');
|
|
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should hide fullname in topic list and topic', function (done) {
|
|
Topics.post({
|
|
uid: uid,
|
|
title: 'Topic hidden',
|
|
content: 'lorem ipsum',
|
|
cid: testCid,
|
|
}, function (err) {
|
|
assert.ifError(err);
|
|
request(nconf.get('url') + '/api/recent', { json: true }, function (err, res, body) {
|
|
assert.ifError(err);
|
|
assert(!body.topics[0].user.hasOwnProperty('fullname'));
|
|
request(nconf.get('url') + '/api/topic/' + body.topics[0].slug, { json: true }, function (err, res, body) {
|
|
assert.ifError(err);
|
|
assert(!body.posts[0].user.hasOwnProperty('fullname'));
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('user blocking methods', function (done) {
|
|
let blockeeUid;
|
|
before(function (done) {
|
|
User.create({
|
|
username: 'blockee',
|
|
email: '[email protected]',
|
|
fullname: 'Block me',
|
|
}, function (err, uid) {
|
|
blockeeUid = uid;
|
|
done(err);
|
|
});
|
|
});
|
|
|
|
describe('.toggle()', function () {
|
|
it('should toggle block', function (done) {
|
|
socketUser.toggleBlock({ uid: 1 }, { blockerUid: 1, blockeeUid: blockeeUid }, function (err) {
|
|
assert.ifError(err);
|
|
User.blocks.is(blockeeUid, 1, function (err, blocked) {
|
|
assert.ifError(err);
|
|
assert(blocked);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should toggle block', function (done) {
|
|
socketUser.toggleBlock({ uid: 1 }, { blockerUid: 1, blockeeUid: blockeeUid }, function (err) {
|
|
assert.ifError(err);
|
|
User.blocks.is(blockeeUid, 1, function (err, blocked) {
|
|
assert.ifError(err);
|
|
assert(!blocked);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('.add()', function () {
|
|
it('should block a uid', function (done) {
|
|
User.blocks.add(blockeeUid, 1, function (err) {
|
|
assert.ifError(err);
|
|
User.blocks.list(1, function (err, blocked_uids) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(Array.isArray(blocked_uids), true);
|
|
assert.strictEqual(blocked_uids.length, 1);
|
|
assert.strictEqual(blocked_uids.includes(blockeeUid), true);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should automatically increment corresponding user field', function (done) {
|
|
db.getObjectField('user:1', 'blocksCount', function (err, count) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(parseInt(count, 10), 1);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error if you try to block the same uid again', function (done) {
|
|
User.blocks.add(blockeeUid, 1, function (err) {
|
|
assert.equal(err.message, '[[error:already-blocked]]');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('.remove()', function () {
|
|
it('should unblock a uid', function (done) {
|
|
User.blocks.remove(blockeeUid, 1, function (err) {
|
|
assert.ifError(err);
|
|
User.blocks.list(1, function (err, blocked_uids) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(Array.isArray(blocked_uids), true);
|
|
assert.strictEqual(blocked_uids.length, 0);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should automatically decrement corresponding user field', function (done) {
|
|
db.getObjectField('user:1', 'blocksCount', function (err, count) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(parseInt(count, 10), 0);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should error if you try to unblock the same uid again', function (done) {
|
|
User.blocks.remove(blockeeUid, 1, function (err) {
|
|
assert.equal(err.message, '[[error:already-unblocked]]');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('.is()', function () {
|
|
before(function (done) {
|
|
User.blocks.add(blockeeUid, 1, done);
|
|
});
|
|
|
|
it('should return a Boolean with blocked status for the queried uid', function (done) {
|
|
User.blocks.is(blockeeUid, 1, function (err, blocked) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(blocked, true);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('.list()', function () {
|
|
it('should return a list of blocked uids', function (done) {
|
|
User.blocks.list(1, function (err, blocked_uids) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(Array.isArray(blocked_uids), true);
|
|
assert.strictEqual(blocked_uids.length, 1);
|
|
assert.strictEqual(blocked_uids.includes(blockeeUid), true);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('.filter()', function () {
|
|
it('should remove entries by blocked uids and return filtered set', function (done) {
|
|
User.blocks.filter(1, [{
|
|
foo: 'foo',
|
|
uid: blockeeUid,
|
|
}, {
|
|
foo: 'bar',
|
|
uid: 1,
|
|
}, {
|
|
foo: 'baz',
|
|
uid: blockeeUid,
|
|
}], function (err, filtered) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(Array.isArray(filtered), true);
|
|
assert.strictEqual(filtered.length, 1);
|
|
assert.equal(filtered[0].uid, 1);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should allow property argument to be passed in to customise checked property', function (done) {
|
|
User.blocks.filter(1, 'fromuid', [{
|
|
foo: 'foo',
|
|
fromuid: blockeeUid,
|
|
}, {
|
|
foo: 'bar',
|
|
fromuid: 1,
|
|
}, {
|
|
foo: 'baz',
|
|
fromuid: blockeeUid,
|
|
}], function (err, filtered) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(Array.isArray(filtered), true);
|
|
assert.strictEqual(filtered.length, 1);
|
|
assert.equal(filtered[0].fromuid, 1);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should not process invalid sets', function (done) {
|
|
User.blocks.filter(1, [{ foo: 'foo' }, { foo: 'bar' }, { foo: 'baz' }], function (err, filtered) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(Array.isArray(filtered), true);
|
|
assert.strictEqual(filtered.length, 3);
|
|
filtered.forEach(function (obj) {
|
|
assert.strictEqual(obj.hasOwnProperty('foo'), true);
|
|
});
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should process plain sets that just contain uids', function (done) {
|
|
User.blocks.filter(1, [1, blockeeUid], function (err, filtered) {
|
|
assert.ifError(err);
|
|
assert.strictEqual(filtered.length, 1);
|
|
assert.strictEqual(filtered[0], 1);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should filter uids that are blocking targetUid', function (done) {
|
|
User.blocks.filterUids(blockeeUid, [1, 2], function (err, filtered) {
|
|
assert.ifError(err);
|
|
assert.deepEqual(filtered, [2]);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('status/online', function () {
|
|
it('should return offline if user is guest', function (done) {
|
|
var status = User.getStatus({ uid: 0 });
|
|
assert.strictEqual(status, 'offline');
|
|
done();
|
|
});
|
|
|
|
it('should return offline if user is guest', async function () {
|
|
assert.strictEqual(await User.isOnline(0), false);
|
|
});
|
|
|
|
it('should return true', async function () {
|
|
assert.strictEqual(await User.isOnline(testUid), true);
|
|
});
|
|
});
|
|
|
|
describe('isPrivilegedOrSelf', function () {
|
|
it('should return not error if self', function (done) {
|
|
User.isPrivilegedOrSelf(1, 1, function (err) {
|
|
assert.ifError(err);
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should not error if privileged', function (done) {
|
|
User.create({ username: 'theadmin' }, function (err, uid) {
|
|
assert.ifError(err);
|
|
groups.join('administrators', uid, function (err) {
|
|
assert.ifError(err);
|
|
User.isPrivilegedOrSelf(uid, 2, function (err) {
|
|
assert.ifError(err);
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should error if not privileged', function (done) {
|
|
User.isPrivilegedOrSelf(0, 1, function (err) {
|
|
assert.equal(err.message, '[[error:no-privileges]]');
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
|
|
it('should get admins and mods', function (done) {
|
|
User.getAdminsandGlobalMods(function (err, data) {
|
|
assert.ifError(err);
|
|
assert(Array.isArray(data));
|
|
done();
|
|
});
|
|
});
|
|
|
|
it('should allow user to login even if password is weak', function (done) {
|
|
User.create({ username: 'weakpwd', password: '123456' }, function (err) {
|
|
assert.ifError(err);
|
|
const oldValue = meta.config.minimumPasswordStrength;
|
|
meta.config.minimumPasswordStrength = 3;
|
|
helpers.loginUser('weakpwd', '123456', function (err, jar, csrfs_token) {
|
|
assert.ifError(err);
|
|
meta.config.minimumPasswordStrength = oldValue;
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
});
|