2022-05-30 13:51:58 +08:00
|
|
|
import { dateRegex } from '@logto/shared';
|
2022-05-26 11:24:52 +08:00
|
|
|
import dayjs, { Dayjs } from 'dayjs';
|
2022-05-30 13:51:58 +08:00
|
|
|
import { object, string } from 'zod';
|
2022-05-26 11:24:52 +08:00
|
|
|
|
2022-05-30 13:51:58 +08:00
|
|
|
import koaGuard from '@/middleware/koa-guard';
|
|
|
|
import {
|
|
|
|
countActiveUsersByTimeInterval,
|
|
|
|
getDailyActiveUserCountsByTimeInterval,
|
|
|
|
getDailyNewUserCountsByTimeInterval,
|
|
|
|
} from '@/queries/log';
|
2022-05-25 16:54:21 +08:00
|
|
|
import { countUsers } from '@/queries/user';
|
|
|
|
|
|
|
|
import { AuthedRouter } from './types';
|
|
|
|
|
2022-05-26 11:24:52 +08:00
|
|
|
const getDateString = (day: Dayjs) => day.format('YYYY-MM-DD');
|
|
|
|
|
2022-06-13 16:18:16 +08:00
|
|
|
const indices = (length: number) => [...Array.from({ length }).keys()];
|
2022-05-30 13:51:58 +08:00
|
|
|
|
2022-06-13 16:18:16 +08:00
|
|
|
const lastTimestampOfDay = (day: Dayjs) => day.endOf('day').valueOf();
|
2022-05-26 11:24:52 +08:00
|
|
|
|
2022-05-25 16:54:21 +08:00
|
|
|
export default function dashboardRoutes<T extends AuthedRouter>(router: T) {
|
|
|
|
router.get('/dashboard/users/total', async (ctx, next) => {
|
|
|
|
const { count: totalUserCount } = await countUsers();
|
|
|
|
ctx.body = { totalUserCount };
|
|
|
|
|
|
|
|
return next();
|
|
|
|
});
|
2022-05-26 11:24:52 +08:00
|
|
|
|
|
|
|
router.get('/dashboard/users/new', async (ctx, next) => {
|
|
|
|
const today = dayjs();
|
|
|
|
const dailyNewUserCounts = await getDailyNewUserCountsByTimeInterval(
|
2022-05-30 13:51:58 +08:00
|
|
|
// (14 days ago 23:59:59.999, today 23:59:59.999]
|
|
|
|
lastTimestampOfDay(today.subtract(14, 'day')),
|
|
|
|
lastTimestampOfDay(today)
|
2022-05-26 11:24:52 +08:00
|
|
|
);
|
|
|
|
|
|
|
|
const last14DaysNewUserCounts = new Map(
|
|
|
|
dailyNewUserCounts.map(({ date, count }) => [date, count])
|
|
|
|
);
|
|
|
|
|
|
|
|
const todayNewUserCount = last14DaysNewUserCounts.get(getDateString(today)) ?? 0;
|
|
|
|
const yesterday = today.subtract(1, 'day');
|
|
|
|
const yesterdayNewUserCount = last14DaysNewUserCounts.get(getDateString(yesterday)) ?? 0;
|
|
|
|
const todayDelta = todayNewUserCount - yesterdayNewUserCount;
|
|
|
|
|
2022-06-13 16:18:16 +08:00
|
|
|
const last7DaysNewUserCount = indices(7)
|
2022-05-26 11:24:52 +08:00
|
|
|
.map((index) => getDateString(today.subtract(index, 'day')))
|
|
|
|
.reduce((sum, date) => sum + (last14DaysNewUserCounts.get(date) ?? 0), 0);
|
2022-06-13 16:18:16 +08:00
|
|
|
const newUserCountFrom13DaysAgoTo7DaysAgo = indices(7)
|
2022-05-26 11:24:52 +08:00
|
|
|
.map((index) => getDateString(today.subtract(7 + index, 'day')))
|
|
|
|
.reduce((sum, date) => sum + (last14DaysNewUserCounts.get(date) ?? 0), 0);
|
|
|
|
const last7DaysDelta = last7DaysNewUserCount - newUserCountFrom13DaysAgoTo7DaysAgo;
|
|
|
|
|
|
|
|
ctx.body = {
|
|
|
|
today: {
|
|
|
|
count: todayNewUserCount,
|
|
|
|
delta: todayDelta,
|
|
|
|
},
|
|
|
|
last7Days: {
|
|
|
|
count: last7DaysNewUserCount,
|
|
|
|
delta: last7DaysDelta,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
return next();
|
|
|
|
});
|
2022-05-30 13:51:58 +08:00
|
|
|
|
|
|
|
router.get(
|
|
|
|
'/dashboard/users/active',
|
|
|
|
koaGuard({
|
|
|
|
query: object({ date: string().regex(dateRegex).optional() }),
|
|
|
|
}),
|
|
|
|
async (ctx, next) => {
|
|
|
|
const {
|
|
|
|
query: { date },
|
|
|
|
} = ctx.guard;
|
|
|
|
|
|
|
|
const targetDay = date ? dayjs(date) : dayjs(); // Defaults to today
|
|
|
|
const [
|
|
|
|
// DAU: Daily Active User
|
|
|
|
last30DauCounts,
|
|
|
|
// WAU: Weekly Active User
|
|
|
|
{ count: previousWAU },
|
|
|
|
{ count: wau },
|
|
|
|
// MAU: Monthly Active User
|
|
|
|
{ count: previousMAU },
|
|
|
|
{ count: mau },
|
|
|
|
] = await Promise.all([
|
|
|
|
getDailyActiveUserCountsByTimeInterval(
|
|
|
|
// (30 days ago 23:59:59.999, target day 23:59:59.999]
|
|
|
|
lastTimestampOfDay(targetDay.subtract(30, 'day')),
|
|
|
|
lastTimestampOfDay(targetDay)
|
|
|
|
),
|
|
|
|
countActiveUsersByTimeInterval(
|
|
|
|
// (14 days ago 23:59:59.999, 7 days ago 23:59:59.999]
|
|
|
|
lastTimestampOfDay(targetDay.subtract(14, 'day')),
|
|
|
|
lastTimestampOfDay(targetDay.subtract(7, 'day'))
|
|
|
|
),
|
|
|
|
countActiveUsersByTimeInterval(
|
|
|
|
// (7 days ago 23:59:59.999, target day 23:59:59.999]
|
|
|
|
lastTimestampOfDay(targetDay.subtract(7, 'day')),
|
|
|
|
lastTimestampOfDay(targetDay)
|
|
|
|
),
|
|
|
|
countActiveUsersByTimeInterval(
|
|
|
|
// (60 days ago 23:59:59.999, 30 days ago 23:59:59.999]
|
|
|
|
lastTimestampOfDay(targetDay.subtract(60, 'day')),
|
|
|
|
lastTimestampOfDay(targetDay.subtract(30, 'day'))
|
|
|
|
),
|
|
|
|
countActiveUsersByTimeInterval(
|
|
|
|
// (30 days ago 23:59:59.999, target day 23:59:59.999]
|
|
|
|
lastTimestampOfDay(targetDay.subtract(30, 'day')),
|
|
|
|
lastTimestampOfDay(targetDay)
|
|
|
|
),
|
|
|
|
]);
|
|
|
|
|
|
|
|
const previousDate = getDateString(targetDay.subtract(1, 'day'));
|
|
|
|
const targetDate = getDateString(targetDay);
|
|
|
|
|
|
|
|
const previousDAU = last30DauCounts.find(({ date }) => date === previousDate)?.count ?? 0;
|
|
|
|
const dau = last30DauCounts.find(({ date }) => date === targetDate)?.count ?? 0;
|
|
|
|
|
2022-06-13 16:18:16 +08:00
|
|
|
const dauCurve = indices(30).map((index) => {
|
|
|
|
const dateString = getDateString(targetDay.subtract(29 - index, 'day'));
|
|
|
|
const count = last30DauCounts.find(({ date }) => date === dateString)?.count ?? 0;
|
|
|
|
|
|
|
|
return { date: dateString, count };
|
|
|
|
});
|
|
|
|
|
2022-05-30 13:51:58 +08:00
|
|
|
ctx.body = {
|
2022-06-13 16:18:16 +08:00
|
|
|
dauCurve,
|
2022-05-30 13:51:58 +08:00
|
|
|
dau: {
|
|
|
|
count: dau,
|
|
|
|
delta: dau - previousDAU,
|
|
|
|
},
|
|
|
|
wau: {
|
|
|
|
count: wau,
|
|
|
|
delta: wau - previousWAU,
|
|
|
|
},
|
|
|
|
mau: {
|
|
|
|
count: mau,
|
|
|
|
delta: mau - previousMAU,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
);
|
2022-05-25 16:54:21 +08:00
|
|
|
}
|