mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-15 13:47:33 +01:00
244 lines
7.8 KiB
JavaScript
244 lines
7.8 KiB
JavaScript
import moment from 'moment';
|
|
|
|
import logger from '../logger';
|
|
import {
|
|
BadRequest,
|
|
NotAuthorized,
|
|
NotFound,
|
|
} from '../errors';
|
|
import payments from './payments'; // eslint-disable-line import/no-cycle
|
|
import { model as User } from '../../models/user'; // eslint-disable-line import/no-cycle
|
|
import { // eslint-disable-line import/no-cycle
|
|
model as Group,
|
|
basicFields as basicGroupFields,
|
|
} from '../../models/group';
|
|
import shared from '../../../common';
|
|
import stripeConstants from './stripe/constants';
|
|
import { checkout } from './stripe/checkout'; // eslint-disable-line import/no-cycle
|
|
import { getStripeApi, setStripeApi } from './stripe/api';
|
|
|
|
const { i18n } = shared;
|
|
|
|
const api = {};
|
|
|
|
api.constants = { ...stripeConstants };
|
|
|
|
api.setStripeApi = setStripeApi;
|
|
|
|
/**
|
|
* Allows for purchasing a user subscription, group subscription or gems with Stripe
|
|
*
|
|
* @param options
|
|
* @param options.token The stripe token generated on the front end
|
|
* @param options.user The user object who is purchasing
|
|
* @param options.gift The gift details if any
|
|
* @param options.sub The subscription data to purchase
|
|
* @param options.groupId The id of the group purchasing a subscription
|
|
* @param options.email The email enter by the user on the Stripe form
|
|
* @param options.headers The request headers to store on analytics
|
|
* @return undefined
|
|
*/
|
|
api.checkout = checkout;
|
|
|
|
/**
|
|
* Edits a subscription created by Stripe
|
|
*
|
|
* @param options
|
|
* @param options.token The stripe token generated on the front end
|
|
* @param options.user The user object who is purchasing
|
|
* @param options.groupId The id of the group purchasing a subscription
|
|
*
|
|
* @return undefined
|
|
*/
|
|
api.editSubscription = async function editSubscription (options, stripeInc) {
|
|
const { token, groupId, user } = options;
|
|
let customerId;
|
|
|
|
// @TODO: We need to mock this, but curently we don't have correct
|
|
// Dependency Injection. And the Stripe Api doesn't seem to be a singleton?
|
|
let stripeApi = getStripeApi();
|
|
if (stripeInc) stripeApi = stripeInc;
|
|
|
|
if (groupId) {
|
|
const groupFields = basicGroupFields.concat(' purchased');
|
|
const group = await Group.getGroup({
|
|
user, groupId, populateLeader: false, groupFields,
|
|
});
|
|
|
|
if (!group) {
|
|
throw new NotFound(i18n.t('groupNotFound'));
|
|
}
|
|
|
|
const allowedManagers = [group.leader, group.purchased.plan.owner];
|
|
|
|
if (allowedManagers.indexOf(user._id) === -1) {
|
|
throw new NotAuthorized(i18n.t('onlyGroupLeaderCanManageSubscription'));
|
|
}
|
|
customerId = group.purchased.plan.customerId;
|
|
} else {
|
|
customerId = user.purchased.plan.customerId;
|
|
}
|
|
|
|
if (!customerId) throw new NotAuthorized(i18n.t('missingSubscription'));
|
|
if (!token) throw new BadRequest('Missing req.body.id');
|
|
|
|
// @TODO: Handle Stripe Error response
|
|
const subscriptions = await stripeApi.subscriptions.list({ customer: customerId });
|
|
const subscriptionId = subscriptions.data[0].id;
|
|
await stripeApi.subscriptions.update(subscriptionId, { card: token });
|
|
};
|
|
|
|
/**
|
|
* Cancels a subscription created by Stripe
|
|
*
|
|
* @param options
|
|
* @param options.user The user object who is purchasing
|
|
* @param options.groupId The id of the group purchasing a subscription
|
|
* @param options.cancellationReason A text string to control sending an email
|
|
*
|
|
* @return undefined
|
|
*/
|
|
api.cancelSubscription = async function cancelSubscription (options, stripeInc) {
|
|
const { groupId, user, cancellationReason } = options;
|
|
let customerId;
|
|
|
|
// @TODO: We need to mock this, but curently we don't have correct
|
|
// Dependency Injection. And the Stripe Api doesn't seem to be a singleton?
|
|
let stripeApi = getStripeApi();
|
|
if (stripeInc) stripeApi = stripeInc;
|
|
|
|
if (groupId) {
|
|
const groupFields = basicGroupFields.concat(' purchased');
|
|
const group = await Group.getGroup({
|
|
user, groupId, populateLeader: false, groupFields,
|
|
});
|
|
|
|
if (!group) {
|
|
throw new NotFound(i18n.t('groupNotFound'));
|
|
}
|
|
|
|
const allowedManagers = [group.leader, group.purchased.plan.owner];
|
|
|
|
if (allowedManagers.indexOf(user._id) === -1) {
|
|
throw new NotAuthorized(i18n.t('onlyGroupLeaderCanManageSubscription'));
|
|
}
|
|
customerId = group.purchased.plan.customerId;
|
|
} else {
|
|
customerId = user.purchased.plan.customerId;
|
|
}
|
|
|
|
if (!customerId) throw new NotAuthorized(i18n.t('missingSubscription'));
|
|
|
|
// @TODO: Handle error response
|
|
const customer = await stripeApi.customers.retrieve(customerId).catch(err => err);
|
|
let nextBill = moment().add(30, 'days').unix() * 1000;
|
|
|
|
if (customer && (customer.subscription || customer.subscriptions)) {
|
|
let { subscription } = customer;
|
|
if (!subscription && customer.subscriptions) {
|
|
[subscription] = customer.subscriptions.data;
|
|
}
|
|
await stripeApi.customers.del(customerId);
|
|
|
|
if (subscription && subscription.current_period_end) {
|
|
nextBill = subscription.current_period_end * 1000; // timestamp in seconds
|
|
}
|
|
}
|
|
|
|
await payments.cancelSubscription({
|
|
user,
|
|
groupId,
|
|
nextBill,
|
|
paymentMethod: this.constants.PAYMENT_METHOD,
|
|
cancellationReason,
|
|
});
|
|
};
|
|
|
|
api.chargeForAdditionalGroupMember = async function chargeForAdditionalGroupMember (group) {
|
|
const stripeApi = getStripeApi();
|
|
const plan = shared.content.subscriptionBlocks.group_monthly;
|
|
|
|
await stripeApi.subscriptions.update(
|
|
group.purchased.plan.subscriptionId,
|
|
{
|
|
plan: plan.key,
|
|
quantity: group.memberCount + plan.quantity - 1,
|
|
},
|
|
);
|
|
|
|
group.purchased.plan.quantity = group.memberCount + plan.quantity - 1;
|
|
};
|
|
|
|
/**
|
|
* Handle webhooks from stripes
|
|
*
|
|
* @param options
|
|
* @param options.user The user object who is purchasing
|
|
* @param options.groupId The id of the group purchasing a subscription
|
|
*
|
|
* @return undefined
|
|
*/
|
|
api.handleWebhooks = async function handleWebhooks (options, stripeInc) {
|
|
const { requestBody } = options;
|
|
|
|
// @TODO: We need to mock this, but curently we don't have correct
|
|
// Dependency Injection. And the Stripe Api doesn't seem to be a singleton?
|
|
let stripeApi = getStripeApi();
|
|
if (stripeInc) stripeApi = stripeInc;
|
|
|
|
// Verify the event by fetching it from Stripe
|
|
const event = await stripeApi.events.retrieve(requestBody.id);
|
|
|
|
switch (event.type) {
|
|
case 'customer.subscription.deleted': {
|
|
// event.request !== null means that the user itself cancelled the subscrioption,
|
|
// the cancellation on our side has been already handled
|
|
if (event.request !== null) break;
|
|
|
|
const subscription = event.data.object;
|
|
const customerId = subscription.customer;
|
|
const isGroupSub = shared.content.subscriptionBlocks[subscription.plan.id].target === 'group';
|
|
|
|
let user;
|
|
let groupId;
|
|
|
|
if (isGroupSub) {
|
|
const groupFields = basicGroupFields.concat(' purchased');
|
|
const group = await Group.findOne({
|
|
'purchased.plan.customerId': customerId,
|
|
'purchased.plan.paymentMethod': this.constants.PAYMENT_METHOD,
|
|
}).select(groupFields).exec();
|
|
|
|
if (!group) throw new NotFound(i18n.t('groupNotFound'));
|
|
groupId = group._id;
|
|
|
|
user = await User.findById(group.leader).exec();
|
|
} else {
|
|
user = await User.findOne({
|
|
'purchased.plan.customerId': customerId,
|
|
'purchased.plan.paymentMethod': this.constants.PAYMENT_METHOD,
|
|
}).exec();
|
|
}
|
|
|
|
if (!user) throw new NotFound(i18n.t('userNotFound'));
|
|
|
|
await stripeApi.customers.del(customerId);
|
|
|
|
await payments.cancelSubscription({
|
|
user,
|
|
groupId,
|
|
paymentMethod: this.constants.PAYMENT_METHOD,
|
|
// Give three extra days to allow the user to resubscribe without losing benefits
|
|
nextBill: moment().add({ days: 3 }).toDate(),
|
|
});
|
|
|
|
break;
|
|
}
|
|
default: {
|
|
logger.error(new Error(`Missing handler for Stripe webhook ${event.type}`), { event });
|
|
}
|
|
}
|
|
};
|
|
|
|
export default api;
|