mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-14 13:17:24 +01:00
258 lines
7.7 KiB
JavaScript
258 lines
7.7 KiB
JavaScript
import moment from 'moment';
|
|
import shared from '../../../common';
|
|
import iap from '../inAppPurchases';
|
|
import payments from './payments';
|
|
import {
|
|
NotAuthorized,
|
|
BadRequest,
|
|
} from '../errors';
|
|
import { model as IapPurchaseReceipt } from '../../models/iapPurchaseReceipt';
|
|
import { model as User } from '../../models/user';
|
|
|
|
const api = {};
|
|
|
|
api.constants = {
|
|
PAYMENT_METHOD_GOOGLE: 'Google',
|
|
PAYMENT_METHOD_GIFT: 'Google (Gift)',
|
|
RESPONSE_INVALID_RECEIPT: 'INVALID_RECEIPT',
|
|
RESPONSE_ALREADY_USED: 'RECEIPT_ALREADY_USED',
|
|
RESPONSE_INVALID_ITEM: 'INVALID_ITEM_PURCHASED',
|
|
RESPONSE_STILL_VALID: 'SUBSCRIPTION_STILL_VALID',
|
|
};
|
|
|
|
api.verifyGemPurchase = async function verifyGemPurchase (options) {
|
|
const {
|
|
gift, user, receipt, signature, headers,
|
|
} = options;
|
|
|
|
if (gift) {
|
|
gift.member = await User.findById(gift.uuid).exec();
|
|
}
|
|
const receiver = gift ? gift.member : user;
|
|
const receiverCanGetGems = await receiver.canGetGems();
|
|
if (!receiverCanGetGems) throw new NotAuthorized(shared.i18n.t('groupPolicyCannotGetGems', user.preferences.language));
|
|
|
|
await iap.setup();
|
|
|
|
const testObj = {
|
|
data: receipt,
|
|
signature,
|
|
};
|
|
|
|
const googleRes = await iap.validate(iap.GOOGLE, testObj);
|
|
|
|
const isValidated = iap.isValidated(googleRes);
|
|
if (!isValidated) throw new NotAuthorized(this.constants.RESPONSE_INVALID_RECEIPT);
|
|
|
|
const receiptObj = typeof testObj.data === 'string' ? JSON.parse(testObj.data) : testObj.data; // passed as a string
|
|
const token = receiptObj.token || receiptObj.purchaseToken;
|
|
|
|
const existingReceipt = await IapPurchaseReceipt.findOne({
|
|
_id: token,
|
|
}).exec();
|
|
if (existingReceipt) throw new NotAuthorized(this.constants.RESPONSE_ALREADY_USED);
|
|
|
|
await IapPurchaseReceipt.create({
|
|
_id: token,
|
|
consumed: true,
|
|
// This should always be the buying user even for a gift.
|
|
userId: user._id,
|
|
});
|
|
|
|
let amount;
|
|
|
|
switch (receiptObj.productId) { // eslint-disable-line default-case
|
|
case 'com.habitrpg.android.habitica.iap.4gems':
|
|
amount = 1;
|
|
break;
|
|
case 'com.habitrpg.android.habitica.iap.20.gems':
|
|
case 'com.habitrpg.android.habitica.iap.21gems':
|
|
amount = 5.25;
|
|
break;
|
|
case 'com.habitrpg.android.habitica.iap.42gems':
|
|
amount = 10.5;
|
|
break;
|
|
case 'com.habitrpg.android.habitica.iap.84gems':
|
|
amount = 21;
|
|
break;
|
|
}
|
|
|
|
if (!amount) throw new NotAuthorized(this.constants.RESPONSE_INVALID_ITEM);
|
|
|
|
await payments.buyGems({
|
|
user: receiver,
|
|
paymentMethod: this.constants.PAYMENT_METHOD_GOOGLE,
|
|
amount,
|
|
headers,
|
|
});
|
|
|
|
return googleRes;
|
|
};
|
|
|
|
api.subscribe = async function subscribe (
|
|
sku, user, receipt, signature,
|
|
headers, nextPaymentProcessing = undefined,
|
|
) {
|
|
if (!sku) throw new BadRequest(shared.i18n.t('missingSubscriptionCode'));
|
|
let subCode;
|
|
switch (sku) { // eslint-disable-line default-case
|
|
case 'com.habitrpg.android.habitica.subscription.1month':
|
|
subCode = 'basic_earned';
|
|
break;
|
|
case 'com.habitrpg.android.habitica.subscription.3month':
|
|
subCode = 'basic_3mo';
|
|
break;
|
|
case 'com.habitrpg.android.habitica.subscription.6month':
|
|
subCode = 'basic_6mo';
|
|
break;
|
|
case 'com.habitrpg.android.habitica.subscription.12month':
|
|
subCode = 'basic_12mo';
|
|
break;
|
|
}
|
|
const sub = subCode ? shared.content.subscriptionBlocks[subCode] : false;
|
|
if (!sub) throw new NotAuthorized(this.constants.RESPONSE_INVALID_ITEM);
|
|
|
|
await iap.setup();
|
|
|
|
const testObj = {
|
|
data: receipt,
|
|
signature,
|
|
};
|
|
|
|
const receiptObj = typeof receipt === 'string' ? JSON.parse(receipt) : receipt; // passed as a string
|
|
const token = receiptObj.token || receiptObj.purchaseToken;
|
|
|
|
const existingUser = await User.findOne({
|
|
'purchased.plan.customerId': token,
|
|
}).exec();
|
|
if (existingUser) throw new NotAuthorized(this.constants.RESPONSE_ALREADY_USED);
|
|
|
|
const googleRes = await iap.validate(iap.GOOGLE, testObj);
|
|
|
|
const isValidated = iap.isValidated(googleRes);
|
|
if (!isValidated) throw new NotAuthorized(this.constants.RESPONSE_INVALID_RECEIPT);
|
|
|
|
nextPaymentProcessing = nextPaymentProcessing || moment.utc().add({ days: 2 }); // eslint-disable-line no-param-reassign, max-len
|
|
|
|
await payments.createSubscription({
|
|
user,
|
|
customerId: token,
|
|
paymentMethod: this.constants.PAYMENT_METHOD_GOOGLE,
|
|
sub,
|
|
headers,
|
|
nextPaymentProcessing,
|
|
additionalData: testObj,
|
|
});
|
|
};
|
|
|
|
api.noRenewSubscribe = async function noRenewSubscribe (options) {
|
|
const {
|
|
sku, gift, user, receipt, signature, headers,
|
|
} = options;
|
|
if (!sku) throw new BadRequest(shared.i18n.t('missingSubscriptionCode'));
|
|
let subCode;
|
|
switch (sku) { // eslint-disable-line default-case
|
|
case 'com.habitrpg.android.habitica.norenew_subscription.1month':
|
|
subCode = 'basic_earned';
|
|
break;
|
|
case 'com.habitrpg.android.habitica.norenew_subscription.3month':
|
|
subCode = 'basic_3mo';
|
|
break;
|
|
case 'com.habitrpg.android.habitica.norenew_subscription.6month':
|
|
subCode = 'basic_6mo';
|
|
break;
|
|
case 'com.habitrpg.android.habitica.norenew_subscription.12month':
|
|
subCode = 'basic_12mo';
|
|
break;
|
|
}
|
|
const sub = subCode ? shared.content.subscriptionBlocks[subCode] : false;
|
|
if (!sub) throw new NotAuthorized(this.constants.RESPONSE_INVALID_ITEM);
|
|
|
|
await iap.setup();
|
|
|
|
const testObj = {
|
|
data: receipt,
|
|
signature,
|
|
};
|
|
|
|
const receiptObj = typeof receipt === 'string' ? JSON.parse(receipt) : receipt; // passed as a string
|
|
const token = receiptObj.token || receiptObj.purchaseToken;
|
|
|
|
const existingReceipt = await IapPurchaseReceipt.findOne({ // eslint-disable-line no-await-in-loop
|
|
_id: token,
|
|
}).exec();
|
|
if (existingReceipt) throw new NotAuthorized(this.constants.RESPONSE_ALREADY_USED);
|
|
|
|
await IapPurchaseReceipt.create({ // eslint-disable-line no-await-in-loop
|
|
_id: token,
|
|
consumed: true,
|
|
// This should always be the buying user even for a gift.
|
|
userId: user._id,
|
|
});
|
|
|
|
const googleRes = await iap.validate(iap.GOOGLE, testObj);
|
|
|
|
const isValidated = iap.isValidated(googleRes);
|
|
if (!isValidated) throw new NotAuthorized(this.constants.RESPONSE_INVALID_RECEIPT);
|
|
|
|
const data = {
|
|
user,
|
|
paymentMethod: this.constants.PAYMENT_METHOD_GOOGLE,
|
|
headers,
|
|
sub,
|
|
autoRenews: false,
|
|
};
|
|
|
|
if (gift) {
|
|
gift.member = await User.findById(gift.uuid).exec();
|
|
gift.subscription = sub;
|
|
data.gift = gift;
|
|
data.paymentMethod = this.constants.PAYMENT_METHOD_GIFT;
|
|
}
|
|
|
|
await payments.createSubscription(data);
|
|
|
|
return googleRes;
|
|
};
|
|
|
|
api.cancelSubscribe = async function cancelSubscribe (user, headers) {
|
|
const { plan } = user.purchased;
|
|
|
|
if (plan.paymentMethod !== api.constants.PAYMENT_METHOD_GOOGLE) throw new NotAuthorized(shared.i18n.t('missingSubscription'));
|
|
|
|
await iap.setup();
|
|
|
|
let dateTerminated;
|
|
|
|
try {
|
|
const googleRes = await iap.validate(iap.GOOGLE, plan.additionalData);
|
|
|
|
const isValidated = iap.isValidated(googleRes);
|
|
if (!isValidated) throw new NotAuthorized(this.constants.RESPONSE_INVALID_RECEIPT);
|
|
|
|
const purchases = iap.getPurchaseData(googleRes);
|
|
if (purchases.length === 0) throw new NotAuthorized(this.constants.RESPONSE_INVALID_RECEIPT);
|
|
const subscriptionData = purchases[0];
|
|
// Check to make sure the sub isn't active anymore.
|
|
if (subscriptionData.autoRenews) return;
|
|
|
|
dateTerminated = new Date(Number(subscriptionData.expirationDate));
|
|
} catch (err) {
|
|
// Status:410 means that the subsctiption isn't active anymore and we can safely delete it
|
|
if (err && err.message === 'Status:410') {
|
|
dateTerminated = new Date();
|
|
} else {
|
|
throw err;
|
|
}
|
|
}
|
|
|
|
await payments.cancelSubscription({
|
|
user,
|
|
nextBill: dateTerminated,
|
|
paymentMethod: this.constants.PAYMENT_METHOD_GOOGLE,
|
|
headers,
|
|
});
|
|
};
|
|
|
|
export default api;
|