mirror of
				https://github.com/HabitRPG/habitica.git
				synced 2025-10-29 20:24:53 +01:00 
			
		
		
		
	* content: add gems blocks * gemsBlocks: include ios and android identifiers * wip: promo code * split common constants into multiple files * add second promo part * geCurrentEvent, refactor promo * fix lint * fix exports, use world state api * start adding world state tests * remove console.log * use gems block for purchases * remove comments * fix most unit tests * restore comment * fix lint * prevent apple/google gift tests from breaking other tests when stub is not reset * fix unit tests, clarify tests names * iap: use gift object when gifting gems * allow gift object with less data * fix iap tests, remove findById stubs * iap: require less data from the mobile apps * apply discounts * add missing worldState file * fix lint * add test event * start removing 20 gems option for web * start adding support for all gems packages on web * fix unit tests for apple, stripe and google * amazon: support all gems blocks * paypal: support all gems blocks * fix payments unit tests, add tests for getGemsBlock * web: add gems plans with discounts, update stripe * fix amazon and paypal clients, payments success modals * amazon pay: disabled state * update icons, start abstracting payments buttons * begin redesign * redesign gems modal * fix buttons * fix hover color for gems modal close icon * add key to world state current event * extend test event length * implement gems modals designs * early test fall2020 * fix header banner position * add missing files * use iso 8601 for dates, minor ui fixes * fix time zones * events: fix ISO8601 format * fix css indentation * start abstracting banners * refactor payments buttons * test spooky, fix group plans box * implement gems promo banners, refactor banners, fixes * fix lint * fix dates * remove unused i18n strings * fix stripe integration test * fix world state integration tests * the current active event * add missing unit tests * add storybook story for payments buttons component * fix typo * fix(stripe): correct label when gifting subscriptions
		
			
				
	
	
		
			322 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			322 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import stripeModule from 'stripe';
 | |
| import cc from 'coupon-code';
 | |
| 
 | |
| import {
 | |
|   generateGroup,
 | |
| } from '../../../../../helpers/api-unit.helper';
 | |
| import { model as User } from '../../../../../../website/server/models/user';
 | |
| import { model as Coupon } from '../../../../../../website/server/models/coupon';
 | |
| import stripePayments from '../../../../../../website/server/libs/payments/stripe';
 | |
| import payments from '../../../../../../website/server/libs/payments/payments';
 | |
| import common from '../../../../../../website/common';
 | |
| 
 | |
| const { i18n } = common;
 | |
| 
 | |
| describe('stripe - checkout with subscription', () => {
 | |
|   const subKey = 'basic_3mo';
 | |
|   const stripe = stripeModule('test');
 | |
|   let user; let group; let data; let gift; let sub;
 | |
|   let groupId; let email; let headers; let coupon;
 | |
|   let customerIdResponse; let subscriptionId; let
 | |
|     token;
 | |
|   let spy;
 | |
|   let stripeCreateCustomerSpy;
 | |
|   let stripePaymentsCreateSubSpy;
 | |
| 
 | |
|   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();
 | |
| 
 | |
|     sub = {
 | |
|       key: 'basic_3mo',
 | |
|     };
 | |
| 
 | |
|     data = {
 | |
|       user,
 | |
|       sub,
 | |
|       customerId: 'customer-id',
 | |
|       paymentMethod: 'Payment Method',
 | |
|     };
 | |
| 
 | |
|     email = 'example@example.com';
 | |
|     customerIdResponse = 'test-id';
 | |
|     subscriptionId = 'test-sub-id';
 | |
|     token = 'test-token';
 | |
| 
 | |
|     spy = sinon.stub(stripe.subscriptions, 'update');
 | |
|     spy.resolves;
 | |
| 
 | |
|     stripeCreateCustomerSpy = sinon.stub(stripe.customers, 'create');
 | |
|     const stripCustomerResponse = {
 | |
|       id: customerIdResponse,
 | |
|       subscriptions: {
 | |
|         data: [{ id: subscriptionId }],
 | |
|       },
 | |
|     };
 | |
|     stripeCreateCustomerSpy.resolves(stripCustomerResponse);
 | |
| 
 | |
|     stripePaymentsCreateSubSpy = sinon.stub(payments, 'createSubscription');
 | |
|     stripePaymentsCreateSubSpy.resolves({});
 | |
| 
 | |
|     data.groupId = group._id;
 | |
|     data.sub.quantity = 3;
 | |
|   });
 | |
| 
 | |
|   afterEach(() => {
 | |
|     stripe.subscriptions.update.restore();
 | |
|     stripe.customers.create.restore();
 | |
|     payments.createSubscription.restore();
 | |
|   });
 | |
| 
 | |
|   it('should throw an error if we are missing a token', async () => {
 | |
|     await expect(stripePayments.checkout({
 | |
|       user,
 | |
|       gift,
 | |
|       sub,
 | |
|       groupId,
 | |
|       email,
 | |
|       headers,
 | |
|       coupon,
 | |
|     }))
 | |
|       .to.eventually.be.rejected.and.to.eql({
 | |
|         httpCode: 400,
 | |
|         name: 'BadRequest',
 | |
|         message: 'Missing req.body.id',
 | |
|       });
 | |
|   });
 | |
| 
 | |
|   it('should throw an error when coupon code is missing', async () => {
 | |
|     sub.discount = 40;
 | |
| 
 | |
|     await expect(stripePayments.checkout({
 | |
|       token,
 | |
|       user,
 | |
|       gift,
 | |
|       sub,
 | |
|       groupId,
 | |
|       email,
 | |
|       headers,
 | |
|       coupon,
 | |
|     }))
 | |
|       .to.eventually.be.rejected.and.to.eql({
 | |
|         httpCode: 400,
 | |
|         name: 'BadRequest',
 | |
|         message: i18n.t('couponCodeRequired'),
 | |
|       });
 | |
|   });
 | |
| 
 | |
|   it('should throw an error when coupon code is invalid', async () => {
 | |
|     sub.discount = 40;
 | |
|     sub.key = 'google_6mo';
 | |
|     coupon = 'example-coupon';
 | |
| 
 | |
|     const couponModel = new Coupon();
 | |
|     couponModel.event = 'google_6mo';
 | |
|     await couponModel.save();
 | |
| 
 | |
|     sinon.stub(cc, 'validate').returns('invalid');
 | |
| 
 | |
|     await expect(stripePayments.checkout({
 | |
|       token,
 | |
|       user,
 | |
|       gift,
 | |
|       sub,
 | |
|       groupId,
 | |
|       email,
 | |
|       headers,
 | |
|       coupon,
 | |
|     }))
 | |
|       .to.eventually.be.rejected.and.to.eql({
 | |
|         httpCode: 400,
 | |
|         name: 'BadRequest',
 | |
|         message: i18n.t('invalidCoupon'),
 | |
|       });
 | |
|     cc.validate.restore();
 | |
|   });
 | |
| 
 | |
|   it('subscribes with stripe with a coupon', async () => {
 | |
|     sub.discount = 40;
 | |
|     sub.key = 'google_6mo';
 | |
|     coupon = 'example-coupon';
 | |
| 
 | |
|     const couponModel = new Coupon();
 | |
|     couponModel.event = 'google_6mo';
 | |
|     const updatedCouponModel = await couponModel.save();
 | |
| 
 | |
|     sinon.stub(cc, 'validate').returns(updatedCouponModel._id);
 | |
| 
 | |
|     await stripePayments.checkout({
 | |
|       token,
 | |
|       user,
 | |
|       gift,
 | |
|       sub,
 | |
|       groupId,
 | |
|       email,
 | |
|       headers,
 | |
|       coupon,
 | |
|     }, stripe);
 | |
| 
 | |
|     expect(stripeCreateCustomerSpy).to.be.calledOnce;
 | |
|     expect(stripeCreateCustomerSpy).to.be.calledWith({
 | |
|       email,
 | |
|       metadata: { uuid: user._id },
 | |
|       card: token,
 | |
|       plan: sub.key,
 | |
|     });
 | |
| 
 | |
|     expect(stripePaymentsCreateSubSpy).to.be.calledOnce;
 | |
|     expect(stripePaymentsCreateSubSpy).to.be.calledWith({
 | |
|       user,
 | |
|       customerId: customerIdResponse,
 | |
|       paymentMethod: 'Stripe',
 | |
|       sub,
 | |
|       headers,
 | |
|       groupId: undefined,
 | |
|       subscriptionId: undefined,
 | |
|     });
 | |
| 
 | |
|     cc.validate.restore();
 | |
|   });
 | |
| 
 | |
|   it('subscribes a user', async () => {
 | |
|     sub = data.sub;
 | |
| 
 | |
|     await stripePayments.checkout({
 | |
|       token,
 | |
|       user,
 | |
|       gift,
 | |
|       sub,
 | |
|       groupId,
 | |
|       email,
 | |
|       headers,
 | |
|       coupon,
 | |
|     }, stripe);
 | |
| 
 | |
|     expect(stripeCreateCustomerSpy).to.be.calledOnce;
 | |
|     expect(stripeCreateCustomerSpy).to.be.calledWith({
 | |
|       email,
 | |
|       metadata: { uuid: user._id },
 | |
|       card: token,
 | |
|       plan: sub.key,
 | |
|     });
 | |
| 
 | |
|     expect(stripePaymentsCreateSubSpy).to.be.calledOnce;
 | |
|     expect(stripePaymentsCreateSubSpy).to.be.calledWith({
 | |
|       user,
 | |
|       customerId: customerIdResponse,
 | |
|       paymentMethod: 'Stripe',
 | |
|       sub,
 | |
|       headers,
 | |
|       groupId: undefined,
 | |
|       subscriptionId: undefined,
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   it('subscribes a group', async () => {
 | |
|     token = 'test-token';
 | |
|     sub = data.sub;
 | |
|     groupId = group._id;
 | |
|     email = 'test@test.com';
 | |
| 
 | |
|     // Add user to group
 | |
|     user.guilds.push(groupId);
 | |
|     await user.save();
 | |
| 
 | |
|     headers = {};
 | |
| 
 | |
|     await stripePayments.checkout({
 | |
|       token,
 | |
|       user,
 | |
|       gift,
 | |
|       sub,
 | |
|       groupId,
 | |
|       email,
 | |
|       headers,
 | |
|       coupon,
 | |
|     }, stripe);
 | |
| 
 | |
|     expect(stripeCreateCustomerSpy).to.be.calledOnce;
 | |
|     expect(stripeCreateCustomerSpy).to.be.calledWith({
 | |
|       email,
 | |
|       metadata: { uuid: user._id },
 | |
|       card: token,
 | |
|       plan: sub.key,
 | |
|       quantity: 3,
 | |
|     });
 | |
| 
 | |
|     expect(stripePaymentsCreateSubSpy).to.be.calledOnce;
 | |
|     expect(stripePaymentsCreateSubSpy).to.be.calledWith({
 | |
|       user,
 | |
|       customerId: customerIdResponse,
 | |
|       paymentMethod: 'Stripe',
 | |
|       sub,
 | |
|       headers,
 | |
|       groupId,
 | |
|       subscriptionId,
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   it('subscribes a group with the correct number of group members', async () => {
 | |
|     token = 'test-token';
 | |
|     sub = data.sub;
 | |
|     groupId = group._id;
 | |
|     email = 'test@test.com';
 | |
|     headers = {};
 | |
| 
 | |
|     // Add user to group
 | |
|     user.guilds.push(groupId);
 | |
|     await user.save();
 | |
| 
 | |
|     user = new User();
 | |
|     user.guilds.push(groupId);
 | |
|     await user.save();
 | |
| 
 | |
|     group.memberCount = 2;
 | |
|     await group.save();
 | |
| 
 | |
|     await stripePayments.checkout({
 | |
|       token,
 | |
|       user,
 | |
|       gift,
 | |
|       sub,
 | |
|       groupId,
 | |
|       email,
 | |
|       headers,
 | |
|       coupon,
 | |
|     }, stripe);
 | |
| 
 | |
|     expect(stripeCreateCustomerSpy).to.be.calledOnce;
 | |
|     expect(stripeCreateCustomerSpy).to.be.calledWith({
 | |
|       email,
 | |
|       metadata: { uuid: user._id },
 | |
|       card: token,
 | |
|       plan: sub.key,
 | |
|       quantity: 4,
 | |
|     });
 | |
| 
 | |
|     expect(stripePaymentsCreateSubSpy).to.be.calledOnce;
 | |
|     expect(stripePaymentsCreateSubSpy).to.be.calledWith({
 | |
|       user,
 | |
|       customerId: customerIdResponse,
 | |
|       paymentMethod: 'Stripe',
 | |
|       sub,
 | |
|       headers,
 | |
|       groupId,
 | |
|       subscriptionId,
 | |
|     });
 | |
|   });
 | |
| });
 |