import moment from 'moment'; import requireAgain from 'require-again'; 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 { model as Blocker } from '../../../../website/server/models/blocker'; import common from '../../../../website/common'; const pathToUserSchema = '../../../../website/server/models/user/schema'; 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); }); }); describe('validates email', () => { it('does not throw an error for a valid email', () => { const user = new User(); user.auth.local.email = 'hello@example.com'; const errors = user.validateSync(); expect(errors.errors['auth.local.email']).to.not.exist; }); it('throws an error if email is not valid', () => { const user = new User(); user.auth.local.email = 'invalid-email'; const errors = user.validateSync(); expect(errors.errors['auth.local.email'].message).to.equal(common.i18n.t('invalidEmail')); }); it('throws an error if email is using a restricted domain', () => { const user = new User(); user.auth.local.email = 'scammer@habitica.com'; const errors = user.validateSync(); expect(errors.errors['auth.local.email'].message).to.equal(common.i18n.t('invalidEmailDomain', { domains: 'habitica.com, habitrpg.com' })); }); it('throws an error if email was blocked specifically', () => { sandbox.stub(Blocker, 'watchBlockers').returns({ on: (event, callback) => { callback({ operation: 'add', blocker: { type: 'email', area: 'full', value: 'blocked@example.com' } }); }, }); const schema = requireAgain(pathToUserSchema).UserSchema; const valid = schema.paths['auth.local.email'].options.validate.every(v => v.validator('blocked@example.com')); expect(valid).to.equal(false); }); it('throws an error if email domain was blocked', () => { sandbox.stub(Blocker, 'watchBlockers').returns({ on: (event, callback) => { callback({ operation: 'add', blocker: { type: 'email', area: 'full', value: '@example.com' } }); }, }); const schema = requireAgain(pathToUserSchema).UserSchema; const valid = schema.paths['auth.local.email'].options.validate.every(v => v.validator('blocked@example.com')); expect(valid).to.equal(false); }); it('throws an error if user portion of email was blocked', () => { sandbox.stub(Blocker, 'watchBlockers').returns({ on: (event, callback) => { callback({ operation: 'add', blocker: { type: 'email', area: 'full', value: 'blocked@' } }); }, }); const schema = requireAgain(pathToUserSchema).UserSchema; const valid = schema.paths['auth.local.email'].options.validate.every(v => v.validator('blocked@example.com')); expect(valid).to.equal(false); }); it('does not throw an error if email is not blocked', () => { sandbox.stub(Blocker, 'watchBlockers').returns({ on: (event, callback) => { callback({ operation: 'add', blocker: { type: 'email', area: 'full', value: '@example.com' } }); callback({ operation: 'add', blocker: { type: 'email', area: 'full', value: 'blocked@' } }); callback({ operation: 'add', blocker: { type: 'email', area: 'full', value: 'bad@test.com' } }); }, }); const schema = requireAgain(pathToUserSchema).UserSchema; const valid = schema.paths['auth.local.email'].options.validate.every(v => v.validator('good@test.com')); expect(valid).to.equal(true); }); }); });