2022-03-23 13:53:32 +03:00
|
|
|
import Service, {inject as service} from '@ember/service';
|
|
|
|
import moment from 'moment';
|
2022-03-24 17:32:34 +03:00
|
|
|
import {task} from 'ember-concurrency';
|
2022-03-23 11:51:53 +03:00
|
|
|
import {tracked} from '@glimmer/tracking';
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
/**
|
|
|
|
* @typedef MrrStat
|
|
|
|
* @type {Object}
|
|
|
|
* @property {string} date The date (YYYY-MM-DD) on which this MRR was recorded
|
|
|
|
* @property {number} mrr The MRR on this date
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef MemberCountStat
|
|
|
|
* @type {Object}
|
|
|
|
* @property {string} date The date (YYYY-MM-DD) on which these counts were recorded
|
|
|
|
* @property {number} paid Amount of paid members
|
|
|
|
* @property {number} free Amount of free members
|
|
|
|
* @property {number} comped Amount of comped members
|
|
|
|
* @property {number} newPaid Amount of new paid members
|
|
|
|
* @property {number} canceledPaid Amount of canceled paid members
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef MemberCounts
|
|
|
|
* @type {Object}
|
|
|
|
* @property {number} total Total amount of members
|
|
|
|
* @property {number} paid Amount of paid members
|
|
|
|
* @property {number} free Amount of free members
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2022-03-28 15:20:07 +03:00
|
|
|
* @typedef EmailOpenRateStat
|
2022-03-23 18:38:16 +03:00
|
|
|
* @type {Object}
|
2022-03-28 15:20:07 +03:00
|
|
|
* @property {string} subject Email title
|
|
|
|
* @property {number} openRate Email openRate
|
|
|
|
* @property {Date} submittedAt Date
|
2022-03-23 18:38:16 +03:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef PaidMembersByCadence
|
|
|
|
* @type {Object}
|
|
|
|
* @property {number} annual Paid memebrs on annual plan
|
|
|
|
* @property {number} monthly Paid memebrs on monthly plan
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef PaidMembersForTier
|
|
|
|
* @type {Object}
|
|
|
|
* @property {Object} tier Tier object
|
|
|
|
* @property {number} members Paid members on this tier
|
|
|
|
*/
|
|
|
|
|
2022-03-24 14:04:18 +03:00
|
|
|
/**
|
|
|
|
* @typedef SiteStatus Contains information on what graphs need to be shown
|
|
|
|
* @type {Object}
|
|
|
|
* @property {boolean} hasPaidTiers Whether the site has paid tiers
|
|
|
|
* @property {boolean} stripeEnabled Whether the site has stripe enabled
|
|
|
|
* @property {boolean} newslettersEnabled Whether the site has newsletters
|
|
|
|
* @property {boolean} membersEnabled Whether the site has members enabled
|
|
|
|
*/
|
|
|
|
|
2022-03-23 11:51:53 +03:00
|
|
|
export default class DashboardStatsService extends Service {
|
2022-03-23 13:53:32 +03:00
|
|
|
@service dashboardMocks;
|
2022-03-24 20:14:51 +03:00
|
|
|
@service store;
|
2022-03-28 11:47:09 +03:00
|
|
|
@service ajax;
|
|
|
|
@service ghostPaths;
|
2022-03-23 11:51:53 +03:00
|
|
|
|
2022-03-24 14:04:18 +03:00
|
|
|
/**
|
|
|
|
* @type {?SiteStatus} Contains information on what graphs need to be shown
|
|
|
|
*/
|
|
|
|
@tracked siteStatus = null;
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
/**
|
|
|
|
* @type {?MemberCounts} memberCounts
|
|
|
|
*/
|
2022-03-23 11:51:53 +03:00
|
|
|
@tracked
|
|
|
|
memberCounts = null;
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
/**
|
|
|
|
* @type {?MemberCountStat[]}
|
|
|
|
*/
|
|
|
|
@tracked
|
|
|
|
memberCountStats = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {?MrrStat[]}
|
|
|
|
*/
|
2022-03-23 11:51:53 +03:00
|
|
|
@tracked
|
2022-03-23 18:38:16 +03:00
|
|
|
mrrStats = null;
|
2022-03-23 11:51:53 +03:00
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
/**
|
|
|
|
* @type {PaidMembersByCadence} Number of members for annual and monthly plans
|
|
|
|
*/
|
2022-03-23 11:51:53 +03:00
|
|
|
@tracked
|
2022-03-23 18:38:16 +03:00
|
|
|
paidMembersByCadence = null;
|
2022-03-23 11:51:53 +03:00
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
/**
|
|
|
|
* @type {PaidMembersForTier[]} Number of members for each tier
|
|
|
|
*/
|
|
|
|
@tracked
|
|
|
|
paidMembersByTier = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {?number} Number of members last seen in last 30 days (could differ if filtered by member status)
|
|
|
|
*/
|
2022-03-23 11:51:53 +03:00
|
|
|
@tracked
|
|
|
|
membersLastSeen30d = null;
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
/**
|
|
|
|
* @type {?number} Number of members last seen in last 7 days (could differ if filtered by member status)
|
|
|
|
*/
|
2022-03-23 11:51:53 +03:00
|
|
|
@tracked
|
|
|
|
membersLastSeen7d = null;
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
/**
|
|
|
|
* @type {?MemberCounts} Number of members that are subscribed (grouped by status)
|
|
|
|
*/
|
2022-03-23 11:51:53 +03:00
|
|
|
@tracked
|
|
|
|
newsletterSubscribers = null;
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
/**
|
|
|
|
* @type {?number} Number of emails sent in last 30 days
|
|
|
|
*/
|
2022-03-23 11:51:53 +03:00
|
|
|
@tracked
|
|
|
|
emailsSent30d = null;
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
/**
|
|
|
|
* @type {?EmailOpenRateStat[]}
|
|
|
|
*/
|
2022-03-23 11:51:53 +03:00
|
|
|
@tracked
|
|
|
|
emailOpenRateStats = null;
|
|
|
|
|
2022-03-24 14:04:18 +03:00
|
|
|
/**
|
|
|
|
* Amount of days to load for member count and MRR related charts
|
|
|
|
*/
|
|
|
|
@tracked chartDays = 7;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Filter last seen by this status
|
|
|
|
* @type {'free'|'paid'|'total'}
|
|
|
|
*/
|
|
|
|
@tracked lastSeenFilterStatus = 'total';
|
|
|
|
|
|
|
|
loadSiteStatus() {
|
2022-03-24 17:32:34 +03:00
|
|
|
return this._loadSiteStatus.perform();
|
|
|
|
}
|
|
|
|
|
|
|
|
@task
|
|
|
|
*_loadSiteStatus() {
|
2022-03-24 14:04:18 +03:00
|
|
|
this.siteStatus = null;
|
|
|
|
if (this.dashboardMocks.enabled) {
|
2022-03-24 17:32:34 +03:00
|
|
|
yield this.dashboardMocks.loadSiteStatus();
|
|
|
|
this.siteStatus = {...this.dashboardMocks.siteStatus};
|
2022-03-24 14:04:18 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Normal implementation
|
|
|
|
// @todo
|
2022-03-24 20:14:51 +03:00
|
|
|
this.siteStatus = {
|
|
|
|
hasPaidTiers: true,
|
|
|
|
stripeEnabled: true,
|
|
|
|
newslettersEnabled: true,
|
|
|
|
membersEnabled: true
|
|
|
|
};
|
2022-03-24 14:04:18 +03:00
|
|
|
}
|
|
|
|
|
2022-03-23 11:51:53 +03:00
|
|
|
loadMembersCounts() {
|
2022-03-24 17:32:34 +03:00
|
|
|
return this._loadMembersCounts.perform();
|
|
|
|
}
|
|
|
|
|
2022-03-24 20:14:51 +03:00
|
|
|
@task({restartable: true})
|
2022-03-24 17:32:34 +03:00
|
|
|
*_loadMembersCounts() {
|
|
|
|
this.memberCounts = null;
|
2022-03-23 13:53:32 +03:00
|
|
|
if (this.dashboardMocks.enabled) {
|
2022-03-24 17:32:34 +03:00
|
|
|
yield this.dashboardMocks.waitRandom();
|
2022-03-24 11:01:30 +03:00
|
|
|
if (this.dashboardMocks.memberCounts === null) {
|
|
|
|
return null;
|
|
|
|
}
|
2022-03-23 18:38:16 +03:00
|
|
|
this.memberCounts = {...this.dashboardMocks.memberCounts};
|
2022-03-23 11:51:53 +03:00
|
|
|
return;
|
|
|
|
}
|
2022-03-28 10:54:21 +03:00
|
|
|
|
|
|
|
// @todo We need to have way to reduce the total number of API requests
|
|
|
|
const paidResult = yield this.store.query('member', {limit: 1, filter: 'status:paid'});
|
|
|
|
const paid = paidResult.meta.pagination.total;
|
|
|
|
|
|
|
|
const freeResult = yield this.store.query('member', {limit: 1, filter: 'status:-paid'});
|
|
|
|
const free = freeResult.meta.pagination.total;
|
|
|
|
|
|
|
|
this.memberCounts = {
|
|
|
|
total: paid + free,
|
|
|
|
paid,
|
|
|
|
free
|
|
|
|
};
|
2022-03-23 11:51:53 +03:00
|
|
|
}
|
|
|
|
|
2022-03-24 17:32:34 +03:00
|
|
|
loadMemberCountStats() {
|
|
|
|
// todo: add proper logic to prevent duplicate calls + reuse results if nothing has changed
|
|
|
|
return this._loadMemberCountStats.perform();
|
|
|
|
}
|
|
|
|
|
2022-03-23 11:51:53 +03:00
|
|
|
/**
|
|
|
|
* Loads the members graphs
|
|
|
|
* - total paid
|
|
|
|
* - total members
|
2022-03-24 14:04:18 +03:00
|
|
|
* for each day in the last chartDays days
|
2022-03-23 11:51:53 +03:00
|
|
|
*/
|
2022-03-24 20:14:51 +03:00
|
|
|
@task({restartable: true})
|
2022-03-24 17:32:34 +03:00
|
|
|
*_loadMemberCountStats() {
|
|
|
|
this.memberCountStats = null;
|
2022-03-23 13:53:32 +03:00
|
|
|
if (this.dashboardMocks.enabled) {
|
2022-03-24 17:32:34 +03:00
|
|
|
yield this.dashboardMocks.waitRandom();
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
if (this.dashboardMocks.memberCountStats === null) {
|
2022-03-24 17:32:34 +03:00
|
|
|
// Note: that this shouldn't happen
|
2022-03-23 18:38:16 +03:00
|
|
|
return null;
|
|
|
|
}
|
2022-03-28 11:47:09 +03:00
|
|
|
this.memberCountStats = this.fillMissingDates(this.dashboardMocks.memberCountStats, {paid: 0, free: 0, comped: 0}, this.chartDays);
|
2022-03-23 11:51:53 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-28 11:47:09 +03:00
|
|
|
// @todo: we need to reuse the result of the call when we reload, because the endpoint returns all available days
|
|
|
|
// at the moment. We can reuse the result to show 7 days, 30 days, ...
|
|
|
|
let statsUrl = this.ghostPaths.url.api('members/stats/count');
|
|
|
|
let stats = yield this.ajax.request(statsUrl);
|
|
|
|
this.memberCountStats = this.fillMissingDates(stats.data, {paid: 0, free: 0, comped: 0}, this.chartDays);
|
2022-03-23 11:51:53 +03:00
|
|
|
}
|
|
|
|
|
2022-03-24 17:32:34 +03:00
|
|
|
loadMrrStats() {
|
|
|
|
// todo: add proper logic to prevent duplicate calls + reuse results if nothing has changed
|
|
|
|
return this._loadMrrStats.perform();
|
|
|
|
}
|
|
|
|
|
2022-03-23 11:51:53 +03:00
|
|
|
/**
|
2022-03-24 14:04:18 +03:00
|
|
|
* Loads the mrr graphs for the current chartDays days
|
2022-03-23 11:51:53 +03:00
|
|
|
*/
|
2022-03-24 20:14:51 +03:00
|
|
|
@task({restartable: true})
|
2022-03-24 17:32:34 +03:00
|
|
|
*_loadMrrStats() {
|
|
|
|
this.mrrStats = null;
|
2022-03-23 13:53:32 +03:00
|
|
|
if (this.dashboardMocks.enabled) {
|
2022-03-24 17:32:34 +03:00
|
|
|
yield this.dashboardMocks.waitRandom();
|
2022-03-23 18:38:16 +03:00
|
|
|
if (this.dashboardMocks.mrrStats === null) {
|
|
|
|
return null;
|
|
|
|
}
|
2022-03-28 12:12:01 +03:00
|
|
|
this.mrrStats = this.fillMissingDates(this.dashboardMocks.mrrStats, {mrr: 0}, this.chartDays);
|
2022-03-23 11:51:53 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-28 12:12:01 +03:00
|
|
|
// @todo: we need to reuse the result of the call when we reload, because the endpoint returns all available days
|
|
|
|
// at the moment. We can reuse the result to show 7 days, 30 days, ...
|
|
|
|
let statsUrl = this.ghostPaths.url.api('members/stats/mrr');
|
|
|
|
let stats = yield this.ajax.request(statsUrl);
|
|
|
|
|
|
|
|
// @todo: add proper support for all different currencies that are returned
|
|
|
|
this.mrrStats = this.fillMissingDates(
|
|
|
|
stats.data[0].data.map((d) => {
|
|
|
|
return {date: d.date, mrr: d.value};
|
|
|
|
}),
|
|
|
|
{mrr: 0},
|
|
|
|
this.chartDays
|
|
|
|
);
|
2022-03-23 11:51:53 +03:00
|
|
|
}
|
|
|
|
|
2022-03-24 17:32:34 +03:00
|
|
|
loadLastSeen() {
|
|
|
|
// todo: add proper logic to prevent duplicate calls + reuse results if nothing has changed
|
|
|
|
return this._loadLastSeen.perform();
|
|
|
|
}
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
/**
|
|
|
|
* Loads the mrr graphs
|
|
|
|
*/
|
2022-03-24 20:14:51 +03:00
|
|
|
@task({restartable: true})
|
2022-03-24 17:32:34 +03:00
|
|
|
*_loadLastSeen() {
|
|
|
|
this.membersLastSeen30d = null;
|
|
|
|
this.membersLastSeen7d = null;
|
|
|
|
|
2022-03-23 13:53:32 +03:00
|
|
|
if (this.dashboardMocks.enabled) {
|
2022-03-24 17:32:34 +03:00
|
|
|
yield this.dashboardMocks.waitRandom();
|
2022-03-24 14:04:18 +03:00
|
|
|
if (this.lastSeenFilterStatus === 'paid') {
|
2022-03-23 18:38:16 +03:00
|
|
|
// @todo
|
|
|
|
}
|
2022-03-23 13:53:32 +03:00
|
|
|
this.membersLastSeen30d = this.dashboardMocks.membersLastSeen30d;
|
|
|
|
this.membersLastSeen7d = this.dashboardMocks.membersLastSeen7d;
|
2022-03-23 11:51:53 +03:00
|
|
|
return;
|
|
|
|
}
|
2022-03-28 10:54:21 +03:00
|
|
|
|
|
|
|
// @todo We need to have way to reduce the total number of API requests
|
|
|
|
|
|
|
|
const start30d = new Date(Date.now() - 30 * 3600 * 1000);
|
|
|
|
const start7d = new Date(Date.now() - 7 * 3600 * 1000);
|
|
|
|
|
|
|
|
let extraFilter = '';
|
|
|
|
if (this.lastSeenFilterStatus === 'paid') {
|
|
|
|
extraFilter = '+status:paid';
|
|
|
|
} else if (this.lastSeenFilterStatus === 'free') {
|
|
|
|
extraFilter = '+status:-paid';
|
|
|
|
}
|
|
|
|
|
|
|
|
// todo: filter by status here
|
|
|
|
const result30d = yield this.store.query('member', {limit: 1, filter: 'last_seen_at:>' + start30d.toISOString() + extraFilter});
|
|
|
|
this.membersLastSeen30d = result30d.meta.pagination.total;
|
|
|
|
|
|
|
|
const result7d = yield this.store.query('member', {limit: 1, filter: 'last_seen_at:>' + start7d.toISOString() + extraFilter});
|
|
|
|
this.membersLastSeen7d = result7d.meta.pagination.total;
|
2022-03-23 11:51:53 +03:00
|
|
|
}
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
loadPaidMembersByCadence() {
|
2022-03-24 17:32:34 +03:00
|
|
|
// todo: add proper logic to prevent duplicate calls + reuse results if nothing has changed
|
|
|
|
return this._loadPaidMembersByCadence.perform();
|
|
|
|
}
|
|
|
|
|
2022-03-24 20:14:51 +03:00
|
|
|
@task({restartable: true})
|
2022-03-24 17:32:34 +03:00
|
|
|
*_loadPaidMembersByCadence() {
|
|
|
|
this.paidMembersByCadence = null;
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
if (this.dashboardMocks.enabled) {
|
2022-03-24 17:32:34 +03:00
|
|
|
yield this.dashboardMocks.waitRandom();
|
|
|
|
this.paidMembersByCadence = {...this.dashboardMocks.paidMembersByCadence};
|
2022-03-23 18:38:16 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Normal implementation
|
|
|
|
// @todo
|
|
|
|
}
|
|
|
|
|
|
|
|
loadPaidMembersByTier() {
|
2022-03-24 17:32:34 +03:00
|
|
|
// todo: add proper logic to prevent duplicate calls + reuse results if nothing has changed
|
|
|
|
return this._loadPaidMembersByTier.perform();
|
|
|
|
}
|
|
|
|
|
2022-03-24 20:14:51 +03:00
|
|
|
@task({restartable: true})
|
2022-03-24 17:32:34 +03:00
|
|
|
*_loadPaidMembersByTier() {
|
|
|
|
this.paidMembersByTier = null;
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
if (this.dashboardMocks.enabled) {
|
2022-03-24 17:32:34 +03:00
|
|
|
yield this.dashboardMocks.waitRandom();
|
|
|
|
this.paidMembersByTier = this.dashboardMocks.paidMembersByTier.slice();
|
2022-03-23 18:38:16 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Normal implementation
|
|
|
|
// @todo
|
|
|
|
}
|
|
|
|
|
|
|
|
loadNewsletterSubscribers() {
|
2022-03-24 17:32:34 +03:00
|
|
|
// todo: add proper logic to prevent duplicate calls + reuse results if nothing has changed
|
|
|
|
return this._loadNewsletterSubscribers.perform();
|
|
|
|
}
|
|
|
|
|
2022-03-24 20:14:51 +03:00
|
|
|
@task({restartable: true})
|
2022-03-24 17:32:34 +03:00
|
|
|
*_loadNewsletterSubscribers() {
|
|
|
|
this.newsletterSubscribers = null;
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
if (this.dashboardMocks.enabled) {
|
2022-03-24 17:32:34 +03:00
|
|
|
yield this.dashboardMocks.waitRandom();
|
2022-03-23 18:38:16 +03:00
|
|
|
this.newsletterSubscribers = this.dashboardMocks.newsletterSubscribers;
|
|
|
|
return;
|
|
|
|
}
|
2022-03-28 14:19:56 +03:00
|
|
|
|
|
|
|
const resultPaid = yield this.store.query('member', {limit: 1, filter: 'subscribed:true+status:paid'});
|
|
|
|
const resultFree = yield this.store.query('member', {limit: 1, filter: 'subscribed:true+status:-paid'});
|
|
|
|
this.newsletterSubscribers = {
|
|
|
|
total: resultFree.meta.pagination.total + resultPaid.meta.pagination.total,
|
|
|
|
free: resultFree.meta.pagination.total,
|
|
|
|
paid: resultPaid.meta.pagination.total
|
|
|
|
};
|
2022-03-23 18:38:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
loadEmailsSent() {
|
2022-03-24 17:32:34 +03:00
|
|
|
// todo: add proper logic to prevent duplicate calls + reuse results if nothing has changed
|
|
|
|
return this._loadEmailsSent.perform();
|
|
|
|
}
|
|
|
|
|
2022-03-24 20:14:51 +03:00
|
|
|
@task({restartable: true})
|
2022-03-24 17:32:34 +03:00
|
|
|
*_loadEmailsSent() {
|
|
|
|
this.emailsSent30d = null;
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
if (this.dashboardMocks.enabled) {
|
2022-03-24 17:32:34 +03:00
|
|
|
yield this.dashboardMocks.waitRandom();
|
2022-03-23 18:38:16 +03:00
|
|
|
this.emailsSent30d = this.dashboardMocks.emailsSent30d;
|
|
|
|
return;
|
|
|
|
}
|
2022-03-28 14:49:44 +03:00
|
|
|
|
|
|
|
const start30d = new Date(Date.now() - 30 * 3600 * 1000);
|
|
|
|
const result = yield this.store.query('email', {limit: 100, filter: 'submitted_at:>' + start30d.toISOString()});
|
|
|
|
this.emailsSent30d = result.reduce((c, email) => c + email.emailCount, 0);
|
2022-03-23 18:38:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
loadEmailOpenRateStats() {
|
2022-03-24 17:32:34 +03:00
|
|
|
// todo: add proper logic to prevent duplicate calls + reuse results if nothing has changed
|
|
|
|
return this._loadEmailOpenRateStats.perform();
|
|
|
|
}
|
|
|
|
|
2022-03-24 20:14:51 +03:00
|
|
|
@task({restartable: true})
|
2022-03-24 17:32:34 +03:00
|
|
|
*_loadEmailOpenRateStats() {
|
|
|
|
this.emailOpenRateStats = null;
|
|
|
|
|
2022-03-23 18:38:16 +03:00
|
|
|
if (this.dashboardMocks.enabled) {
|
2022-03-24 17:32:34 +03:00
|
|
|
yield this.dashboardMocks.waitRandom();
|
2022-03-23 18:38:16 +03:00
|
|
|
this.emailOpenRateStats = this.dashboardMocks.emailOpenRateStats;
|
|
|
|
return;
|
|
|
|
}
|
2022-03-24 20:14:51 +03:00
|
|
|
|
2022-03-28 15:20:07 +03:00
|
|
|
const limit = 8;
|
|
|
|
let query = {
|
|
|
|
filter: 'email_count:-0',
|
|
|
|
order: 'submitted_at desc',
|
|
|
|
limit: limit
|
|
|
|
};
|
|
|
|
const results = yield this.store.query('email', query);
|
|
|
|
const data = results.toArray();
|
|
|
|
let stats = data.map((d) => {
|
|
|
|
return {
|
|
|
|
subject: d.subject,
|
|
|
|
submittedAt: moment(d.submittedAtUTC).format('YYYY-MM-DD'),
|
|
|
|
openRate: d.openRate
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
const paddedResults = [];
|
|
|
|
if (data.length < limit) {
|
|
|
|
const pad = limit - data.length;
|
|
|
|
const lastSubmittedAt = data.length > 0 ? data[results.length - 1].submittedAtUTC : moment();
|
|
|
|
for (let i = 0; i < pad; i++) {
|
|
|
|
paddedResults.push({
|
|
|
|
subject: '',
|
|
|
|
submittedAt: moment(lastSubmittedAt).subtract(i + 1, 'days').format('YYYY-MM-DD'),
|
|
|
|
openRate: 0
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
stats = stats.concat(paddedResults);
|
|
|
|
stats.reverse();
|
|
|
|
this.emailOpenRateStats = stats;
|
2022-03-23 18:38:16 +03:00
|
|
|
}
|
|
|
|
|
2022-03-23 13:53:32 +03:00
|
|
|
/**
|
|
|
|
* For now this is only used when reloading all the graphs after changing the mocked data
|
|
|
|
* @todo: reload only data that we loaded earlier
|
|
|
|
*/
|
2022-03-24 14:04:18 +03:00
|
|
|
reloadAll() {
|
2022-03-23 13:53:32 +03:00
|
|
|
this.loadMembersCounts();
|
2022-03-24 14:04:18 +03:00
|
|
|
this.loadMrrStats();
|
|
|
|
this.loadMemberCountStats();
|
|
|
|
this.loadLastSeen();
|
2022-03-23 18:38:16 +03:00
|
|
|
this.loadPaidMembersByCadence();
|
|
|
|
this.loadPaidMembersByTier();
|
|
|
|
|
|
|
|
this.loadNewsletterSubscribers();
|
|
|
|
this.loadEmailsSent();
|
|
|
|
this.loadEmailOpenRateStats();
|
2022-03-23 13:53:32 +03:00
|
|
|
}
|
2022-03-23 11:51:53 +03:00
|
|
|
|
2022-03-23 13:53:32 +03:00
|
|
|
/**
|
|
|
|
* Fill data to match a given amount of days
|
2022-03-23 18:38:16 +03:00
|
|
|
* @param {MemberCountStat[]|MrrStat[]} data
|
|
|
|
* @param {MemberCountStat|MrrStat} defaultData
|
|
|
|
* @param {number} days Amount of days to fill the graph with
|
2022-03-23 13:53:32 +03:00
|
|
|
*/
|
|
|
|
fillMissingDates(data, defaultData, days) {
|
|
|
|
let currentRangeDate = moment().subtract(days, 'days');
|
|
|
|
|
|
|
|
let endDate = moment().add(1, 'hour');
|
|
|
|
const output = [];
|
|
|
|
const firstDateInRangeIndex = data.findIndex((val) => {
|
|
|
|
return moment(val.date).isAfter(currentRangeDate);
|
|
|
|
});
|
|
|
|
let initialDateInRangeVal = firstDateInRangeIndex > 0 ? data[firstDateInRangeIndex - 1] : null;
|
|
|
|
if (firstDateInRangeIndex === 0 && !initialDateInRangeVal) {
|
|
|
|
initialDateInRangeVal = data[firstDateInRangeIndex];
|
|
|
|
}
|
|
|
|
if (data.length > 0 && !initialDateInRangeVal && firstDateInRangeIndex !== 0) {
|
|
|
|
initialDateInRangeVal = data[data.length - 1];
|
|
|
|
}
|
2022-03-23 11:51:53 +03:00
|
|
|
|
2022-03-23 13:53:32 +03:00
|
|
|
let lastVal = initialDateInRangeVal ? initialDateInRangeVal : defaultData;
|
2022-03-23 11:51:53 +03:00
|
|
|
|
2022-03-23 13:53:32 +03:00
|
|
|
while (currentRangeDate.isBefore(endDate)) {
|
|
|
|
let dateStr = currentRangeDate.format('YYYY-MM-DD');
|
|
|
|
const dataOnDate = data.find(d => d.date === dateStr);
|
2022-03-24 11:46:16 +03:00
|
|
|
lastVal = dataOnDate ? dataOnDate : {...lastVal, date: dateStr};
|
2022-03-23 13:53:32 +03:00
|
|
|
output.push(lastVal);
|
|
|
|
currentRangeDate = currentRangeDate.add(1, 'day');
|
|
|
|
}
|
|
|
|
return output;
|
|
|
|
}
|
2022-03-23 11:51:53 +03:00
|
|
|
}
|