fix isValidated and port iOS iap code

This commit is contained in:
Matteo Pagliazzi
2016-07-04 16:12:43 +02:00
parent dff9356778
commit d4cff995e8
2 changed files with 96 additions and 104 deletions

View File

@@ -8,6 +8,7 @@ import {
NotAuthorized, NotAuthorized,
} from '../../../libs/api-v3/errors'; } from '../../../libs/api-v3/errors';
import { model as IapPurchaseReceipt } from '../../../models/iapPurchaseReceipt'; import { model as IapPurchaseReceipt } from '../../../models/iapPurchaseReceipt';
import logger from '../../../libs/api-v3/logger';
let api = {}; let api = {};
@@ -35,15 +36,17 @@ api.iapAndroidVerify = {
let googleRes = await iap.validate(iap.GOOGLE, testObj); let googleRes = await iap.validate(iap.GOOGLE, testObj);
if (iap.isValidated(googleRes)) { let isValidated = iap.isValidated(googleRes);
if (!isValidated) throw new NotAuthorized('INVALID_RECEIPT');
let receiptObj = JSON.parse(testObj.data); // passed as a string let receiptObj = JSON.parse(testObj.data); // passed as a string
let token = receiptObj.token || receiptObj.purchaseToken; let token = receiptObj.token || receiptObj.purchaseToken;
let existingReceipt = await IapPurchaseReceipt.findOne({ let existingReceipt = await IapPurchaseReceipt.findOne({
_id: token, _id: token,
}).exec(); }).exec();
if (existingReceipt) throw new NotAuthorized('RECEIPT_ALREADY_USED');
if (!existingReceipt) {
await IapPurchaseReceipt.create({ await IapPurchaseReceipt.create({
_id: token, _id: token,
consumed: true, consumed: true,
@@ -55,12 +58,6 @@ api.iapAndroidVerify = {
paymentMethod: 'IAP GooglePlay', paymentMethod: 'IAP GooglePlay',
amount: 5.25, amount: 5.25,
}); });
} else {
throw new NotAuthorized('RECEIPT_ALREADY_USED');
}
} else {
throw new NotAuthorized('INVALID_RECEIPT');
}
res.respond(200, googleRes); res.respond(200, googleRes);
}, },
@@ -80,11 +77,79 @@ api.iapiOSVerify = {
url: '/iap/ios/verify', url: '/iap/ios/verify',
middlewares: [authWithHeaders()], middlewares: [authWithHeaders()],
async handler (req, res) { async handler (req, res) {
let resObject = await iapIOSVerify(res.locals.user, req.body); let user = res.locals.user;
let iapBody = req.body;
return res let appleRes;
.status(resObject.ok === true ? 200 : 500) let token;
.json(resObject);
try {
await iap.setup();
appleRes = await iap.validate(iap.APPLE, iapBody.transaction.receipt);
let isValidated = iap.isValidated(appleRes);
if (!isValidated) throw new Error('INVALID_RECEIPT');
let purchaseDataList = iap.getPurchaseData(appleRes);
if (purchaseDataList.length === 0) throw new Error('NO_ITEM_PURCHASED');
let correctReceipt = true;
// Purchasing one item at a time (processing of await(s) below is sequential not parallel)
for (let purchaseData of purchaseDataList) {
token = purchaseData.transactionId;
let existingReceipt = await IapPurchaseReceipt.findOne({ // eslint-disable-line babel/no-await-in-loop
_id: token,
}).exec();
if (!existingReceipt) {
await IapPurchaseReceipt.create({ // eslint-disable-line babel/no-await-in-loop
_id: token,
consumed: true,
userId: user._id,
});
} else {
throw new Error('RECEIPT_ALREADY_USED');
}
switch (purchaseData.productId) {
case 'com.habitrpg.ios.Habitica.4gems':
await payments.buyGems({user, paymentMethod: 'IAP AppleStore', amount: 1}); // eslint-disable-line babel/no-await-in-loop
break;
case 'com.habitrpg.ios.Habitica.8gems':
await payments.buyGems({user, paymentMethod: 'IAP AppleStore', amount: 2}); // eslint-disable-line babel/no-await-in-loop
break;
case 'com.habitrpg.ios.Habitica.20gems':
case 'com.habitrpg.ios.Habitica.21gems':
await payments.buyGems({user, paymentMethod: 'IAP AppleStore', amount: 5.25}); // eslint-disable-line babel/no-await-in-loop
break;
case 'com.habitrpg.ios.Habitica.42gems':
await payments.buyGems({user, paymentMethod: 'IAP AppleStore', amount: 10.5}); // eslint-disable-line babel/no-await-in-loop
break;
default:
correctReceipt = false;
}
}
if (!correctReceipt) throw new Error('INVALID_ITEM_PURCHASED');
return res.status(200).json({
ok: true,
data: appleRes,
});
} catch (err) {
logger.error(err, {
userId: user._id,
iapBody,
appleRes,
});
return res.status(500).json({
ok: false,
data: 'An error occurred while processing the purchase.',
});
}
}, },
}; };

View File

@@ -1,9 +1,6 @@
import nconf from 'nconf'; import nconf from 'nconf';
import iap from 'in-app-purchase'; import iap from 'in-app-purchase';
import payments from './payments';
import { model as IapPurchaseReceipt } from '../../models/iapPurchaseReceipt';
import Bluebird from 'bluebird'; import Bluebird from 'bluebird';
import logger from './logger';
// Validation ERROR Codes // Validation ERROR Codes
// const INVALID_PAYLOAD = 6778001; // const INVALID_PAYLOAD = 6778001;
@@ -18,78 +15,8 @@ iap.config({
module.exports = { module.exports = {
setup: Bluebird.promisify(iap.setup, { context: iap }), setup: Bluebird.promisify(iap.setup, { context: iap }),
validate: Bluebird.promisify(iap.validate, { context: iap }), validate: Bluebird.promisify(iap.validate, { context: iap }),
isValidated: iap.isValidated,
getPurchaseData: iap.getPurchaseData,
GOOGLE: iap.GOOGLE, GOOGLE: iap.GOOGLE,
APPLE: iap.APPLE,
}; };
async function iapIOSVerify (user, iapBody) {
// Defining these 2 variables here so they can be logged in case of error
let token;
let appleRes;
try {
await iapSetup();
appleRes = await iapValidate(iap.APPLE, iapBody.transaction.receipt);
if (iap.isValidated(appleRes)) {
token = appleRes.receipt.transactionIdentifier;
console.log(JSON.stringify(appleRes), JSON.stringify(appleRes.receipt));
let existingReceipt = await IapPurchaseReceipt.findOne({
_id: token,
}).exec();
if (!existingReceipt) {
await IapPurchaseReceipt.create({
_id: token,
consumed: true,
userId: user._id,
});
} else {
throw new Error('RECEIPT_ALREADY_USED');
}
let purchaseDataList = iap.getPurchaseData(appleRes);
if (purchaseDataList.length === 0) throw new Error('NO_ITEM_PURCHASED');
let correctReceipt = true;
// Purchasing one item at a time (processing of await(s) below is sequential not parallel)
for (let index in purchaseDataList) {
switch (purchaseDataList[index].productId) {
case 'com.habitrpg.ios.Habitica.4gems':
await payments.buyGems({user, paymentMethod: 'IAP AppleStore', amount: 1}); // eslint-disable-line babel/no-await-in-loop
break;
case 'com.habitrpg.ios.Habitica.8gems':
await payments.buyGems({user, paymentMethod: 'IAP AppleStore', amount: 2}); // eslint-disable-line babel/no-await-in-loop
break;
case 'com.habitrpg.ios.Habitica.20gems':
case 'com.habitrpg.ios.Habitica.21gems':
await payments.buyGems({user, paymentMethod: 'IAP AppleStore', amount: 5.25}); // eslint-disable-line babel/no-await-in-loop
break;
case 'com.habitrpg.ios.Habitica.42gems':
await payments.buyGems({user, paymentMethod: 'IAP AppleStore', amount: 10.5}); // eslint-disable-line babel/no-await-in-loop
break;
default:
correctReceipt = false;
}
}
if (!correctReceipt) throw new Error('INVALID_ITEM_PURCHASED');
} else {
throw new Error('INVALID_RECEIPT');
}
} catch (err) {
logger.error(err, {
userId: user._id,
iapBody,
appleRes,
token,
});
return {
ok: false,
data: 'An error occurred while processing the purchase.',
};
}
}