mirror of
https://github.com/codeninjasllc/codecombat.git
synced 2025-04-02 16:21:01 -04:00
Add sponsored subs to admin dashboard
This commit is contained in:
parent
92332c37f3
commit
59de47069a
4 changed files with 96 additions and 97 deletions
app
styles/admin
templates/admin
views/admin
server/payments
|
@ -47,7 +47,14 @@
|
|||
background-color: blanchedalmond
|
||||
font-size: 10pt
|
||||
|
||||
.subscribers-thead
|
||||
font-size: 10pt
|
||||
th
|
||||
padding: 2px
|
||||
|
||||
.subscribers-tbody
|
||||
font-size: 9pt
|
||||
td
|
||||
padding: 2px
|
||||
max-width: 220px
|
||||
overflow: hidden
|
||||
|
|
|
@ -36,14 +36,12 @@ block content
|
|||
each value in point.values
|
||||
div #{value}
|
||||
|
||||
div *Stripe APIs do not return information about inactive subs.
|
||||
|
||||
h2 Recent Subscribers
|
||||
if !subscribers || subscribers.length < 1
|
||||
h4 Fetching recent subscribers...
|
||||
else
|
||||
table.table.table-striped.table-condensed
|
||||
thead
|
||||
thead.subscribers-thead
|
||||
tr
|
||||
th Sub ID
|
||||
th User Start
|
||||
|
@ -52,19 +50,22 @@ block content
|
|||
th Cancelled
|
||||
else
|
||||
th
|
||||
th
|
||||
//- th Name
|
||||
if subscriberSponsored
|
||||
th Sponsored
|
||||
else
|
||||
th
|
||||
th Email
|
||||
th Hero
|
||||
th Level
|
||||
th Last Level
|
||||
th Age
|
||||
th Spoken
|
||||
th Clans
|
||||
tbody.subscribers-tbody
|
||||
each subscriber in subscribers
|
||||
tr
|
||||
td
|
||||
a(href="https://dashboard.stripe.com/customers/#{subscriber.customerID}", target="_blank")= subscriber.subscriptionID
|
||||
a(href="https://dashboard.stripe.com/customers/#{subscriber.customerID}", target="_blank")= subscriber.subID
|
||||
td= subscriber.user.dateCreated.substring(0, 10)
|
||||
td= subscriber.start.substring(0, 10)
|
||||
td
|
||||
|
@ -72,15 +73,17 @@ block content
|
|||
span= subscriber.cancel.substring(0, 10)
|
||||
td
|
||||
if subscriber.user.stripe.sponsorID
|
||||
span Sponsored
|
||||
//- td
|
||||
//- a(href="/user/#{subscriber.user._id}")= subscriber.user.name || 'Anoner'
|
||||
span Yes
|
||||
td= subscriber.user.emailLower
|
||||
td= subscriber.hero
|
||||
td= subscriber.level
|
||||
td= subscriber.user.lastLevel
|
||||
td= subscriber.user.ageRange
|
||||
td= subscriber.user.preferredLanguage
|
||||
if subscriber.user.clans
|
||||
td= subscriber.user.clans.length
|
||||
else
|
||||
td
|
||||
|
||||
h2 Subscriptions
|
||||
if !subs || subs.length < 1
|
||||
|
@ -93,8 +96,9 @@ block content
|
|||
th Total
|
||||
th Started
|
||||
th Cancelled
|
||||
th Net
|
||||
th Net (cancelled)
|
||||
th Ended
|
||||
th Net (ended)
|
||||
tbody
|
||||
each sub in subs
|
||||
tr
|
||||
|
@ -104,3 +108,4 @@ block content
|
|||
td= sub.cancelled
|
||||
td= sub.started - sub.cancelled
|
||||
td= sub.ended
|
||||
td= sub.started - sub.ended
|
||||
|
|
|
@ -10,7 +10,7 @@ require 'vendor/d3'
|
|||
module.exports = class AnalyticsSubscriptionsView extends RootView
|
||||
id: 'admin-analytics-subscriptions-view'
|
||||
template: template
|
||||
targetSubCount: 2000
|
||||
targetSubCount: 1200
|
||||
|
||||
constructor: (options) ->
|
||||
super options
|
||||
|
@ -25,6 +25,7 @@ module.exports = class AnalyticsSubscriptionsView extends RootView
|
|||
context.subs = _.cloneDeep(@subs ? []).reverse()
|
||||
context.subscribers = @subscribers ? []
|
||||
context.subscriberCancelled = _.find context.subscribers, (subscriber) -> subscriber.cancel
|
||||
context.subscriberSponsored = _.find context.subscribers, (subscriber) -> subscriber.user?.stripe?.sponsorID
|
||||
context.total = @total ? 0
|
||||
context.cancelled = @cancelled ? 0
|
||||
context.monthlyChurn = @monthlyChurn ? 0.0
|
||||
|
@ -46,9 +47,9 @@ module.exports = class AnalyticsSubscriptionsView extends RootView
|
|||
refreshData: ->
|
||||
return unless me.isAdmin()
|
||||
@resetSubscriptionsData()
|
||||
@getSubscribers()
|
||||
@getCancellations (cancellations) =>
|
||||
@getSubscriptions(cancellations)
|
||||
@getSubscriptions cancellations, (subscriptions) =>
|
||||
@getSubscribers(subscriptions)
|
||||
|
||||
getCancellations: (done) ->
|
||||
options =
|
||||
|
@ -62,25 +63,34 @@ module.exports = class AnalyticsSubscriptionsView extends RootView
|
|||
done(cancellations)
|
||||
@supermodel.addRequestResource('get_cancellations', options, 0).load()
|
||||
|
||||
getSubscribers: ->
|
||||
getSubscribers: (subscriptions) ->
|
||||
maxSubscribers = 40
|
||||
|
||||
subscribers = _.filter subscriptions, (a) -> a.userID?
|
||||
subscribers.sort (a, b) -> b.start.localeCompare(a.start)
|
||||
subscribers = subscribers.slice(0, maxSubscribers)
|
||||
subscriberUserIDs = _.map subscribers, (a) -> a.userID
|
||||
|
||||
options =
|
||||
url: '/db/subscription/-/subscribers'
|
||||
method: 'POST'
|
||||
data: {maxCount: 30}
|
||||
data: {ids: subscriberUserIDs}
|
||||
options.error = (model, response, options) =>
|
||||
return if @destroyed
|
||||
console.error 'Failed to get subscribers', response
|
||||
options.success = (subscribers, response, options) =>
|
||||
options.success = (userMap, response, options) =>
|
||||
return if @destroyed
|
||||
@subscribers = subscribers
|
||||
for subscriber in @subscribers
|
||||
for subscriber in subscribers
|
||||
continue unless subscriber.userID of userMap
|
||||
subscriber.user = userMap[subscriber.userID]
|
||||
subscriber.level = User.levelFromExp subscriber.user.points
|
||||
if hero = subscriber.user.heroConfig?.thangType
|
||||
subscriber.hero = _.invert(ThangType.heroes)[hero]
|
||||
@subscribers = subscribers
|
||||
@render?()
|
||||
@supermodel.addRequestResource('get_subscribers', options, 0).load()
|
||||
|
||||
getSubscriptions: (cancellations=[]) ->
|
||||
getSubscriptions: (cancellations=[], done) ->
|
||||
options =
|
||||
url: '/db/subscription/-/subscriptions'
|
||||
method: 'GET'
|
||||
|
@ -102,6 +112,7 @@ module.exports = class AnalyticsSubscriptionsView extends RootView
|
|||
subDayMap[endDay]['end']++
|
||||
for cancellation in cancellations
|
||||
if cancellation.subID is sub.subID
|
||||
sub.cancel = cancellation.cancel
|
||||
cancelDay = cancellation.cancel.substring(0, 10)
|
||||
subDayMap[cancelDay] ?= {}
|
||||
subDayMap[cancelDay]['cancel'] ?= 0
|
||||
|
@ -132,6 +143,7 @@ module.exports = class AnalyticsSubscriptionsView extends RootView
|
|||
@monthlyGrowth = (endMonthTotal / startMonthTotal - 1) * 100
|
||||
@updateAnalyticsGraphData()
|
||||
@render?()
|
||||
done(subs)
|
||||
@supermodel.addRequestResource('get_subscriptions', options, 0).load()
|
||||
|
||||
updateAnalyticsGraphData: ->
|
||||
|
@ -170,7 +182,7 @@ module.exports = class AnalyticsSubscriptionsView extends RootView
|
|||
color: 'red'
|
||||
strokeWidth: 1
|
||||
lineMetadata[netSubsID] =
|
||||
description: '7-day Average Net Subscriptions'
|
||||
description: '7-day Average Net Subscriptions (started - ended)'
|
||||
color: 'black'
|
||||
strokeWidth: 4
|
||||
|
||||
|
|
|
@ -67,7 +67,7 @@ class SubscriptionHandler extends Handler
|
|||
(done) =>
|
||||
stripe.customers.retrieveSubscription customerID, subscriptionID, (err, subscription) =>
|
||||
return done() if err
|
||||
return unless subscription?.cancel_at_period_end
|
||||
return done() unless subscription?.cancel_at_period_end
|
||||
cancellations.push
|
||||
cancel: new Date(subscription.canceled_at * 1000)
|
||||
subID: subscription.id
|
||||
|
@ -82,74 +82,18 @@ class SubscriptionHandler extends Handler
|
|||
getSubscribers: (req, res) ->
|
||||
# console.log 'subscription_handler getSubscribers'
|
||||
return @sendForbiddenError(res) unless req.user?.isAdmin()
|
||||
|
||||
maxReturnCount = req.body.maxCount or 20
|
||||
|
||||
# @subscribers ?= []
|
||||
# return @sendSuccess(res, @subscribers) unless _.isEmpty(@subscribers)
|
||||
@subscribers = []
|
||||
|
||||
subscriberIDs = []
|
||||
|
||||
customersProcessed = 0
|
||||
nextBatch = (starting_after, done) =>
|
||||
options = limit: 100
|
||||
options.starting_after = starting_after if starting_after
|
||||
stripe.customers.list options, (err, customers) =>
|
||||
return done(err) if err
|
||||
customersProcessed += customers.data.length
|
||||
|
||||
for customer in customers.data
|
||||
break unless @subscribers.length < maxReturnCount
|
||||
continue unless customer?.subscriptions?.data?.length > 0
|
||||
for subscription in customer.subscriptions.data
|
||||
continue unless subscription.plan.id is 'basic'
|
||||
|
||||
amount = subscription.plan.amount
|
||||
if subscription?.discount?.coupon?
|
||||
if subscription.discount.coupon.percent_off
|
||||
amount = amount * (100 - subscription.discount.coupon.percent_off) / 100;
|
||||
else if subscription.discount.coupon.amount_off
|
||||
amount -= subscription.discount.coupon.amount_off
|
||||
else if customer.discount?.coupon?
|
||||
if customer.discount.coupon.percent_off
|
||||
amount = amount * (100 - customer.discount.coupon.percent_off) / 100
|
||||
else if customer.discount.coupon.amount_off
|
||||
amount -= customer.discount.coupon.amount_off
|
||||
|
||||
continue unless amount > 0
|
||||
|
||||
subscriber =
|
||||
customerID: customer.id
|
||||
start: new Date(subscription.start * 1000)
|
||||
subscriptionID: subscription.id
|
||||
if subscription.metadata?.id?
|
||||
subscriber.userID = subscription.metadata.id
|
||||
subscriberIDs.push subscription.metadata.id
|
||||
if subscription.cancel_at_period_end
|
||||
subscriber.cancel = new Date(subscription.canceled_at * 1000)
|
||||
subscriber.end = new Date(subscription.current_period_end * 1000)
|
||||
@subscribers.push(subscriber)
|
||||
|
||||
if customers.has_more and @subscribers.length < maxReturnCount
|
||||
return nextBatch(customers.data[customers.data.length - 1].id, done)
|
||||
else
|
||||
return done()
|
||||
nextBatch null, (err) =>
|
||||
subscriberUserIDs = req.body.ids or []
|
||||
User.find {_id: {$in: subscriberUserIDs}}, (err, users) =>
|
||||
return @sendDatabaseError(res, err) if err
|
||||
User.find {_id: {$in: subscriberIDs}}, (err, users) =>
|
||||
return @sendDatabaseError(res, err) if err
|
||||
for user in users
|
||||
subscriber.user = user for subscriber in @subscribers when subscriber.userID is user.id
|
||||
@sendSuccess(res, @subscribers)
|
||||
userMap = {}
|
||||
userMap[user.id] = user for user in users
|
||||
@sendSuccess(res, userMap)
|
||||
|
||||
getSubscriptions: (req, res) ->
|
||||
# console.log 'subscription_handler getSubscriptions'
|
||||
# Returns a list of active subscriptions
|
||||
# TODO: does not track sponsored subs, only basic
|
||||
# TODO: does not return free subs
|
||||
# TODO: add tests
|
||||
# TODO: aggregate this data daily instead of providing it on demand
|
||||
# TODO: take date range as input
|
||||
# TODO: are ended counts correct for today? E.g. retries may complicate things.
|
||||
|
||||
|
@ -169,11 +113,13 @@ class SubscriptionHandler extends Handler
|
|||
continue unless invoice.subscription
|
||||
continue unless invoice.total > 0
|
||||
continue unless invoice.lines?.data?[0]?.plan?.id is 'basic'
|
||||
newInvoices.push
|
||||
newInvoice =
|
||||
customerID: invoice.customer
|
||||
invoiceID: invoice.id
|
||||
subscriptionID: invoice.subscription
|
||||
date: new Date(invoice.date * 1000)
|
||||
newInvoice.userID = invoice.lines.data[0].metadata.id if invoice.lines?.data?[0]?.metadata?.id
|
||||
newInvoices.push newInvoice
|
||||
if invoices.has_more
|
||||
# console.log 'Fetching more invoices', @invoices.length, newInvoices.length
|
||||
return processInvoices(invoices.data[invoices.data.length - 1].id, done)
|
||||
|
@ -193,20 +139,49 @@ class SubscriptionHandler extends Handler
|
|||
first: invoice.date
|
||||
last: invoice.date
|
||||
customerID: invoice.customerID
|
||||
subs = []
|
||||
for subID of subMap
|
||||
sub =
|
||||
start: subMap[subID].first
|
||||
subID: subID
|
||||
customerID: subMap[subID].customerID
|
||||
sub.cancel = subMap[subID].cancel if subMap[subID].cancel
|
||||
oneMonthAgo = new Date()
|
||||
oneMonthAgo.setUTCMonth(oneMonthAgo.getUTCMonth() - 1)
|
||||
if subMap[subID].last < oneMonthAgo
|
||||
sub.end = new Date(subMap[subID].last)
|
||||
sub.end.setUTCMonth(sub.end.getUTCMonth() + 1)
|
||||
subs.push sub
|
||||
@sendSuccess(res, subs)
|
||||
subMap[subID].userID = invoice.userID if invoice.userID
|
||||
|
||||
# Check sponsored subscriptions
|
||||
User.find {"stripe.sponsorSubscriptionID": {$exists: true}}, (err, sponsors) =>
|
||||
return @sendDatabaseError(res, err) if err
|
||||
|
||||
createCheckSubFn = (customerID, subscriptionID) =>
|
||||
(done) =>
|
||||
stripe.customers.retrieveSubscription customerID, subscriptionID, (err, subscription) =>
|
||||
return done() if err
|
||||
return done() unless subscription?
|
||||
subMap[subscription.id] =
|
||||
first: new Date(subscription.start * 1000)
|
||||
subMap[subscription.id].userID = subscription.metadata.id if subscription.metadata?.id?
|
||||
if subscription.cancel_at_period_end
|
||||
subMap[subscription.id].cancel = new Date(subscription.canceled_at * 1000)
|
||||
subMap[subscription.id].end = new Date(subscription.current_period_end * 1000)
|
||||
done()
|
||||
|
||||
tasks = []
|
||||
for user in sponsors
|
||||
for recipient in user.get("stripe")?.recipients
|
||||
tasks.push createCheckSubFn(user.get('stripe')?.customerID, recipient.subscriptionID)
|
||||
async.parallel tasks, (err, results) =>
|
||||
return @sendDatabaseError(res, err) if err
|
||||
|
||||
subs = []
|
||||
for subID of subMap
|
||||
sub =
|
||||
start: subMap[subID].first
|
||||
subID: subID
|
||||
customerID: subMap[subID].customerID
|
||||
sub.cancel = subMap[subID].cancel if subMap[subID].cancel
|
||||
oneMonthAgo = new Date()
|
||||
oneMonthAgo.setUTCMonth(oneMonthAgo.getUTCMonth() - 1)
|
||||
if subMap[subID].end?
|
||||
sub.end = subMap[subID].end
|
||||
else if subMap[subID].last < oneMonthAgo
|
||||
sub.end = new Date(subMap[subID].last)
|
||||
sub.end.setUTCMonth(sub.end.getUTCMonth() + 1)
|
||||
sub.userID = subMap[subID].userID if subMap[subID].userID
|
||||
subs.push sub
|
||||
@sendSuccess(res, subs)
|
||||
|
||||
subscribeUser: (req, user, done) ->
|
||||
if (not req.user) or req.user.isAnonymous() or user.isAnonymous()
|
||||
|
|
Loading…
Add table
Reference in a new issue