Skip to content
This repository was archived by the owner on Jun 12, 2025. It is now read-only.
Merged

Dev #20

Show file tree
Hide file tree
Changes from all commits
Commits
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
4 changes: 3 additions & 1 deletion src/classes/MainServer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import { authLoginRoute } from '../routes/auth/login';
import { authRegisterRoute } from '../routes/auth/register';
import { classroomsRoute } from '../routes/classroom';
import { classroomActivitiesRoute } from '../routes/classroom/activities';
import { classroomActivitySubmissionsRoute } from '../routes/classroom/activities/submissions';
import { userRoute } from '../routes/user';

export class MainServer {
Expand All @@ -25,7 +26,8 @@ export class MainServer {
.use(authRegisterRoute)
.use(userRoute)
.use(classroomsRoute)
.use(classroomActivitiesRoute);
.use(classroomActivitiesRoute)
.use(classroomActivitySubmissionsRoute);

if (process.env['ENABLE_SWAGGER'] === 'true') {
console.log('Swagger enabled.');
Expand Down
62 changes: 62 additions & 0 deletions src/models/SubmissionModel.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
import { Schema, Types, model } from 'mongoose';

export interface ISubmission {
id: string;
activity: Types.ObjectId;
user: Types.ObjectId;
files: Array<{
name: string;
url: string;
}>;
submittedAt: string;
comment?: string;
}

const fileSchema = new Schema(
{
name: {
type: String,
required: true
},
url: {
type: String,
required: true
}
},
{ _id: false }
);

const submissionSchema = new Schema<ISubmission>(
{
activity: {
type: Schema.Types.ObjectId,
ref: 'activity',
required: true,
index: true
},
user: {
type: Schema.Types.ObjectId,
ref: 'user',
required: true,
index: true
},
files: {
type: [fileSchema],
required: true,
default: []
},
submittedAt: {
type: String,
required: true
},
comment: {
type: String,
required: false
}
},
{
timestamps: true
}
);

export const SubmissionModel = model<ISubmission>('Submission', submissionSchema);
206 changes: 206 additions & 0 deletions src/routes/classroom/activities/submissions/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,206 @@
import { Elysia, t } from 'elysia';
import { minio } from '../../../..';
import { ActivityModel } from '../../../../models/ActivityModel';
import { SubmissionModel } from '../../../../models/SubmissionModel';
import { UserModel } from '../../../../models/UserModel';
import { headersPlugin } from '../../../../plugins/headers';

export const classroomActivitySubmissionsRoute = new Elysia({
name: 'routes:classroomActivitySubmissionsRoute',
prefix: '/classrooms/:classroomId/activities/:activityId/submissions'
})
.use(headersPlugin)
.get(
'/',
async ({ params: { classroomId, activityId }, status, token }) => {
const user = await UserModel.findOne({ token }, { __v: 0 })
.lean()
.exec()
.catch(() => null);

if (!user) return status(400, 'Bad Request');
if (!user.classroomIds?.some((cId) => cId.toString() === classroomId)) return status(403, 'Forbidden');

const activity = await ActivityModel.findOne(
{
_id: activityId,
classroom: classroomId
},
{ __v: 0 }
)
.lean()
.exec()
.catch(() => null);

if (!activity) return status(404, 'Activity not found');

const submission = await SubmissionModel.findOne(
{
activity: activityId,
user: user._id
},
{ __v: 0 }
)
.lean()
.exec()
.catch(() => null);

if (!submission) return status(400, 'Bad Request');

return ({
...submission,
id: submission._id.toString(),
_id: undefined
})
},
{
params: t.Object({
classroomId: t.String({ minLength: 1 }),
activityId: t.String({ minLength: 1 })
})
}
)
.get(
'/all',
async ({ params: { classroomId, activityId }, status, token }) => {
const user = await UserModel.findOne({ token }, { __v: 0 })
.lean()
.exec()
.catch(() => null);

if (!user) return status(400, 'Bad Request');

const activity = await ActivityModel.findOne(
{
_id: activityId,
classroom: classroomId
},
{ __v: 0 }
)
.lean()
.exec()
.catch(() => null);

if (!activity) return status(404, 'Activity not found');

if (user._id.toString() !== activity.owner.toString()) return status(403, 'Forbidden');

const submissions = await SubmissionModel.find(
{
activity: activityId,
},
{ __v: 0 }
)
.populate('user', { __v: 0, token: 0 })
.lean()
.exec()
.catch(() => null);

if (!submissions) return status(400, 'Bad Request');

return submissions.map((submission) => ({
...submission,
id: submission._id.toString(),
_id: undefined,
user: {
...submission.user,
id: submission.user._id.toString(),
_id: undefined
}
}));
},
{
params: t.Object({
classroomId: t.String({ minLength: 1 }),
activityId: t.String({ minLength: 1 })
})
}
)
.post(
'/',
async ({ body, params: { classroomId, activityId }, status, token }) => {
if (!body) return status(400, 'Bad Request');

const user = await UserModel.findOne({ token }, { __v: 0 })
.lean()
.exec()
.catch(() => null);

if (!user) return status(400, 'Bad Request');
if (!user.classroomIds?.some((cId) => cId.toString() === classroomId)) return status(403, 'Forbidden');

const activity = await ActivityModel.findOne(
{
_id: activityId,
classroom: classroomId
},
{ __v: 0 }
)
.lean()
.exec()
.catch(() => null);

if (!activity) return status(404, 'Activity not found');

const { filename, comment } = body;

const url = await minio.presignedPutObject(
'submissions',
`${user._id.toString()}/${activityId}/${filename}`,
1 * 60 * 60
);

if (!url) return status(400, 'Error');

let submission = await SubmissionModel.findOne(
{
activity: activityId,
user: user._id
},
{ __v: 0 }
)
.exec()
.catch(() => null);

if (!submission) {
submission = new SubmissionModel({
activity: activityId,
user: user._id,
files: [],
comment: comment,
submittedAt: new Date().toISOString()
});
}

submission.files.push({
name: filename,
url: `${user._id.toString()}/${activityId}/${filename}`
});

if (comment !== undefined) {
submission.comment = comment;
}

const save = await submission.save().catch(() => null);

if (!save) return status(400, 'Bad Request');

return { url };
},
{
parse: 'json',
params: t.Object({
classroomId: t.String({ minLength: 1 }),
activityId: t.String({ minLength: 1 })
}),
body: t.Object(
{
filename: t.String({ minLength: 1 }),
comment: t.Optional(t.String({ minLength: 1 }))
},
{
additionalProperties: false
}
)
}
);