Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Added feature to filter stats at a glance by user type #122

Closed
wants to merge 18 commits into from
Closed
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
18 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 0 additions & 3 deletions components/AdminStatsCard.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,3 @@
import { SvgIconTypeMap } from '@mui/material';
import { OverridableComponent } from '@mui/material/OverridableComponent';

interface AdminStatsCardProps {
title: string;
value: number;
Expand Down
27 changes: 22 additions & 5 deletions components/StatsBarChart.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,16 +7,33 @@ import {
Tooltip,
ValueAxis,
} from '@devexpress/dx-react-chart-material-ui';
import { useState } from 'react';

interface StatsBarChartProps {
name: string;
items: Array<{
itemName: string;
itemCount: number;
}>;
fieldName: string;
statsData: Record<string, GeneralStats>;
rolesDict: Record<string, boolean>;
}

export default function StatsBarChart({ name, items }: StatsBarChartProps) {
export default function StatsBarChart({
name,
fieldName,
statsData,
rolesDict,
}: StatsBarChartProps) {
const items = Object.entries(
Object.entries(statsData)
.filter(([k, _]) => rolesDict[k])
.map(([k, v]) => v[fieldName])
.reduce((acc: Record<string, number>, curr: Record<string, number>) => {
for (let key of Object.keys(curr)) {
if (!acc.hasOwnProperty(key)) acc[key] = curr[key];
else acc[key] += curr[key];
}
return acc;
}, {}) as Record<string, number>,
).map(([k, v]) => ({ itemName: k, itemCount: v }));
const coordinates = [];
/**
*
Expand Down
29 changes: 23 additions & 6 deletions components/StatsPieChart.tsx
Original file line number Diff line number Diff line change
@@ -1,16 +1,33 @@
import { Animation, EventTracker, Palette } from '@devexpress/dx-react-chart';
import { Chart, Legend, PieSeries, Title, Tooltip } from '@devexpress/dx-react-chart-material-ui';
import React from 'react';
import React, { useState } from 'react';

interface StatsPieChartProps {
name: string;
items: Array<{
itemName: string;
itemCount: number;
}>;
fieldName: string;
statsData: Record<string, GeneralStats>;
rolesDict: Record<string, boolean>;
}

export default function StatsPieChart({ name, items }: StatsPieChartProps) {
export default function StatsPieChart({
name,
fieldName,
statsData,
rolesDict,
}: StatsPieChartProps) {
const items = Object.entries(
Object.entries(statsData)
.filter(([k, _]) => rolesDict[k])
.map(([k, v]) => v[fieldName])
.reduce((acc: Record<string, number>, curr: Record<string, number>) => {
for (let key of Object.keys(curr)) {
if (!acc.hasOwnProperty(key)) acc[key] = curr[key];
else acc[key] += curr[key];
}
return acc;
}, {}) as Record<string, number>,
).map(([k, v]) => ({ itemName: k, itemCount: v }));

return (
<div className="w-full flex-grow border-2 my-2 rounded-2xl p-6">
<Chart data={items}>
Expand Down
4 changes: 1 addition & 3 deletions lib/types.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -236,10 +236,8 @@ type Sponsor = {
};

type GeneralStats = {
superAdminCount: number;
count: number;
checkedInCount: number;
hackerCount: number;
adminCount: number;
scans: Record<string, number>;
companies: Record<string, number>;
dietary: Record<string, number>;
Expand Down
71 changes: 56 additions & 15 deletions pages/admin/stats.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ import SupervisorAccountIcon from '@mui/icons-material/SupervisorAccount';
import EngineeringIcon from '@mui/icons-material/Engineering';
import StatsPieChart from '../../components/StatsPieChart';
import { fieldToName } from '../../lib/stats/field';
import FilterComponent from '../../components/FilterComponent';

function isAuthorized(user): boolean {
if (!user || !user.permissions) return false;
Expand All @@ -22,11 +23,16 @@ function isAuthorized(user): boolean {
export default function AdminStatsPage() {
const [loading, setLoading] = useState(true);
const { user, isSignedIn } = useAuthContext();
const [statsData, setStatsData] = useState<GeneralStats>();
const [statsData, setStatsData] = useState<Record<string, GeneralStats>>();
const [roles, setRoles] = useState<Record<string, boolean>>({
hacker: true,
admin: true,
super_admin: true,
});

useEffect(() => {
async function getData() {
const { data } = await RequestHelper.get<GeneralStats>('/api/stats', {
const { data } = await RequestHelper.get<Record<string, GeneralStats>>('/api/stats', {
headers: {
Authorization: user.token,
},
Expand All @@ -37,6 +43,12 @@ export default function AdminStatsPage() {
getData();
}, []);

const updateFilter = (name: string) => {
const newFilter = { ...roles };
newFilter[name] = !newFilter[name];
setRoles(newFilter);
};

if (!isSignedIn || !isAuthorized(user)) {
return <div className="text-2xl font-black text-center">Unauthorized</div>;
}
Expand All @@ -53,46 +65,75 @@ export default function AdminStatsPage() {
</Head>
<AdminHeader />
<div className="w-full xl:w-3/5 mx-auto p-6 flex flex-col gap-y-6">
<div className="border-2 rounded-xl p-3">
<h1 className="text-center text-xl font-bold">Filter Stats by:</h1>

<FilterComponent
checked={roles['hacker']}
onCheck={() => {
updateFilter('hacker');
}}
title="Hackers"
/>
<FilterComponent
checked={roles['admin']}
onCheck={() => {
updateFilter('admin');
}}
title="Admin"
/>
<FilterComponent
checked={roles['super_admin']}
onCheck={() => {
updateFilter('super_admin');
}}
title="Super Admin"
/>
</div>
<div className="flex-col gap-y-3 w-full md:flex-row flex justify-around gap-x-2">
<AdminStatsCard icon={<CheckIcon />} title="Check-Ins" value={statsData.checkedInCount} />
<AdminStatsCard
icon={<CheckIcon />}
title="Check-Ins"
value={Object.entries(statsData)
.filter(([k, v]) => roles[k])
.reduce((acc, [k, v]) => acc + v.checkedInCount, 0)}
/>
<AdminStatsCard
icon={<AccountCircleIcon />}
title="Hackers"
value={statsData.hackerCount}
value={statsData['hacker'].count}
/>
<AdminStatsCard
icon={<SupervisorAccountIcon />}
title="Admins"
value={statsData.adminCount}
value={statsData['admin'].count}
/>
<AdminStatsCard
icon={<EngineeringIcon />}
title="Super Admin"
value={statsData.superAdminCount}
value={statsData['super_admin'].count}
/>
</div>
{Object.entries(statsData)
{Object.entries(statsData['hacker'])
.filter(([k, v]) => typeof v === 'object')
.map(([key, value]) => {
if (Object.keys(value).length <= 6)
return (
<StatsPieChart
key={key}
name={fieldToName[key]}
items={Object.entries(statsData[key] as Record<any, any>).map(([k, v]) => ({
itemName: k,
itemCount: v,
}))}
fieldName={key}
statsData={statsData}
rolesDict={roles}
/>
);
return (
<StatsBarChart
key={key}
name={fieldToName[key]}
items={Object.entries(statsData[key] as Record<any, any>).map(([k, v]) => ({
itemName: k,
itemCount: v,
}))}
fieldName={key}
statsData={statsData}
rolesDict={roles}
/>
);
})}
Expand Down
71 changes: 30 additions & 41 deletions pages/api/stats.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,65 +22,54 @@ async function getCheckInEventName() {
async function getStatsData() {
const checkInEventName = await getCheckInEventName();
// const swagData: Record<string, number> = {};
const generalStats: GeneralStats = {
superAdminCount: 0,
checkedInCount: 0,
hackerCount: 0,
adminCount: 0,
scans: {},
age: {},
companies: {},
dietary: {},
ethnicity: {},
gender: {},
hackathonExperience: {},
heardFrom: {},
race: {},
size: {},
softwareExperience: {},
studyLevel: {},
university: {},
};
const generalStats: Record<string, GeneralStats> = {};
for (const role of ['hacker', 'admin', 'super_admin']) {
generalStats[role] = {
count: 0,
checkedInCount: 0,
scans: {},
age: {},
companies: {},
dietary: {},
ethnicity: {},
gender: {},
hackathonExperience: {},
heardFrom: {},
race: {},
size: {},
softwareExperience: {},
studyLevel: {},
university: {},
};
}

const snapshot = await db.collection(USERS_COLLECTION).get();
snapshot.forEach((doc) => {
const userData = doc.data();
const userPermission = userData.user.permissions[0];

for (let arrayField of arrayFields) {
if (!userData[arrayField]) continue;
userData[arrayField].forEach((data: string) => {
if (arrayField === 'scans' && data === checkInEventName) generalStats.checkedInCount++;
if (arrayField === 'scans' && data === checkInEventName)
generalStats[userPermission].checkedInCount++;
else {
if (!generalStats[arrayField].hasOwnProperty(data)) generalStats[arrayField][data] = 0;
generalStats[arrayField][data]++;
if (!generalStats[userPermission][arrayField].hasOwnProperty(data))
generalStats[userPermission][arrayField][data] = 0;
generalStats[userPermission][arrayField][data]++;
}
});
}

for (let singleField of singleFields) {
if (!userData[singleField] || userData[singleField] === '') continue;
if (!generalStats[singleField].hasOwnProperty(userData[singleField])) {
generalStats[singleField][userData[singleField]] = 0;
if (!generalStats[userPermission][singleField].hasOwnProperty(userData[singleField])) {
generalStats[userPermission][singleField][userData[singleField]] = 0;
}
generalStats[singleField][userData[singleField]]++;
generalStats[userPermission][singleField][userData[singleField]]++;
}

const userPermission = userData.user.permissions[0];

switch (userPermission) {
case 'super_admin': {
generalStats.superAdminCount++;
break;
}
case 'admin': {
generalStats.adminCount++;
break;
}
case 'hacker': {
generalStats.hackerCount++;
break;
}
}
generalStats[userPermission].count++;
});

return generalStats;
Expand Down