mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-10-28 03:32:29 +01:00
commit dd0a410fa6c3741dc0d6793283cf4df3c37790a5 Author: Kalista Payne <sabrecat@gmail.com> Date: Mon Nov 4 14:24:30 2024 -0600 fix(subs): center next hourglass message commit 72d92ffd76bb43fee8ba2bbabd211e595afbd664 Author: Kalista Payne <sabrecat@gmail.com> Date: Fri Nov 1 14:17:59 2024 -0500 fix(subs): don't hide HG preview entirely commit ea0ecb0c3d519ed3d5c42266367eaaa7283ac5de Author: Kalista Payne <sabrecat@gmail.com> Date: Fri Nov 1 13:01:06 2024 -0500 fix(subs): Google wording and HG escape commit 2bd2c69e18e37c8c8c7106c62f186c372d25c5d2 Author: Kalista Payne <sabrecat@gmail.com> Date: Fri Nov 1 09:25:30 2024 -0500 fix(layout): tighten cancellation note commiteb2fc40d24Author: Kalista Payne <sabrecat@gmail.com> Date: Thu Oct 24 15:41:43 2024 -0500 fix(g1g1): don't try to find Gems promo during bogo commitd3eea86bd7Author: Kalista Payne <sabrecat@gmail.com> Date: Thu Oct 24 15:00:09 2024 -0500 fix(subs): fix typeError commite3ae9a2d67Author: Kalista Payne <sabrecat@gmail.com> Date: Thu Oct 24 13:57:27 2024 -0500 fix(subs): also redirect to subs after gift sub commit690163a0deAuthor: Phillip Thelen <phillip@habitica.com> Date: Wed Oct 23 16:42:38 2024 +0200 fix test commit2ad7541fc0Author: Phillip Thelen <phillip@habitica.com> Date: Wed Oct 23 16:34:52 2024 +0200 fix test commit7e337a9e59Author: Phillip Thelen <phillip@habitica.com> Date: Wed Oct 23 11:54:15 2024 +0200 remove only commit7462b8a57fAuthor: Phillip Thelen <phillip@habitica.com> Date: Wed Oct 23 11:51:25 2024 +0200 fix bug with incorrectly giving HG bonus commitacd6183e95Author: Kalista Payne <sabrecat@gmail.com> Date: Mon Oct 21 17:22:26 2024 -0500 fix(subs): unhovery and un-12-monthy commit935e9fd6ecAuthor: Kalista Payne <sabrecat@gmail.com> Date: Fri Oct 18 14:50:17 2024 -0500 fix(subs): try again on gifts commit6e1fb7df38Author: Kalista Payne <sabrecat@gmail.com> Date: Thu Oct 17 18:19:20 2024 -0500 fix(lint): do negate object ig commit71d434b94eAuthor: Kalista Payne <sabrecat@gmail.com> Date: Thu Oct 17 18:15:11 2024 -0500 fix(lint): unnecessary ternary commitb90b0bb9c3Author: Kalista Payne <sabrecat@gmail.com> Date: Thu Oct 17 17:34:24 2024 -0500 fix(subs): gifts DON't renew commit19469304c5Author: Kalista Payne <sabrecat@gmail.com> Date: Thu Oct 17 17:13:29 2024 -0500 fix(subs): pass autoRenews through Stripe commit6819e7b7e5Author: Kalista Payne <sabrecat@gmail.com> Date: Thu Oct 17 16:03:25 2024 -0500 fix(subscriptions): minor visual updates commit74633b5e5eAuthor: Kalista Payne <sabrecat@gmail.com> Date: Wed Oct 16 17:27:09 2024 -0500 fix(subscriptions): more gift layout revisions commita90ccb89deAuthor: Kalista Payne <sabrecat@gmail.com> Date: Wed Oct 16 15:37:50 2024 -0500 fix(subscription): update layout when gifting commitc24b2db8dcAuthor: Phillip Thelen <phillip@habitica.com> Date: Mon Oct 14 16:11:46 2024 +0200 fix issue with promo hourglasses commit7a61c72b47Author: Phillip Thelen <phillip@habitica.com> Date: Mon Oct 14 15:59:40 2024 +0200 don’t give additional HG for new sub if they already got one this month commitf14cb09026Author: Phillip Thelen <phillip@habitica.com> Date: Mon Oct 14 10:38:01 2024 +0200 Admin panel display fixes commitf4cff698cfAuthor: Kalista Payne <sabrecat@gmail.com> Date: Thu Oct 3 17:58:59 2024 -0500 fix(stripe): correct redirect after success commitc468b58f3fAuthor: Kalista Payne <sabrecat@gmail.com> Date: Thu Oct 3 17:35:37 2024 -0500 fix(subs): correct border-radius and redirect commit78fb9e31d6Author: Kalista Payne <sabrecat@gmail.com> Date: Wed Oct 2 17:41:49 2024 -0500 fix(css): correct and refactor heights and selection states commite2babe8053Author: Kalista Payne <sabrecat@gmail.com> Date: Mon Sep 30 16:45:29 2024 -0500 feat(subscription): max Gems progress readout commit61af8302a3Author: Phillip Thelen <phillip@habitica.com> Date: Fri Sep 27 15:11:22 2024 +0200 fix test commitef8ff0ea9eAuthor: Phillip Thelen <phillip@habitica.com> Date: Fri Sep 27 14:14:44 2024 +0200 show date for hourglass bonus if it was received commit4bafafdc8dAuthor: Phillip Thelen <phillip@habitica.com> Date: Fri Sep 27 14:12:52 2024 +0200 add new field for cumulative subscription count commit30096247b7Author: Phillip Thelen <phillip@habitica.com> Date: Fri Sep 27 13:39:49 2024 +0200 fix missing transaction type commit70872651b0Author: Phillip Thelen <phillip@habitica.com> Date: Fri Sep 27 13:31:40 2024 +0200 fix admin panel strings commitf3398db65fAuthor: Kalista Payne <sabrecat@gmail.com> Date: Thu Sep 26 23:11:16 2024 -0500 WIP(subs): extant Stripe state commitc6b2020109Author: Phillip Thelen <phillip@habitica.com> Date: Thu Sep 26 11:41:55 2024 +0200 fix admin panel display commitd9afc96d2dAuthor: Phillip Thelen <phillip@habitica.com> Date: Thu Sep 26 11:40:16 2024 +0200 Fix hourglass logic for upgrades commit6e2c8eeb64Author: Phillip Thelen <phillip@habitica.com> Date: Wed Sep 25 17:48:54 2024 +0200 fix hourglass count commitcd752fbdceAuthor: Kalista Payne <sabrecat@gmail.com> Date: Fri Sep 20 12:24:21 2024 -0500 WIP(frontend): draft of main subs page view commit0102b29d59Author: Kalista Payne <sabe@habitica.com> Date: Wed Sep 18 15:29:08 2024 -0500 fix(admin): correct logic and style for shrimple subs commit5469a5c5c3Author: Kalista Payne <sabe@habitica.com> Date: Wed Sep 18 15:07:36 2024 -0500 fix(test): short circuit this. commit526193ee6cAuthor: Phillip Thelen <phillip@habitica.com> Date: Wed Sep 18 14:42:06 2024 +0200 fix gem limit commit19cf1636aaAuthor: Phillip Thelen <phillip@habitica.com> Date: Tue Aug 13 17:00:40 2024 +0200 return nextHourglassDate again commiteea36e3ed5Author: Phillip Thelen <phillip@habitica.com> Date: Tue Aug 13 13:11:22 2024 +0200 subscription test improvements commitca78e74330Author: Phillip Thelen <phillip@habitica.com> Date: Mon Aug 12 15:46:15 2024 +0200 add more subscription tests commitf4c4f93a08Author: Phillip Thelen <phillip@habitica.com> Date: Fri Aug 9 13:35:22 2024 +0200 finish basic implementation of new logic commite036742048Author: Phillip Thelen <phillip@habitica.com> Date: Fri Aug 9 11:37:44 2024 +0200 cleanup commit6431865688Author: Phillip Thelen <phillip@habitica.com> Date: Wed Aug 7 05:41:18 2024 -0400 update cron tests commit930d875ae9Author: Phillip Thelen <phillip@habitica.com> Date: Thu Aug 8 10:36:50 2024 +0200 begin refactoring commit96623608d0Author: Phillip Thelen <phillip@habitica.com> Date: Tue Aug 6 16:28:16 2024 +0200 begin removing obsolete tests
440 lines
14 KiB
JavaScript
440 lines
14 KiB
JavaScript
import cc from 'coupon-code';
|
|
import stripeModule from 'stripe';
|
|
|
|
import { model as Coupon } from '../../../../../../website/server/models/coupon';
|
|
import common from '../../../../../../website/common';
|
|
import {
|
|
checkSubData,
|
|
applySubscription,
|
|
chargeForAdditionalGroupMember,
|
|
handlePaymentMethodChange,
|
|
} from '../../../../../../website/server/libs/payments/stripe/subscriptions';
|
|
import {
|
|
generateGroup,
|
|
} from '../../../../../helpers/api-unit.helper';
|
|
import { model as User } from '../../../../../../website/server/models/user';
|
|
import payments from '../../../../../../website/server/libs/payments/payments';
|
|
import stripePayments from '../../../../../../website/server/libs/payments/stripe';
|
|
|
|
const { i18n } = common;
|
|
|
|
describe('Stripe - Subscriptions', () => {
|
|
describe('checkSubData', () => {
|
|
it('does not throw if the subscription can be used', async () => {
|
|
const sub = common.content.subscriptionBlocks['basic_3mo']; // eslint-disable-line dot-notation
|
|
const res = await checkSubData(sub);
|
|
expect(res).to.equal(undefined);
|
|
});
|
|
|
|
it('throws if the subscription does not exists', async () => {
|
|
await expect(checkSubData())
|
|
.to.eventually.be.rejected.and.to.eql({
|
|
httpCode: 400,
|
|
name: 'BadRequest',
|
|
message: i18n.t('missingSubscriptionCode'),
|
|
});
|
|
});
|
|
|
|
it('throws if the subscription can\'t be used', async () => {
|
|
const sub = common.content.subscriptionBlocks['group_plan_auto']; // eslint-disable-line dot-notation
|
|
await expect(checkSubData(sub, true))
|
|
.to.eventually.be.rejected.and.to.eql({
|
|
httpCode: 400,
|
|
name: 'BadRequest',
|
|
message: i18n.t('missingSubscriptionCode'),
|
|
});
|
|
});
|
|
|
|
it('throws if the subscription targets a group and an user is making the request', async () => {
|
|
const sub = common.content.subscriptionBlocks['group_monthly']; // eslint-disable-line dot-notation
|
|
await expect(checkSubData(sub, false))
|
|
.to.eventually.be.rejected.and.to.eql({
|
|
httpCode: 400,
|
|
name: 'BadRequest',
|
|
message: i18n.t('missingSubscriptionCode'),
|
|
});
|
|
});
|
|
|
|
it('throws if the subscription targets an user and a group is making the request', async () => {
|
|
const sub = common.content.subscriptionBlocks['basic_3mo']; // eslint-disable-line dot-notation
|
|
await expect(checkSubData(sub, true))
|
|
.to.eventually.be.rejected.and.to.eql({
|
|
httpCode: 400,
|
|
name: 'BadRequest',
|
|
message: i18n.t('missingSubscriptionCode'),
|
|
});
|
|
});
|
|
|
|
it('throws if the coupon is required but not passed', async () => {
|
|
const sub = common.content.subscriptionBlocks['google_6mo']; // eslint-disable-line dot-notation
|
|
await expect(checkSubData(sub, false))
|
|
.to.eventually.be.rejected.and.to.eql({
|
|
httpCode: 400,
|
|
name: 'BadRequest',
|
|
message: i18n.t('couponCodeRequired'),
|
|
});
|
|
});
|
|
|
|
it('throws if the coupon is required but does not exist', async () => {
|
|
const coupon = 'not-valid';
|
|
const sub = common.content.subscriptionBlocks['google_6mo']; // eslint-disable-line dot-notation
|
|
await expect(checkSubData(sub, false, coupon))
|
|
.to.eventually.be.rejected.and.to.eql({
|
|
httpCode: 400,
|
|
name: 'BadRequest',
|
|
message: i18n.t('invalidCoupon'),
|
|
});
|
|
});
|
|
|
|
it('throws if the coupon is required but is invalid', async () => {
|
|
const couponModel = new Coupon();
|
|
couponModel.event = 'google_6mo';
|
|
await couponModel.save();
|
|
|
|
sandbox.stub(cc, 'validate').returns('invalid');
|
|
|
|
const sub = common.content.subscriptionBlocks['google_6mo']; // eslint-disable-line dot-notation
|
|
await expect(checkSubData(sub, false, couponModel._id))
|
|
.to.eventually.be.rejected.and.to.eql({
|
|
httpCode: 400,
|
|
name: 'BadRequest',
|
|
message: i18n.t('invalidCoupon'),
|
|
});
|
|
});
|
|
|
|
it('works if the coupon is required and valid', async () => {
|
|
const couponModel = new Coupon();
|
|
couponModel.event = 'google_6mo';
|
|
await couponModel.save();
|
|
|
|
sandbox.stub(cc, 'validate').returns(couponModel._id);
|
|
|
|
const sub = common.content.subscriptionBlocks['google_6mo']; // eslint-disable-line dot-notation
|
|
await checkSubData(sub, false, couponModel._id);
|
|
});
|
|
});
|
|
|
|
describe('applySubscription', () => {
|
|
let user; let group; let sub;
|
|
let groupId;
|
|
let customerId; let subscriptionId;
|
|
let subKey;
|
|
let userFindByIdStub;
|
|
let stripePaymentsCreateSubSpy;
|
|
|
|
beforeEach(async () => {
|
|
subKey = 'basic_3mo';
|
|
sub = common.content.subscriptionBlocks[subKey];
|
|
|
|
user = new User();
|
|
await user.save();
|
|
|
|
const execStub = sandbox.stub().resolves(user);
|
|
userFindByIdStub = sandbox.stub(User, 'findById');
|
|
userFindByIdStub.withArgs(user._id).returns({ exec: execStub });
|
|
|
|
group = generateGroup({
|
|
name: 'test group',
|
|
type: 'guild',
|
|
privacy: 'public',
|
|
leader: user._id,
|
|
});
|
|
groupId = group._id;
|
|
await group.save();
|
|
|
|
// Add user to group
|
|
user.guilds.push(groupId);
|
|
await user.save();
|
|
|
|
customerId = 'test-id';
|
|
subscriptionId = 'test-sub-id';
|
|
|
|
stripePaymentsCreateSubSpy = sandbox.stub(payments, 'createSubscription');
|
|
stripePaymentsCreateSubSpy.resolves({});
|
|
});
|
|
|
|
it('subscribes a user', async () => {
|
|
await applySubscription({
|
|
customer: customerId,
|
|
subscription: subscriptionId,
|
|
metadata: {
|
|
sub: JSON.stringify(sub),
|
|
userId: user._id,
|
|
groupId: null,
|
|
},
|
|
user,
|
|
});
|
|
|
|
expect(stripePaymentsCreateSubSpy).to.be.calledOnce;
|
|
expect(stripePaymentsCreateSubSpy).to.be.calledWith({
|
|
user,
|
|
customerId,
|
|
subscriptionId,
|
|
paymentMethod: 'Stripe',
|
|
sub: sinon.match({ ...sub }),
|
|
groupId: null,
|
|
autoRenews: true,
|
|
});
|
|
});
|
|
|
|
it('subscribes a group', async () => {
|
|
sub = common.content.subscriptionBlocks['group_monthly']; // eslint-disable-line dot-notation
|
|
await applySubscription({
|
|
customer: customerId,
|
|
subscription: subscriptionId,
|
|
metadata: {
|
|
sub: JSON.stringify(sub),
|
|
userId: user._id,
|
|
groupId,
|
|
},
|
|
user,
|
|
});
|
|
|
|
expect(stripePaymentsCreateSubSpy).to.be.calledOnce;
|
|
expect(stripePaymentsCreateSubSpy).to.be.calledWith({
|
|
user,
|
|
customerId,
|
|
subscriptionId,
|
|
paymentMethod: 'Stripe',
|
|
sub: sinon.match({ ...sub }),
|
|
groupId,
|
|
autoRenews: true,
|
|
});
|
|
});
|
|
|
|
it('subscribes a group with multiple users', async () => {
|
|
const user2 = new User();
|
|
user2.guilds.push(groupId);
|
|
await user2.save();
|
|
|
|
const execStub2 = sandbox.stub().resolves(user);
|
|
userFindByIdStub.withArgs(user2._id).returns({ exec: execStub2 });
|
|
|
|
group.memberCount = 2;
|
|
await group.save();
|
|
|
|
sub = common.content.subscriptionBlocks['group_monthly']; // eslint-disable-line dot-notation
|
|
await applySubscription({
|
|
customer: customerId,
|
|
subscription: subscriptionId,
|
|
metadata: {
|
|
sub: JSON.stringify(sub),
|
|
userId: user._id,
|
|
groupId,
|
|
},
|
|
user,
|
|
});
|
|
|
|
expect(stripePaymentsCreateSubSpy).to.be.calledOnce;
|
|
expect(stripePaymentsCreateSubSpy).to.be.calledWith({
|
|
user,
|
|
customerId,
|
|
subscriptionId,
|
|
paymentMethod: 'Stripe',
|
|
sub: sinon.match({ ...sub }),
|
|
groupId,
|
|
autoRenews: true,
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('handlePaymentMethodChange', () => {
|
|
const stripe = stripeModule('test');
|
|
|
|
it('updates the plan quantity based on the number of group members', async () => {
|
|
const stripeIntentRetrieveStub = sandbox.stub(stripe.setupIntents, 'retrieve').resolves({
|
|
payment_method: 1,
|
|
metadata: {
|
|
subscription_id: 2,
|
|
},
|
|
});
|
|
const stripeSubUpdateStub = sandbox.stub(stripe.subscriptions, 'update');
|
|
|
|
await handlePaymentMethodChange({}, stripe);
|
|
expect(stripeIntentRetrieveStub).to.be.calledOnce;
|
|
expect(stripeSubUpdateStub).to.be.calledOnce;
|
|
expect(stripeSubUpdateStub).to.be.calledWith(2, {
|
|
default_payment_method: 1,
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('chargeForAdditionalGroupMember', () => {
|
|
const stripe = stripeModule('test');
|
|
let stripeUpdateSubStub;
|
|
const plan = common.content.subscriptionBlocks['group_monthly']; // eslint-disable-line dot-notation
|
|
|
|
let user; let group;
|
|
|
|
beforeEach(async () => {
|
|
user = new User();
|
|
|
|
group = generateGroup({
|
|
name: 'test group',
|
|
type: 'guild',
|
|
privacy: 'public',
|
|
leader: user._id,
|
|
});
|
|
group.purchased.plan.customerId = 'customer-id';
|
|
group.purchased.plan.planId = plan.key;
|
|
group.purchased.plan.subscriptionId = 'sub-id';
|
|
await group.save();
|
|
|
|
stripeUpdateSubStub = sandbox.stub(stripe.subscriptions, 'update').resolves({});
|
|
});
|
|
|
|
it('updates the plan quantity based on the number of group members', async () => {
|
|
group.memberCount = 4;
|
|
const newQuantity = group.memberCount + plan.quantity - 1;
|
|
|
|
await chargeForAdditionalGroupMember(group, stripe);
|
|
expect(stripeUpdateSubStub).to.be.calledWithMatch(
|
|
group.purchased.plan.subscriptionId,
|
|
sinon.match({
|
|
plan: group.purchased.plan.planId,
|
|
quantity: newQuantity,
|
|
}),
|
|
);
|
|
expect(group.purchased.plan.quantity).to.equal(newQuantity);
|
|
});
|
|
});
|
|
|
|
describe('cancelSubscription', () => {
|
|
const subKey = 'basic_3mo';
|
|
const stripe = stripeModule('test');
|
|
let user; let groupId; let
|
|
group;
|
|
|
|
beforeEach(async () => {
|
|
user = new User();
|
|
user.profile.name = 'sender';
|
|
user.purchased.plan.customerId = 'customer-id';
|
|
user.purchased.plan.planId = subKey;
|
|
user.purchased.plan.lastBillingDate = new Date();
|
|
|
|
group = generateGroup({
|
|
name: 'test group',
|
|
type: 'guild',
|
|
privacy: 'public',
|
|
leader: user._id,
|
|
});
|
|
group.purchased.plan.customerId = 'customer-id';
|
|
group.purchased.plan.planId = subKey;
|
|
await group.save();
|
|
|
|
groupId = group._id;
|
|
});
|
|
|
|
it('throws an error if there is no customer id', async () => {
|
|
user.purchased.plan.customerId = undefined;
|
|
|
|
await expect(stripePayments.cancelSubscription({
|
|
user,
|
|
groupId: undefined,
|
|
}))
|
|
.to.eventually.be.rejected.and.to.eql({
|
|
httpCode: 401,
|
|
name: 'NotAuthorized',
|
|
message: i18n.t('missingSubscription'),
|
|
});
|
|
});
|
|
|
|
it('throws an error if the group is not found', async () => {
|
|
await expect(stripePayments.cancelSubscription({
|
|
user,
|
|
groupId: 'fake-group',
|
|
}))
|
|
.to.eventually.be.rejected.and.to.eql({
|
|
httpCode: 404,
|
|
name: 'NotFound',
|
|
message: i18n.t('groupNotFound'),
|
|
});
|
|
});
|
|
|
|
it('throws an error if user is not the group leader', async () => {
|
|
const nonLeader = new User();
|
|
nonLeader.guilds.push(groupId);
|
|
await nonLeader.save();
|
|
|
|
await expect(stripePayments.cancelSubscription({
|
|
user: nonLeader,
|
|
groupId,
|
|
}))
|
|
.to.eventually.be.rejected.and.to.eql({
|
|
httpCode: 401,
|
|
name: 'NotAuthorized',
|
|
message: i18n.t('onlyGroupLeaderCanManageSubscription'),
|
|
});
|
|
});
|
|
|
|
describe('success', () => {
|
|
let stripeDeleteCustomerStub; let paymentsCancelSubStub;
|
|
let stripeRetrieveStub; let subscriptionId; let
|
|
currentPeriodEndTimeStamp;
|
|
|
|
beforeEach(() => {
|
|
subscriptionId = 'subId';
|
|
stripeDeleteCustomerStub = sinon.stub(stripe.customers, 'del').resolves({});
|
|
paymentsCancelSubStub = sinon.stub(payments, 'cancelSubscription').resolves({});
|
|
|
|
currentPeriodEndTimeStamp = (new Date()).getTime();
|
|
stripeRetrieveStub = sinon.stub(stripe.customers, 'retrieve')
|
|
.resolves({
|
|
subscriptions: {
|
|
data: [{
|
|
id: subscriptionId,
|
|
current_period_end: currentPeriodEndTimeStamp,
|
|
}], // eslint-disable-line camelcase
|
|
},
|
|
});
|
|
});
|
|
|
|
afterEach(() => {
|
|
stripe.customers.del.restore();
|
|
stripe.customers.retrieve.restore();
|
|
payments.cancelSubscription.restore();
|
|
});
|
|
|
|
it('cancels a user subscription', async () => {
|
|
await stripePayments.cancelSubscription({
|
|
user,
|
|
groupId: undefined,
|
|
}, stripe);
|
|
|
|
expect(stripeDeleteCustomerStub).to.be.calledOnce;
|
|
expect(stripeDeleteCustomerStub).to.be.calledWith(user.purchased.plan.customerId);
|
|
expect(stripeRetrieveStub).to.be.calledOnce;
|
|
expect(stripeRetrieveStub).to.be.calledWith(user.purchased.plan.customerId);
|
|
expect(paymentsCancelSubStub).to.be.calledOnce;
|
|
expect(paymentsCancelSubStub).to.be.calledWith({
|
|
user,
|
|
groupId: undefined,
|
|
nextBill: currentPeriodEndTimeStamp * 1000, // timestamp in seconds
|
|
paymentMethod: 'Stripe',
|
|
cancellationReason: undefined,
|
|
});
|
|
});
|
|
|
|
it('cancels a group subscription', async () => {
|
|
await stripePayments.cancelSubscription({
|
|
user,
|
|
groupId,
|
|
}, stripe);
|
|
|
|
expect(stripeDeleteCustomerStub).to.be.calledOnce;
|
|
expect(stripeDeleteCustomerStub).to.be.calledWith(group.purchased.plan.customerId);
|
|
expect(stripeRetrieveStub).to.be.calledOnce;
|
|
expect(stripeRetrieveStub).to.be.calledWith(user.purchased.plan.customerId);
|
|
expect(paymentsCancelSubStub).to.be.calledOnce;
|
|
expect(paymentsCancelSubStub).to.be.calledWith({
|
|
user,
|
|
groupId,
|
|
nextBill: currentPeriodEndTimeStamp * 1000, // timestamp in seconds
|
|
paymentMethod: 'Stripe',
|
|
cancellationReason: undefined,
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|