mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-16 06:07:21 +01:00
* begin moving to new fcm library * Add error handling * Add opening notification to correct screen * Fix tests and make async * lint fix * Rename pushNotificationstest..js to pushNotifications.test.js * fix(potions): remove Fungi Potion time banner * 5.24.3 * update(content): add 2024-06 content prebuild (#15231) * update sprites * add 2024-06 content * add 2024-06 enchanted armoire items * update sprites * update sprites * fix errors found in testing * Fix liveliness probes being rate limited (#15236) * Do not rate limit any liveliness probes * update example config * Translated using Weblate (German) Currently translated at 96.2% (181 of 188 strings) Translated using Weblate (Japanese) Currently translated at 99.4% (769 of 773 strings) Translated using Weblate (German) Currently translated at 93.6% (176 of 188 strings) Translated using Weblate (Japanese) Currently translated at 96.2% (2972 of 3089 strings) Translated using Weblate (Japanese) Currently translated at 100.0% (232 of 232 strings) Translated using Weblate (Japanese) Currently translated at 96.8% (841 of 868 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (94 of 94 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (113 of 113 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (131 of 131 strings) Translated using Weblate (Japanese) Currently translated at 100.0% (286 of 286 strings) Translated using Weblate (German) Currently translated at 86.7% (163 of 188 strings) Translated using Weblate (German) Currently translated at 85.1% (160 of 188 strings) Translated using Weblate (German) Currently translated at 84.0% (158 of 188 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (94 of 94 strings) Translated using Weblate (German) Currently translated at 83.5% (157 of 188 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (113 of 113 strings) Translated using Weblate (German) Currently translated at 82.9% (156 of 188 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (113 of 113 strings) Translated using Weblate (German) Currently translated at 81.9% (154 of 188 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (113 of 113 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (8 of 8 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (8 of 8 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (188 of 188 strings) Translated using Weblate (German) Currently translated at 79.2% (149 of 188 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (189 of 189 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (8 of 8 strings) Translated using Weblate (German) Currently translated at 90.6% (2799 of 3089 strings) Translated using Weblate (German) Currently translated at 77.6% (146 of 188 strings) Translated using Weblate (German) Currently translated at 90.5% (2797 of 3089 strings) Translated using Weblate (German) Currently translated at 90.4% (2794 of 3089 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (131 of 131 strings) Translated using Weblate (German) Currently translated at 90.1% (2786 of 3089 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (188 of 188 strings) Translated using Weblate (German) Currently translated at 77.1% (145 of 188 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 98.7% (763 of 773 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (868 of 868 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (131 of 131 strings) Translated using Weblate (Spanish) Currently translated at 100.0% (286 of 286 strings) Translated using Weblate (German) Currently translated at 90.0% (2782 of 3089 strings) Translated using Weblate (Spanish) Currently translated at 100.0% (773 of 773 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (378 of 378 strings) Translated using Weblate (Spanish) Currently translated at 100.0% (868 of 868 strings) Translated using Weblate (Spanish) Currently translated at 100.0% (167 of 167 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (259 of 259 strings) Translated using Weblate (Japanese) Currently translated at 100.0% (259 of 259 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (131 of 131 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (286 of 286 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (239 of 239 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (188 of 188 strings) Translated using Weblate (French) Currently translated at 100.0% (188 of 188 strings) Translated using Weblate (German) Currently translated at 75.0% (141 of 188 strings) Translated using Weblate (Spanish) Currently translated at 99.0% (766 of 773 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (189 of 189 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (868 of 868 strings) Translated using Weblate (Japanese) Currently translated at 98.8% (764 of 773 strings) Translated using Weblate (Japanese) Currently translated at 99.6% (258 of 259 strings) Translated using Weblate (Japanese) Currently translated at 100.0% (378 of 378 strings) Translated using Weblate (Japanese) Currently translated at 100.0% (188 of 188 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (140 of 140 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (131 of 131 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (286 of 286 strings) Translated using Weblate (Ukrainian) Currently translated at 62.5% (1931 of 3089 strings) Translated using Weblate (German) Currently translated at 89.8% (2777 of 3089 strings) Translated using Weblate (French) Currently translated at 100.0% (188 of 188 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 98.5% (762 of 773 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (868 of 868 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (286 of 286 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (286 of 286 strings) Translated using Weblate (French) Currently translated at 82.9% (156 of 188 strings) Translated using Weblate (German) Currently translated at 93.0% (241 of 259 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (286 of 286 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (8 of 8 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (427 of 427 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 98.5% (762 of 773 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (868 of 868 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (167 of 167 strings) Translated using Weblate (Japanese) Currently translated at 99.2% (257 of 259 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 98.5% (762 of 773 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (868 of 868 strings) Translated using Weblate (German) Currently translated at 92.2% (239 of 259 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (286 of 286 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (239 of 239 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 100.0% (188 of 188 strings) Translated using Weblate (German) Currently translated at 91.8% (238 of 259 strings) Translated using Weblate (Ukrainian) Currently translated at 100.0% (131 of 131 strings) Translated using Weblate (Chinese (Simplified)) Currently translated at 98.5% (762 of 773 strings) Translated using Weblate (German) Currently translated at 90.3% (234 of 259 strings) Co-authored-by: Finrod <963505255@qq.com> Co-authored-by: Jaime Martí <jaumemarti77@icloud.com> Co-authored-by: Kem Kembo <medamamef@gmail.com> Co-authored-by: Sophie LE MASLE <sophiesuff@gmail.com> Co-authored-by: TOMA Mitsuru <toma0001@gmail.com> Co-authored-by: Tetiana <merekka13@gmail.com> Co-authored-by: Toro Mor <thomas.bizer@gmx.de> Co-authored-by: Weblate <noreply@weblate.org> Translate-URL: https://translate.habitica.com/projects/habitica/achievements/es/ Translate-URL: https://translate.habitica.com/projects/habitica/achievements/zh_Hans/ Translate-URL: https://translate.habitica.com/projects/habitica/backgrounds/es/ Translate-URL: https://translate.habitica.com/projects/habitica/backgrounds/ja/ Translate-URL: https://translate.habitica.com/projects/habitica/backgrounds/zh_Hans/ Translate-URL: https://translate.habitica.com/projects/habitica/character/uk/ Translate-URL: https://translate.habitica.com/projects/habitica/character/zh_Hans/ Translate-URL: https://translate.habitica.com/projects/habitica/content/ja/ Translate-URL: https://translate.habitica.com/projects/habitica/content/uk/ Translate-URL: https://translate.habitica.com/projects/habitica/faq/de/ Translate-URL: https://translate.habitica.com/projects/habitica/faq/fr/ Translate-URL: https://translate.habitica.com/projects/habitica/faq/ja/ Translate-URL: https://translate.habitica.com/projects/habitica/faq/zh_Hans/ Translate-URL: https://translate.habitica.com/projects/habitica/gear/de/ Translate-URL: https://translate.habitica.com/projects/habitica/gear/ja/ Translate-URL: https://translate.habitica.com/projects/habitica/gear/uk/ Translate-URL: https://translate.habitica.com/projects/habitica/generic/zh_Hans/ Translate-URL: https://translate.habitica.com/projects/habitica/groups/zh_Hans/ Translate-URL: https://translate.habitica.com/projects/habitica/inventory/uk/ Translate-URL: https://translate.habitica.com/projects/habitica/limited/es/ Translate-URL: https://translate.habitica.com/projects/habitica/limited/ja/ Translate-URL: https://translate.habitica.com/projects/habitica/limited/uk/ Translate-URL: https://translate.habitica.com/projects/habitica/limited/zh_Hans/ Translate-URL: https://translate.habitica.com/projects/habitica/npc/uk/ Translate-URL: https://translate.habitica.com/projects/habitica/overview/uk/ Translate-URL: https://translate.habitica.com/projects/habitica/pets/uk/ Translate-URL: https://translate.habitica.com/projects/habitica/quests/uk/ Translate-URL: https://translate.habitica.com/projects/habitica/questscontent/es/ Translate-URL: https://translate.habitica.com/projects/habitica/questscontent/ja/ Translate-URL: https://translate.habitica.com/projects/habitica/questscontent/zh_Hans/ Translate-URL: https://translate.habitica.com/projects/habitica/settings/de/ Translate-URL: https://translate.habitica.com/projects/habitica/settings/ja/ Translate-URL: https://translate.habitica.com/projects/habitica/settings/uk/ Translate-URL: https://translate.habitica.com/projects/habitica/subscriber/ja/ Translate-URL: https://translate.habitica.com/projects/habitica/tasks/zh_Hans/ Translation: Habitica/Achievements Translation: Habitica/Backgrounds Translation: Habitica/Character Translation: Habitica/Content Translation: Habitica/Faq Translation: Habitica/Gear Translation: Habitica/Generic Translation: Habitica/Groups Translation: Habitica/Inventory Translation: Habitica/Limited Translation: Habitica/Npc Translation: Habitica/Overview Translation: Habitica/Pets Translation: Habitica/Quests Translation: Habitica/Questscontent Translation: Habitica/Settings Translation: Habitica/Subscriber Translation: Habitica/Tasks * 5.25.0 * Fix dockerfile (#15241) * Fix issue with l4p not resetting properly (#15240) * actually clear out seeking field on user. Even when creating a party * Add tests to ensure party.seeking is cleared * fix(lint): don't assign unused const --------- Co-authored-by: Sabe Jones <sabe@habitica.com> --------- Co-authored-by: Sabe Jones <sabe@habitica.com> Co-authored-by: Natalie <78037386+CuriousMagpie@users.noreply.github.com> Co-authored-by: Weblate <noreply@weblate.org> Co-authored-by: Finrod <963505255@qq.com> Co-authored-by: Jaime Martí <jaumemarti77@icloud.com> Co-authored-by: Kem Kembo <medamamef@gmail.com> Co-authored-by: Sophie LE MASLE <sophiesuff@gmail.com> Co-authored-by: TOMA Mitsuru <toma0001@gmail.com> Co-authored-by: Tetiana <merekka13@gmail.com> Co-authored-by: Toro Mor <thomas.bizer@gmx.de> Co-authored-by: Rafał Jagielski <jagielski.rafal.uwm@gmail.com>
488 lines
15 KiB
JavaScript
488 lines
15 KiB
JavaScript
// TODO these files need to refactored.
|
|
|
|
import _ from 'lodash';
|
|
import moment from 'moment';
|
|
|
|
import { getAnalyticsServiceByEnvironment } from '../analyticsService';
|
|
import * as slack from '../slack'; // eslint-disable-line import/no-cycle
|
|
import { // eslint-disable-line import/no-cycle
|
|
getUserInfo,
|
|
sendTxn as txnEmail,
|
|
} from '../email';
|
|
import { // eslint-disable-line import/no-cycle
|
|
model as Group,
|
|
basicFields as basicGroupFields,
|
|
} from '../../models/group';
|
|
import {
|
|
NotAuthorized,
|
|
NotFound,
|
|
} from '../errors';
|
|
import shared from '../../../common';
|
|
import { sendNotification as sendPushNotification } from '../pushNotifications'; // eslint-disable-line import/no-cycle
|
|
import calculateSubscriptionTerminationDate from './calculateSubscriptionTerminationDate';
|
|
import { getCurrentEventList } from '../worldState'; // eslint-disable-line import/no-cycle
|
|
import { paymentConstants } from './constants';
|
|
import { addSubscriptionToGroupUsers, cancelGroupUsersSubscription } from './groupPayments'; // eslint-disable-line import/no-cycle
|
|
|
|
// @TODO: Abstract to shared/constant
|
|
const JOINED_GROUP_PLAN = 'joined group plan';
|
|
const analytics = getAnalyticsServiceByEnvironment();
|
|
|
|
function _findMysteryItems (user, dateMoment) {
|
|
const pushedItems = [];
|
|
_.each(shared.content.gear.flat, item => {
|
|
if (
|
|
item.klass === 'mystery'
|
|
&& shared.content.mystery[item.mystery]
|
|
&& dateMoment.isSameOrAfter(shared.content.mystery[item.mystery].start)
|
|
&& dateMoment.isSameOrBefore(moment(shared.content.mystery[item.mystery].end).endOf('day'))
|
|
&& !user.items.gear.owned[item.key]
|
|
&& user.purchased.plan.mysteryItems.indexOf(item.key) === -1
|
|
) {
|
|
user.purchased.plan.mysteryItems.push(item.key);
|
|
pushedItems.push(item.key);
|
|
}
|
|
});
|
|
return pushedItems;
|
|
}
|
|
|
|
function revealMysteryItems (user, elapsedMonths = 1) {
|
|
let monthsToCheck = elapsedMonths;
|
|
let pushedItems = [];
|
|
|
|
do {
|
|
monthsToCheck -= 1;
|
|
pushedItems = pushedItems.concat(_findMysteryItems(user, moment().subtract(monthsToCheck, 'months')));
|
|
}
|
|
while (monthsToCheck > 0);
|
|
|
|
if (pushedItems.length > 0) {
|
|
user.addNotification('NEW_MYSTERY_ITEMS', { items: pushedItems });
|
|
}
|
|
}
|
|
|
|
// @TODO: Abstract to payment helper
|
|
function _dateDiff (earlyDate, lateDate) {
|
|
if (!earlyDate || !lateDate || moment(lateDate).isBefore(earlyDate)) return 0;
|
|
|
|
return moment(lateDate).diff(earlyDate, 'months', true);
|
|
}
|
|
|
|
async function prepareSubscriptionValues (data) {
|
|
let recipient = data.gift ? data.gift.member : data.user;
|
|
const block = shared.content.subscriptionBlocks[data.gift
|
|
? data.gift.subscription.key
|
|
: data.sub.key];
|
|
const autoRenews = data.autoRenews !== undefined ? data.autoRenews : true;
|
|
const updatedFrom = data.updatedFrom
|
|
? shared.content.subscriptionBlocks[data.updatedFrom.key]
|
|
: undefined;
|
|
let months;
|
|
if (updatedFrom && Number(updatedFrom.months) !== 1) {
|
|
if (Number(updatedFrom.months) > Number(block.months)) {
|
|
months = 0;
|
|
} else if (data.updatedFrom.logic === 'payDifference') {
|
|
months = Math.max(0, Number(block.months) - Number(updatedFrom.months));
|
|
} else if (data.updatedFrom.logic === 'payFull') {
|
|
months = Number(block.months);
|
|
} else if (data.updatedFrom.logic === 'refundAndRepay') {
|
|
const originalMonths = Number(updatedFrom.months);
|
|
let currentCycleBegin = moment(recipient.purchased.plan.dateCurrentTypeCreated);
|
|
const today = moment();
|
|
while (currentCycleBegin.isBefore()) {
|
|
currentCycleBegin = currentCycleBegin.add({ months: originalMonths });
|
|
}
|
|
// Subtract last iteration again, because we overshot
|
|
currentCycleBegin = currentCycleBegin.subtract({ months: originalMonths });
|
|
// For simplicity we round every month to 30 days since moment can not add half months
|
|
if (currentCycleBegin.add({ days: (originalMonths * 30) / 2.0 }).isBefore(today)) {
|
|
// user is in second half of their subscription cycle. Give them full benefits.
|
|
months = Number(block.months);
|
|
} else {
|
|
// user is in first half of their subscription cycle. Give them the difference.
|
|
months = Math.max(0, Number(block.months) - Number(updatedFrom.months));
|
|
}
|
|
}
|
|
}
|
|
if (months === undefined) {
|
|
months = Number(block.months);
|
|
}
|
|
const today = new Date();
|
|
let group;
|
|
let groupId;
|
|
let itemPurchased = 'Subscription';
|
|
let purchaseType = 'subscribe';
|
|
let emailType = 'subscription-begins';
|
|
let recipientIsSubscribed = recipient.isSubscribed();
|
|
const isNewSubscription = !recipientIsSubscribed;
|
|
|
|
// If we are buying a group subscription
|
|
if (data.groupId) {
|
|
const groupFields = basicGroupFields.concat(' purchased');
|
|
group = await Group.getGroup({
|
|
user: data.user, groupId: data.groupId, populateLeader: false, groupFields,
|
|
});
|
|
|
|
if (group) {
|
|
analytics.track(
|
|
data.groupID,
|
|
data.demographics,
|
|
);
|
|
}
|
|
|
|
if (!group) {
|
|
throw new NotFound(shared.i18n.t('groupNotFound'));
|
|
}
|
|
|
|
if (!group.leader === data.user._id) {
|
|
throw new NotAuthorized(shared.i18n.t('onlyGroupLeaderCanManageSubscription'));
|
|
}
|
|
|
|
if (group.privacy !== 'private') {
|
|
throw new NotAuthorized(shared.i18n.t('onlyPrivateGuildsCanUpgrade'));
|
|
}
|
|
|
|
recipient = group;
|
|
itemPurchased = 'Group-Subscription';
|
|
purchaseType = 'group-subscribe';
|
|
emailType = 'group-subscription-begins';
|
|
recipientIsSubscribed = group.hasActiveGroupPlan();
|
|
groupId = group._id;
|
|
recipient.purchased.plan.quantity = data.sub.quantity;
|
|
|
|
await addSubscriptionToGroupUsers(group);
|
|
}
|
|
|
|
const { plan } = recipient.purchased;
|
|
|
|
if (isNewSubscription) {
|
|
plan.perkMonthCount = 0;
|
|
}
|
|
|
|
if (data.gift || !autoRenews) {
|
|
if (plan.customerId && !plan.dateTerminated) { // User has active plan
|
|
plan.extraMonths += months;
|
|
} else {
|
|
if (!recipientIsSubscribed || !plan.dateUpdated) {
|
|
plan.dateUpdated = today;
|
|
}
|
|
|
|
if (moment(plan.dateTerminated).isAfter()) {
|
|
plan.dateTerminated = moment(plan.dateTerminated).add({ months }).toDate();
|
|
} else {
|
|
plan.dateTerminated = moment().add({ months }).toDate();
|
|
plan.dateCreated = today;
|
|
}
|
|
plan.dateCurrentTypeCreated = today;
|
|
}
|
|
|
|
if (!plan.customerId) {
|
|
plan.customerId = 'Gift';
|
|
}
|
|
|
|
// don't override existing customer, but all sub need a customerId
|
|
} else {
|
|
if (!plan.dateTerminated) {
|
|
plan.dateTerminated = today;
|
|
}
|
|
|
|
Object.assign(plan, { // override plan with new values
|
|
planId: block.key,
|
|
customerId: data.customerId,
|
|
dateUpdated: today,
|
|
dateCurrentTypeCreated: today,
|
|
paymentMethod: data.paymentMethod,
|
|
extraMonths: Number(plan.extraMonths) + _dateDiff(today, plan.dateTerminated),
|
|
dateTerminated: null,
|
|
// Specify a lastBillingDate just for Amazon Payments
|
|
// Resetted every time the subscription restarts
|
|
lastBillingDate: data.paymentMethod === 'Amazon Payments' ? today : undefined,
|
|
nextPaymentProcessing: data.nextPaymentProcessing,
|
|
nextBillingDate: data.nextBillingDate,
|
|
additionalData: data.additionalData,
|
|
lastReminderDate: null,
|
|
owner: data.user._id,
|
|
});
|
|
|
|
// allow non-override if a plan was previously used
|
|
if (!plan.gemsBought) {
|
|
plan.gemsBought = 0;
|
|
}
|
|
|
|
if (!plan.dateCreated) {
|
|
plan.dateCreated = today;
|
|
}
|
|
|
|
if (!plan.mysteryItems) {
|
|
plan.mysteryItems = [];
|
|
}
|
|
|
|
if (data.subscriptionId) {
|
|
plan.subscriptionId = data.subscriptionId;
|
|
}
|
|
}
|
|
|
|
return {
|
|
block,
|
|
months,
|
|
plan,
|
|
recipient,
|
|
autoRenews,
|
|
group,
|
|
groupId,
|
|
itemPurchased,
|
|
purchaseType,
|
|
emailType,
|
|
isNewSubscription,
|
|
};
|
|
}
|
|
|
|
async function createSubscription (data) {
|
|
const {
|
|
block,
|
|
months,
|
|
plan,
|
|
recipient,
|
|
autoRenews,
|
|
group,
|
|
groupId,
|
|
itemPurchased,
|
|
purchaseType,
|
|
emailType,
|
|
isNewSubscription,
|
|
} = await prepareSubscriptionValues(data);
|
|
|
|
// Block sub perks
|
|
if (months > 1 && (!data.gift || !isNewSubscription)) {
|
|
if (!data.gift && !groupId) {
|
|
plan.consecutive.offset = block.months;
|
|
}
|
|
} else if (months === 1) {
|
|
plan.consecutive.offset = 0;
|
|
}
|
|
if (months > 1 || data.gift) {
|
|
await plan.incrementPerkCounterAndReward(recipient._id, months);
|
|
} else {
|
|
// Make sure the perkMonthCount field is initialized.
|
|
await plan.incrementPerkCounterAndReward(recipient._id, 0);
|
|
}
|
|
|
|
if (recipient !== group) {
|
|
recipient.items.pets['Jackalope-RoyalPurple'] = 5;
|
|
recipient.markModified('items.pets');
|
|
revealMysteryItems(recipient);
|
|
}
|
|
|
|
// @TODO: Create a factory pattern for use cases
|
|
if (!data.gift && data.customerId !== paymentConstants.GROUP_PLAN_CUSTOMER_ID) {
|
|
txnEmail(data.user, emailType);
|
|
}
|
|
|
|
if (!group && !data.promo) data.user.purchased.txnCount += 1;
|
|
|
|
if (!data.promo) {
|
|
analytics.trackPurchase({
|
|
uuid: data.user._id,
|
|
groupId,
|
|
itemPurchased,
|
|
sku: `${data.paymentMethod.toLowerCase()}-subscription`,
|
|
purchaseType,
|
|
paymentMethod: data.paymentMethod,
|
|
quantity: 1,
|
|
gift: Boolean(data.gift),
|
|
purchaseValue: block.price,
|
|
headers: data.headers || { 'x-client': 'habitica-web' },
|
|
firstPurchase: !group && data.user.purchased.txnCount === 1,
|
|
});
|
|
}
|
|
|
|
if (data.gift) {
|
|
const byUserName = getUserInfo(data.user, ['name']).name;
|
|
|
|
// generate the message in both languages, so both users can understand it
|
|
const languages = [data.user.preferences.language, data.gift.member.preferences.language];
|
|
if (data.promo) {
|
|
let receiverMsg = shared.i18n.t(`giftedSubscription${data.promo}Promo`, {
|
|
username: data.gift.member.profile.name,
|
|
monthCount: shared.content.subscriptionBlocks[data.gift.subscription.key].months,
|
|
}, languages[0]);
|
|
|
|
receiverMsg = `\`${receiverMsg}\``;
|
|
data.user.sendMessage(data.gift.member, { receiverMsg, save: false });
|
|
} else {
|
|
let senderMsg = shared.i18n.t('giftedSubscriptionFull', {
|
|
username: data.gift.member.profile.name,
|
|
sender: byUserName,
|
|
monthCount: shared.content.subscriptionBlocks[data.gift.subscription.key].months,
|
|
}, languages[0]);
|
|
senderMsg = `\`${senderMsg}\``;
|
|
|
|
let receiverMsg = shared.i18n.t('giftedSubscriptionFull', {
|
|
username: data.gift.member.profile.name,
|
|
sender: byUserName,
|
|
monthCount: shared.content.subscriptionBlocks[data.gift.subscription.key].months,
|
|
}, languages[1]);
|
|
receiverMsg = `\`${receiverMsg}\``;
|
|
|
|
if (data.gift.message) {
|
|
receiverMsg += ` ${data.gift.message}`;
|
|
senderMsg += ` ${data.gift.message}`;
|
|
}
|
|
|
|
data.user.sendMessage(data.gift.member, { receiverMsg, senderMsg, save: false });
|
|
}
|
|
|
|
if (data.gift.member.preferences.emailNotifications.giftedSubscription !== false) {
|
|
if (data.promo) {
|
|
txnEmail(data.gift.member, 'gift-one-get-one', [
|
|
{ name: 'GIFTEE_USERNAME', content: data.promoUsername },
|
|
{ name: 'X_MONTHS_SUBSCRIPTION', content: months },
|
|
]);
|
|
} else {
|
|
txnEmail(data.gift.member, 'gifted-subscription', [
|
|
{ name: 'GIFTER', content: byUserName },
|
|
{ name: 'X_MONTHS_SUBSCRIPTION', content: months },
|
|
]);
|
|
}
|
|
}
|
|
|
|
// Only send push notifications if sending to a user other than yourself
|
|
if (data.gift.member._id !== data.user._id) {
|
|
const currentEventList = getCurrentEventList();
|
|
const currentEvent = _.find(currentEventList, event => Boolean(event.promo));
|
|
if (currentEvent && currentEvent.promo === 'g1g1') {
|
|
const promoData = {
|
|
user: data.user,
|
|
gift: {
|
|
member: data.user,
|
|
subscription: {
|
|
key: data.gift.subscription.key,
|
|
},
|
|
},
|
|
paymentMethod: data.paymentMethod,
|
|
promo: 'Winter',
|
|
promoUsername: data.gift.member.auth.local.username,
|
|
};
|
|
await createSubscription(promoData);
|
|
}
|
|
|
|
if (data.gift.member.preferences.pushNotifications.giftedSubscription !== false) {
|
|
await sendPushNotification(
|
|
data.gift.member,
|
|
{
|
|
title: shared.i18n.t('giftedSubscription', languages[1]),
|
|
message: shared.i18n.t('giftedSubscriptionInfo', { months, name: byUserName }, languages[1]),
|
|
identifier: 'giftedSubscription',
|
|
payload: { replyTo: data.user._id },
|
|
},
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (group) await group.save();
|
|
if (data.user && data.user.isModified()) await data.user.save();
|
|
if (data.gift) await data.gift.member.save();
|
|
|
|
slack.sendSubscriptionNotification({
|
|
buyer: {
|
|
id: data.user._id,
|
|
name: data.user.profile.name,
|
|
email: getUserInfo(data.user, ['email']).email,
|
|
},
|
|
recipient: data.gift ? {
|
|
id: data.gift.member._id,
|
|
name: data.gift.member.profile.name,
|
|
email: getUserInfo(data.gift.member, ['email']).email,
|
|
} : {},
|
|
paymentMethod: data.paymentMethod,
|
|
months: group ? 1 : months,
|
|
groupId,
|
|
autoRenews,
|
|
});
|
|
}
|
|
|
|
// Cancels a subscription or group plan, setting termination to happen later
|
|
async function cancelSubscription (data) {
|
|
let plan;
|
|
let group;
|
|
let cancelType = 'unsubscribe';
|
|
let groupId;
|
|
let emailType;
|
|
const emailMergeData = [];
|
|
let sendEmail = true;
|
|
|
|
if (data.groupId) {
|
|
// cancelling a group plan
|
|
const groupFields = basicGroupFields.concat(' purchased');
|
|
group = await Group.getGroup({
|
|
user: data.user, groupId: data.groupId, populateLeader: false, groupFields,
|
|
});
|
|
|
|
if (!group) {
|
|
throw new NotFound(shared.i18n.t('groupNotFound'));
|
|
}
|
|
|
|
const allowedManagers = [group.leader, group.purchased.plan.owner];
|
|
|
|
if (allowedManagers.indexOf(data.user._id) === -1) {
|
|
throw new NotAuthorized(shared.i18n.t('onlyGroupLeaderCanManageSubscription'));
|
|
}
|
|
plan = group.purchased.plan;
|
|
emailType = 'group-cancel-subscription';
|
|
emailMergeData.push({ name: 'GROUP_NAME', content: group.name });
|
|
|
|
await cancelGroupUsersSubscription(group);
|
|
} else {
|
|
// cancelling a user subscription
|
|
plan = data.user.purchased.plan;
|
|
emailType = 'cancel-subscription';
|
|
// When cancelling because the user joined a group plan, no cancel-subscription email is sent
|
|
// because the group-member-join email says the subscription is cancelled.
|
|
if (data.cancellationReason && data.cancellationReason === JOINED_GROUP_PLAN) sendEmail = false;
|
|
}
|
|
|
|
if (plan.customerId === paymentConstants.GROUP_PLAN_CUSTOMER_ID) {
|
|
sendEmail = false; // because group-member-cancel email has already been sent
|
|
}
|
|
|
|
plan.dateTerminated = calculateSubscriptionTerminationDate(
|
|
data.nextBill,
|
|
plan,
|
|
paymentConstants.GROUP_PLAN_CUSTOMER_ID,
|
|
);
|
|
|
|
// clear extra time. If they subscribe again, it'll be recalculated from p.dateTerminated
|
|
plan.extraMonths = 0;
|
|
|
|
if (group) {
|
|
await group.save();
|
|
} else {
|
|
await data.user.save();
|
|
}
|
|
|
|
if (sendEmail) {
|
|
txnEmail(data.user, emailType, emailMergeData);
|
|
}
|
|
|
|
if (group) {
|
|
cancelType = 'group-unsubscribe';
|
|
groupId = group._id;
|
|
}
|
|
|
|
analytics.track(cancelType, {
|
|
uuid: data.user._id,
|
|
groupId,
|
|
gaCategory: 'commerce',
|
|
gaLabel: data.paymentMethod,
|
|
paymentMethod: data.paymentMethod,
|
|
headers: data.headers,
|
|
});
|
|
}
|
|
|
|
export {
|
|
createSubscription,
|
|
cancelSubscription,
|
|
revealMysteryItems,
|
|
};
|