From 8db98b5cf4f8ce1bc30f49bc98edda73a55c9dc8 Mon Sep 17 00:00:00 2001 From: Julian Lam Date: Fri, 27 Apr 2018 12:51:04 -0400 Subject: [PATCH] additional UCP integration for #6463 --- public/language/en-GB/user.json | 1 + public/src/client/account/blocks.js | 13 + src/controllers/accounts/blocks.js | 31 +- src/controllers/accounts/helpers.js | 1 + src/user/blocks.js | 14 +- test/user.js | 2994 ++++++++++++++------------- 6 files changed, 1570 insertions(+), 1484 deletions(-) create mode 100644 public/src/client/account/blocks.js diff --git a/public/language/en-GB/user.json b/public/language/en-GB/user.json index 7d6bbb63a1..3ff3c47cac 100644 --- a/public/language/en-GB/user.json +++ b/public/language/en-GB/user.json @@ -96,6 +96,7 @@ "has_no_upvoted_posts": "This user hasn't upvoted any posts yet.", "has_no_downvoted_posts": "This user hasn't downvoted any posts yet.", "has_no_voted_posts": "This user has no voted posts", + "has_no_blocks": "You have blocked no users.", "email_hidden": "Email Hidden", "hidden": "hidden", diff --git a/public/src/client/account/blocks.js b/public/src/client/account/blocks.js new file mode 100644 index 0000000000..4e6230e00e --- /dev/null +++ b/public/src/client/account/blocks.js @@ -0,0 +1,13 @@ +'use strict'; + +define('forum/account/blocks', ['forum/account/header'], function (header) { + var Blocks = {}; + + Blocks.init = function () { + header.init(); + + console.log('derpp'); + }; + + return Blocks; +}); diff --git a/src/controllers/accounts/blocks.js b/src/controllers/accounts/blocks.js index c97ecca6ca..f37590ca53 100644 --- a/src/controllers/accounts/blocks.js +++ b/src/controllers/accounts/blocks.js @@ -2,13 +2,22 @@ var async = require('async'); +var helpers = require('../helpers'); var accountHelpers = require('./helpers'); +var pagination = require('../../pagination'); +var user = require('../../user'); +var plugins = require('../../plugins'); var blocksController = {}; blocksController.getBlocks = function (req, res, callback) { var userData; + var page = parseInt(req.query.page, 10) || 1; + var resultsPerPage = 50; + var start = Math.max(0, page - 1) * resultsPerPage; + var stop = start + resultsPerPage - 1; + async.waterfall([ function (next) { accountHelpers.getUserDataByUserSlug(req.params.userslug, req.uid, next); @@ -19,13 +28,31 @@ blocksController.getBlocks = function (req, res, callback) { return callback(); } - next(); + user.blocks.list(res.locals.uid, next); + }, + function (uids, next) { + plugins.fireHook('filter:user.getBlocks', { + uids: uids, + uid: res.locals.uid, + start: start, + stop: stop, + }, next); }, - ], function (err) { + function (data, next) { + user.getUsers(data.uids, res.locals.uid, next); + }, + ], function (err, users) { if (err) { return callback(err); } + userData.users = users; + userData.title = '[[pages:account/blocks, ' + userData.username + ']]'; + var count = userData.blocksCount; + var pageCount = Math.ceil(count / resultsPerPage); + userData.pagination = pagination.create(page, pageCount); + userData.breadcrumbs = helpers.buildBreadcrumbs([{ text: userData.username, url: '/user/' + userData.userslug }, { text: '[[user:blocks]]' }]); + res.render('account/blocks', userData); }); }; diff --git a/src/controllers/accounts/helpers.js b/src/controllers/accounts/helpers.js index bc43213de0..955432c652 100644 --- a/src/controllers/accounts/helpers.js +++ b/src/controllers/accounts/helpers.js @@ -157,6 +157,7 @@ helpers.getUserDataByUserSlug = function (userslug, callerUID, callback) { userData.websiteName = userData.website.replace(validator.escape('http://'), '').replace(validator.escape('https://'), ''); userData.followingCount = parseInt(userData.followingCount, 10) || 0; userData.followerCount = parseInt(userData.followerCount, 10) || 0; + userData.blocksCount = parseInt(userData.blocksCount, 10) || 0; userData.email = validator.escape(String(userData.email || '')); userData.fullname = validator.escape(String(userData.fullname || '')); diff --git a/src/user/blocks.js b/src/user/blocks.js index 093ea518f3..1d3720c20e 100644 --- a/src/user/blocks.js +++ b/src/user/blocks.js @@ -37,8 +37,10 @@ module.exports = function (User) { User.blocks.add = function (targetUid, uid, callback) { async.waterfall([ + async.apply(this.stateCheck, true, targetUid, uid), async.apply(db.sortedSetAdd.bind(db), 'uid:' + uid + ':blocked_uids', Date.now(), targetUid), - function (next) { + async.apply(User.incrementUserFieldBy, uid, 'blocksCount', 1), + function (_blank, next) { User.blocks._cache.del(uid); setImmediate(next); }, @@ -48,8 +50,10 @@ module.exports = function (User) { User.blocks.remove = function (targetUid, uid, callback) { async.waterfall([ + async.apply(this.stateCheck, false, targetUid, uid), async.apply(db.sortedSetRemove.bind(db), 'uid:' + uid + ':blocked_uids', targetUid), - function (next) { + async.apply(User.decrementUserFieldBy, uid, 'blocksCount', 1), + function (_blank, next) { User.blocks._cache.del(uid); setImmediate(next); }, @@ -57,6 +61,12 @@ module.exports = function (User) { ], callback); }; + User.blocks.stateCheck = function (block, targetUid, uid, callback) { + User.blocks.is(targetUid, uid, function (err, is) { + callback(err || (is === block ? new Error('[[error:already-' + (block ? 'blocked' : 'unblocked') + ']]') : null)); + }); + }; + User.blocks.filter = function (uid, property, set, callback) { // Given whatever is passed in, iterates through it, and removes entries made by blocked uids // property is optional diff --git a/test/user.js b/test/user.js index bf91604514..a7d266138a 100644 --- a/test/user.js +++ b/test/user.js @@ -272,1501 +272,1528 @@ describe('User', function () { }); }); - describe('.search()', function () { - var uid; - 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) { - meta.config.allowGuestUserSearching = 0; - socketUser.search({ uid: 0 }, { query: 'john' }, function (err) { - assert.equal(err.message, '[[error:not-logged-in]]'); - meta.config.allowGuestUserSearching = 1; - done(); - }); - }); - - it('should error with invalid data', function (done) { - socketUser.search({ uid: testUid }, null, function (err) { - assert.equal(err.message, '[[error:invalid-data]]'); - done(); - }); - }); - - it('should search users by ip', function (done) { - User.create({ username: 'ipsearch' }, function (err, uid) { - assert.ifError(err); - db.sortedSetAdd('ip:1.1.1.1:uid', [1, 1], [testUid, uid], function (err) { - assert.ifError(err); - socketUser.search({ uid: testUid }, { query: '1.1.1.1', searchBy: 'ip' }, function (err, data) { - assert.ifError(err); - assert(Array.isArray(data.users)); - assert.equal(data.users.length, 2); - done(); - }); - }); - }); - }); - - it('should search users by ip', 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 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: testUid }, { - query: 'ipsearch', - onlineOnly: true, - bannedOnly: true, - flaggedOnly: true, - }, 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: 'delete@me.com' }, function (err, newUid) { - assert.ifError(err); - uid = newUid; - done(); - }); - }); - - it('should delete a user account', function (done) { - User.delete(1, uid, function (err) { - assert.ifError(err); - User.existsBySlug('usertodelete', function (err, exists) { - assert.ifError(err); - assert.equal(exists, false); - done(); - }); - }); - }); - }); - - describe('passwordReset', function () { - var uid; - var code; - before(function (done) { - User.create({ username: 'resetuser', password: '123456', email: 'reset@me.com' }, function (err, newUid) { - assert.ifError(err); - uid = newUid; - done(); - }); - }); - - it('.generate() should generate a new reset code', function (done) { - User.reset.generate(uid, function (err, _code) { - assert.ifError(err); - assert(_code); - - code = _code; - done(); - }); - }); - - it('.validate() should ensure that this new code is valid', function (done) { - User.reset.validate(code, function (err, valid) { - assert.ifError(err); - assert.strictEqual(valid, true); - done(); - }); - }); - - it('.validate() should correctly identify an invalid code', function (done) { - User.reset.validate(code + 'abcdef', function (err, valid) { - assert.ifError(err); - assert.strictEqual(valid, false); - done(); - }); - }); - - it('.send() should create a new reset code and reset password', function (done) { - User.reset.send('reset@me.com', function (err) { - 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); - - db.getObject('user:' + uid, function (err, userData) { - assert.ifError(err); - Password.compare('newpassword', userData.password, function (err, match) { - assert.ifError(err); - assert(match); - assert.equal(parseInt(userData['email:confirmed'], 10), 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: 'sorry@forgot.com', 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('john@example.com', function (err, uid) { - assert.ifError(err); - assert.equal(parseInt(uid, 10), parseInt(testUid, 10)); - done(); - }); - }); - - it('should return uid from username', function (done) { - User.getUidByUsername('John Smith', function (err, uid) { - assert.ifError(err); - assert.equal(parseInt(uid, 10), parseInt(testUid, 10)); - done(); - }); - }); - - it('should return uid from userslug', function (done) { - User.getUidByUserslug('john-smith', function (err, uid) { - assert.ifError(err); - assert.equal(parseInt(uid, 10), parseInt(testUid, 10)); - done(); - }); - }); - - 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.equal(userData.postcount, '123'); - 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(); - }); - }); - }); - - describe('not logged in', function () { - it('should return error if not logged in', function (done) { - socketUser.updateProfile({ uid: 0 }, {}, 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: 'update@me.com', 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) { - var data = { - uid: uid, - username: 'updatedUserName', - email: 'updatedEmail@me.com', - fullname: 'updatedFullname', - website: 'http://nodebb.org', - location: 'izmir', - groupTitle: 'testGroup', - birthday: '01/01/1980', - signature: 'nodebb is good', - }; - socketUser.updateProfile({ uid: uid }, data, function (err, result) { - assert.ifError(err); - - assert.equal(result.username, 'updatedUserName'); - assert.equal(result.userslug, 'updatedusername'); - assert.equal(result.email, 'updatedEmail@me.com'); - - db.getObject('user:' + uid, function (err, userData) { - assert.ifError(err); - Object.keys(data).forEach(function (key) { - assert.equal(data[key], userData[key]); - }); - done(); - }); - }); - }); - - it('should change a user\'s password', function (done) { - 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', function (err, correct) { - assert.ifError(err); - assert(correct); - done(); - }); - }); - }); - }); - - 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 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(data[0].startsWith('updatedAgain')); - assert(data[1].startsWith('updatedUserName')); - done(); - }); - }); - }); - - it('should change email', function (done) { - socketUser.changeUsernameEmail({ uid: uid }, { uid: uid, email: 'updatedAgain@me.com', password: '123456' }, function (err) { - assert.ifError(err); - db.getObjectField('user:' + uid, 'email', function (err, email) { - assert.ifError(err); - assert.equal(email, 'updatedAgain@me.com'); - done(); - }); - }); - }); - - it('should update cover image', function (done) { - var imageData = ''; - var position = '50.0301% 19.2464%'; - 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 upload profile picture', function (done) { - helpers.copyFile( - path.join(nconf.get('base_dir'), 'test/files/test.png'), - path.join(nconf.get('base_dir'), 'test/files/test_copy.png'), - function (err) { - assert.ifError(err); - var picture = { - path: path.join(nconf.get('base_dir'), 'test/files/test_copy.png'), - size: 7189, - name: 'test_copy.png', - type: 'image/png', - }; - User.uploadCroppedPicture({ - uid: uid, - file: picture, - }, function (err, uploadedPicture) { - assert.ifError(err); - assert.equal(uploadedPicture.url, '/assets/uploads/profile/' + uid + '-profileavatar.png'); - assert.equal(uploadedPicture.path, path.join(nconf.get('upload_path'), 'profile', uid + '-profileavatar.png')); - 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.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]]'); - done(); - }); - }); - - it('should return error if profile image is too big', function (done) { - meta.config.allowProfileImageUploads = 1; - var picture = { - path: path.join(nconf.get('base_dir'), 'test/files/test.png'), - size: 265000, - name: 'test.png', - type: 'image/png', - }; - - User.uploadCroppedPicture({ - uid: uid, - file: picture, - }, function (err) { - assert.equal(err.message, '[[error:file-too-big, 256]]'); - done(); - }); - }); - - it('should return error if profile image has no mime type', function (done) { - var picture = { - path: path.join(nconf.get('base_dir'), 'test/files/test.png'), - size: 7189, - name: 'test', - }; - User.uploadCroppedPicture({ - uid: uid, - file: picture, - }, function (err) { - assert.equal(err.message, '[[error:invalid-image]]'); - done(); - }); - }); - - describe('user.uploadCroppedPicture', function () { - var goodImage = ''; - var badImage = 'data:audio/mp3;base64,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'; - 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', function (done) { - var now = Date.now(); - db.sortedSetAdd('user:' + testUid + ':usernames', now, 'derp:' + now, function (err) { - assert.ifError(err); - User.getHistory('user:' + testUid + ':usernames', function (err, data) { - assert.ifError(err); - assert.equal(data[0].value, 'derp'); - assert.equal(data[0].timestamp, now); - done(); - }); - }); - }); - - it('should return the correct ban reason', function (done) { - async.series([ - function (next) { - User.ban(testUid, 0, '', function (err) { - assert.ifError(err); - next(err); - }); - }, - function (next) { - User.getModerationHistory(testUid, function (err, data) { - assert.ifError(err); - assert.equal(data.bans.length, 1, 'one ban'); - assert.equal(data.bans[0].reason, '[[user:info.banned-no-reason]]', 'no ban reason'); - - next(err); - }); - }, - ], function (err) { - assert.ifError(err); - User.unban(testUid, function (err) { - assert.ifError(err); - done(); - }); - }); - }); - - it('should ban user permanently', function (done) { - User.ban(testUid, function (err) { - assert.ifError(err); - User.isBanned(testUid, function (err, isBanned) { - assert.ifError(err); - assert.equal(isBanned, true); - User.unban(testUid, done); - }); - }); - }); - - it('should ban user temporarily', function (done) { - User.ban(testUid, Date.now() + 2000, function (err) { - assert.ifError(err); - - User.isBanned(testUid, function (err, isBanned) { - assert.ifError(err); - assert.equal(isBanned, true); - setTimeout(function () { - User.isBanned(testUid, function (err, isBanned) { - assert.ifError(err); - assert.equal(isBanned, false); - User.unban(testUid, done); - }); - }, 3000); - }); - }); - }); - - it('should error if until is NaN', function (done) { - User.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: 'test@example.com' }, next); - }, - function (_uid, next) { - uid = _uid; - User.updateDigestSetting(uid, 'day', next); - }, - function (next) { - User.setSetting(uid, 'dailyDigestFreq', 'day', 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('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 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: 'john@example.com' }, 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: 'does@not.exist' }, 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 }, 'john@example.com', 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 }, 'doestnot@exist.com', 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: 1, - settings: { - bootswatchSkin: 'default', - homePageRoute: 'none', - homePageCustom: '', - openOutgoingLinksInNewTab: 0, - scrollToMyPost: 1, - delayImageLoading: 1, - userLang: 'en-GB', - usePagination: 1, - topicsPerPage: '10', - postsPerPage: '5', - showemail: 1, - showfullname: 1, - restrictChat: 0, - followTopicsOnCreate: 1, - followTopicsOnReply: 1, - notificationSound: '', - incomingChatSound: '', - outgoingChatSound: '', - }, - }; - 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 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: 'alert("ok")