Skip to content
Snippets Groups Projects
Select Git revision
  • d7409df1a14b8ddbe8442e3ebc327f63b94289d8
  • main default protected
  • jw_sonar
  • v6.0.0 protected
  • bedran_exercise-list
  • ask-user-to-delete-exercises-on-duplicates
  • update-dependencies
  • jw_sonar_backup
  • add_route_assignments
  • 6.0.0-dev
  • 5.0.1
  • 5.0.0
  • 4.1.0
  • 4.0.0
  • 3.5.3
  • 3.5.2
  • 3.5.1
  • 3.5.0
  • 3.4.2
  • 3.4.1
  • 3.4.0
  • 3.3.0
  • 3.2.0
  • 3.1.3
  • 3.1.2
  • 3.1.1
  • 3.1.0
  • 3.0.1
  • 3.0.0
29 results

ExerciseRoutes.ts

Blame
  • ExerciseRoutes.ts 11.88 KiB
    import { Express }                    from 'express-serve-static-core';
    import express                        from 'express';
    import * as ExpressValidator          from 'express-validator';
    import { StatusCodes }                from 'http-status-codes';
    import RoutesManager                  from '../express/RoutesManager';
    import ParamsValidatorMiddleware      from '../middlewares/ParamsValidatorMiddleware';
    import SecurityMiddleware             from '../middlewares/SecurityMiddleware';
    import GitlabUser                     from '../shared/types/Gitlab/GitlabUser';
    import GitlabManager                  from '../managers/GitlabManager';
    import Config                         from '../config/Config';
    import GitlabRepository               from '../shared/types/Gitlab/GitlabRepository';
    import { AxiosError, HttpStatusCode } from 'axios';
    import logger                         from '../shared/logging/WinstonLogger';
    import DojoValidators                 from '../helpers/DojoValidators';
    import { v4 as uuidv4 }               from 'uuid';
    import GitlabMember                   from '../shared/types/Gitlab/GitlabMember';
    import GitlabAccessLevel              from '../shared/types/Gitlab/GitlabAccessLevel';
    import { Prisma }                     from '@prisma/client';
    import { Assignment, Exercise }       from '../types/DatabaseTypes';
    import db                             from '../helpers/DatabaseHelper';
    import SecurityCheckType              from '../types/SecurityCheckType';
    import GitlabTreeFile                 from '../shared/types/Gitlab/GitlabTreeFile';
    import GitlabFile                     from '../shared/types/Gitlab/GitlabFile';
    import GitlabTreeFileType             from '../shared/types/Gitlab/GitlabTreeFileType';
    import JSON5                          from 'json5';
    import fs                             from 'fs';
    import path                           from 'path';
    import AssignmentFile                 from '../shared/types/Dojo/AssignmentFile';
    import ExerciseResultsFile            from '../shared/types/Dojo/ExerciseResultsFile';
    
    
    class ExerciseRoutes implements RoutesManager {
        private readonly exerciseValidator: ExpressValidator.Schema = {
            members: {
                trim: true, notEmpty: true, customSanitizer: DojoValidators.jsonSanitizer
            }
        };
    
        private readonly resultValidator: ExpressValidator.Schema = {
            exitCode        : {
                isInt: true, toInt: true
            }, commit       : {
                trim: true, notEmpty: true, customSanitizer: DojoValidators.jsonSanitizer
            }, results      : {
                trim: true, notEmpty: true, custom: DojoValidators.exerciseResultsValidator, customSanitizer: DojoValidators.jsonSanitizer
            }, files        : {
                trim: true, notEmpty: true, customSanitizer: DojoValidators.jsonSanitizer
            }, archiveBase64: {
                isBase64: true, notEmpty: true
            }
        };
    
        registerOnBackend(backend: Express) {
            backend.post('/assignments/:assignmentNameOrUrl/exercises', SecurityMiddleware.check(true, SecurityCheckType.ASSIGNMENT_IS_PUBLISHED), ParamsValidatorMiddleware.validate(this.exerciseValidator), this.createExercise.bind(this));
    
            backend.get('/exercises/:exerciseId/assignment', SecurityMiddleware.check(false, SecurityCheckType.EXERCISE_SECRET), this.getAssignment.bind(this));
    
            backend.post('/exercises/:exerciseId/results', SecurityMiddleware.check(false, SecurityCheckType.EXERCISE_SECRET), ParamsValidatorMiddleware.validate(this.resultValidator), this.createResult.bind(this));
        }
    
        private getExerciseName(assignment: Assignment, members: Array<GitlabUser>, suffix: number): string {
            return `DojoEx - ${ assignment.name } - ${ members.map(member => member.username).join(' + ') }${ suffix > 0 ? ` - ${ suffix }` : '' }`;
        }
    
        private getExercisePath(assignment: Assignment, exerciseId: string): string {
            return `dojo-ex_${ (assignment.gitlabLastInfo as unknown as GitlabRepository).path }_${ exerciseId }`;
        }
    
        private async createExercise(req: express.Request, res: express.Response) {
            const params: { members: Array<GitlabUser> } = req.body;
            params.members = [ await req.session.profile.gitlabProfile!.value, ...params.members ].removeObjectDuplicates(gitlabUser => gitlabUser.id);
            const assignment: Assignment = req.boundParams.assignment!;
    
    
            const exerciseId: string = uuidv4();
            const secret: string = uuidv4();
            let repository!: GitlabRepository;
    
            let suffix: number = 0;
            do {
                try {
                    repository = await GitlabManager.forkRepository((assignment.gitlabCreationInfo as unknown as GitlabRepository).id, this.getExerciseName(assignment, params.members, suffix), this.getExercisePath(req.boundParams.assignment!, exerciseId), Config.exercise.default.description.replace('{{ASSIGNMENT_NAME}}', assignment.name), Config.exercise.default.visibility, Config.gitlab.group.exercises);
    
                    await GitlabManager.protectBranch(repository.id, '*', false, GitlabAccessLevel.DEVELOPER, GitlabAccessLevel.DEVELOPER, GitlabAccessLevel.OWNER);
    
                    await GitlabManager.addRepositoryVariable(repository.id, 'DOJO_EXERCISE_ID', exerciseId, false, true);
                    await GitlabManager.addRepositoryVariable(repository.id, 'DOJO_SECRET', secret, false, true);
                    await GitlabManager.addRepositoryVariable(repository.id, 'DOJO_RESULTS_FOLDER', Config.exercise.pipelineResultsFolder, false, false);
    
                    await GitlabManager.addRepositoryBadge(repository.id, Config.gitlab.badges.pipeline.link, Config.gitlab.badges.pipeline.imageUrl, 'Pipeline Status');
    
                    break;
                } catch ( error ) {
                    if ( error instanceof AxiosError ) {
                        if ( error.response?.data.message.name && error.response.data.message.name == 'has already been taken' ) {
                            suffix++;
                        } else {
                            return res.status(error.response?.status ?? HttpStatusCode.InternalServerError).send();
                        }
                    } else {
                        return res.status(StatusCodes.INTERNAL_SERVER_ERROR).send();
                    }
                }
            } while ( suffix < Config.exercise.maxSameName );
    
            if ( suffix >= Config.exercise.maxSameName ) {
                return res.status(StatusCodes.INSUFFICIENT_SPACE_ON_RESOURCE).send();
            }
    
            try {
                await GitlabManager.createFile(repository.id, '.gitlab-ci.yml', fs.readFileSync(path.join(__dirname, '../../assets/exercise_gitlab_ci.yml'), 'base64'), 'Add .gitlab-ci.yml (DO NOT MODIFY THIS FILE)');
            } catch ( error ) {
                logger.error(error);
    
                if ( error instanceof AxiosError ) {
                    return res.status(error.response?.status ?? HttpStatusCode.InternalServerError).send();
                }
    
                return res.status(StatusCodes.INTERNAL_SERVER_ERROR).send();
            }
    
            try {
                await Promise.all([ ...new Set([ ...assignment.staff.map(user => user.gitlabId), ...params.members.map(member => member.id) ]) ].map(async (memberId: number): Promise<GitlabMember | false> => {
                    try {
                        return await GitlabManager.addRepositoryMember(repository.id, memberId, GitlabAccessLevel.DEVELOPER);
                    } catch ( e ) {
                        return false;
                    }
                }));
    
                const exercise: Exercise = await db.exercise.create({
                                                                        data: {
                                                                            id: exerciseId, assignmentName: assignment.name, name: repository.name, secret: secret, gitlabId: repository.id, gitlabLink: repository.web_url, gitlabCreationInfo: repository as unknown as Prisma.JsonObject, gitlabLastInfo: repository as unknown as Prisma.JsonObject, gitlabLastInfoDate: new Date(), members: {
                                                                                connectOrCreate: [ ...params.members.map(gitlabUser => {
                                                                                    return {
                                                                                        create  : {
                                                                                            gitlabId: gitlabUser.id, firstname: gitlabUser.name
                                                                                        }, where: {
                                                                                            gitlabId: gitlabUser.id
                                                                                        }
                                                                                    };
                                                                                }) ]
                                                                            }
                                                                        }
                                                                    }) as unknown as Exercise;
    
                return req.session.sendResponse(res, StatusCodes.OK, exercise);
            } catch ( error ) {
                logger.error(error);
    
                if ( error instanceof AxiosError ) {
                    return res.status(error.response?.status ?? HttpStatusCode.InternalServerError).send();
                }
    
                return res.status(StatusCodes.INTERNAL_SERVER_ERROR).send();
            }
        }
    
        private async getAssignment(req: express.Request, res: express.Response) {
            const repoTree: Array<GitlabTreeFile> = await GitlabManager.getRepositoryTree(req.boundParams.exercise!.assignment.gitlabId);
    
            let assignmentHjsonFile!: GitlabFile;
            let immutableFiles: Array<GitlabFile> = await Promise.all(Config.assignment.baseFiles.map(async (baseFile: string) => {
                let file = await GitlabManager.getFile(req.boundParams.exercise!.assignment.gitlabId, baseFile);
    
                if ( baseFile === Config.assignment.filename ) {
                    assignmentHjsonFile = file;
                }
    
                return file;
            }));
    
            const dojoAssignmentFile: AssignmentFile = JSON5.parse(atob(assignmentHjsonFile.content)) as AssignmentFile;
    
            const immutablePaths = dojoAssignmentFile.immutable.map(fileDescriptor => fileDescriptor.path);
    
            await Promise.all(repoTree.map(async gitlabTreeFile => {
                if ( gitlabTreeFile.type == GitlabTreeFileType.BLOB ) {
                    for ( const immutablePath of immutablePaths ) {
                        if ( gitlabTreeFile.path.startsWith(immutablePath) ) {
                            immutableFiles.push(await GitlabManager.getFile(req.boundParams.exercise!.assignment.gitlabId, gitlabTreeFile.path));
                            break;
                        }
                    }
                }
            }));
    
            return req.session.sendResponse(res, StatusCodes.OK, {
                assignment: (req.boundParams.exercise as Exercise).assignment, assignmentFile: dojoAssignmentFile, immutable: immutableFiles
            });
        }
    
        private async createResult(req: express.Request, res: express.Response) {
            const params: { exitCode: number, commit: any, results: ExerciseResultsFile, files: any, archiveBase64: string } = req.body;
            const exercise: Exercise = req.boundParams.exercise!;
    
            const result = await db.result.create({
                                                      data: {
                                                          exerciseId: exercise.id, exitCode: params.exitCode, success: params.results.success, commit: params.commit, results: params.results as unknown as Prisma.JsonObject, files: params.files
                                                      }
                                                  });
    
            fs.writeFileSync(path.join(Config.getResultsFolder(exercise), `${ result.dateTime.toISOString().replace(/:/g, '_') }.tar.gz`), params.archiveBase64, 'base64');
    
            req.session.sendResponse(res, StatusCodes.OK);
        }
    }
    
    
    export default new ExerciseRoutes();