mirror of
https://github.com/sudoxnym/habitica.git
synced 2026-04-14 19:56:23 +00:00
* Begin refactoring news API to return individual markdown posts
* Implement simple bailey CMS
* Prevented users with lvl less than 10 from seeing mana
* Added in class checks and notification tests
* Added getter use
* Fixed class check
* chore(i18n): update locales
* 4.60.2
* remove tests that are no longer needed because we won't be purging private messages (#10670)
Ref: this comment from paglias: https://github.com/HabitRPG/habitica/issues/7940#issuecomment-406489506
* remove .only
* allow challenge leader/owner to view/join/modify challenge in private group they've left - fixes #9753 (#10606)
* rename hasAccess to canJoin for challenges
This is so the function won't be used accidentally for other
purposes, since hasAccess could be misinterpretted.
* add isLeader function for challenges
* allow challenge leader to join/modify/end challenge when they're not in the private group it's in
* delete duplicate test
* clarify title of existing tests
* add tests and adjust existing tests to reduce privileges of test users
* fix lint errors
* remove pointless isLeader check (it's checked in canJoin)
* Correct Challenges tooltip in Guild view (#10667)
* Fix new party member cannot join pending quest (#10648)
* Saved sort selection into local storage for later use - fixes #10432 (#10655)
* Saved sort selection into local storage for later use
* Updated code to use userLocalManager module
* Fix initial position item info when selecting one item after another (fixes #10077) (#10661)
* Update lastMouseMoveEvent even when dragging an egg or potion.
* Update lastMouseMoveEvent even when dragging a food item.
* Refactor/market vue (#10601)
* extract inventoryDrawer from market
* show scrollbar only if needed
* extract featuredItemsHeader / pinUtils
* extract pageLayout
* extract layoutSection / filterDropdown - fix sortByNumber
* rollback sortByNumber order-fix
* move equipment lists out of the layout-section (for now)
* refactor sellModal
* extract checkbox
* extract equipment section
* extract category row
* revert scroll - remove sellModal item template
* fix(lint): commas and semis
* Created category item component (#10613)
* extract filter sidebar
* fix gemCount - fix raising the item count if the item wasn't previously owned
* fixes #10659
* remove unneeded method
* fix typo when importing component
* feat(content): Forest Friends Quest Bundle
* chore(sprites): compile
* chore(i18n): update locales
* 4.60.3
* fix(bcrypt): install fork compatible with Node 8
* chore(i18n): update locales
* 4.60.4
* add swear words - TRIGGER / CONTENT WARNING: assault, slurs, swearwords, etc
* add pinUtils-mixin - fixes #10682 (#10683)
* chore(news): Bailey
* chore(i18n): update locales
* 4.60.5
* Improve rendering banner about sleeping in the inn
See #10695
* Display settings in one column
* Small Updates (#10701)
* small updates
* fix client unit test
* fix uuid validation
* Revert "Small Updates (#10701)" (#10702)
This reverts commit dd7fa73961.
* feat(event): Fall Festival 2018
* chore(sprites): compile
* chore(i18n): update locales
* 4.61.0
* Move inbox to its own model (#10428)
* shared model for chat and inbox
* disable inbox schema
* inbox: use separate model
* remove old code that used group.chat
* add back chat field (not used) and remove old tests
* remove inbox exclusions when loading user
* add GET /api/v3/inbox/messages
* add comment
* implement DELETE /inbox/messages/:messageid in v4
* implement GET /inbox/messages in v4 and update tests
* implement DELETE /api/v4/inbox/clear
* fix url
* fix doc
* update /export/inbox.html
* update other data exports
* add back messages in user schema
* add user.toJSONWithInbox
* add compativility until migration is done
* more compatibility
* fix tojson called twice
* add compatibility methods
* fix common tests
* fix v4 integration tests
* v3 get user -> with inbox
* start to fix tests
* fix v3 integration tests
* wip
* wip, client use new route
* update tests for members/send-private-message
* tests for get user in v4
* add tests for DELETE /inbox/messages/:messageId
* add tests for DELETE /inbox/clear in v4
* update docs
* fix tests
* initial migration
* fix migration
* fix migration
* migration fixes
* migrate api.enterCouponCode
* migrate api.castSpell
* migrate reset, reroll, rebirth
* add routes to v4 version
* fix tests
* fixes
* api.updateUser
* remove .only
* get user -> userLib
* refactor inbox.vue to work with new data model
* fix return message when messaging yourself
* wip fix bug with new conversation
* wip
* fix remaining ui issues
* move api.registerLocal, fixes
* keep only v3 version of GET /inbox/messages
* Fix API early Stat Point allocation (#10680)
* Refactor hasClass check to common so it can be used in shared & server-side code
* Check that user has selected class before allocating stat points
* chore(event): end Ember Hatching Potions
* chore(analytics): reenable navigation tracking
* update bcrypt
* Point achievement modal links to main site (#10709)
* Animal ears after death (#10691)
* Animal Ears purchasable with Gold if lost in Death
* remove ears from pinned items when set is bought
* standardise css and error handling for gems and coins
* revert accidental new line
* fix client tests
* Reduce margin-bottom of checklist-item from 10px to -3px. (#10684)
* chore(i18n): update locales
* 4.61.1
* Position inn banner when window is resized
* feat(content): Subscriber Items and Magic Potions
* chore(sprites): compile
* chore(i18n): update locales
* 4.62.0
* Update inn banner handling
* Fix banner offset on initial load
* Fix minor issues.
* Issue: 10660 - Fixed. Changed default to Please Enter A Value (#10718)
* Issue: 10660 - Fixed. Changed default to Please Enter A Value
* Issue: 10660 - Fixed/revision 2 Changed default to Enter A Value
* chore(news): Bailey announcements
* chore(i18n): update locales
* 4.62.1
* adjust wiki link for usernameInfo string
https://github.com/HabitRPG/habitica-private/issues/7#issuecomment-425405425
* raise coverage for tasks api calls (#10029)
* - updates a group task - approval is required
- updates a group task with checklist
* add expect to test the new checklist length
* - moves tasks to a specified position out of length
* remove unused line
* website getter tasks tests
* re-add sanitizeUserChallengeTask
* change config.json.example variable to be a string not a boolean
* fix tests - pick the text / up/down props too
* fix test - remove changes on text/up/down - revert sanitize condition - revert sanitization props
* chore(i18n): update locales
* 4.62.2
* chore(news): Bailey
* chore(i18n): update locales
* 4.62.3
* inbox: fix avatar display and order
* Username announcement (#10729)
* Change update username API call
The call no longer requires a password and also validates the username.
* Implement API call to verify username without setting it
* Improve coding style
* Apply username verification to registration
* Update error messages
* Validate display names.
* Fix API early Stat Point allocation (#10680)
* Refactor hasClass check to common so it can be used in shared & server-side code
* Check that user has selected class before allocating stat points
* chore(event): end Ember Hatching Potions
* chore(analytics): reenable navigation tracking
* update bcrypt
* Point achievement modal links to main site (#10709)
* Animal ears after death (#10691)
* Animal Ears purchasable with Gold if lost in Death
* remove ears from pinned items when set is bought
* standardise css and error handling for gems and coins
* revert accidental new line
* fix client tests
* Reduce margin-bottom of checklist-item from 10px to -3px. (#10684)
* chore(i18n): update locales
* 4.61.1
* feat(content): Subscriber Items and Magic Potions
* chore(sprites): compile
* chore(i18n): update locales
* 4.62.0
* Display notification for users to confirm their username
* fix typo
* WIP(usernames): Changes to address #10694
* WIP(usernames): Further changes for #10694
* fix(usernames): don't show spurious headings
* Change verify username notification to new version
* Improve feedback for invalid usernames
* Allow user to set their username again to confirm it
* Improve validation display for usernames
* Temporarily move display name validation outside of schema
* Improve rendering banner about sleeping in the inn
See #10695
* Display settings in one column
* Position inn banner when window is resized
* Update inn banner handling
* Fix banner offset on initial load
* Fix minor issues.
* Issue: 10660 - Fixed. Changed default to Please Enter A Value (#10718)
* Issue: 10660 - Fixed. Changed default to Please Enter A Value
* Issue: 10660 - Fixed/revision 2 Changed default to Enter A Value
* chore(news): Bailey announcements
* chore(i18n): update locales
* 4.62.1
* adjust wiki link for usernameInfo string
https://github.com/HabitRPG/habitica-private/issues/7#issuecomment-425405425
* raise coverage for tasks api calls (#10029)
* - updates a group task - approval is required
- updates a group task with checklist
* add expect to test the new checklist length
* - moves tasks to a specified position out of length
* remove unused line
* website getter tasks tests
* re-add sanitizeUserChallengeTask
* change config.json.example variable to be a string not a boolean
* fix tests - pick the text / up/down props too
* fix test - remove changes on text/up/down - revert sanitize condition - revert sanitization props
* Change update username API call
The call no longer requires a password and also validates the username.
* feat(content): Subscriber Items and Magic Potions
* Re-add register call
* Fix merge issue
* Fix issue with setting username
* Implement new alert style
* Display username confirmation status in settings
* Add disclaimer to change username field
* validate username in settings
* Allow specific fields to be focused when opening site settings
* Implement requested changes.
* Fix merge issue
* Fix failing tests
* verify username when users register with username and password
* Set ID for change username notification
* Disable submit button if username is invalid
* Improve username confirmation handling
* refactor(settings): address remaining code comments on auth form
* Revert "refactor(settings): address remaining code comments on auth form"
This reverts commit 9b6609ad646b23d9e3e394c1856f149d9a2d0995.
* Social user username (#10620)
* Refactored private functions to library
* Refactored social login code
* Added username to social registration
* Changed id library
* Added new local auth check
* Fixed export error. Fixed password check error
* fix(settings): password not available on client
* refactor(settings): more sensible placement of methods
* chore(migration): script to hand out procgen usernames
* fix(migration): don't give EVERYONE new names you doofus
* fix(migration): limit data retrieved, be extra careful about updates
* fix(migration): use missing field, not migration tag, for query
* fix(migration): unused var
* fix(usernames): only generate 20 characters
* fix(migration): set lowerCaseUsername
* fix(lint): comma
* fix(lint): comma spacing
* chore(i18n): update locales
* 4.63.0
* chore(news): Bailey
* chore(i18n): update locales
* 4.63.1
* fix(usernames): various
Reword invalid characters error
Correct typo in slur error
Remove extraneous Confirm button
Reset username field if empty on blur
Restore ability to add local auth to social login
* fix(auth): account for new username paradigm in add-local flow
* fix(auth): alert on successful addLocal
* chore(i18n): update locales
* 4.63.2
* fix(auth): Don't try to check existing username on new reg
* 4.63.3
* feat(content): Armoire and BGs 2018/10
* chore(sprites): compile
* fix(passport): use graph API v2.8
* chore(i18n): update locales
* 4.64.0
* Begin refactoring news API to return individual markdown posts
* Implement simple bailey CMS
* remove old news markdown
* Correctly display images in bailey modal
* Remove need for newStuff migration
* Add basic tests
* Fix authentication issue
* Fix tests
* Update news model
* add API route to get single post
* remove news admin frontend code
* fix lint error
* Fix merge mixups
* Fix lint errors
* fix api call
* fix lint error
* Fix issues caused by merging
* remove console log
* Improve news display
* Correctly update users notifications
* Fix date display for news posts
* Fix tests
* remove old cache file
* correctly create date
* correctly create promise
* Better check for existance.
* Improve docs
* Fix minor issues
* Add method to get latest post
* fix lint errors
* use correct call for 404
* add comment about old newStuff field
* paginate news
* Fix lint errors
* Remove unnecessary await
* Fix broken tests
* ...
* correct existence check
* fix database queries
* change approach to cached news posts
* fix tests
* Change how news posts are cached
* Fetch last news post at an interval
* Fix typos and other small things
* add new permission for modifying bailey posts
* add test for ensureNewsPoster
* return last news post with legacy api
* Fix test
* Hopefully fix test
* change fields to _id
* Fixes
* Fixes
* fix test
* Fixes
* make all tests pass
* fix lint
* id -> _id
* _id -> id
* remove identical tell me later route from api v4
* fix lint
* user model: fix issues with newStuff
* improve user#toJSONTransform
* fix typo
* improve newsPost.js
* fix(integration tests): do not return flags.newStuff if it was not selected
* fix news controller
* server side fixes, start refactoring client
* more client fixes
* automatically set author
* new stuff: show one post per user + drafts
* change default border radius for modals to 8px
* required fields and defaults
* slit news into its own component and fix static page
* noNewsPoster: move from i18n to apiError
* remove unused strings
* fix unit tests
* update apidocs
* add backward comparibility for flags.newStuff in api v3
* fix integration tests
* POST news: make integration test independent of number of posts
* api v3 news: render markdown
* static new-stuff: add padding and fix when user not logged in
* test flags.newStuff
* api v3: test setting flags.newStuff on PUT /user
* refactor news post cache and add tests
* remove new locales file
* more resilient tests
* more resilient tests
* refactor tests for NewsPost.updateLastNewsPost
* api v4: fix tests
* api v3: fix tests
* can set flags.newStuff in api v4
Co-authored-by: Keith Holliday <keithrholliday@gmail.com>
Co-authored-by: Sabe Jones <sabrecat@gmail.com>
Co-authored-by: Alys <Alys@users.noreply.github.com>
Co-authored-by: Matteo Pagliazzi <matteopagliazzi@gmail.com>
Co-authored-by: Carl Vuorinen <carl.vuorinen@gmail.com>
Co-authored-by: Rene Cordier <rene.cordier@gmail.com>
Co-authored-by: Forrest Hatfield <github@forresthatfield.com>
Co-authored-by: lucubro <88whacko@gmail.com>
Co-authored-by: negue <negue@users.noreply.github.com>
Co-authored-by: Alys <alice.harris@oldgods.net>
Co-authored-by: J.D. Sandifer <sandifer.jd@gmail.com>
Co-authored-by: Kirsty <kirsty-tortoise@users.noreply.github.com>
Co-authored-by: beatscribe <rattjp@gmail.com>
Co-authored-by: Phillip Thelen <phillip@habitica.com>
875 lines
31 KiB
JavaScript
875 lines
31 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.update({
|
|
$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.update({
|
|
$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.update({
|
|
$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.update({
|
|
$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.update({
|
|
$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('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_BEAST_MASTER')).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_BEAST_MASTER')).to.exist;
|
|
expect(user.notifications.find(notification => notification.type === 'ACHIEVEMENT_MOUNT_MASTER')).to.exist;
|
|
expect(user.notifications.find(notification => notification.type === 'ACHIEVEMENT_TRIAD_BINGO')).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('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.contributor.newsPoster = 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);
|
|
});
|
|
});
|
|
});
|