mirror of
				https://github.com/HabitRPG/habitica.git
				synced 2025-10-30 20:52:29 +01:00 
			
		
		
		
	* remove some unused dependencies * update mongoose version * make common tests pass * Make unit tests pass * make api v3 integration tests pass * fix lint issues * fix issue with package-lock * fix(lint): we don't need no .js * fix(lint): update to latest config-habitrpg * chore(npm): update package locks * fix(test): replace deprecated fn * chore(package): update eslint-habitrpg again * fix(lint): server linting * fix(lint): client linting * fix(client): correct mangled common imports * chore(npm): update package-locks * fix(lint): punctuation, module --------- Co-authored-by: SabreCat <sabrecat@gmail.com> Co-authored-by: SabreCat <sabe@habitica.com>
		
			
				
	
	
		
			916 lines
		
	
	
		
			32 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			916 lines
		
	
	
		
			32 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import moment from 'moment';
 | |
| import { model as User } from '../../../../website/server/models/user';
 | |
| import { model as NewsPost } from '../../../../website/server/models/newsPost';
 | |
| import { model as Group } from '../../../../website/server/models/group';
 | |
| import common from '../../../../website/common';
 | |
| 
 | |
| describe('User Model', () => {
 | |
|   describe('.toJSON()', () => {
 | |
|     it('keeps user._tmp when calling .toJSON', () => {
 | |
|       const user = new User({
 | |
|         auth: {
 | |
|           local: {
 | |
|             username: 'username',
 | |
|             lowerCaseUsername: 'username',
 | |
|             email: 'email@email.email',
 | |
|             salt: 'salt',
 | |
|             hashed_password: 'hashed_password', // eslint-disable-line camelcase
 | |
|           },
 | |
|         },
 | |
|       });
 | |
| 
 | |
|       user._tmp = { ok: true };
 | |
|       user._nonTmp = { ok: true };
 | |
| 
 | |
|       expect(user._tmp).to.eql({ ok: true });
 | |
|       expect(user._nonTmp).to.eql({ ok: true });
 | |
| 
 | |
|       const toObject = user.toObject();
 | |
|       const toJSON = user.toJSON();
 | |
| 
 | |
|       expect(toObject).to.not.have.keys('_tmp');
 | |
|       expect(toObject).to.not.have.keys('_nonTmp');
 | |
| 
 | |
|       expect(toJSON).to.have.any.key('_tmp');
 | |
|       expect(toJSON._tmp).to.eql({ ok: true });
 | |
|       expect(toJSON).to.not.have.keys('_nonTmp');
 | |
|     });
 | |
| 
 | |
|     it('can add computed stats to a JSONified user object', () => {
 | |
|       const user = new User();
 | |
|       const userToJSON = user.toJSON();
 | |
| 
 | |
|       expect(userToJSON.stats.maxMP).to.not.exist;
 | |
|       expect(userToJSON.stats.maxHealth).to.not.exist;
 | |
|       expect(userToJSON.stats.toNextLevel).to.not.exist;
 | |
| 
 | |
|       User.addComputedStatsToJSONObj(userToJSON.stats, userToJSON);
 | |
| 
 | |
|       expect(userToJSON.stats.maxMP).to.exist;
 | |
|       expect(userToJSON.stats.maxHealth).to.equal(common.maxHealth);
 | |
|       expect(userToJSON.stats.toNextLevel).to.equal(common.tnl(user.stats.lvl));
 | |
|     });
 | |
| 
 | |
|     it('can transform user object without mongoose helpers', async () => {
 | |
|       const user = new User();
 | |
|       await user.save();
 | |
|       const userToJSON = await User.findById(user._id).lean().exec();
 | |
| 
 | |
|       expect(userToJSON.stats.maxMP).to.not.exist;
 | |
|       expect(userToJSON.stats.maxHealth).to.not.exist;
 | |
|       expect(userToJSON.stats.toNextLevel).to.not.exist;
 | |
|       expect(userToJSON.id).to.not.exist;
 | |
| 
 | |
|       User.transformJSONUser(userToJSON);
 | |
| 
 | |
|       expect(userToJSON.id).to.equal(userToJSON._id);
 | |
|       expect(userToJSON.stats.maxMP).to.not.exist;
 | |
|       expect(userToJSON.stats.maxHealth).to.not.exist;
 | |
|       expect(userToJSON.stats.toNextLevel).to.not.exist;
 | |
|     });
 | |
| 
 | |
|     it('can transform user object without mongoose helpers (including computed stats)', async () => {
 | |
|       const user = new User();
 | |
|       await user.save();
 | |
|       const userToJSON = await User.findById(user._id).lean().exec();
 | |
| 
 | |
|       expect(userToJSON.stats.maxMP).to.not.exist;
 | |
|       expect(userToJSON.stats.maxHealth).to.not.exist;
 | |
|       expect(userToJSON.stats.toNextLevel).to.not.exist;
 | |
| 
 | |
|       User.transformJSONUser(userToJSON, true);
 | |
| 
 | |
|       expect(userToJSON.id).to.equal(userToJSON._id);
 | |
|       expect(userToJSON.stats.maxMP).to.exist;
 | |
|       expect(userToJSON.stats.maxHealth).to.equal(common.maxHealth);
 | |
|       expect(userToJSON.stats.toNextLevel).to.equal(common.tnl(user.stats.lvl));
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   context('achievements', () => {
 | |
|     it('can add an achievement', () => {
 | |
|       const user = new User();
 | |
|       const originalUserToJSON = user.toJSON({ minimize: false });
 | |
|       expect(originalUserToJSON.achievements.createdTask).to.not.eql(true);
 | |
|       const notificationsN = originalUserToJSON.notifications.length;
 | |
| 
 | |
|       user.addAchievement('createdTask');
 | |
| 
 | |
|       const userToJSON = user.toJSON();
 | |
|       expect(user.notifications.length).to.equal(notificationsN + 1);
 | |
|       expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|       expect(userToJSON.notifications[0].type).to.equal('ACHIEVEMENT');
 | |
|       expect(userToJSON.notifications[0].data).to.eql({
 | |
|         achievement: 'createdTask',
 | |
|       });
 | |
|       expect(userToJSON.notifications[0].seen).to.eql(false);
 | |
| 
 | |
|       expect(userToJSON.achievements.createdTask).to.eql(true);
 | |
|     });
 | |
| 
 | |
|     it('throws an error if the achievement is not valid', () => {
 | |
|       const user = new User();
 | |
|       expect(() => user.addAchievement('notAnAchievement')).to.throw;
 | |
|     });
 | |
| 
 | |
|     context('static push method', () => {
 | |
|       it('throws an error if the achievement is not valid', async () => {
 | |
|         const user = new User();
 | |
|         await user.save();
 | |
| 
 | |
|         await expect(User.addAchievementUpdate({ _id: user._id }, 'notAnAchievement'))
 | |
|           .to.eventually.be.rejected;
 | |
| 
 | |
|         expect(() => user.addAchievement('notAnAchievement')).to.throw;
 | |
|       });
 | |
| 
 | |
|       it('adds an achievement for a single member via static method', async () => {
 | |
|         let user = new User();
 | |
|         await user.save();
 | |
| 
 | |
|         const originalUserToJSON = user.toJSON({ minimize: false });
 | |
|         expect(originalUserToJSON.achievements.createdTask).to.not.eql(true);
 | |
|         const notificationsN = originalUserToJSON.notifications.length;
 | |
| 
 | |
|         await User.addAchievementUpdate({ _id: user._id }, 'createdTask');
 | |
| 
 | |
|         user = await User.findOne({ _id: user._id }).exec();
 | |
| 
 | |
|         const userToJSON = user.toJSON();
 | |
|         expect(user.notifications.length).to.equal(notificationsN + 1);
 | |
|         expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|         expect(userToJSON.notifications[0].type).to.equal('ACHIEVEMENT');
 | |
|         expect(userToJSON.notifications[0].data).to.eql({
 | |
|           achievement: 'createdTask',
 | |
|         });
 | |
|         expect(userToJSON.notifications[0].seen).to.eql(false);
 | |
| 
 | |
|         expect(userToJSON.achievements.createdTask).to.eql(true);
 | |
|       });
 | |
| 
 | |
|       it('adds an achievement for all given users via static method', async () => {
 | |
|         let user = new User();
 | |
|         const otherUser = new User();
 | |
|         await Promise.all([user.save(), otherUser.save()]);
 | |
| 
 | |
|         await User.addAchievementUpdate({ _id: { $in: [user._id, otherUser._id] } }, 'createdTask');
 | |
| 
 | |
|         user = await User.findOne({ _id: user._id }).exec();
 | |
| 
 | |
|         let userToJSON = user.toJSON();
 | |
|         expect(user.notifications.length).to.equal(1);
 | |
|         expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|         expect(userToJSON.notifications[0].type).to.equal('ACHIEVEMENT');
 | |
|         expect(userToJSON.notifications[0].data).to.eql({
 | |
|           achievement: 'createdTask',
 | |
|         });
 | |
|         expect(userToJSON.notifications[0].seen).to.eql(false);
 | |
| 
 | |
|         expect(userToJSON.achievements.createdTask).to.eql(true);
 | |
| 
 | |
|         user = await User.findOne({ _id: otherUser._id }).exec();
 | |
| 
 | |
|         userToJSON = user.toJSON();
 | |
|         expect(user.notifications.length).to.equal(1);
 | |
|         expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|         expect(userToJSON.notifications[0].type).to.equal('ACHIEVEMENT');
 | |
|         expect(userToJSON.notifications[0].data).to.eql({
 | |
|           achievement: 'createdTask',
 | |
|         });
 | |
|         expect(userToJSON.notifications[0].seen).to.eql(false);
 | |
| 
 | |
|         expect(userToJSON.achievements.createdTask).to.eql(true);
 | |
|       });
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   context('post init', () => {
 | |
|     it('removes invalid tags when loading the user', async () => {
 | |
|       let user = new User();
 | |
|       await user.save();
 | |
|       await user.updateOne({
 | |
|         $set: {
 | |
|           tags: [
 | |
|             null, // invalid, not an object
 | |
|             // { name: '123' }, // invalid, no id - generated automatically
 | |
|             { id: '123' }, // invalid, no name
 | |
|             { name: 'ABC', id: '1234' }, // valid
 | |
|           ],
 | |
|         },
 | |
|       }).exec();
 | |
| 
 | |
|       user = await User.findById(user._id).exec();
 | |
| 
 | |
|       const userToJSON = user.toJSON();
 | |
|       expect(userToJSON.tags.length).to.equal(1);
 | |
| 
 | |
|       expect(userToJSON.tags[0]).to.have.all.keys(['id', 'name']);
 | |
|       expect(userToJSON.tags[0].id).to.equal('1234');
 | |
|       expect(userToJSON.tags[0].name).to.equal('ABC');
 | |
|     });
 | |
| 
 | |
|     it('removes invalid push devices when loading the user', async () => {
 | |
|       let user = new User();
 | |
|       await user.save();
 | |
|       await user.updateOne({
 | |
|         $set: {
 | |
|           pushDevices: [
 | |
|             null, // invalid, not an object
 | |
|             { regId: '123' }, // invalid, no type
 | |
|             { type: 'android' }, // invalid, no regId
 | |
|             { type: 'android', regId: '1234' }, // valid
 | |
|           ],
 | |
|         },
 | |
|       }).exec();
 | |
| 
 | |
|       user = await User.findById(user._id).exec();
 | |
| 
 | |
|       const userToJSON = user.toJSON();
 | |
|       expect(userToJSON.pushDevices.length).to.equal(1);
 | |
| 
 | |
|       expect(userToJSON.pushDevices[0]).to.have.all.keys(['regId', 'type', 'createdAt', 'updatedAt']);
 | |
|       expect(userToJSON.pushDevices[0].type).to.equal('android');
 | |
|       expect(userToJSON.pushDevices[0].regId).to.equal('1234');
 | |
|     });
 | |
| 
 | |
|     it('removes duplicate push devices when loading the user', async () => {
 | |
|       let user = new User();
 | |
|       await user.save();
 | |
|       await user.updateOne({
 | |
|         $set: {
 | |
|           pushDevices: [
 | |
|             { type: 'android', regId: '1234' },
 | |
|             { type: 'android', regId: '1234' },
 | |
|           ],
 | |
|         },
 | |
|       }).exec();
 | |
| 
 | |
|       user = await User.findById(user._id).exec();
 | |
| 
 | |
|       const userToJSON = user.toJSON();
 | |
|       expect(userToJSON.pushDevices.length).to.equal(1);
 | |
| 
 | |
|       expect(userToJSON.pushDevices[0]).to.have.all.keys(['regId', 'type', 'createdAt', 'updatedAt']);
 | |
|       expect(userToJSON.pushDevices[0].type).to.equal('android');
 | |
|       expect(userToJSON.pushDevices[0].regId).to.equal('1234');
 | |
|     });
 | |
| 
 | |
|     it('removes invalid notifications when loading the user', async () => {
 | |
|       let user = new User();
 | |
|       await user.save();
 | |
|       await user.updateOne({
 | |
|         $set: {
 | |
|           notifications: [
 | |
|             null, // invalid, not an object
 | |
|             { seen: true }, // invalid, no type or id
 | |
|             { id: 123 }, // invalid, no type
 | |
|             // invalid, no id, not included here because the id would be added automatically
 | |
|             // {type: 'ABC'},
 | |
|             { type: 'ABC', id: '123' }, // valid
 | |
|           ],
 | |
|         },
 | |
|       }).exec();
 | |
| 
 | |
|       user = await User.findById(user._id).exec();
 | |
| 
 | |
|       const userToJSON = user.toJSON();
 | |
|       expect(userToJSON.notifications.length).to.equal(1);
 | |
| 
 | |
|       expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|       expect(userToJSON.notifications[0].type).to.equal('ABC');
 | |
|       expect(userToJSON.notifications[0].id).to.equal('123');
 | |
|     });
 | |
| 
 | |
|     it('removes multiple NEW_CHAT_MESSAGE for the same group', async () => {
 | |
|       let user = new User();
 | |
|       await user.save();
 | |
|       await user.updateOne({
 | |
|         $set: {
 | |
|           notifications: [
 | |
|             {
 | |
|               type: 'NEW_CHAT_MESSAGE',
 | |
|               id: 123,
 | |
|               data: { group: { id: 12345 } },
 | |
|             },
 | |
|             {
 | |
|               type: 'NEW_CHAT_MESSAGE',
 | |
|               id: 1234,
 | |
|               data: { group: { id: 12345 } },
 | |
|             },
 | |
|             {
 | |
|               type: 'NEW_CHAT_MESSAGE',
 | |
|               id: 123,
 | |
|               data: { group: { id: 123456 } },
 | |
|             }, // not duplicate, different group
 | |
|             {
 | |
|               type: 'NEW_CHAT_MESSAGE_DIFF',
 | |
|               id: 123,
 | |
|               data: { group: { id: 12345 } },
 | |
|             }, // not duplicate, different type
 | |
|           ],
 | |
|         },
 | |
|       }).exec();
 | |
| 
 | |
|       user = await User.findById(user._id).exec();
 | |
| 
 | |
|       const userToJSON = user.toJSON();
 | |
|       expect(userToJSON.notifications.length).to.equal(3);
 | |
| 
 | |
|       expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|       expect(userToJSON.notifications[0].type).to.equal('NEW_CHAT_MESSAGE');
 | |
|       expect(userToJSON.notifications[0].id).to.equal('123');
 | |
|       expect(userToJSON.notifications[0].data).to.deep.equal({ group: { id: 12345 } });
 | |
|       expect(userToJSON.notifications[0].seen).to.equal(false);
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   context('notifications', () => {
 | |
|     it('can add notifications without data', () => {
 | |
|       const user = new User();
 | |
| 
 | |
|       user.addNotification('CRON');
 | |
| 
 | |
|       const userToJSON = user.toJSON();
 | |
|       expect(user.notifications.length).to.equal(1);
 | |
|       expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|       expect(userToJSON.notifications[0].type).to.equal('CRON');
 | |
|       expect(userToJSON.notifications[0].data).to.eql({});
 | |
|       expect(userToJSON.notifications[0].seen).to.eql(false);
 | |
|     });
 | |
| 
 | |
|     it('can add notifications with data and already marked as seen', () => {
 | |
|       const user = new User();
 | |
| 
 | |
|       user.addNotification('CRON', { field: 1 }, true);
 | |
| 
 | |
|       const userToJSON = user.toJSON();
 | |
|       expect(user.notifications.length).to.equal(1);
 | |
|       expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|       expect(userToJSON.notifications[0].type).to.equal('CRON');
 | |
|       expect(userToJSON.notifications[0].data).to.eql({ field: 1 });
 | |
|       expect(userToJSON.notifications[0].seen).to.eql(true);
 | |
|     });
 | |
| 
 | |
|     context('static push method', () => {
 | |
|       it('adds notifications for a single member via static method', async () => {
 | |
|         let user = new User();
 | |
|         await user.save();
 | |
| 
 | |
|         await User.pushNotification({ _id: user._id }, 'CRON');
 | |
| 
 | |
|         user = await User.findOne({ _id: user._id }).exec();
 | |
| 
 | |
|         const userToJSON = user.toJSON();
 | |
|         expect(user.notifications.length).to.equal(1);
 | |
|         expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|         expect(userToJSON.notifications[0].type).to.equal('CRON');
 | |
|         expect(userToJSON.notifications[0].data).to.eql({});
 | |
|       });
 | |
| 
 | |
|       it('validates notifications via static method', async () => {
 | |
|         const user = new User();
 | |
|         await user.save();
 | |
| 
 | |
|         expect(User.pushNotification({ _id: user._id }, 'BAD_TYPE')).to.eventually.be.rejected;
 | |
|         expect(User.pushNotification({ _id: user._id }, 'CRON', null, 'INVALID_SEEN')).to.eventually.be.rejected;
 | |
|       });
 | |
| 
 | |
|       it('adds notifications without data for all given users via static method', async () => {
 | |
|         let user = new User();
 | |
|         const otherUser = new User();
 | |
|         await Promise.all([user.save(), otherUser.save()]);
 | |
| 
 | |
|         await User.pushNotification({ _id: { $in: [user._id, otherUser._id] } }, 'CRON');
 | |
| 
 | |
|         user = await User.findOne({ _id: user._id }).exec();
 | |
| 
 | |
|         let userToJSON = user.toJSON();
 | |
|         expect(user.notifications.length).to.equal(1);
 | |
|         expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|         expect(userToJSON.notifications[0].type).to.equal('CRON');
 | |
|         expect(userToJSON.notifications[0].data).to.eql({});
 | |
|         expect(userToJSON.notifications[0].seen).to.eql(false);
 | |
| 
 | |
|         user = await User.findOne({ _id: otherUser._id }).exec();
 | |
| 
 | |
|         userToJSON = user.toJSON();
 | |
|         expect(user.notifications.length).to.equal(1);
 | |
|         expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|         expect(userToJSON.notifications[0].type).to.equal('CRON');
 | |
|         expect(userToJSON.notifications[0].data).to.eql({});
 | |
|         expect(userToJSON.notifications[0].seen).to.eql(false);
 | |
|       });
 | |
| 
 | |
|       it('adds notifications with data and seen status for all given users via static method', async () => {
 | |
|         let user = new User();
 | |
|         const otherUser = new User();
 | |
|         await Promise.all([user.save(), otherUser.save()]);
 | |
| 
 | |
|         await User.pushNotification({ _id: { $in: [user._id, otherUser._id] } }, 'CRON', { field: 1 }, true);
 | |
| 
 | |
|         user = await User.findOne({ _id: user._id }).exec();
 | |
| 
 | |
|         let userToJSON = user.toJSON();
 | |
|         expect(user.notifications.length).to.equal(1);
 | |
|         expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|         expect(userToJSON.notifications[0].type).to.equal('CRON');
 | |
|         expect(userToJSON.notifications[0].data).to.eql({ field: 1 });
 | |
|         expect(userToJSON.notifications[0].seen).to.eql(true);
 | |
| 
 | |
|         user = await User.findOne({ _id: otherUser._id }).exec();
 | |
| 
 | |
|         userToJSON = user.toJSON();
 | |
|         expect(user.notifications.length).to.equal(1);
 | |
|         expect(userToJSON.notifications[0]).to.have.all.keys(['data', 'id', 'type', 'seen']);
 | |
|         expect(userToJSON.notifications[0].type).to.equal('CRON');
 | |
|         expect(userToJSON.notifications[0].data).to.eql({ field: 1 });
 | |
|         expect(userToJSON.notifications[0].seen).to.eql(true);
 | |
|       });
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   context('isSubscribed', () => {
 | |
|     let user;
 | |
|     beforeEach(() => {
 | |
|       user = new User();
 | |
|     });
 | |
| 
 | |
|     it('returns false if user does not have customer id', () => {
 | |
|       expect(user.isSubscribed()).to.be.undefined;
 | |
|     });
 | |
| 
 | |
|     it('returns true if user does not have plan.dateTerminated', () => {
 | |
|       user.purchased.plan.customerId = 'test-id';
 | |
| 
 | |
|       expect(user.isSubscribed()).to.be.true;
 | |
|     });
 | |
| 
 | |
|     it('returns true if user if plan.dateTerminated is after today', () => {
 | |
|       user.purchased.plan.customerId = 'test-id';
 | |
|       user.purchased.plan.dateTerminated = moment().add(1, 'days').toDate();
 | |
| 
 | |
|       expect(user.isSubscribed()).to.be.true;
 | |
|     });
 | |
| 
 | |
|     it('returns false if user if plan.dateTerminated is before today', () => {
 | |
|       user.purchased.plan.customerId = 'test-id';
 | |
|       user.purchased.plan.dateTerminated = moment().subtract(1, 'days').toDate();
 | |
| 
 | |
|       expect(user.isSubscribed()).to.be.false;
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   context('canGetGems', () => {
 | |
|     let user;
 | |
|     let group;
 | |
|     beforeEach(() => {
 | |
|       user = new User();
 | |
|       const leader = new User();
 | |
|       group = new Group({
 | |
|         name: 'test',
 | |
|         type: 'guild',
 | |
|         privacy: 'private',
 | |
|         leader: leader._id,
 | |
|       });
 | |
|     });
 | |
| 
 | |
|     it('returns true if user is not subscribed', async () => {
 | |
|       expect(await user.canGetGems()).to.equal(true);
 | |
|     });
 | |
| 
 | |
|     it('returns true if user is not subscribed with a group plan', async () => {
 | |
|       user.purchased.plan.customerId = 123;
 | |
|       expect(await user.canGetGems()).to.equal(true);
 | |
|     });
 | |
| 
 | |
|     it('returns true if user is subscribed with a group plan', async () => {
 | |
|       user.purchased.plan.customerId = 'group-plan';
 | |
|       expect(await user.canGetGems()).to.equal(true);
 | |
|     });
 | |
| 
 | |
|     it('returns true if user is part of a group', async () => {
 | |
|       user.guilds.push(group._id);
 | |
|       expect(await user.canGetGems()).to.equal(true);
 | |
|     });
 | |
| 
 | |
|     it('returns true if user is part of a group with a subscription', async () => {
 | |
|       user.guilds.push(group._id);
 | |
|       user.purchased.plan.customerId = 'group-plan';
 | |
|       group.purchased.plan.customerId = 123;
 | |
|       await group.save();
 | |
|       expect(await user.canGetGems()).to.equal(true);
 | |
|     });
 | |
| 
 | |
|     it('returns true if leader is part of a group with a subscription and canGetGems: false', async () => {
 | |
|       user.guilds.push(group._id);
 | |
|       user.purchased.plan.customerId = 'group-plan';
 | |
|       group.purchased.plan.customerId = 123;
 | |
|       group.leader = user._id;
 | |
|       group.leaderOnly.getGems = true;
 | |
|       await group.save();
 | |
|       expect(await user.canGetGems()).to.equal(true);
 | |
|     });
 | |
| 
 | |
|     it('returns true if user is part of a group with no subscription but canGetGems: false', async () => {
 | |
|       user.guilds.push(group._id);
 | |
|       user.purchased.plan.customerId = 'group-plan';
 | |
|       group.leaderOnly.getGems = true;
 | |
|       await group.save();
 | |
|       expect(await user.canGetGems()).to.equal(true);
 | |
|     });
 | |
| 
 | |
|     it('returns false if user is part of a group with a subscription and canGetGems: false', async () => {
 | |
|       user.guilds.push(group._id);
 | |
|       user.purchased.plan.customerId = 'group-plan';
 | |
|       group.purchased.plan.customerId = 123;
 | |
|       group.leaderOnly.getGems = true;
 | |
|       await group.save();
 | |
|       expect(await user.canGetGems()).to.equal(false);
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   context('hasNotCancelled', () => {
 | |
|     let user;
 | |
|     beforeEach(() => {
 | |
|       user = new User();
 | |
|     });
 | |
| 
 | |
|     it('returns false if user does not have customer id', () => {
 | |
|       expect(user.hasNotCancelled()).to.be.false;
 | |
|     });
 | |
| 
 | |
|     it('returns true if user does not have plan.dateTerminated', () => {
 | |
|       user.purchased.plan.customerId = 'test-id';
 | |
| 
 | |
|       expect(user.hasNotCancelled()).to.be.true;
 | |
|     });
 | |
| 
 | |
|     it('returns false if user if plan.dateTerminated is after today', () => {
 | |
|       user.purchased.plan.customerId = 'test-id';
 | |
|       user.purchased.plan.dateTerminated = moment().add(1, 'days').toDate();
 | |
| 
 | |
|       expect(user.hasNotCancelled()).to.be.false;
 | |
|     });
 | |
| 
 | |
|     it('returns false if user if plan.dateTerminated is before today', () => {
 | |
|       user.purchased.plan.customerId = 'test-id';
 | |
|       user.purchased.plan.dateTerminated = moment().subtract(1, 'days').toDate();
 | |
| 
 | |
|       expect(user.hasNotCancelled()).to.be.false;
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   context('hasCancelled', () => {
 | |
|     let user;
 | |
|     beforeEach(() => {
 | |
|       user = new User();
 | |
|     });
 | |
| 
 | |
|     it('returns false if user does not have customer id', () => {
 | |
|       expect(user.hasCancelled()).to.be.false;
 | |
|     });
 | |
| 
 | |
|     it('returns false if user does not have plan.dateTerminated', () => {
 | |
|       user.purchased.plan.customerId = 'test-id';
 | |
| 
 | |
|       expect(user.hasCancelled()).to.be.false;
 | |
|     });
 | |
| 
 | |
|     it('returns true if user if plan.dateTerminated is after today', () => {
 | |
|       user.purchased.plan.customerId = 'test-id';
 | |
|       user.purchased.plan.dateTerminated = moment().add(1, 'days').toDate();
 | |
| 
 | |
|       expect(user.hasCancelled()).to.be.true;
 | |
|     });
 | |
| 
 | |
|     it('returns false if user if plan.dateTerminated is before today', () => {
 | |
|       user.purchased.plan.customerId = 'test-id';
 | |
|       user.purchased.plan.dateTerminated = moment().subtract(1, 'days').toDate();
 | |
| 
 | |
|       expect(user.hasCancelled()).to.be.false;
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   context('pre-save hook', () => {
 | |
|     it('marks the last news post as read for new users', async () => {
 | |
|       const lastNewsPost = { _id: '1' };
 | |
|       sandbox.stub(NewsPost, 'lastNewsPost').returns(lastNewsPost);
 | |
| 
 | |
|       let user = new User();
 | |
|       expect(user.isNew).to.equal(true);
 | |
|       user = await user.save();
 | |
| 
 | |
|       expect(user.checkNewStuff()).to.equal(false);
 | |
|       expect(user.toJSON().flags.newStuff).to.equal(false);
 | |
|       expect(user.flags.lastNewStuffRead).to.equal(lastNewsPost._id);
 | |
|     });
 | |
| 
 | |
|     it('does not mark the last news post as read for existing users', async () => {
 | |
|       const lastNewsPost = { _id: '1' };
 | |
|       const lastNewsPostStub = sandbox.stub(NewsPost, 'lastNewsPost');
 | |
|       lastNewsPostStub.returns(lastNewsPost);
 | |
| 
 | |
|       let user = new User();
 | |
|       user = await user.save();
 | |
| 
 | |
|       expect(user.isNew).to.equal(false);
 | |
|       user.profile.name = 'new name';
 | |
| 
 | |
|       lastNewsPostStub.returns({ _id: '2' });
 | |
|       user = await user.save();
 | |
| 
 | |
|       expect(user.flags.lastNewStuffRead).to.equal(lastNewsPost._id); // not _id: 2
 | |
|     });
 | |
| 
 | |
|     it('does not try to award achievements when achievements or items not selected in query', async () => {
 | |
|       let user = new User();
 | |
|       user = await user.save(); // necessary for user.isSelected to work correctly
 | |
| 
 | |
|       // Create conditions for the Beast Master achievement to be awarded
 | |
|       user.achievements.beastMasterCount = 3;
 | |
|       // verify that it was not awarded initially
 | |
|       expect(user.achievements.beastMaster).to.not.equal(true);
 | |
| 
 | |
|       user = await user.save();
 | |
|       // verify that it's been awarded
 | |
|       expect(user.achievements.beastMaster).to.equal(true);
 | |
|       expect(user.notifications.find(notification => notification.type === 'ACHIEVEMENT_STABLE')).to.exist;
 | |
| 
 | |
|       // reset the user
 | |
|       user.achievements.beastMasterCount = 0;
 | |
|       user.achievements.beastMaster = false;
 | |
| 
 | |
|       user = await user.save();
 | |
|       // verify it's been removed
 | |
|       expect(user.achievements.beastMaster).to.equal(false);
 | |
| 
 | |
|       // fetch the user without selecting the 'items' field
 | |
|       user = await User.findById(user._id).select('-items').exec();
 | |
|       expect(user.isSelected('items')).to.equal(false);
 | |
| 
 | |
|       // create the conditions for the beast master achievement
 | |
|       // but this time it should not be awarded
 | |
|       user.achievements.beastMasterCount = 3;
 | |
|       user = await user.save();
 | |
|       expect(user.achievements.beastMaster).to.equal(false);
 | |
| 
 | |
|       // reset
 | |
|       user.achievements.beastMasterCount = 0;
 | |
|       user = await user.save();
 | |
| 
 | |
|       // this time with achievements not selected
 | |
|       user = await User.findById(user._id).select('-achievements').exec();
 | |
|       expect(user.isSelected('achievements')).to.equal(false);
 | |
|       user.achievements.beastMasterCount = 3;
 | |
|       user = await user.save();
 | |
|       expect(user.achievements.beastMaster).to.not.equal(true);
 | |
|     });
 | |
| 
 | |
|     it('adds achievements to notification list', async () => {
 | |
|       let user = new User();
 | |
|       user = await user.save(); // necessary for user.isSelected to work correctly
 | |
| 
 | |
|       // Create conditions for achievements to be awarded
 | |
|       user.achievements.beastMasterCount = 3;
 | |
|       user.achievements.mountMasterCount = 3;
 | |
|       user.achievements.triadBingoCount = 3;
 | |
|       // verify that it was not awarded initially
 | |
|       expect(user.achievements.beastMaster).to.not.equal(true);
 | |
|       // verify that it was not awarded initially
 | |
|       expect(user.achievements.mountMaster).to.not.equal(true);
 | |
|       // verify that it was not awarded initially
 | |
|       expect(user.achievements.triadBingo).to.not.equal(true);
 | |
| 
 | |
|       user = await user.save();
 | |
|       // verify that it's been awarded
 | |
|       expect(user.notifications.find(
 | |
|         notification => notification.type === 'ACHIEVEMENT_STABLE',
 | |
|       )).to.exist;
 | |
|     });
 | |
| 
 | |
|     context('manage unallocated stats points notifications', () => {
 | |
|       it('doesn\'t add a notification if there are no points to allocate', async () => {
 | |
|         let user = new User();
 | |
| 
 | |
|         user.flags.classSelected = true;
 | |
|         user.preferences.disableClasses = false;
 | |
|         user.stats.class = 'warrior';
 | |
|         user = await user.save(); // necessary for user.isSelected to work correctly
 | |
| 
 | |
|         const oldNotificationsCount = user.notifications.length;
 | |
| 
 | |
|         user.stats.points = 0;
 | |
|         user = await user.save();
 | |
| 
 | |
|         expect(user.notifications.length).to.equal(oldNotificationsCount);
 | |
|       });
 | |
| 
 | |
|       it('removes a notification if there are no more points to allocate', async () => {
 | |
|         let user = new User();
 | |
| 
 | |
|         user.flags.classSelected = true;
 | |
|         user.preferences.disableClasses = false;
 | |
|         user.stats.class = 'warrior';
 | |
|         user.stats.points = 9;
 | |
|         user = await user.save(); // necessary for user.isSelected to work correctly
 | |
| 
 | |
|         expect(user.notifications[0].type).to.equal('UNALLOCATED_STATS_POINTS');
 | |
|         const oldNotificationsCount = user.notifications.length;
 | |
| 
 | |
|         user.stats.points = 0;
 | |
|         user = await user.save();
 | |
| 
 | |
|         expect(user.notifications.length).to.equal(oldNotificationsCount - 1);
 | |
|       });
 | |
| 
 | |
|       it('adds a notification if there are points to allocate', async () => {
 | |
|         let user = new User();
 | |
|         user.flags.classSelected = true;
 | |
|         user.preferences.disableClasses = false;
 | |
|         user.stats.class = 'warrior';
 | |
|         user = await user.save(); // necessary for user.isSelected to work correctly
 | |
|         const oldNotificationsCount = user.notifications.length;
 | |
| 
 | |
|         user.stats.points = 9;
 | |
|         user = await user.save();
 | |
| 
 | |
|         expect(user.notifications.length).to.equal(oldNotificationsCount + 1);
 | |
|         expect(user.notifications[0].type).to.equal('UNALLOCATED_STATS_POINTS');
 | |
|         expect(user.notifications[0].data.points).to.equal(9);
 | |
|       });
 | |
| 
 | |
|       it('adds a notification if the points to allocate have changed', async () => {
 | |
|         let user = new User();
 | |
|         user.stats.points = 9;
 | |
|         user.flags.classSelected = true;
 | |
|         user.preferences.disableClasses = false;
 | |
|         user.stats.class = 'warrior';
 | |
|         user = await user.save(); // necessary for user.isSelected to work correctly
 | |
| 
 | |
|         const oldNotificationsCount = user.notifications.length;
 | |
|         const oldNotificationsUUID = user.notifications[0].id;
 | |
|         expect(user.notifications[0].type).to.equal('UNALLOCATED_STATS_POINTS');
 | |
|         expect(user.notifications[0].data.points).to.equal(9);
 | |
| 
 | |
|         user.stats.points = 11;
 | |
|         user = await user.save();
 | |
| 
 | |
|         expect(user.notifications.length).to.equal(oldNotificationsCount);
 | |
|         expect(user.notifications[0].type).to.equal('UNALLOCATED_STATS_POINTS');
 | |
|         expect(user.notifications[0].data.points).to.equal(11);
 | |
|         expect(user.notifications[0].id).to.not.equal(oldNotificationsUUID);
 | |
|       });
 | |
| 
 | |
|       it('does not add a notification if the user has disabled classes', async () => {
 | |
|         let user = new User();
 | |
|         user.stats.points = 9;
 | |
|         user.flags.classSelected = true;
 | |
|         user.preferences.disableClasses = true;
 | |
|         user.stats.class = 'warrior';
 | |
|         user = await user.save(); // necessary for user.isSelected to work correctly
 | |
| 
 | |
|         const oldNotificationsCount = user.notifications.length;
 | |
| 
 | |
|         user.stats.points = 9;
 | |
|         user = await user.save();
 | |
| 
 | |
|         expect(user.notifications.length).to.equal(oldNotificationsCount);
 | |
|       });
 | |
| 
 | |
|       it('does not add a notification if the user has not selected a class', async () => {
 | |
|         let user = new User();
 | |
|         user.stats.points = 9;
 | |
|         user.flags.classSelected = false;
 | |
|         user.stats.class = 'warrior';
 | |
|         user = await user.save(); // necessary for user.isSelected to work correctly
 | |
| 
 | |
|         const oldNotificationsCount = user.notifications.length;
 | |
| 
 | |
|         user.stats.points = 9;
 | |
|         user = await user.save();
 | |
| 
 | |
|         expect(user.notifications.length).to.equal(oldNotificationsCount);
 | |
|       });
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   describe('daysUserHasMissed', () => {
 | |
|     // http://forbrains.co.uk/international_tools/earth_timezones
 | |
|     let user;
 | |
| 
 | |
|     beforeEach(() => {
 | |
|       user = new User();
 | |
|     });
 | |
| 
 | |
|     it('correctly calculates days missed since lastCron', () => {
 | |
|       const now = moment();
 | |
|       user.lastCron = moment(now).subtract(5, 'days');
 | |
| 
 | |
|       const { daysMissed } = user.daysUserHasMissed(now);
 | |
| 
 | |
|       expect(daysMissed).to.eql(5);
 | |
|     });
 | |
| 
 | |
|     it('correctly handles a cron that did not complete', () => {
 | |
|       const now = moment();
 | |
|       user.lastCron = moment(now).subtract(2, 'days');
 | |
|       user.auth.timestamps.loggedIn = moment(now).subtract(5, 'days');
 | |
| 
 | |
|       const { daysMissed } = user.daysUserHasMissed(now);
 | |
| 
 | |
|       expect(daysMissed).to.eql(5);
 | |
|     });
 | |
| 
 | |
|     it('uses timezone from preferences to calculate days missed', () => {
 | |
|       const now = moment('2017-07-08 01:00:00Z');
 | |
|       user.lastCron = moment('2017-07-04 13:00:00Z');
 | |
|       user.preferences.timezoneOffset = 120;
 | |
| 
 | |
|       const { daysMissed } = user.daysUserHasMissed(now);
 | |
| 
 | |
|       expect(daysMissed).to.eql(3);
 | |
|     });
 | |
| 
 | |
|     it('uses timezone at last cron to calculate days missed', () => {
 | |
|       const now = moment('2017-09-08 13:00:00Z');
 | |
|       user.lastCron = moment('2017-09-06 01:00:00+02:00');
 | |
|       user.preferences.timezoneOffset = 0;
 | |
|       user.preferences.timezoneOffsetAtLastCron = -120;
 | |
| 
 | |
|       const { daysMissed } = user.daysUserHasMissed(now);
 | |
| 
 | |
|       expect(daysMissed).to.eql(2);
 | |
|     });
 | |
| 
 | |
|     it('respects new timezone that drags time into same day', () => {
 | |
|       user.lastCron = moment('2017-12-05T00:00:00.000-06:00');
 | |
|       user.preferences.timezoneOffset = 360;
 | |
|       const today = moment('2017-12-06T00:00:00.000-06:00');
 | |
|       const requestWithMinus7Timezone = { header: () => 420 };
 | |
| 
 | |
|       const { daysMissed } = user.daysUserHasMissed(today, requestWithMinus7Timezone);
 | |
| 
 | |
|       expect(user.preferences.timezoneOffset).to.eql(420);
 | |
|       expect(daysMissed).to.eql(0);
 | |
|     });
 | |
| 
 | |
|     it('should not cron early when going back a timezone with a custom day start', () => {
 | |
|       const yesterday = moment('2017-12-05T02:00:00.000-08:00');
 | |
|       const timezoneOffset = 480;
 | |
|       user.lastCron = yesterday;
 | |
|       user.preferences.timezoneOffset = timezoneOffset;
 | |
|       user.preferences.dayStart = 2;
 | |
| 
 | |
|       const today = moment('2017-12-06T02:00:00.000-08:00');
 | |
|       const req = {};
 | |
|       req.header = () => timezoneOffset + 60;
 | |
| 
 | |
|       const { daysMissed } = user.daysUserHasMissed(today, req);
 | |
| 
 | |
|       expect(daysMissed).to.eql(0);
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   it('isNewsPoster', async () => {
 | |
|     const user = new User();
 | |
|     await user.save();
 | |
| 
 | |
|     expect(user.isNewsPoster()).to.equal(false);
 | |
| 
 | |
|     user.permissions = { news: true };
 | |
|     expect(user.isNewsPoster()).to.equal(true);
 | |
|   });
 | |
| 
 | |
|   describe('checkNewStuff', () => {
 | |
|     let user;
 | |
| 
 | |
|     beforeEach(() => {
 | |
|       user = new User();
 | |
|     });
 | |
| 
 | |
|     afterEach(() => {
 | |
|       sandbox.restore();
 | |
|     });
 | |
| 
 | |
|     it('no last news post', () => {
 | |
|       sandbox.stub(NewsPost, 'lastNewsPost').returns(null);
 | |
|       expect(user.checkNewStuff()).to.equal(false);
 | |
|       expect(user.toJSON().flags.newStuff).to.equal(false);
 | |
|     });
 | |
| 
 | |
|     it('last news post read', () => {
 | |
|       sandbox.stub(NewsPost, 'lastNewsPost').returns({ _id: '123' });
 | |
|       user.flags.lastNewStuffRead = '123';
 | |
|       expect(user.checkNewStuff()).to.equal(false);
 | |
|       expect(user.toJSON().flags.newStuff).to.equal(false);
 | |
|     });
 | |
| 
 | |
|     it('last news post not read', () => {
 | |
|       sandbox.stub(NewsPost, 'lastNewsPost').returns({ _id: '123' });
 | |
|       user.flags.lastNewStuffRead = '124';
 | |
|       expect(user.checkNewStuff()).to.equal(true);
 | |
|       expect(user.toJSON().flags.newStuff).to.equal(true);
 | |
|     });
 | |
|   });
 | |
| });
 |