mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-16 22:27:26 +01: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 commitdd7fa73961. * 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 commit9b6609ad64. * 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>
528 lines
19 KiB
JavaScript
528 lines
19 KiB
JavaScript
import moment from 'moment';
|
|
import {
|
|
defaults, map, flatten, flow, compact, uniq, partialRight,
|
|
} from 'lodash';
|
|
import common from '../../../common';
|
|
|
|
import { // eslint-disable-line import/no-cycle
|
|
TAVERN_ID,
|
|
model as Group,
|
|
} from '../group';
|
|
|
|
import {
|
|
messageDefaults,
|
|
setUserStyles,
|
|
inboxModel as Inbox,
|
|
} from '../message';
|
|
|
|
import { model as UserNotification } from '../userNotification';
|
|
import schema from './schema'; // eslint-disable-line import/no-cycle
|
|
import payments from '../../libs/payments/payments'; // eslint-disable-line import/no-cycle
|
|
import * as inboxLib from '../../libs/inbox'; // eslint-disable-line import/no-cycle
|
|
import amazonPayments from '../../libs/payments/amazon'; // eslint-disable-line import/no-cycle
|
|
import stripePayments from '../../libs/payments/stripe'; // eslint-disable-line import/no-cycle
|
|
import paypalPayments from '../../libs/payments/paypal'; // eslint-disable-line import/no-cycle
|
|
import { model as NewsPost } from '../newsPost';
|
|
|
|
const { daysSince } = common;
|
|
|
|
schema.methods.isSubscribed = function isSubscribed () {
|
|
const now = new Date();
|
|
const { plan } = this.purchased;
|
|
return plan && plan.customerId
|
|
&& (!plan.dateTerminated || moment(plan.dateTerminated).isAfter(now));
|
|
};
|
|
|
|
schema.methods.hasNotCancelled = function hasNotCancelled () {
|
|
const { plan } = this.purchased;
|
|
return Boolean(this.isSubscribed() && !plan.dateTerminated);
|
|
};
|
|
|
|
schema.methods.hasCancelled = function hasCancelled () {
|
|
const { plan } = this.purchased;
|
|
return Boolean(this.isSubscribed() && plan.dateTerminated);
|
|
};
|
|
|
|
// Get an array of groups ids the user is member of
|
|
schema.methods.getGroups = function getUserGroups () {
|
|
const userGroups = this.guilds.slice(0); // clone this.guilds so we don't modify the original
|
|
if (this.party._id) userGroups.push(this.party._id);
|
|
userGroups.push(TAVERN_ID);
|
|
return userGroups;
|
|
};
|
|
|
|
/* eslint-disable no-unused-vars */
|
|
// The checks below all get access to sndr and rcvr, but not all use both
|
|
const INTERACTION_CHECKS = Object.freeze({
|
|
always: [
|
|
// Revoked chat privileges block all interactions
|
|
// to prevent the evading of harassment protections
|
|
// See issue #7971 for some discussion
|
|
(sndr, rcvr) => sndr.flags.chatRevoked && 'chatPrivilegesRevoked',
|
|
|
|
// Direct user blocks prevent all interactions
|
|
(sndr, rcvr) => rcvr.inbox.blocks.includes(sndr._id) && 'notAuthorizedToSendMessageToThisUser',
|
|
(sndr, rcvr) => sndr.inbox.blocks.includes(rcvr._id) && 'blockedToSendToThisUser',
|
|
],
|
|
|
|
'send-private-message': [
|
|
// Private messaging has an opt-out, which does not affect other interactions
|
|
(sndr, rcvr) => rcvr.inbox.optOut && 'notAuthorizedToSendMessageToThisUser',
|
|
|
|
// We allow a player to message themselves so they can test how PMs work
|
|
// or send their own notes to themselves
|
|
],
|
|
|
|
'transfer-gems': [
|
|
// Unlike private messages, gems can't be sent to oneself
|
|
(sndr, rcvr) => rcvr._id === sndr._id && 'cannotSendGemsToYourself',
|
|
],
|
|
|
|
'group-invitation': [
|
|
// uses the checks that are in the 'always' array
|
|
],
|
|
});
|
|
/* eslint-enable no-unused-vars */
|
|
|
|
export const KNOWN_INTERACTIONS = Object.freeze(Object.keys(INTERACTION_CHECKS) // eslint-disable-line import/prefer-default-export, max-len
|
|
.filter(key => key !== 'always'));
|
|
|
|
// Get an array of error message keys that would be thrown if the given interaction was attempted
|
|
schema.methods.getObjectionsToInteraction = function getObjectionsToInteraction (interaction, receiver) { // eslint-disable-line max-len
|
|
if (!KNOWN_INTERACTIONS.includes(interaction)) {
|
|
throw new Error(`Unknown kind of interaction: "${interaction}", expected one of ${KNOWN_INTERACTIONS.join(', ')}`);
|
|
}
|
|
|
|
const sender = this;
|
|
const checks = [
|
|
INTERACTION_CHECKS.always,
|
|
INTERACTION_CHECKS[interaction],
|
|
];
|
|
|
|
const executeChecks = partialRight(map, check => check(sender, receiver));
|
|
|
|
return flow(
|
|
flatten,
|
|
executeChecks,
|
|
compact, // Remove passed checks (passed checks return falsy; failed checks return message keys)
|
|
uniq,
|
|
)(checks);
|
|
};
|
|
|
|
/**
|
|
* Sends a message to a user. Archives a copy in sender's inbox.
|
|
*
|
|
* @param userToReceiveMessage The receiver
|
|
* @param options
|
|
* @param options.receiverMsg The message to send to the receiver
|
|
* @param options.senderMsg The message to archive instead of receiverMsg
|
|
* @return N/A
|
|
*/
|
|
schema.methods.sendMessage = async function sendMessage (userToReceiveMessage, options) {
|
|
const sender = this;
|
|
const senderMsg = options.senderMsg || options.receiverMsg;
|
|
// whether to save users after sending the message, defaults to true
|
|
const saveUsers = options.save !== false;
|
|
|
|
const newReceiverMessage = new Inbox({
|
|
ownerId: userToReceiveMessage._id,
|
|
});
|
|
Object.assign(newReceiverMessage, messageDefaults(options.receiverMsg, sender));
|
|
setUserStyles(newReceiverMessage, sender);
|
|
|
|
userToReceiveMessage.inbox.newMessages += 1;
|
|
userToReceiveMessage._v += 1;
|
|
|
|
/* @TODO disabled until mobile is ready
|
|
|
|
let excerpt;
|
|
|
|
if (!options.receiverMsg) {
|
|
excerpt = '';
|
|
} else if (options.receiverMsg.length < 100) {
|
|
excerpt = options.receiverMsg;
|
|
} else {
|
|
excerpt = options.receiverMsg.substring(0, 100);
|
|
}
|
|
userToReceiveMessage.addNotification('NEW_INBOX_MESSAGE', {
|
|
sender: {
|
|
id: sender._id,
|
|
name: sender.profile.name,
|
|
},
|
|
excerpt,
|
|
messageId: newMessage.id,
|
|
});
|
|
|
|
*/
|
|
|
|
const sendingToYourself = userToReceiveMessage._id === sender._id;
|
|
|
|
// Do not add the message twice when sending it to yourself
|
|
let newSenderMessage;
|
|
|
|
if (!sendingToYourself) {
|
|
newSenderMessage = new Inbox({
|
|
sent: true,
|
|
ownerId: sender._id,
|
|
});
|
|
Object.assign(newSenderMessage, messageDefaults(senderMsg, userToReceiveMessage));
|
|
setUserStyles(newSenderMessage, sender);
|
|
}
|
|
|
|
const promises = [newReceiverMessage.save()];
|
|
if (!sendingToYourself) promises.push(newSenderMessage.save());
|
|
|
|
if (saveUsers) {
|
|
promises.push(sender.save());
|
|
if (!sendingToYourself) promises.push(userToReceiveMessage.save());
|
|
}
|
|
|
|
await Promise.all(promises);
|
|
|
|
return sendingToYourself ? newReceiverMessage : newSenderMessage;
|
|
};
|
|
|
|
/**
|
|
* Creates a notification based on the input parameters and adds
|
|
* it to the local user notifications array.
|
|
* This does not save the notification to the database or interact with the database in any way.
|
|
*
|
|
* @param type The type of notification to add to the this.
|
|
* Possible values are defined in the UserNotificaiton Schema
|
|
* @param data The data to add to the notification
|
|
* @param seen If the notification should be marked as seen
|
|
*/
|
|
schema.methods.addNotification = function addUserNotification (type, data = {}, seen = false) {
|
|
this.notifications.push({
|
|
type,
|
|
data,
|
|
seen,
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Creates a notification based on the type and data input parameters
|
|
and saves that new notification
|
|
* to the database directly using an update statement.
|
|
* The local copy of these users are not updated by
|
|
* this operation. Use this function when you want to add a notification to a user(s),
|
|
* but do not have
|
|
* the user document(s) opened.
|
|
*
|
|
* @param query A Mongoose query defining the users to add the notification to.
|
|
* @param type The type of notification to add to the this.
|
|
* Possible values are defined in the UserNotificaiton Schema
|
|
* @param data The data to add to the notification
|
|
*/
|
|
schema.statics.pushNotification = async function pushNotification (
|
|
query, type, data = {}, seen = false,
|
|
) {
|
|
const newNotification = new UserNotification({ type, data, seen });
|
|
|
|
const validationResult = newNotification.validateSync();
|
|
if (validationResult) {
|
|
throw validationResult;
|
|
}
|
|
|
|
await this.update(
|
|
query,
|
|
{ $push: { notifications: newNotification.toObject() } },
|
|
{ multi: true },
|
|
).exec();
|
|
};
|
|
|
|
/**
|
|
* Adds an achievement and a related notification to the user.
|
|
*
|
|
* @param achievement The key identifying the achievement to award.
|
|
*/
|
|
schema.methods.addAchievement = function addAchievement (achievement) {
|
|
const achievementData = common.content.achievements[achievement];
|
|
if (!achievementData) throw new Error(`Achievement ${achievement} does not exist.`);
|
|
|
|
this.achievements[achievement] = true;
|
|
|
|
this.notifications.push({
|
|
type: 'ACHIEVEMENT',
|
|
data: {
|
|
achievement,
|
|
},
|
|
seen: false,
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Adds an achievement and a related notification to the user, saving it directly to the database
|
|
* To be used when the user object is not loaded or we don't want to use `user.save`
|
|
*
|
|
* @param query A Mongoose query defining the users to add the notification to.
|
|
* @param achievement The key identifying the achievement to award.
|
|
*/
|
|
schema.statics.addAchievementUpdate = async function addAchievementUpdate (query, achievement) {
|
|
const achievementData = common.content.achievements[achievement];
|
|
if (!achievementData) throw new Error(`Achievement ${achievement} does not exist.`);
|
|
|
|
const newNotification = new UserNotification({
|
|
type: 'ACHIEVEMENT',
|
|
data: {
|
|
achievement,
|
|
},
|
|
seen: false,
|
|
});
|
|
|
|
const validationResult = newNotification.validateSync();
|
|
if (validationResult) throw validationResult;
|
|
|
|
await this.update(
|
|
query,
|
|
{
|
|
$push: { notifications: newNotification.toObject() },
|
|
$set: { [`achievements.${achievement}`]: true },
|
|
},
|
|
{ multi: true },
|
|
).exec();
|
|
};
|
|
|
|
// Static method to add/remove properties to a JSON User object,
|
|
// For example for when the user is returned using `.lean()` and thus doesn't
|
|
// have access to any mongoose helper
|
|
schema.statics.transformJSONUser = function transformJSONUser (jsonUser, addComputedStats = false) {
|
|
// Add id property
|
|
jsonUser.id = jsonUser._id;
|
|
|
|
// Remove username if not verified
|
|
if (!jsonUser.flags.verifiedUsername) jsonUser.auth.local.username = null;
|
|
|
|
if (addComputedStats) this.addComputedStatsToJSONObj(jsonUser.stats, jsonUser);
|
|
};
|
|
|
|
// Returns true if the user has read the last news post
|
|
schema.methods.checkNewStuff = function checkNewStuff () {
|
|
const lastNewsPost = NewsPost.lastNewsPost();
|
|
return Boolean(lastNewsPost && this.flags && this.flags.lastNewStuffRead !== lastNewsPost._id);
|
|
};
|
|
|
|
// Add stats.toNextLevel, stats.maxMP and stats.maxHealth
|
|
// to a JSONified User stats object
|
|
schema.statics.addComputedStatsToJSONObj = function addComputedStatsToUserJSONObj (
|
|
userStatsJSON,
|
|
user,
|
|
) {
|
|
// NOTE: if an item is manually added to this.stats then
|
|
// common/fns/predictableRandom must be tweaked so the new item is not considered.
|
|
// Otherwise the client will have it while the server won't and the results will be different.
|
|
userStatsJSON.toNextLevel = common.tnl(user.stats.lvl);
|
|
userStatsJSON.maxHealth = common.maxHealth;
|
|
userStatsJSON.maxMP = common.statsComputed(user).maxMP;
|
|
|
|
return userStatsJSON;
|
|
};
|
|
|
|
/**
|
|
* Cancels a subscription.
|
|
*
|
|
* @param options
|
|
* @param options.user The user object who is purchasing
|
|
* @param options.groupId The id of the group purchasing a subscription
|
|
* @param options.headers The request headers (only for Amazon subscriptions)
|
|
* @param options.cancellationReason A text string to control sending an email
|
|
*
|
|
* @return a Promise from api.cancelSubscription()
|
|
*/
|
|
// @TODO: There is currently a three way relation between the user,
|
|
// payment methods and the payment helper
|
|
// This creates some odd Dependency Injection issues. To counter that,
|
|
// we use the user as the third layer
|
|
// To negotiate between the payment providers and the payment helper
|
|
// (which probably has too many responsibilities)
|
|
// In summary, currently is is best practice to use this method to cancel a user subscription,
|
|
// rather than calling the
|
|
// payment helper.
|
|
schema.methods.cancelSubscription = async function cancelSubscription (options = {}) {
|
|
const { plan } = this.purchased;
|
|
|
|
options.user = this;
|
|
if (plan.paymentMethod === amazonPayments.constants.PAYMENT_METHOD) {
|
|
return amazonPayments.cancelSubscription(options);
|
|
} if (plan.paymentMethod === stripePayments.constants.PAYMENT_METHOD) {
|
|
return stripePayments.cancelSubscription(options);
|
|
} if (plan.paymentMethod === paypalPayments.constants.PAYMENT_METHOD) {
|
|
return paypalPayments.subscribeCancel(options);
|
|
}
|
|
// Android and iOS subscriptions cannot be cancelled by Habitica.
|
|
|
|
return payments.cancelSubscription(options);
|
|
};
|
|
|
|
schema.methods.getUtcOffset = function getUtcOffset () {
|
|
return common.fns.getUtcOffset(this);
|
|
};
|
|
|
|
schema.methods.daysUserHasMissed = function daysUserHasMissed (now, req = {}) {
|
|
// If the user's timezone has changed (due to travel or daylight savings),
|
|
// cron can be triggered twice in one day, so we check for that and use
|
|
// both timezones to work out if cron should run.
|
|
// CDS = Custom Day Start time.
|
|
let timezoneUtcOffsetFromUserPrefs = this.getUtcOffset();
|
|
const timezoneUtcOffsetAtLastCron = Number.isFinite(this.preferences.timezoneOffsetAtLastCron)
|
|
? -this.preferences.timezoneOffsetAtLastCron
|
|
: timezoneUtcOffsetFromUserPrefs;
|
|
|
|
let timezoneUtcOffsetFromBrowser = typeof req.header === 'function' && -Number(req.header('x-user-timezoneoffset'));
|
|
timezoneUtcOffsetFromBrowser = Number.isFinite(timezoneUtcOffsetFromBrowser)
|
|
? timezoneUtcOffsetFromBrowser
|
|
: timezoneUtcOffsetFromUserPrefs;
|
|
// NB: All timezone offsets can be 0, so can't use `... || ...` to apply non-zero defaults
|
|
|
|
if (timezoneUtcOffsetFromBrowser !== timezoneUtcOffsetFromUserPrefs) {
|
|
// The user's browser has just told Habitica that the user's timezone has
|
|
// changed so store and use the new zone.
|
|
this.preferences.timezoneOffset = -timezoneUtcOffsetFromBrowser;
|
|
timezoneUtcOffsetFromUserPrefs = timezoneUtcOffsetFromBrowser;
|
|
}
|
|
|
|
// How many days have we missed using the user's current timezone:
|
|
let daysMissed = daysSince(this.lastCron, defaults({ now }, this.preferences));
|
|
|
|
if (timezoneUtcOffsetAtLastCron !== timezoneUtcOffsetFromUserPrefs) {
|
|
// Give the user extra time based on the difference in timezones
|
|
if (timezoneUtcOffsetAtLastCron > timezoneUtcOffsetFromUserPrefs) {
|
|
const differenceBetweenTimezonesInMinutes = timezoneUtcOffsetAtLastCron - timezoneUtcOffsetFromUserPrefs; // eslint-disable-line max-len
|
|
now = moment(now).subtract(differenceBetweenTimezonesInMinutes, 'minutes'); // eslint-disable-line no-param-reassign, max-len
|
|
}
|
|
|
|
// Since cron last ran, the user's timezone has changed.
|
|
// How many days have we missed using the old timezone:
|
|
const daysMissedNewZone = daysMissed;
|
|
const daysMissedOldZone = daysSince(this.lastCron, defaults({
|
|
now,
|
|
timezoneUtcOffsetOverride: timezoneUtcOffsetAtLastCron,
|
|
}, this.preferences));
|
|
|
|
if (timezoneUtcOffsetAtLastCron > timezoneUtcOffsetFromUserPrefs) {
|
|
// The timezone change was in the unsafe direction.
|
|
// E.g., timezone changes from UTC+1 (utcOffset 60) to UTC+0 (offset 0).
|
|
// or timezone changes from UTC-4 (utcOffset -240) to UTC-5 (utcOffset -300).
|
|
// Local time changed from, for example, 03:00 to 02:00.
|
|
|
|
if (daysMissedOldZone > 0 && daysMissedNewZone > 0) {
|
|
// Both old and new timezones indicate that we SHOULD run cron, so
|
|
// it is safe to do so immediately.
|
|
daysMissed = Math.min(daysMissedOldZone, daysMissedNewZone);
|
|
// use minimum value to be nice to user
|
|
} else if (daysMissedOldZone > 0) {
|
|
// The old timezone says that cron should run; the new timezone does not.
|
|
// This should be impossible for this direction of timezone change, but
|
|
// just in case I'm wrong...
|
|
// TODO
|
|
// console.log("zone has changed - old zone says run cron,
|
|
// NEW zone says no - stop cron now only -- SHOULD NOT HAVE GOT TO HERE",
|
|
// timezoneOffsetAtLastCron, timezoneOffsetFromUserPrefs, now);
|
|
// used in production for confirming this never happens
|
|
} else if (daysMissedNewZone > 0) {
|
|
// The old timezone says that cron should NOT run -- i.e., cron has
|
|
// already run today, from the old timezone's point of view.
|
|
// The new timezone says that cron SHOULD run, but this is almost
|
|
// certainly incorrect.
|
|
// This happens when cron occurred at a time soon after the CDS. When
|
|
// you reinterpret that time in the new timezone, it looks like it
|
|
// was before the CDS, because local time has stepped backwards.
|
|
// To fix this, rewrite the cron time to a time that the new
|
|
// timezone interprets as being in today.
|
|
|
|
daysMissed = 0; // prevent cron running now
|
|
const timezoneOffsetDiff = timezoneUtcOffsetFromUserPrefs - timezoneUtcOffsetAtLastCron;
|
|
// e.g., for dangerous zone change: -300 - -240 = -60 or 600 - 660= -60
|
|
|
|
this.lastCron = moment(this.lastCron).subtract(timezoneOffsetDiff, 'minutes');
|
|
// NB: We don't change this.auth.timestamps.loggedin so that will still record
|
|
// the time that the previous cron actually ran.
|
|
// From now on we can ignore the old timezone:
|
|
// This is still timezoneOffset for backwards compatibility reasons.
|
|
this.preferences.timezoneOffsetAtLastCron = -timezoneUtcOffsetAtLastCron;
|
|
} else {
|
|
// Both old and new timezones indicate that cron should
|
|
// NOT run.
|
|
daysMissed = 0; // prevent cron running now
|
|
}
|
|
} else if (timezoneUtcOffsetAtLastCron < timezoneUtcOffsetFromUserPrefs) {
|
|
daysMissed = daysMissedNewZone;
|
|
// TODO: Either confirm that there is nothing that could possibly go wrong
|
|
// here and remove the need for this else branch, or fix stuff.
|
|
// There are probably situations where the Dailies do not reset early enough
|
|
// for a user who was expecting the zone change and wants to use all their Dailies
|
|
// immediately in the new zone;
|
|
// if so, we should provide an option for easy reset of Dailies
|
|
// (can't be automatic because there will be other situations where
|
|
// the user was not prepared).
|
|
}
|
|
}
|
|
|
|
return { daysMissed, timezoneUtcOffsetFromUserPrefs };
|
|
};
|
|
|
|
async function getUserGroupData (user) {
|
|
const userGroups = user.getGroups();
|
|
|
|
const groups = await Group
|
|
.find({
|
|
_id: { $in: userGroups },
|
|
})
|
|
.select('leaderOnly leader purchased')
|
|
.exec();
|
|
|
|
return groups;
|
|
}
|
|
|
|
// Determine if the user can get gems: some groups restrict their members ability to obtain them.
|
|
// User is allowed to buy gems if no group has `leaderOnly.getGems` === true or if
|
|
// its the group leader
|
|
schema.methods.canGetGems = async function canObtainGems () {
|
|
const user = this;
|
|
const { plan } = user.purchased;
|
|
|
|
if (!user.isSubscribed() || plan.customerId !== payments.constants.GROUP_PLAN_CUSTOMER_ID) {
|
|
return true;
|
|
}
|
|
|
|
const groups = await getUserGroupData(user);
|
|
|
|
return groups
|
|
.every(g => !g.hasActiveGroupPlan() || g.leader === user._id || g.leaderOnly.getGems !== true);
|
|
};
|
|
|
|
schema.methods.isMemberOfGroupPlan = async function isMemberOfGroupPlan () {
|
|
const groups = await getUserGroupData(this);
|
|
|
|
return groups.some(g => g.hasActiveGroupPlan());
|
|
};
|
|
|
|
schema.methods.isAdmin = function isAdmin () {
|
|
return Boolean(this.contributor && this.contributor.admin);
|
|
};
|
|
|
|
schema.methods.isNewsPoster = function isNewsPoster () {
|
|
return Boolean(this.contributor && this.contributor.newsPoster);
|
|
};
|
|
|
|
// When converting to json add inbox messages from the Inbox collection
|
|
// for backward compatibility in API v3.
|
|
schema.methods.toJSONWithInbox = async function userToJSONWithInbox () {
|
|
const user = this;
|
|
const toJSON = user.toJSON();
|
|
|
|
if (toJSON.inbox) {
|
|
toJSON.inbox.messages = await inboxLib.getUserInbox(user, {
|
|
asArray: false,
|
|
});
|
|
}
|
|
|
|
return toJSON;
|
|
};
|
|
|
|
schema.methods.getSecretData = function getSecretData () {
|
|
const user = this;
|
|
|
|
return user.secret;
|
|
};
|