fix(tests): leave/reject/quests

This commit is contained in:
SabreCat
2023-08-01 16:53:48 -05:00
parent 150cd16b1c
commit d7071d6b4d
3 changed files with 194 additions and 273 deletions

View File

@@ -5,7 +5,6 @@ import {
generateChallenge, generateChallenge,
checkExistence, checkExistence,
createAndPopulateGroup, createAndPopulateGroup,
sleep,
generateUser, generateUser,
translate as t, translate as t,
} from '../../../../helpers/api-integration/v3'; } from '../../../../helpers/api-integration/v3';
@@ -14,253 +13,182 @@ import payments from '../../../../../website/server/libs/payments/payments';
import calculateSubscriptionTerminationDate from '../../../../../website/server/libs/payments/calculateSubscriptionTerminationDate'; import calculateSubscriptionTerminationDate from '../../../../../website/server/libs/payments/calculateSubscriptionTerminationDate';
describe('POST /groups/:groupId/leave', () => { describe('POST /groups/:groupId/leave', () => {
const typesOfGroups = { let groupToLeave;
'public guild': { type: 'guild', privacy: 'public' }, let leader;
'private guild': { type: 'guild', privacy: 'private' }, let member;
party: { type: 'party', privacy: 'private' }, let members;
}; let memberCount;
each(typesOfGroups, (groupDetails, groupType) => { context('Leaving a Group Plan', () => {
context(`Leaving a ${groupType}`, () => { beforeEach(async () => {
let groupToLeave; ({ group: groupToLeave, groupLeader: leader, members } = await createAndPopulateGroup({
let leader; type: 'guild',
let member; privacy: 'private',
let memberCount; members: 1,
upgradeToGroupPlan: true,
}));
member = members[0]; // eslint-disable-line prefer-destructuring
memberCount = groupToLeave.memberCount;
await leader.update({ 'auth.timestamps.created': new Date('2022-01-01') });
});
it('prevents non members from leaving', async () => {
const user = await generateUser();
await expect(user.post(`/groups/${groupToLeave._id}/leave`)).to.eventually.be.rejected.and.eql({
code: 404,
error: 'NotFound',
message: t('groupNotFound'),
});
});
it('lets user leave', async () => {
await member.post(`/groups/${groupToLeave._id}/leave`);
const userThatLeftGroup = await member.get('/user');
expect(userThatLeftGroup.guilds).to.be.empty;
expect(userThatLeftGroup.party._id).to.not.exist;
await groupToLeave.sync();
expect(groupToLeave.memberCount).to.equal(memberCount - 1);
});
it('removes new messages for that group from user', async () => {
await leader.post(`/groups/${groupToLeave._id}/chat`, { message: 'Some message' });
await member.sync();
expect(member.notifications.find(n => n.type === 'NEW_CHAT_MESSAGE' && n.data.group.id === groupToLeave._id)).to.exist;
expect(member.newMessages[groupToLeave._id]).to.not.be.empty;
await member.post(`/groups/${groupToLeave._id}/leave`);
await member.sync();
expect(member.notifications.find(n => n.type === 'NEW_CHAT_MESSAGE' && n.data.group.id === groupToLeave._id)).to.not.exist;
expect(member.newMessages[groupToLeave._id]).to.be.undefined;
});
context('with challenges', () => {
let challenge;
beforeEach(async () => { beforeEach(async () => {
const { group, groupLeader, members } = await createAndPopulateGroup({ challenge = await generateChallenge(leader, groupToLeave);
groupDetails, await member.post(`/challenges/${challenge._id}/join`);
members: 1,
});
groupToLeave = group; await leader.post(`/tasks/challenge/${challenge._id}`, {
leader = groupLeader; text: 'test habit',
member = members[0]; // eslint-disable-line prefer-destructuring type: 'habit',
memberCount = group.memberCount;
await members[0].update({ 'auth.timestamps.created': new Date('2022-01-01') });
});
it('prevents non members from leaving', async () => {
const user = await generateUser();
await expect(user.post(`/groups/${groupToLeave._id}/leave`)).to.eventually.be.rejected.and.eql({
code: 404,
error: 'NotFound',
message: t('groupNotFound'),
}); });
}); });
it(`lets user leave a ${groupType}`, async () => { it('removes all challenge tasks when keep parameter is set to remove', async () => {
await member.post(`/groups/${groupToLeave._id}/leave?keep=remove-all`);
const userWithoutChallengeTasks = await member.get('/user');
expect(userWithoutChallengeTasks.challenges).to.not.include(challenge._id);
expect(userWithoutChallengeTasks.tasksOrder.habits).to.be.empty;
});
it('keeps all challenge tasks when keep parameter is not set', async () => {
await member.post(`/groups/${groupToLeave._id}/leave`); await member.post(`/groups/${groupToLeave._id}/leave`);
const userThatLeftGroup = await member.get('/user'); const userWithChallengeTasks = await member.get('/user');
expect(userThatLeftGroup.guilds).to.be.empty; expect(userWithChallengeTasks.tasksOrder.habits).to.not.be.empty;
expect(userThatLeftGroup.party._id).to.not.exist;
await groupToLeave.sync();
expect(groupToLeave.memberCount).to.equal(memberCount - 1);
}); });
it(`sets a new group leader when leader leaves a ${groupType}`, async () => { it('keeps the user in the challenge when the keepChallenges parameter is set to remain-in-challenges', async () => {
await leader.post(`/groups/${groupToLeave._id}/leave`); await member.post(`/groups/${groupToLeave._id}/leave`, { keepChallenges: 'remain-in-challenges' });
await groupToLeave.sync(); const userWithChallengeTasks = await member.get('/user');
expect(groupToLeave.memberCount).to.equal(memberCount - 1);
expect(groupToLeave.leader).to.equal(member._id); expect(userWithChallengeTasks.challenges).to.include(challenge._id);
}); });
it('removes new messages for that group from user', async () => { it('drops the user in the challenge when the keepChallenges parameter isn\'t set', async () => {
await member.post(`/groups/${groupToLeave._id}/chat`, { message: 'Some message' }); await member.post(`/groups/${groupToLeave._id}/leave`);
await sleep(0.5); const userWithChallengeTasks = await member.get('/user');
await leader.sync(); expect(userWithChallengeTasks.challenges).to.not.include(challenge._id);
expect(leader.notifications.find(n => n.type === 'NEW_CHAT_MESSAGE' && n.data.group.id === groupToLeave._id)).to.exist;
expect(leader.newMessages[groupToLeave._id]).to.not.be.empty;
await leader.post(`/groups/${groupToLeave._id}/leave`);
await leader.sync();
expect(leader.notifications.find(n => n.type === 'NEW_CHAT_MESSAGE' && n.data.group.id === groupToLeave._id)).to.not.exist;
expect(leader.newMessages[groupToLeave._id]).to.be.undefined;
}); });
context('with challenges', () => {
let challenge;
beforeEach(async () => {
challenge = await generateChallenge(leader, groupToLeave);
await leader.post(`/challenges/${challenge._id}/join`);
await leader.post(`/tasks/challenge/${challenge._id}`, {
text: 'test habit',
type: 'habit',
});
await sleep(0.5);
});
it('removes all challenge tasks when keep parameter is set to remove', async () => {
await leader.post(`/groups/${groupToLeave._id}/leave?keep=remove-all`);
const userWithoutChallengeTasks = await leader.get('/user');
expect(userWithoutChallengeTasks.challenges).to.not.include(challenge._id);
expect(userWithoutChallengeTasks.tasksOrder.habits).to.be.empty;
});
it('keeps all challenge tasks when keep parameter is not set', async () => {
await leader.post(`/groups/${groupToLeave._id}/leave`);
const userWithChallengeTasks = await leader.get('/user');
// @TODO find elegant way to assert against the task existing
expect(userWithChallengeTasks.tasksOrder.habits).to.not.be.empty;
});
it('keeps the user in the challenge when the keepChallenges parameter is set to remain-in-challenges', async () => {
await leader.post(`/groups/${groupToLeave._id}/leave`, { keepChallenges: 'remain-in-challenges' });
const userWithChallengeTasks = await leader.get('/user');
expect(userWithChallengeTasks.challenges).to.include(challenge._id);
});
it('drops the user in the challenge when the keepChallenges parameter isn\'t set', async () => {
await leader.post(`/groups/${groupToLeave._id}/leave`);
const userWithChallengeTasks = await leader.get('/user');
expect(userWithChallengeTasks.challenges).to.not.include(challenge._id);
});
});
it('prevents quest leader from leaving a groupToLeave');
it('prevents a user from leaving during an active quest');
}); });
}); });
context('Leaving a group as the last member', () => { context('Leaving a Party', () => {
context('private guild', () => { let invitees;
let privateGuild; let invitedUser;
let leader;
let invitedUser;
beforeEach(async () => { beforeEach(async () => {
const { group, groupLeader, invitees } = await createAndPopulateGroup({ ({ group: groupToLeave, groupLeader: leader, members, invitees } = await createAndPopulateGroup({
groupDetails: { type: 'party',
name: 'Test Private Guild', privacy: 'private',
type: 'guild', members: 1,
}, invites: 1,
invites: 1, }));
leaderDetails: {
'auth.timestamps.created': new Date('2022-01-01'),
balance: 10,
},
});
privateGuild = group; member = members[0]; // eslint-disable-line prefer-destructuring
leader = groupLeader; invitedUser = invitees[0];
invitedUser = invitees[0]; // eslint-disable-line prefer-destructuring memberCount = groupToLeave.memberCount;
await leader.update({ 'auth.timestamps.created': new Date('2022-01-01') });
});
await leader.post(`/groups/${group._id}/chat`, { message: 'Some message' }); it('prevents non members from leaving', async () => {
}); const user = await generateUser();
await expect(user.post(`/groups/${groupToLeave._id}/leave`)).to.eventually.be.rejected.and.eql({
it('removes a group when the last member leaves', async () => { code: 404,
await leader.post(`/groups/${privateGuild._id}/leave`); error: 'NotFound',
message: t('groupNotFound'),
await expect(checkExistence('groups', privateGuild._id)).to.eventually.equal(false);
});
it('removes invitations when the last member leaves', async () => {
await leader.post(`/groups/${privateGuild._id}/leave`);
const userWithoutInvitation = await invitedUser.get('/user');
expect(userWithoutInvitation.invitations.guilds).to.be.empty;
}); });
}); });
context('public guild', () => { it('lets user leave', async () => {
let publicGuild; await member.post(`/groups/${groupToLeave._id}/leave`);
let leader;
let invitedUser;
beforeEach(async () => { const userThatLeftGroup = await member.get('/user');
const { group, groupLeader, invitees } = await createAndPopulateGroup({
groupDetails: {
name: 'Test Public Guild',
type: 'guild',
privacy: 'public',
},
invites: 1,
});
publicGuild = group; expect(userThatLeftGroup.guilds).to.be.empty;
leader = groupLeader; expect(userThatLeftGroup.party._id).to.not.exist;
invitedUser = invitees[0]; // eslint-disable-line prefer-destructuring await groupToLeave.sync();
}); expect(groupToLeave.memberCount).to.equal(memberCount - 1);
it('keeps the group when the last member leaves', async () => {
await leader.post(`/groups/${publicGuild._id}/leave`);
await expect(checkExistence('groups', publicGuild._id)).to.eventually.equal(true);
});
it('keeps the invitations when the last member leaves a public guild', async () => {
await leader.post(`/groups/${publicGuild._id}/leave`);
const userWithoutInvitation = await invitedUser.get('/user');
expect(userWithoutInvitation.invitations.guilds).to.not.be.empty;
});
it('deletes non existent guild from user when user tries to leave', async () => {
const nonExistentGuildId = generateUUID();
const userWithNonExistentGuild = await generateUser({ guilds: [nonExistentGuildId] });
expect(userWithNonExistentGuild.guilds).to.contain(nonExistentGuildId);
await expect(userWithNonExistentGuild.post(`/groups/${nonExistentGuildId}/leave`))
.to.eventually.be.rejected;
await userWithNonExistentGuild.sync();
expect(userWithNonExistentGuild.guilds).to.not.contain(nonExistentGuildId);
});
}); });
context('party', () => { it('sets a new group leader when leader leaves', async () => {
let party; await leader.post(`/groups/${groupToLeave._id}/leave`);
let leader;
let invitedUser;
beforeEach(async () => { await groupToLeave.sync();
const { group, groupLeader, invitees } = await createAndPopulateGroup({ expect(groupToLeave.memberCount).to.equal(memberCount - 1);
groupDetails: { expect(groupToLeave.leader).to.equal(member._id);
name: 'Test Party', });
type: 'party',
},
invites: 1,
});
party = group; it('removes new messages for that group from user', async () => {
leader = groupLeader; await leader.post(`/groups/${groupToLeave._id}/chat`, { message: 'Some message' });
invitedUser = invitees[0]; // eslint-disable-line prefer-destructuring await member.sync();
});
it('removes a group when the last member leaves a party', async () => { expect(member.notifications.find(n => n.type === 'NEW_CHAT_MESSAGE' && n.data.group.id === groupToLeave._id)).to.exist;
await leader.post(`/groups/${party._id}/leave`); expect(member.newMessages[groupToLeave._id]).to.not.be.empty;
await expect(checkExistence('party', party._id)).to.eventually.equal(false); await member.post(`/groups/${groupToLeave._id}/leave`);
}); await member.sync();
it('removes invitations when the last member leaves a party', async () => { expect(member.notifications.find(n => n.type === 'NEW_CHAT_MESSAGE' && n.data.group.id === groupToLeave._id)).to.not.exist;
await leader.post(`/groups/${party._id}/leave`); expect(member.newMessages[groupToLeave._id]).to.be.undefined;
});
const userWithoutInvitation = await invitedUser.get('/user'); it('removes a party when the last member leaves', async () => {
await member.post(`/groups/${groupToLeave._id}/leave`);
await leader.post(`/groups/${groupToLeave._id}/leave`);
expect(userWithoutInvitation.invitations.parties[0]).to.be.undefined; await expect(checkExistence('party', groupToLeave._id)).to.eventually.equal(false);
}); });
it('removes invitations when the last member leaves a party', async () => {
await member.post(`/groups/${groupToLeave._id}/leave`);
await leader.post(`/groups/${groupToLeave._id}/leave`);
const userWithoutInvitation = await invitedUser.get('/user');
expect(userWithoutInvitation.invitations.parties[0]).to.be.undefined;
}); });
it('deletes non existent party from user when user tries to leave', async () => { it('deletes non existent party from user when user tries to leave', async () => {
@@ -275,11 +203,62 @@ describe('POST /groups/:groupId/leave', () => {
expect(userWithNonExistentParty.party).to.eql({}); expect(userWithNonExistentParty.party).to.eql({});
}); });
context('with challenges', () => {
let challenge;
beforeEach(async () => {
challenge = await generateChallenge(leader, groupToLeave);
await member.post(`/challenges/${challenge._id}/join`);
await leader.post(`/tasks/challenge/${challenge._id}`, {
text: 'test habit',
type: 'habit',
});
});
it('removes all challenge tasks when keep parameter is set to remove', async () => {
await member.post(`/groups/${groupToLeave._id}/leave?keep=remove-all`);
const userWithoutChallengeTasks = await member.get('/user');
expect(userWithoutChallengeTasks.challenges).to.not.include(challenge._id);
expect(userWithoutChallengeTasks.tasksOrder.habits).to.be.empty;
});
it('keeps all challenge tasks when keep parameter is not set', async () => {
await member.post(`/groups/${groupToLeave._id}/leave`);
const userWithChallengeTasks = await member.get('/user');
expect(userWithChallengeTasks.tasksOrder.habits).to.not.be.empty;
});
it('keeps the user in the challenge when the keepChallenges parameter is set to remain-in-challenges', async () => {
await member.post(`/groups/${groupToLeave._id}/leave`, { keepChallenges: 'remain-in-challenges' });
const userWithChallengeTasks = await member.get('/user');
expect(userWithChallengeTasks.challenges).to.include(challenge._id);
});
it('drops the user in the challenge when the keepChallenges parameter isn\'t set', async () => {
await member.post(`/groups/${groupToLeave._id}/leave`);
const userWithChallengeTasks = await member.get('/user');
expect(userWithChallengeTasks.challenges).to.not.include(challenge._id);
});
});
}); });
const typesOfGroups = {
'private guild': { type: 'guild', privacy: 'private' },
party: { type: 'party', privacy: 'private' },
};
each(typesOfGroups, (groupDetails, groupType) => { each(typesOfGroups, (groupDetails, groupType) => {
context(`Leaving a group plan when the group is a ${groupType}`, () => { context(`Leaving a group plan when the group is a ${groupType}`, () => {
if (groupDetails.privacy === 'public') return; // public guilds cannot be group plans
let groupWithPlan; let groupWithPlan;
let leader; let leader;
let member; let member;
@@ -288,6 +267,7 @@ describe('POST /groups/:groupId/leave', () => {
const { group, groupLeader, members } = await createAndPopulateGroup({ const { group, groupLeader, members } = await createAndPopulateGroup({
groupDetails, groupDetails,
members: 1, members: 1,
upgradeToGroupPlan: true,
}); });
leader = groupLeader; leader = groupLeader;
member = members[0]; // eslint-disable-line prefer-destructuring member = members[0]; // eslint-disable-line prefer-destructuring
@@ -321,27 +301,6 @@ describe('POST /groups/:groupId/leave', () => {
await member.sync(); await member.sync();
expect(member.purchased.plan.dateTerminated).to.exist; expect(member.purchased.plan.dateTerminated).to.exist;
}); });
it('preserves the free subscription when leaving a any other group without a plan', async () => {
// Joining a guild without a group plan
const { group: groupWithNoPlan } = await createAndPopulateGroup({
groupDetails: {
name: 'Group Without Plan',
type: 'guild',
privacy: 'public',
},
});
await member.post(`/groups/${groupWithNoPlan._id}/join`);
await member.sync();
expect(member.purchased.plan.planId).to.equal('group_plan_auto');
expect(member.purchased.plan.dateTerminated).to.not.exist;
// Leaving the guild without a group plan
await member.post(`/groups/${groupWithNoPlan._id}/leave`);
await member.sync();
expect(member.purchased.plan.dateTerminated).to.not.exist;
});
}); });
}); });

View File

@@ -5,43 +5,6 @@ import {
} from '../../../../helpers/api-integration/v3'; } from '../../../../helpers/api-integration/v3';
describe('POST /group/:groupId/reject-invite', () => { describe('POST /group/:groupId/reject-invite', () => {
context('Rejecting a public guild invite', () => {
let publicGuild; let
invitedUser;
beforeEach(async () => {
const { group, invitees } = await createAndPopulateGroup({
groupDetails: {
name: 'Test Guild',
type: 'guild',
privacy: 'public',
},
invites: 1,
});
publicGuild = group;
invitedUser = invitees[0]; // eslint-disable-line prefer-destructuring
});
it('returns error when user is not invited', async () => {
const userWithoutInvite = await generateUser();
await expect(userWithoutInvite.post(`/groups/${publicGuild._id}/reject-invite`)).to.eventually.be.rejected.and.eql({
code: 401,
error: 'NotAuthorized',
message: t('messageGroupRequiresInvite'),
});
});
it('clears invitation from user', async () => {
await invitedUser.post(`/groups/${publicGuild._id}/reject-invite`);
await expect(invitedUser.get('/user'))
.to.eventually.have.nested.property('invitations.guilds')
.to.not.include({ id: publicGuild._id });
});
});
context('Rejecting a private guild invite', () => { context('Rejecting a private guild invite', () => {
let invitedUser; let let invitedUser; let
guild; guild;
@@ -54,6 +17,7 @@ describe('POST /group/:groupId/reject-invite', () => {
privacy: 'private', privacy: 'private',
}, },
invites: 1, invites: 1,
upgradeToGroupPlan: true,
}); });
guild = group; guild = group;

View File

@@ -51,15 +51,13 @@ describe('POST /groups/:groupId/quests/invite/:questKey', () => {
}); });
it('does not issue invites for Guilds', async () => { it('does not issue invites for Guilds', async () => {
const { group } = await createAndPopulateGroup({ const { group, groupLeader } = await createAndPopulateGroup({
groupDetails: { type: 'guild', privacy: 'private' }, groupDetails: { type: 'guild', privacy: 'private' },
members: 1, members: 1,
upgradeToGroupPlan: true, upgradeToGroupPlan: true,
}); });
const alternateGroup = group; await expect(groupLeader.post(`/groups/${group._id}/quests/invite/${PET_QUEST}`)).to.eventually.be.rejected.and.eql({
await expect(leader.post(`/groups/${alternateGroup._id}/quests/invite/${PET_QUEST}`)).to.eventually.be.rejected.and.eql({
code: 401, code: 401,
error: 'NotAuthorized', error: 'NotAuthorized',
message: t('guildQuestsNotSupported'), message: t('guildQuestsNotSupported'),