'use strict'; var assert = require('assert'); var async = require('async'); var path = require('path'); var nconf = require('nconf'); var db = require('./mocks/databasemock'); var helpers = require('./helpers'); var Groups = require('../src/groups'); var User = require('../src/user'); var socketGroups = require('../src/socket.io/groups'); var meta = require('../src/meta'); describe('Groups', function () { var adminUid; var testUid; before(function (done) { async.series([ function (next) { // Create a group to play around with Groups.create({ name: 'Test', description: 'Foobar!', }, next); }, function (next) { Groups.create({ name: 'PrivateNoJoin', description: 'Private group', private: 1, disableJoinRequests: 1, }, next); }, function (next) { Groups.create({ name: 'PrivateCanJoin', description: 'Private group', private: 1, disableJoinRequests: 0, }, next); }, async () => { await Groups.create({ name: 'PrivateNoLeave', description: 'Private group', private: 1, disableLeave: 1, }); }, function (next) { // Create a new user User.create({ username: 'testuser', email: 'b@c.com', }, next); }, function (next) { User.create({ username: 'admin', email: 'admin@admin.com', password: '123456', }, next); }, function (next) { // Also create a hidden group Groups.join('Hidden', 'Test', next); }, function (next) { // create another group that starts with test for search/sort Groups.create({ name: 'Test2', description: 'Foobar!' }, next); }, ], function (err, results) { assert.ifError(err); testUid = results[4]; adminUid = results[5]; Groups.join('administrators', adminUid, done); }); }); describe('.list()', function () { it('should list the groups present', function (done) { Groups.getGroupsFromSet('groups:visible:createtime', 0, -1, function (err, groups) { assert.ifError(err); assert.equal(groups.length, 5); done(); }); }); }); describe('.get()', function () { before(function (done) { Groups.join('Test', testUid, done); }); it('with no options, should show group information', function (done) { Groups.get('Test', {}, function (err, groupObj) { assert.ifError(err); assert.equal(typeof groupObj, 'object'); assert(Array.isArray(groupObj.members)); assert.strictEqual(groupObj.name, 'Test'); assert.strictEqual(groupObj.description, 'Foobar!'); assert.strictEqual(groupObj.memberCount, 1); assert.equal(typeof groupObj.members[0], 'object'); done(); }); }); it('should return null if group does not exist', function (done) { Groups.get('doesnotexist', {}, function (err, groupObj) { assert.ifError(err); assert.strictEqual(groupObj, null); done(); }); }); }); describe('.search()', function () { var socketGroups = require('../src/socket.io/groups'); it('should return empty array if query is falsy', function (done) { Groups.search(null, {}, function (err, groups) { assert.ifError(err); assert.equal(0, groups.length); done(); }); }); it('should return the groups when search query is empty', function (done) { socketGroups.search({ uid: adminUid }, { query: '' }, function (err, groups) { assert.ifError(err); assert.equal(5, groups.length); done(); }); }); it('should return the "Test" group when searched for', function (done) { socketGroups.search({ uid: adminUid }, { query: 'test' }, function (err, groups) { assert.ifError(err); assert.equal(2, groups.length); assert.strictEqual('Test', groups[0].name); done(); }); }); it('should return the "Test" group when searched for and sort by member count', function (done) { Groups.search('test', { filterHidden: true, sort: 'count' }, function (err, groups) { assert.ifError(err); assert.equal(2, groups.length); assert.strictEqual('Test', groups[0].name); done(); }); }); it('should return the "Test" group when searched for and sort by creation time', function (done) { Groups.search('test', { filterHidden: true, sort: 'date' }, function (err, groups) { assert.ifError(err); assert.equal(2, groups.length); assert.strictEqual('Test', groups[1].name); done(); }); }); it('should return all users if no query', function (done) { function createAndJoinGroup(username, email, callback) { async.waterfall([ function (next) { User.create({ username: username, email: email }, next); }, function (uid, next) { Groups.join('Test', uid, next); }, ], callback); } async.series([ function (next) { createAndJoinGroup('newuser', 'newuser@b.com', next); }, function (next) { createAndJoinGroup('bob', 'bob@b.com', next); }, ], function (err) { assert.ifError(err); socketGroups.searchMembers({ uid: adminUid }, { groupName: 'Test', query: '' }, function (err, data) { assert.ifError(err); assert.equal(data.users.length, 3); done(); }); }); }); it('should search group members', function (done) { socketGroups.searchMembers({ uid: adminUid }, { groupName: 'Test', query: 'test' }, function (err, data) { assert.ifError(err); assert.strictEqual('testuser', data.users[0].username); done(); }); }); it('should not return hidden groups', async function () { await Groups.create({ name: 'hiddenGroup', hidden: '1', }); const result = await socketGroups.search({ uid: testUid }, { query: 'hiddenGroup' }); assert.equal(result.length, 0); }); }); describe('.isMember()', function () { it('should return boolean true when a user is in a group', function (done) { Groups.isMember(1, 'Test', function (err, isMember) { assert.ifError(err); assert.strictEqual(isMember, true); done(); }); }); it('should return boolean false when a user is not in a group', function (done) { Groups.isMember(2, 'Test', function (err, isMember) { assert.ifError(err); assert.strictEqual(isMember, false); done(); }); }); it('should return true for uid 0 and guests group', function (done) { Groups.isMembers([1, 0], 'guests', function (err, isMembers) { assert.ifError(err); assert.deepStrictEqual(isMembers, [false, true]); done(); }); }); it('should return true for uid 0 and guests group', function (done) { Groups.isMemberOfGroups(0, ['guests', 'registered-users'], function (err, isMembers) { assert.ifError(err); assert.deepStrictEqual(isMembers, [true, false]); done(); }); }); }); describe('.isMemberOfGroupList', function () { it('should report that a user is part of a groupList, if they are', function (done) { Groups.isMemberOfGroupList(1, 'Hidden', function (err, isMember) { assert.ifError(err); assert.strictEqual(isMember, true); done(); }); }); it('should report that a user is not part of a groupList, if they are not', function (done) { Groups.isMemberOfGroupList(2, 'Hidden', function (err, isMember) { assert.ifError(err); assert.strictEqual(isMember, false); done(); }); }); }); describe('.exists()', function () { it('should verify that the test group exists', function (done) { Groups.exists('Test', function (err, exists) { assert.ifError(err); assert.strictEqual(exists, true); done(); }); }); it('should verify that a fake group does not exist', function (done) { Groups.exists('Derp', function (err, exists) { assert.ifError(err); assert.strictEqual(exists, false); done(); }); }); it('should check if group exists using an array', function (done) { Groups.exists(['Test', 'Derp'], function (err, groupsExists) { assert.ifError(err); assert.strictEqual(groupsExists[0], true); assert.strictEqual(groupsExists[1], false); done(); }); }); }); describe('.create()', function () { it('should create another group', function (done) { Groups.create({ name: 'foo', description: 'bar', }, function (err) { assert.ifError(err); Groups.get('foo', {}, done); }); }); it('should create a hidden group if hidden is 1', function (done) { Groups.create({ name: 'hidden group', hidden: '1', }, function (err) { assert.ifError(err); db.isSortedSetMember('groups:visible:memberCount', 'visible group', function (err, isMember) { assert.ifError(err); assert(!isMember); done(); }); }); }); it('should create a visible group if hidden is 0', function (done) { Groups.create({ name: 'visible group', hidden: '0', }, function (err) { assert.ifError(err); db.isSortedSetMember('groups:visible:memberCount', 'visible group', function (err, isMember) { assert.ifError(err); assert(isMember); done(); }); }); }); it('should create a visible group if hidden is not passed in', function (done) { Groups.create({ name: 'visible group 2', }, function (err) { assert.ifError(err); db.isSortedSetMember('groups:visible:memberCount', 'visible group 2', function (err, isMember) { assert.ifError(err); assert(isMember); done(); }); }); }); it('should fail to create group with duplicate group name', function (done) { Groups.create({ name: 'foo' }, function (err) { assert(err); assert.equal(err.message, '[[error:group-already-exists]]'); done(); }); }); it('should fail to create group if slug is empty', function (done) { Groups.create({ name: '>>>>' }, function (err) { assert.equal(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should fail if group name is invalid', function (done) { Groups.create({ name: 'not/valid' }, function (err) { assert.equal(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should fail if group name is invalid', function (done) { Groups.create({ name: ['array/'] }, function (err) { assert.equal(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should fail if group name is invalid', function (done) { socketGroups.create({ uid: adminUid }, { name: ['test', 'administrators'] }, function (err) { assert.equal(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should not create a system group', function (done) { socketGroups.create({ uid: adminUid }, { name: 'mysystemgroup', system: true }, function (err) { assert.ifError(err); Groups.getGroupData('mysystemgroup', function (err, data) { assert.ifError(err); assert.strictEqual(data.system, 0); done(); }); }); }); it('should fail if group name is invalid', function (done) { Groups.create({ name: 'not:valid' }, function (err) { assert.equal(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should return falsy for userTitleEnabled', function (done) { Groups.create({ name: 'userTitleEnabledGroup' }, function (err) { assert.ifError(err); Groups.setGroupField('userTitleEnabledGroup', 'userTitleEnabled', 0, function (err) { assert.ifError(err); Groups.getGroupData('userTitleEnabledGroup', function (err, data) { assert.ifError(err); assert.strictEqual(data.userTitleEnabled, 0); done(); }); }); }); }); }); describe('.hide()', function () { it('should mark the group as hidden', function (done) { Groups.hide('foo', function (err) { assert.ifError(err); Groups.get('foo', {}, function (err, groupObj) { assert.ifError(err); assert.strictEqual(1, groupObj.hidden); done(); }); }); }); }); describe('.update()', function () { before(function (done) { Groups.create({ name: 'updateTestGroup', description: 'bar', system: 0, hidden: 0, }, done); }); it('should change an aspect of a group', function (done) { Groups.update('updateTestGroup', { description: 'baz', }, function (err) { assert.ifError(err); Groups.get('updateTestGroup', {}, function (err, groupObj) { assert.ifError(err); assert.strictEqual('baz', groupObj.description); done(); }); }); }); it('should rename a group if the name was updated', function (done) { Groups.update('updateTestGroup', { name: 'updateTestGroup?', }, function (err) { assert.ifError(err); Groups.get('updateTestGroup?', {}, function (err, groupObj) { assert.ifError(err); assert.strictEqual('updateTestGroup?', groupObj.name); assert.strictEqual('updatetestgroup', groupObj.slug); done(); }); }); }); it('should fail if system groups is being renamed', function (done) { Groups.update('administrators', { name: 'administrators_fail', }, function (err) { assert.equal(err.message, '[[error:not-allowed-to-rename-system-group]]'); done(); }); }); it('should fail to rename if group name is invalid', function (done) { socketGroups.update({ uid: adminUid }, { groupName: ['updateTestGroup?'], values: {} }, function (err) { assert.strictEqual(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should fail to rename if group name is too short', function (done) { socketGroups.update({ uid: adminUid }, { groupName: 'updateTestGroup?', values: { name: '' } }, function (err) { assert.strictEqual(err.message, '[[error:group-name-too-short]]'); done(); }); }); it('should fail to rename if group name is invalid', function (done) { socketGroups.update({ uid: adminUid }, { groupName: 'updateTestGroup?', values: { name: ['invalid'] } }, function (err) { assert.strictEqual(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should fail to rename if group name is invalid', function (done) { socketGroups.update({ uid: adminUid }, { groupName: 'updateTestGroup?', values: { name: 'cid:0:privileges:ban' } }, function (err) { assert.strictEqual(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should fail to rename if group name is too long', function (done) { socketGroups.update({ uid: adminUid }, { groupName: 'updateTestGroup?', values: { name: 'verylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstringverylongstring' } }, function (err) { assert.strictEqual(err.message, '[[error:group-name-too-long]]'); done(); }); }); it('should fail to rename if group name is invalid', function (done) { socketGroups.update({ uid: adminUid }, { groupName: 'updateTestGroup?', values: { name: 'test:test' } }, function (err) { assert.strictEqual(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should fail to rename if group name is invalid', function (done) { socketGroups.update({ uid: adminUid }, { groupName: 'updateTestGroup?', values: { name: 'another/test' } }, function (err) { assert.strictEqual(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should fail to rename if group name is invalid', function (done) { socketGroups.update({ uid: adminUid }, { groupName: 'updateTestGroup?', values: { name: '---' } }, function (err) { assert.strictEqual(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should fail to rename group to an existing group', function (done) { Groups.create({ name: 'group2', system: 0, hidden: 0, }, function (err) { assert.ifError(err); Groups.update('group2', { name: 'updateTestGroup?', }, function (err) { assert.equal(err.message, '[[error:group-already-exists]]'); done(); }); }); }); }); describe('.destroy()', function () { before(function (done) { Groups.join('foobar?', 1, done); }); it('should destroy a group', function (done) { Groups.destroy('foobar?', function (err) { assert.ifError(err); Groups.get('foobar?', {}, function (err, groupObj) { assert.ifError(err); assert.strictEqual(groupObj, null); done(); }); }); }); it('should also remove the members set', function (done) { db.exists('group:foo:members', function (err, exists) { assert.ifError(err); assert.strictEqual(false, exists); done(); }); }); it('should remove group from privilege groups', function (done) { const privileges = require('../src/privileges'); const cid = 1; const groupName = '1'; const uid = 1; async.waterfall([ function (next) { Groups.create({ name: groupName }, next); }, function (groupData, next) { privileges.categories.give(['groups:topics:create'], cid, groupName, next); }, function (next) { Groups.isMember(groupName, 'cid:1:privileges:groups:topics:create', next); }, function (isMember, next) { assert(isMember); Groups.destroy(groupName, next); }, function (next) { Groups.isMember(groupName, 'cid:1:privileges:groups:topics:create', next); }, function (isMember, next) { assert(!isMember); Groups.isMember(uid, 'registered-users', next); }, function (isMember, next) { assert(isMember); next(); }, ], done); }); }); describe('.join()', function () { before(function (done) { Groups.leave('Test', testUid, done); }); it('should add a user to a group', function (done) { Groups.join('Test', testUid, function (err) { assert.ifError(err); Groups.isMember(testUid, 'Test', function (err, isMember) { assert.ifError(err); assert.strictEqual(true, isMember); done(); }); }); }); it('should fail to add user to admin group', async function () { const oldValue = meta.config.allowPrivateGroups; try { meta.config.allowPrivateGroups = false; const newUid = await User.create({ username: 'newadmin' }); await socketGroups.join({ uid: newUid }, { groupName: ['test', 'administrators'], uid: newUid }, 1); const isMember = await Groups.isMember(newUid, 'administrators'); assert(!isMember); } catch (err) { assert.strictEqual(err.message, '[[error:invalid-group-name]]'); } meta.config.allowPrivateGroups = oldValue; }); it('should fail to add user to group if group name is invalid', function (done) { Groups.join(0, 1, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); Groups.join(null, 1, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); Groups.join(undefined, 1, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); }); }); it('should fail to add user to group if uid is invalid', function (done) { Groups.join('Test', 0, function (err) { assert.equal(err.message, '[[error:invalid-uid]]'); Groups.join('Test', null, function (err) { assert.equal(err.message, '[[error:invalid-uid]]'); Groups.join('Test', undefined, function (err) { assert.equal(err.message, '[[error:invalid-uid]]'); done(); }); }); }); }); it('should add user to multiple groups', function (done) { var groupNames = ['test-hidden1', 'Test', 'test-hidden2', 'empty group']; Groups.create({ name: 'empty group' }, function (err) { assert.ifError(err); Groups.join(groupNames, testUid, function (err) { assert.ifError(err); Groups.isMemberOfGroups(testUid, groupNames, function (err, isMembers) { assert.ifError(err); assert(isMembers.every(Boolean)); db.sortedSetScores('groups:visible:memberCount', groupNames, function (err, memberCounts) { assert.ifError(err); // hidden groups are not in "groups:visible:memberCount" so they are null assert.deepEqual(memberCounts, [null, 3, null, 1]); done(); }); }); }); }); }); it('should set group title when user joins the group', function (done) { var groupName = 'this will be title'; User.create({ username: 'needstitle' }, function (err, uid) { assert.ifError(err); Groups.create({ name: groupName }, function (err) { assert.ifError(err); Groups.join([groupName], uid, function (err) { assert.ifError(err); User.getUserData(uid, function (err, data) { assert.ifError(err); assert.equal(data.groupTitle, '["' + groupName + '"]'); assert.deepEqual(data.groupTitleArray, [groupName]); done(); }); }); }); }); }); }); describe('.leave()', function () { it('should remove a user from a group', function (done) { Groups.leave('Test', testUid, function (err) { assert.ifError(err); Groups.isMember(testUid, 'Test', function (err, isMember) { assert.ifError(err); assert.strictEqual(false, isMember); done(); }); }); }); }); describe('.leaveAllGroups()', function () { it('should remove a user from all groups', function (done) { Groups.leaveAllGroups(testUid, function (err) { assert.ifError(err); var groups = ['Test', 'Hidden']; async.every(groups, function (group, next) { Groups.isMember(testUid, group, function (err, isMember) { next(err, !isMember); }); }, function (err, result) { assert.ifError(err); assert(result); done(); }); }); }); }); describe('.show()', function () { it('should make a group visible', function (done) { Groups.show('Test', function (err) { assert.ifError(err); assert.equal(arguments.length, 1); db.isSortedSetMember('groups:visible:createtime', 'Test', function (err, isMember) { assert.ifError(err); assert.strictEqual(isMember, true); done(); }); }); }); }); describe('.hide()', function () { it('should make a group hidden', function (done) { Groups.hide('Test', function (err) { assert.ifError(err); assert.equal(arguments.length, 1); db.isSortedSetMember('groups:visible:createtime', 'Test', function (err, isMember) { assert.ifError(err); assert.strictEqual(isMember, false); done(); }); }); }); }); describe('socket methods', function () { it('should error if data is null', function (done) { socketGroups.before({ uid: 0 }, 'groups.join', null, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should not error if data is valid', function (done) { socketGroups.before({ uid: 0 }, 'groups.join', {}, function (err) { assert.ifError(err); done(); }); }); it('should return error if not logged in', function (done) { socketGroups.join({ uid: 0 }, {}, function (err) { assert.equal(err.message, '[[error:invalid-uid]]'); done(); }); }); it('should return error if group name is special', function (done) { socketGroups.join({ uid: adminUid }, { groupName: 'administrators' }, function (err) { assert.equal(err.message, '[[error:not-allowed]]'); done(); }); }); it('should error if group does not exist', function (done) { socketGroups.join({ uid: adminUid }, { groupName: 'doesnotexist' }, function (err) { assert.equal(err.message, '[[error:no-group]]'); done(); }); }); it('should join test group', function (done) { meta.config.allowPrivateGroups = 0; socketGroups.join({ uid: adminUid }, { groupName: 'Test' }, function (err) { assert.ifError(err); Groups.isMember(adminUid, 'Test', function (err, isMember) { assert.ifError(err); assert(isMember); done(); }); }); }); it('should error if not logged in', function (done) { socketGroups.leave({ uid: 0 }, {}, function (err) { assert.equal(err.message, '[[error:invalid-uid]]'); done(); }); }); it('should return error if group name is special', function (done) { socketGroups.leave({ uid: adminUid }, { groupName: 'administrators' }, function (err) { assert.equal(err.message, '[[error:cant-remove-self-as-admin]]'); done(); }); }); it('should leave test group', function (done) { socketGroups.leave({ uid: adminUid }, { groupName: 'Test' }, function (err) { assert.ifError(err); Groups.isMember('Test', adminUid, function (err, isMember) { assert.ifError(err); assert(!isMember); done(); }); }); }); it('should fail to join if group is private and join requests are disabled', function (done) { meta.config.allowPrivateGroups = 1; socketGroups.join({ uid: testUid }, { groupName: 'PrivateNoJoin' }, function (err) { assert.equal(err.message, '[[error:group-join-disabled]]'); done(); }); }); it('should fail to leave if group is private and leave is disabled', async () => { await socketGroups.join({ uid: testUid }, { groupName: 'PrivateNoLeave' }); try { await socketGroups.leave({ uid: testUid }, { groupName: 'PrivateNoLeave' }); } catch (err) { assert.equal(err.message, '[[error:group-leave-disabled]]'); } }); it('should join if user is admin', function (done) { socketGroups.join({ uid: adminUid }, { groupName: 'PrivateCanJoin' }, function (err) { assert.ifError(err); Groups.isMember(adminUid, 'PrivateCanJoin', function (err, isMember) { assert.ifError(err); assert(isMember); done(); }); }); }); it('should request membership for regular user', function (done) { socketGroups.join({ uid: testUid }, { groupName: 'PrivateCanJoin' }, function (err) { assert.ifError(err); Groups.isPending(testUid, 'PrivateCanJoin', function (err, isPending) { assert.ifError(err); assert(isPending); done(); }); }); }); it('should reject membership of user', function (done) { socketGroups.reject({ uid: adminUid }, { groupName: 'PrivateCanJoin', toUid: testUid }, function (err) { assert.ifError(err); Groups.isInvited(testUid, 'PrivateCanJoin', function (err, invited) { assert.ifError(err); assert.equal(invited, false); done(); }); }); }); it('should error if not owner or admin', function (done) { socketGroups.accept({ uid: 0 }, { groupName: 'PrivateCanJoin', toUid: testUid }, function (err) { assert.equal(err.message, '[[error:no-privileges]]'); done(); }); }); it('should accept membership of user', function (done) { socketGroups.join({ uid: testUid }, { groupName: 'PrivateCanJoin' }, function (err) { assert.ifError(err); socketGroups.accept({ uid: adminUid }, { groupName: 'PrivateCanJoin', toUid: testUid }, function (err) { assert.ifError(err); Groups.isMember(testUid, 'PrivateCanJoin', function (err, isMember) { assert.ifError(err); assert(isMember); done(); }); }); }); }); it('should reject/accept all memberships requests', function (done) { function requestMembership(uids, callback) { async.series([ function (next) { socketGroups.join({ uid: uids.uid1 }, { groupName: 'PrivateCanJoin' }, next); }, function (next) { socketGroups.join({ uid: uids.uid2 }, { groupName: 'PrivateCanJoin' }, next); }, ], function (err) { callback(err); }); } var uids; async.waterfall([ function (next) { async.parallel({ uid1: function (next) { User.create({ username: 'groupuser1' }, next); }, uid2: function (next) { User.create({ username: 'groupuser2' }, next); }, }, next); }, function (results, next) { uids = results; requestMembership(results, next); }, function (next) { socketGroups.rejectAll({ uid: adminUid }, { groupName: 'PrivateCanJoin' }, next); }, function (next) { Groups.getPending('PrivateCanJoin', next); }, function (pending, next) { assert.equal(pending.length, 0); requestMembership(uids, next); }, function (next) { socketGroups.acceptAll({ uid: adminUid }, { groupName: 'PrivateCanJoin' }, next); }, function (next) { Groups.isMembers([uids.uid1, uids.uid2], 'PrivateCanJoin', next); }, function (isMembers, next) { assert(isMembers[0]); assert(isMembers[1]); next(); }, ], function (err) { done(err); }); }); it('should issue invite to user', function (done) { User.create({ username: 'invite1' }, function (err, uid) { assert.ifError(err); socketGroups.issueInvite({ uid: adminUid }, { groupName: 'PrivateCanJoin', toUid: uid }, function (err) { assert.ifError(err); Groups.isInvited(uid, 'PrivateCanJoin', function (err, isInvited) { assert.ifError(err); assert(isInvited); done(); }); }); }); }); it('should fail with invalid data', function (done) { socketGroups.issueMassInvite({ uid: adminUid }, { groupName: 'PrivateCanJoin', usernames: null }, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should issue mass invite to users', function (done) { User.create({ username: 'invite2' }, function (err, uid) { assert.ifError(err); socketGroups.issueMassInvite({ uid: adminUid }, { groupName: 'PrivateCanJoin', usernames: 'invite1, invite2' }, function (err) { assert.ifError(err); Groups.isInvited([adminUid, uid], 'PrivateCanJoin', function (err, isInvited) { assert.ifError(err); assert.deepStrictEqual(isInvited, [false, true]); done(); }); }); }); }); it('should rescind invite', function (done) { User.create({ username: 'invite3' }, function (err, uid) { assert.ifError(err); socketGroups.issueInvite({ uid: adminUid }, { groupName: 'PrivateCanJoin', toUid: uid }, function (err) { assert.ifError(err); socketGroups.rescindInvite({ uid: adminUid }, { groupName: 'PrivateCanJoin', toUid: uid }, function (err) { assert.ifError(err); Groups.isInvited(uid, 'PrivateCanJoin', function (err, isInvited) { assert.ifError(err); assert(!isInvited); done(); }); }); }); }); }); it('should error if user is not invited', function (done) { socketGroups.acceptInvite({ uid: adminUid }, { groupName: 'PrivateCanJoin' }, function (err) { assert.equal(err.message, '[[error:not-invited]]'); done(); }); }); it('should accept invite', function (done) { User.create({ username: 'invite4' }, function (err, uid) { assert.ifError(err); socketGroups.issueInvite({ uid: adminUid }, { groupName: 'PrivateCanJoin', toUid: uid }, function (err) { assert.ifError(err); socketGroups.acceptInvite({ uid: uid }, { groupName: 'PrivateCanJoin' }, function (err) { assert.ifError(err); Groups.isMember(uid, 'PrivateCanJoin', function (err, isMember) { assert.ifError(err); assert(isMember); done(); }); }); }); }); }); it('should reject invite', function (done) { User.create({ username: 'invite5' }, function (err, uid) { assert.ifError(err); socketGroups.issueInvite({ uid: adminUid }, { groupName: 'PrivateCanJoin', toUid: uid }, function (err) { assert.ifError(err); socketGroups.rejectInvite({ uid: uid }, { groupName: 'PrivateCanJoin' }, function (err) { assert.ifError(err); Groups.isInvited(uid, 'PrivateCanJoin', function (err, isInvited) { assert.ifError(err); assert(!isInvited); done(); }); }); }); }); }); it('should grant ownership to user', function (done) { socketGroups.grant({ uid: adminUid }, { groupName: 'PrivateCanJoin', toUid: testUid }, function (err) { assert.ifError(err); Groups.ownership.isOwner(testUid, 'PrivateCanJoin', function (err, isOwner) { assert.ifError(err); assert(isOwner); done(); }); }); }); it('should rescind ownership from user', function (done) { socketGroups.rescind({ uid: adminUid }, { groupName: 'PrivateCanJoin', toUid: testUid }, function (err) { assert.ifError(err); Groups.ownership.isOwner(testUid, 'PrivateCanJoin', function (err, isOwner) { assert.ifError(err); assert(!isOwner); done(); }); }); }); it('should fail to kick user with invalid data', function (done) { socketGroups.kick({ uid: adminUid }, { groupName: 'PrivateCanJoin', uid: adminUid }, function (err) { assert.equal(err.message, '[[error:cant-kick-self]]'); done(); }); }); it('should kick user from group', function (done) { socketGroups.kick({ uid: adminUid }, { groupName: 'PrivateCanJoin', uid: testUid }, function (err) { assert.ifError(err); Groups.isMember(testUid, 'PrivateCanJoin', function (err, isMember) { assert.ifError(err); assert(!isMember); done(); }); }); }); it('should fail to create group with invalid data', function (done) { socketGroups.create({ uid: 0 }, {}, function (err) { assert.equal(err.message, '[[error:no-privileges]]'); done(); }); }); it('should fail to create group if group creation is disabled', function (done) { socketGroups.create({ uid: testUid }, { name: 'avalidname' }, function (err) { assert.equal(err.message, '[[error:no-privileges]]'); done(); }); }); it('should fail to create group if name is privilege group', function (done) { socketGroups.create({ uid: 1 }, { name: 'cid:1:privileges:groups:find' }, function (err) { assert.equal(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should create/update group', function (done) { socketGroups.create({ uid: adminUid }, { name: 'createupdategroup' }, function (err, groupData) { assert.ifError(err); assert(groupData); var data = { groupName: 'createupdategroup', values: { name: 'renamedupdategroup', description: 'cat group', userTitle: 'cats', userTitleEnabled: 1, disableJoinRequests: 1, hidden: 1, private: 0, }, }; socketGroups.update({ uid: adminUid }, data, function (err) { assert.ifError(err); Groups.get('renamedupdategroup', {}, function (err, groupData) { assert.ifError(err); assert.equal(groupData.name, 'renamedupdategroup'); assert.equal(groupData.userTitle, 'cats'); assert.equal(groupData.description, 'cat group'); assert.equal(groupData.hidden, true); assert.equal(groupData.disableJoinRequests, true); assert.equal(groupData.private, false); done(); }); }); }); }); it('should fail to create a group with name guests', function (done) { socketGroups.create({ uid: adminUid }, { name: 'guests' }, function (err) { assert.equal(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should fail to rename guests group', function (done) { var data = { groupName: 'guests', values: { name: 'guests2', }, }; socketGroups.update({ uid: adminUid }, data, function (err) { assert.equal(err.message, '[[error:no-group]]'); done(); }); }); it('should delete group', function (done) { socketGroups.delete({ uid: adminUid }, { groupName: 'renamedupdategroup' }, function (err) { assert.ifError(err); Groups.exists('renamedupdategroup', function (err, exists) { assert.ifError(err); assert(!exists); done(); }); }); }); it('should fail to delete group if name is special', function (done) { socketGroups.delete({ uid: adminUid }, { groupName: 'administrators' }, function (err) { assert.equal(err.message, '[[error:not-allowed]]'); done(); }); }); it('should fail to delete group if name is special', function (done) { socketGroups.delete({ uid: adminUid }, { groupName: 'registered-users' }, function (err) { assert.equal(err.message, '[[error:not-allowed]]'); done(); }); }); it('should fail to delete group if name is special', function (done) { socketGroups.delete({ uid: adminUid }, { groupName: 'Global Moderators' }, function (err) { assert.equal(err.message, '[[error:not-allowed]]'); done(); }); }); it('should fail to delete group if name is special', function (done) { socketGroups.delete({ uid: adminUid }, { groupName: 'guests' }, function (err) { assert.equal(err.message, '[[error:not-allowed]]'); done(); }); }); it('should fail to load more groups with invalid data', function (done) { socketGroups.loadMore({ uid: adminUid }, {}, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should load more groups', function (done) { socketGroups.loadMore({ uid: adminUid }, { after: 0, sort: 'count' }, function (err, data) { assert.ifError(err); assert(Array.isArray(data.groups)); done(); }); }); it('should fail to load more members with invalid data', function (done) { socketGroups.loadMoreMembers({ uid: adminUid }, {}, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should load more members', function (done) { socketGroups.loadMoreMembers({ uid: adminUid }, { after: 0, groupName: 'PrivateCanJoin' }, function (err, data) { assert.ifError(err); assert(Array.isArray(data.users)); done(); }); }); }); describe('admin socket methods', function () { var socketGroups = require('../src/socket.io/admin/groups'); it('should fail to create group with invalid data', function (done) { socketGroups.create({ uid: adminUid }, null, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should fail to create group if group name is privilege group', function (done) { socketGroups.create({ uid: adminUid }, { name: 'cid:1:privileges:read' }, function (err) { assert.equal(err.message, '[[error:invalid-group-name]]'); done(); }); }); it('should create a group', function (done) { socketGroups.create({ uid: adminUid }, { name: 'newgroup', description: 'group created by admin' }, function (err, groupData) { assert.ifError(err); assert.equal(groupData.name, 'newgroup'); assert.equal(groupData.description, 'group created by admin'); assert.equal(groupData.ownerUid, adminUid); assert.equal(groupData.private, 1); assert.equal(groupData.hidden, 0); assert.equal(groupData.memberCount, 1); done(); }); }); it('should fail to join with invalid data', function (done) { socketGroups.join({ uid: adminUid }, null, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should add user to group', function (done) { socketGroups.join({ uid: adminUid }, { uid: testUid, groupName: 'newgroup' }, function (err) { assert.ifError(err); Groups.isMember(testUid, 'newgroup', function (err, isMember) { assert.ifError(err); assert(isMember); done(); }); }); }); it('should fail to if user is already member', function (done) { socketGroups.join({ uid: adminUid }, { uid: testUid, groupName: 'newgroup' }, function (err) { assert.equal(err.message, '[[error:group-already-member]]'); done(); }); }); it('it should fail with invalid data', function (done) { socketGroups.leave({ uid: adminUid }, null, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('it should fail if admin tries to remove self', function (done) { socketGroups.leave({ uid: adminUid }, { uid: adminUid, groupName: 'administrators' }, function (err) { assert.equal(err.message, '[[error:cant-remove-self-as-admin]]'); done(); }); }); it('should fail if user is not member', function (done) { socketGroups.leave({ uid: adminUid }, { uid: 3, groupName: 'newgroup' }, function (err) { assert.equal(err.message, '[[error:group-not-member]]'); done(); }); }); it('should remove user from group', function (done) { socketGroups.leave({ uid: adminUid }, { uid: testUid, groupName: 'newgroup' }, function (err) { assert.ifError(err); Groups.isMember(testUid, 'newgroup', function (err, isMember) { assert.ifError(err); assert(!isMember); done(); }); }); }); it('should fail with invalid data', function (done) { socketGroups.update({ uid: adminUid }, null, function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should update group', function (done) { var data = { groupName: 'newgroup', values: { name: 'renamedgroup', description: 'cat group', userTitle: 'cats', userTitleEnabled: 1, disableJoinRequests: 1, hidden: 1, private: 0, }, }; socketGroups.update({ uid: adminUid }, data, function (err) { assert.ifError(err); Groups.get('renamedgroup', {}, function (err, groupData) { assert.ifError(err); assert.equal(groupData.name, 'renamedgroup'); assert.equal(groupData.userTitle, 'cats'); assert.equal(groupData.description, 'cat group'); assert.equal(groupData.hidden, true); assert.equal(groupData.disableJoinRequests, true); assert.equal(groupData.private, false); done(); }); }); }); }); describe('groups cover', function () { var socketGroups = require('../src/socket.io/groups'); var regularUid; var logoPath = path.join(__dirname, '../test/files/test.png'); var imagePath = path.join(__dirname, '../test/files/groupcover.png'); before(function (done) { User.create({ username: 'regularuser', password: '123456' }, function (err, uid) { assert.ifError(err); regularUid = uid; async.series([ function (next) { Groups.join('Test', adminUid, next); }, function (next) { Groups.join('Test', regularUid, next); }, function (next) { helpers.copyFile(logoPath, imagePath, next); }, ], done); }); }); it('should fail if user is not logged in or not owner', function (done) { socketGroups.cover.update({ uid: 0 }, {}, function (err) { assert.equal(err.message, '[[error:no-privileges]]'); socketGroups.cover.update({ uid: regularUid }, { groupName: 'Test' }, function (err) { assert.equal(err.message, '[[error:no-privileges]]'); done(); }); }); }); it('should upload group cover image from file', function (done) { var data = { groupName: 'Test', file: { path: imagePath, type: 'image/png', }, }; socketGroups.cover.update({ uid: adminUid }, data, function (err, data) { assert.ifError(err); Groups.getGroupFields('Test', ['cover:url'], function (err, groupData) { assert.ifError(err); assert.equal(nconf.get('relative_path') + data.url, groupData['cover:url']); if (nconf.get('relative_path')) { assert(!data.url.startsWith(nconf.get('relative_path'))); assert(groupData['cover:url'].startsWith(nconf.get('relative_path')), groupData['cover:url']); } done(); }); }); }); it('should upload group cover image from data', function (done) { var data = { groupName: 'Test', imageData: '', }; socketGroups.cover.update({ uid: adminUid }, data, function (err, data) { assert.ifError(err); Groups.getGroupFields('Test', ['cover:url'], function (err, groupData) { assert.ifError(err); assert.equal(nconf.get('relative_path') + data.url, groupData['cover:url']); done(); }); }); }); it('should fail to upload group cover with invalid image', function (done) { var data = { groupName: 'Test', imageData: '', }; socketGroups.cover.update({ uid: adminUid }, data, function (err, data) { assert.equal(err.message, '[[error:invalid-image]]'); done(); }); }); it('should update group cover position', function (done) { var data = { groupName: 'Test', position: '50% 50%', }; socketGroups.cover.update({ uid: adminUid }, data, function (err) { assert.ifError(err); Groups.getGroupFields('Test', ['cover:position'], function (err, groupData) { assert.ifError(err); assert.equal('50% 50%', groupData['cover:position']); done(); }); }); }); it('should fail to update cover position if group name is missing', function (done) { Groups.updateCoverPosition('', '50% 50%', function (err) { assert.equal(err.message, '[[error:invalid-data]]'); done(); }); }); it('should fail to remove cover if not logged in', function (done) { socketGroups.cover.remove({ uid: 0 }, { groupName: 'Test' }, function (err) { assert.equal(err.message, '[[error:no-privileges]]'); done(); }); }); it('should fail to remove cover if not owner', function (done) { socketGroups.cover.remove({ uid: regularUid }, { groupName: 'Test' }, function (err) { assert.equal(err.message, '[[error:no-privileges]]'); done(); }); }); it('should remove cover', function (done) { socketGroups.cover.remove({ uid: adminUid }, { groupName: 'Test' }, function (err) { assert.ifError(err); db.getObjectFields('group:Test', ['cover:url'], function (err, groupData) { assert.ifError(err); assert(!groupData['cover:url']); done(); }); }); }); }); });