mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-18 15:17:25 +01:00
141 lines
5.5 KiB
JavaScript
141 lines
5.5 KiB
JavaScript
/* @see ./routes.coffee for routing*/
|
|
var _ = require('lodash');
|
|
var shared = require('../../../common');
|
|
var nconf = require('nconf');
|
|
var utils = require('./../../utils');
|
|
var moment = require('moment');
|
|
var isProduction = nconf.get("NODE_ENV") === "production";
|
|
var stripe = require('./stripe');
|
|
var paypal = require('./paypal');
|
|
var members = require('../members')
|
|
var async = require('async');
|
|
var iap = require('./iap');
|
|
var mongoose= require('mongoose');
|
|
var cc = require('coupon-code');
|
|
|
|
function revealMysteryItems(user) {
|
|
_.each(shared.content.gear.flat, function(item) {
|
|
if (
|
|
item.klass === 'mystery' &&
|
|
moment().isAfter(shared.content.mystery[item.mystery].start) &&
|
|
moment().isBefore(shared.content.mystery[item.mystery].end) &&
|
|
!user.items.gear.owned[item.key] &&
|
|
!~user.purchased.plan.mysteryItems.indexOf(item.key)
|
|
) {
|
|
user.purchased.plan.mysteryItems.push(item.key);
|
|
}
|
|
});
|
|
}
|
|
|
|
exports.createSubscription = function(data, cb) {
|
|
var recipient = data.gift ? data.gift.member : data.user;
|
|
//if (!recipient.purchased.plan) recipient.purchased.plan = {}; // FIXME double-check, this should never be the case
|
|
var p = recipient.purchased.plan;
|
|
var block = shared.content.subscriptionBlocks[data.gift ? data.gift.subscription.key : data.sub.key];
|
|
var months = +block.months;
|
|
|
|
if (data.gift) {
|
|
if (p.customerId && !p.dateTerminated) { // User has active plan
|
|
p.extraMonths += months;
|
|
} else {
|
|
p.dateTerminated = moment(p.dateTerminated).add({months: months}).toDate();
|
|
if (!p.dateUpdated) p.dateUpdated = new Date();
|
|
}
|
|
if (!p.customerId) p.customerId = 'Gift'; // don't override existing customer, but all sub need a customerId
|
|
} else {
|
|
_(p).merge({ // override with these values
|
|
planId: block.key,
|
|
customerId: data.customerId,
|
|
dateUpdated: new Date(),
|
|
gemsBought: 0,
|
|
paymentMethod: data.paymentMethod,
|
|
extraMonths: +p.extraMonths
|
|
+ +(p.dateTerminated ? moment(p.dateTerminated).diff(new Date(),'months',true) : 0),
|
|
dateTerminated: null
|
|
}).defaults({ // allow non-override if a plan was previously used
|
|
dateCreated: new Date(),
|
|
mysteryItems: []
|
|
});
|
|
}
|
|
|
|
// Block sub perks
|
|
var perks = Math.floor(months/3);
|
|
if (perks) {
|
|
p.consecutive.offset += months;
|
|
p.consecutive.gemCapExtra += perks*5;
|
|
if (p.consecutive.gemCapExtra > 25) p.consecutive.gemCapExtra = 25;
|
|
p.consecutive.trinkets += perks;
|
|
}
|
|
revealMysteryItems(recipient);
|
|
if(isProduction) {
|
|
if (!data.gift) utils.txnEmail(data.user, 'subscription-begins');
|
|
utils.ga.event('subscribe', data.paymentMethod).send();
|
|
utils.ga.transaction(data.user._id, block.price).item(block.price, 1, data.paymentMethod.toLowerCase() + '-subscription', data.paymentMethod).send();
|
|
}
|
|
data.user.purchased.txnCount++;
|
|
if (data.gift) members.sendMessage(data.user, data.gift.member, data.gift);
|
|
async.parallel([
|
|
function(cb2){data.user.save(cb2)},
|
|
function(cb2){data.gift ? data.gift.member.save(cb2) : cb2(null);}
|
|
], cb);
|
|
}
|
|
|
|
/**
|
|
* Sets their subscription to be cancelled later
|
|
*/
|
|
exports.cancelSubscription = function(data, cb) {
|
|
var p = data.user.purchased.plan,
|
|
now = moment(),
|
|
remaining = data.nextBill ? moment(data.nextBill).diff(new Date, 'days') : 30;
|
|
|
|
p.dateTerminated =
|
|
moment( now.format('MM') + '/' + moment(p.dateUpdated).format('DD') + '/' + now.format('YYYY') )
|
|
.add({days: remaining}) // end their subscription 1mo from their last payment
|
|
.add({months: Math.ceil(p.extraMonths)})// plus any extra time (carry-over, gifted subscription, etc) they have. FIXME: moment can't add months in fractions...
|
|
.toDate();
|
|
p.extraMonths = 0; // clear extra time. If they subscribe again, it'll be recalculated from p.dateTerminated
|
|
|
|
data.user.save(cb);
|
|
if(isProduction) utils.txnEmail(data.user, 'cancel-subscription');
|
|
utils.ga.event('unsubscribe', data.paymentMethod).send();
|
|
}
|
|
|
|
exports.buyGems = function(data, cb) {
|
|
var amt = data.gift ? data.gift.gems.amount/4 : 5;
|
|
(data.gift ? data.gift.member : data.user).balance += amt;
|
|
data.user.purchased.txnCount++;
|
|
if(isProduction) {
|
|
if (!data.gift) utils.txnEmail(data.user, 'donation');
|
|
utils.ga.event('checkout', data.paymentMethod).send();
|
|
//TODO ga.transaction to reflect whether this is gift or self-purchase
|
|
utils.ga.transaction(data.user._id, amt).item(amt, 1, data.paymentMethod.toLowerCase() + "-checkout", "Gems > " + data.paymentMethod).send();
|
|
}
|
|
if (data.gift) members.sendMessage(data.user, data.gift.member, data.gift);
|
|
async.parallel([
|
|
function(cb2){data.user.save(cb2)},
|
|
function(cb2){data.gift ? data.gift.member.save(cb2) : cb2(null);}
|
|
], cb);
|
|
}
|
|
|
|
exports.validCoupon = function(req, res, next){
|
|
mongoose.model('Coupon').findOne({_id:cc.validate(req.params.code), event:'google_6mo'}, function(err, coupon){
|
|
if (err) return next(err);
|
|
if (!coupon) return res.json(401, {err:"Invalid coupon code"});
|
|
return res.send(200);
|
|
});
|
|
}
|
|
|
|
exports.stripeCheckout = stripe.checkout;
|
|
exports.stripeSubscribeCancel = stripe.subscribeCancel;
|
|
exports.stripeSubscribeEdit = stripe.subscribeEdit;
|
|
|
|
exports.paypalSubscribe = paypal.createBillingAgreement;
|
|
exports.paypalSubscribeSuccess = paypal.executeBillingAgreement;
|
|
exports.paypalSubscribeCancel = paypal.cancelSubscription;
|
|
exports.paypalCheckout = paypal.createPayment;
|
|
exports.paypalCheckoutSuccess = paypal.executePayment;
|
|
exports.paypalIPN = paypal.ipn;
|
|
|
|
exports.iapAndroidVerify = iap.androidVerify;
|
|
exports.iapIosVerify = iap.iosVerify;
|