1
0
mirror of https://github.com/matt-fidd/stratos.git synced 2026-01-01 20:19:30 +00:00
Files
stratos/routes/singleClass.js
2022-04-22 04:06:50 +00:00

297 lines
6.8 KiB
JavaScript

'use strict';
const crypto = require('crypto');
const express = require('express');
const router = express.Router();
const validator = require('../lib/validator');
const User = require('../lib/User');
router.get('/:id', async (req, res) => {
const c = req.class;
const linkRoot = `/admin/class/${c.id}`;
const upcomingTests = await c.getTests({ range: 'after' });
const recentTests = await c.getTests({ range: 'before' });
const testCount = recentTests.length + upcomingTests.length;
const trs = (await Promise.all(recentTests.map(async t => {
if (req.session.userType === 'account')
return t.getAveragePercentage();
const tr = (await t.getTestResults())
.filter(tr => tr.student.id === req.session.userId);
return tr?.[0]?.percentage;
}))).filter(tr => typeof tr !== 'undefined');
const averagePercentage = trs
.reduce((a, b) => a + b, 0) / trs.length;
return res.render('class', {
...req.hbsContext,
title: `Stratos - ${c.name}`,
current: 'Classes',
className: c.name,
teachers: c.teachers,
members: c.students,
recentTests: recentTests,
upcomingTests: upcomingTests,
contactLink: `${linkRoot}/contact`,
testsLink: `${linkRoot}/tests`,
reportsLink: `${linkRoot}/reports`,
deleteLink: `${linkRoot}/delete`,
membersLink: `${linkRoot}/members`,
teachersLink: `${linkRoot}/teachers`,
stats: [
{
value: testCount,
text: 'Test' + (testCount !== 1 ? 's' : '')
},
{
value: recentTests.length,
text: 'Completed test' +
(recentTests.length !== 1 ? 's' : '')
},
{
value: upcomingTests.length,
text: 'Upcoming test' +
(upcomingTests.length !== 1 ? 's' : '')
},
{
value: `${Math.round(averagePercentage)}%`,
text: (req.session.userType !== 'account' ?
'Your' : 'Class') +
' average percentage'
}
]
});
});
router.post('/:id/delete', async (req, res) => {
if (req.session.userType !== 'account')
return res.redirect('/admin/classes');
const c = req.class;
await c.delete();
res.redirect('/admin/classes');
});
router.get('/:id/:memberType(members|teachers)', (req, res) => {
const c = req.class;
const linkRoot = `/admin/class/${c.id}`;
let users, addLink, addContent, pageTitle;
if (req.params.memberType === 'members') {
users = c.students;
addLink = `${linkRoot}/members/add`;
addContent = 'Add new students';
pageTitle = 'Students';
} else {
users = c.teachers;
addLink = `${linkRoot}/teachers/add`;
addContent = 'Add new teachers';
pageTitle = 'Teachers';
}
return res.render('classUsers', {
...req.hbsContext,
title: `Stratos - ${c.name}`,
current: 'Classes',
className: c.name,
users: users,
addLink: addLink,
addContent: addContent,
pageTitle: pageTitle
});
});
router.get('/:id/:userType(members|teachers)/add', (req, res) => {
if (req.session.userType !== 'account')
return res.redirect('/admin/classes');
const c = req.class;
const userType =
req.params.userType === 'teachers' ?
'teachers' :
'students';
const errors = [];
req.query.err && req.query.err.split(',').forEach(e => {
switch (e) {
case 'no_user':
errors.push({
msg: 'No user with that email ' +
'address can be found, they need to ' +
'create an account before you can ' +
'add them to a class'
});
break;
case 'dup_user':
errors.push({
msg: 'This user is already assigned ' +
'to the class'
});
break;
}
});
return res.render('addClassUser', {
...req.hbsContext,
title: `Stratos - ${c.name}`,
current: 'Classes',
className: c.name,
postLink: `/admin/class/${c.id}/${req.params.userType}/add`,
newType: userType.slice(0, -1),
pageTitle: `Add a new ${userType.slice(0, -1)}`,
errors: errors
});
});
router.post('/:id/:userType(members|teachers)/add', async (req, res) => {
if (req.session.userType !== 'account')
return res.redirect('/admin/classes');
const c = req.class;
const userType = req.params.userType;
const rejectURL = `/admin/class/${c.id}/${userType}/add`;
let fields;
try {
fields = validator.validate(req.body,
[
'email'
], {
email: 'email'
}
).fields;
} catch (e) {
console.error(e);
return res.redirect(rejectURL);
}
const u = await User.getUserByEmail(req.db, fields.get('email'));
if (!u) {
if (userType === 'teachers')
return res.redirect(`${rejectURL}/?err=no_user`);
return res.render('addClassUser2', {
...req.hbsContext,
title: `Stratos - ${c.name}`,
current: 'Classes',
className: c.name,
postLink: `/admin/class/${c.id}/members/add2`,
newType: 'student',
pageTitle: 'Add a new student',
email: fields.get('email')
});
}
try {
await c.addUser(u);
return res.redirect(`/admin/class/${c.id}/${userType}`);
} catch (e) {
return res.redirect(`${rejectURL}/?err=dup_user`);
}
});
router.post('/:id/members/add2', async (req, res) => {
if (req.session.userType !== 'account')
return res.redirect('/admin/classes');
const c = req.class;
const rejectURL = `/admin/class/${c.id}/students/add`;
let fields;
try {
fields = validator.validate(req.body,
[
'fname',
'lname',
'email'
], {
email: 'email'
}
).fields;
} catch (e) {
console.error(e);
return res.redirect(rejectURL);
}
const password = crypto.randomBytes(20).toString('base64').slice(0, 20);
const u = await User.createUser(
req.db,
'student',
fields.get('fname'),
fields.get('onames'),
fields.get('lname'),
fields.get('email'),
password
);
await c.addUser(u);
return res.redirect(`/admin/class/${c.id}/members`);
});
router.get('/:id/:userType(members|teachers)/:userId/remove',
async (req, res) => {
if (req.session.userType !== 'account')
return res.redirect('/admin/classes');
const c = req.class;
const u = await new User(req.db, req.params.userId);
const userType =
req.params.userType === 'teachers' ?
'teachers' :
'students';
const postLink =
`/admin/class/${c.id}/${req.params.userType}` +
`/${u.id}/remove`;
return res.render('removeClassUser', {
...req.hbsContext,
title: `Stratos - ${c.name}`,
current: 'Classes',
u: u,
postLink: postLink,
pageTitle: `Remove a ${userType.slice(0, -1)}`
});
}
);
router.post('/:id/:userType(members|teachers)/:userId/remove',
async (req, res) => {
if (req.session.userType !== 'account')
return res.redirect('/admin/classes');
const c = req.class;
const u = await new User(req.db, req.params.userId);
const userType = req.params.userType;
try {
await c.removeUser(u);
return res.redirect(`/admin/class/${c.id}/${userType}`);
} catch (e) {
console.error(e);
return res.render('error', {
...req.hbsContext,
title: 'Stratos - Error',
current: 'Classes',
msg: `Can not remove this user: ${e.message}`
});
}
}
);
module.exports = {
priority: 50,
root: '/admin/class',
router: router
};