Merge branch 'fix/mark-as-completed-bij-leerobjecten-werkt-niet-altijd-#253' into fix/progress-bar
Merge fix/mark-as-completed-bij-leerobjecten-werkt-niet-altijd-#253 into fix/progress-bar
This commit is contained in:
commit
779ebb28ee
9 changed files with 415 additions and 10 deletions
|
@ -62,6 +62,11 @@ export async function getAllSubmissionsHandler(req: Request, res: Response): Pro
|
||||||
|
|
||||||
// TODO: gerald moet nog dingen toevoegen aan de databank voor dat dit gefinaliseerd kan worden
|
// TODO: gerald moet nog dingen toevoegen aan de databank voor dat dit gefinaliseerd kan worden
|
||||||
export async function createSubmissionHandler(req: Request, res: Response): Promise<void> {
|
export async function createSubmissionHandler(req: Request, res: Response): Promise<void> {
|
||||||
|
const submitter = req.body.submitter;
|
||||||
|
const usernameSubmitter = req.body.submitter.username;
|
||||||
|
const group = req.body.group;
|
||||||
|
requireFields({ group, submitter, usernameSubmitter });
|
||||||
|
|
||||||
const submissionDTO = req.body as SubmissionDTO;
|
const submissionDTO = req.body as SubmissionDTO;
|
||||||
const submission = await createSubmission(submissionDTO);
|
const submission = await createSubmission(submissionDTO);
|
||||||
|
|
||||||
|
|
|
@ -4,7 +4,7 @@ import { getLearningPathRepository } from '../../data/repositories.js';
|
||||||
import learningObjectService from '../learning-objects/learning-object-service.js';
|
import learningObjectService from '../learning-objects/learning-object-service.js';
|
||||||
import { LearningPathNode } from '../../entities/content/learning-path-node.entity.js';
|
import { LearningPathNode } from '../../entities/content/learning-path-node.entity.js';
|
||||||
import { LearningPathTransition } from '../../entities/content/learning-path-transition.entity.js';
|
import { LearningPathTransition } from '../../entities/content/learning-path-transition.entity.js';
|
||||||
import { getLastSubmissionForGroup, isTransitionPossible } from './learning-path-personalization-util.js';
|
import { getLastSubmissionForGroup, idFromLearningPathNode, isTransitionPossible } from './learning-path-personalization-util.js';
|
||||||
import {
|
import {
|
||||||
FilteredLearningObject,
|
FilteredLearningObject,
|
||||||
LearningObjectNode,
|
LearningObjectNode,
|
||||||
|
@ -97,7 +97,7 @@ async function convertNode(
|
||||||
personalizedFor: Group | undefined,
|
personalizedFor: Group | undefined,
|
||||||
nodesToLearningObjects: Map<LearningPathNode, FilteredLearningObject>
|
nodesToLearningObjects: Map<LearningPathNode, FilteredLearningObject>
|
||||||
): Promise<LearningObjectNode> {
|
): Promise<LearningObjectNode> {
|
||||||
const lastSubmission = personalizedFor ? await getLastSubmissionForGroup(node, personalizedFor) : null;
|
const lastSubmission = personalizedFor ? await getLastSubmissionForGroup(idFromLearningPathNode(node), personalizedFor) : null;
|
||||||
const transitions = node.transitions
|
const transitions = node.transitions
|
||||||
.filter(
|
.filter(
|
||||||
(trans) =>
|
(trans) =>
|
||||||
|
|
|
@ -3,11 +3,33 @@ import { DWENGO_API_BASE } from '../../config.js';
|
||||||
import { LearningPathProvider } from './learning-path-provider.js';
|
import { LearningPathProvider } from './learning-path-provider.js';
|
||||||
import { getLogger, Logger } from '../../logging/initalize.js';
|
import { getLogger, Logger } from '../../logging/initalize.js';
|
||||||
import { LearningPath, LearningPathResponse } from '@dwengo-1/common/interfaces/learning-content';
|
import { LearningPath, LearningPathResponse } from '@dwengo-1/common/interfaces/learning-content';
|
||||||
|
import { Group } from '../../entities/assignments/group.entity.js';
|
||||||
|
import { getLastSubmissionForGroup, idFromLearningObjectNode } from './learning-path-personalization-util.js';
|
||||||
|
|
||||||
const logger: Logger = getLogger();
|
const logger: Logger = getLogger();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds progress information to the learning path. Modifies the learning path in-place.
|
||||||
|
* @param learningPath The learning path to add progress to.
|
||||||
|
* @param personalizedFor The group whose progress should be shown.
|
||||||
|
* @returns the modified learning path.
|
||||||
|
*/
|
||||||
|
async function addProgressToLearningPath(learningPath: LearningPath, personalizedFor: Group): Promise<LearningPath> {
|
||||||
|
await Promise.all(
|
||||||
|
learningPath.nodes.map(async (node) => {
|
||||||
|
const lastSubmission = personalizedFor ? await getLastSubmissionForGroup(idFromLearningObjectNode(node), personalizedFor) : null;
|
||||||
|
node.done = Boolean(lastSubmission);
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
learningPath.num_nodes = learningPath.nodes.length;
|
||||||
|
learningPath.num_nodes_left = learningPath.nodes.filter((it) => !it.done).length;
|
||||||
|
|
||||||
|
return learningPath;
|
||||||
|
}
|
||||||
|
|
||||||
const dwengoApiLearningPathProvider: LearningPathProvider = {
|
const dwengoApiLearningPathProvider: LearningPathProvider = {
|
||||||
async fetchLearningPaths(hruids: string[], language: string, source: string): Promise<LearningPathResponse> {
|
async fetchLearningPaths(hruids: string[], language: string, source: string, personalizedFor: Group): Promise<LearningPathResponse> {
|
||||||
if (hruids.length === 0) {
|
if (hruids.length === 0) {
|
||||||
return {
|
return {
|
||||||
success: false,
|
success: false,
|
||||||
|
@ -32,17 +54,24 @@ const dwengoApiLearningPathProvider: LearningPathProvider = {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await Promise.all(learningPaths?.map(async (it) => addProgressToLearningPath(it, personalizedFor)));
|
||||||
|
|
||||||
return {
|
return {
|
||||||
success: true,
|
success: true,
|
||||||
source,
|
source,
|
||||||
data: learningPaths,
|
data: learningPaths,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
async searchLearningPaths(query: string, language: string): Promise<LearningPath[]> {
|
async searchLearningPaths(query: string, language: string, personalizedFor: Group): Promise<LearningPath[]> {
|
||||||
const apiUrl = `${DWENGO_API_BASE}/learningPath/search`;
|
const apiUrl = `${DWENGO_API_BASE}/learningPath/search`;
|
||||||
const params = { all: query, language };
|
const params = { all: query, language };
|
||||||
|
|
||||||
const searchResults = await fetchWithLogging<LearningPath[]>(apiUrl, `Search learning paths with query "${query}"`, { params });
|
const searchResults = await fetchWithLogging<LearningPath[]>(apiUrl, `Search learning paths with query "${query}"`, { params });
|
||||||
|
|
||||||
|
if (searchResults) {
|
||||||
|
await Promise.all(searchResults?.map(async (it) => addProgressToLearningPath(it, personalizedFor)));
|
||||||
|
}
|
||||||
|
|
||||||
return searchResults ?? [];
|
return searchResults ?? [];
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
|
@ -5,18 +5,36 @@ import { getSubmissionRepository } from '../../data/repositories.js';
|
||||||
import { LearningObjectIdentifier } from '../../entities/content/learning-object-identifier.js';
|
import { LearningObjectIdentifier } from '../../entities/content/learning-object-identifier.js';
|
||||||
import { LearningPathTransition } from '../../entities/content/learning-path-transition.entity.js';
|
import { LearningPathTransition } from '../../entities/content/learning-path-transition.entity.js';
|
||||||
import { JSONPath } from 'jsonpath-plus';
|
import { JSONPath } from 'jsonpath-plus';
|
||||||
|
import { LearningObjectNode } from '@dwengo-1/common/interfaces/learning-content';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the last submission for the learning object associated with the given node and for the group
|
* Returns the last submission for the learning object associated with the given node and for the group
|
||||||
*/
|
*/
|
||||||
export async function getLastSubmissionForGroup(node: LearningPathNode, pathFor: Group): Promise<Submission | null> {
|
export async function getLastSubmissionForGroup(learningObjectId: LearningObjectIdentifier, pathFor: Group): Promise<Submission | null> {
|
||||||
const submissionRepo = getSubmissionRepository();
|
const submissionRepo = getSubmissionRepository();
|
||||||
const learningObjectId: LearningObjectIdentifier = {
|
return await submissionRepo.findMostRecentSubmissionForGroup(learningObjectId, pathFor);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a LearningObjectIdentifier describing the specified node.
|
||||||
|
*/
|
||||||
|
export function idFromLearningObjectNode(node: LearningObjectNode): LearningObjectIdentifier {
|
||||||
|
return {
|
||||||
|
hruid: node.learningobject_hruid,
|
||||||
|
language: node.language,
|
||||||
|
version: node.version,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a LearningObjectIdentifier describing the specified node.
|
||||||
|
*/
|
||||||
|
export function idFromLearningPathNode(node: LearningPathNode): LearningObjectIdentifier {
|
||||||
|
return {
|
||||||
hruid: node.learningObjectHruid,
|
hruid: node.learningObjectHruid,
|
||||||
language: node.language,
|
language: node.language,
|
||||||
version: node.version,
|
version: node.version,
|
||||||
};
|
};
|
||||||
return await submissionRepo.findMostRecentSubmissionForGroup(learningObjectId, pathFor);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -42,7 +42,7 @@ export async function fetchStudent(username: string): Promise<Student> {
|
||||||
const user = await studentRepository.findByUsername(username);
|
const user = await studentRepository.findByUsername(username);
|
||||||
|
|
||||||
if (!user) {
|
if (!user) {
|
||||||
throw new NotFoundException('Student with username not found');
|
throw new NotFoundException(`Student with username ${username} not found`);
|
||||||
}
|
}
|
||||||
|
|
||||||
return user;
|
return user;
|
||||||
|
|
76
backend/tests/controllers/assignments.test.ts
Normal file
76
backend/tests/controllers/assignments.test.ts
Normal file
|
@ -0,0 +1,76 @@
|
||||||
|
import { setupTestApp } from '../setup-tests.js';
|
||||||
|
import { describe, it, expect, beforeAll, beforeEach, vi, Mock } from 'vitest';
|
||||||
|
import { Request, Response } from 'express';
|
||||||
|
import { getAssignmentHandler, getAllAssignmentsHandler, getAssignmentsSubmissionsHandler } from '../../src/controllers/assignments.js';
|
||||||
|
import { NotFoundException } from '../../src/exceptions/not-found-exception';
|
||||||
|
import { getClass01 } from '../test_assets/classes/classes.testdata';
|
||||||
|
import { getAssignment01 } from '../test_assets/assignments/assignments.testdata';
|
||||||
|
|
||||||
|
function createRequestObject(
|
||||||
|
classid: string,
|
||||||
|
assignmentid: string
|
||||||
|
): {
|
||||||
|
query: { full: string };
|
||||||
|
params: { classid: string; id: string };
|
||||||
|
} {
|
||||||
|
return {
|
||||||
|
params: {
|
||||||
|
classid: classid,
|
||||||
|
id: assignmentid,
|
||||||
|
},
|
||||||
|
query: {
|
||||||
|
full: 'true',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('Assignment controllers', () => {
|
||||||
|
let req: Partial<Request>;
|
||||||
|
let res: Partial<Response>;
|
||||||
|
|
||||||
|
let jsonMock: Mock;
|
||||||
|
let statusMock: Mock;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
await setupTestApp();
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
jsonMock = vi.fn();
|
||||||
|
statusMock = vi.fn().mockReturnThis();
|
||||||
|
|
||||||
|
res = {
|
||||||
|
json: jsonMock,
|
||||||
|
status: statusMock,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
it('return error non-existing assignment', async () => {
|
||||||
|
req = createRequestObject('doesnotexist', '43000'); // Should not exist
|
||||||
|
|
||||||
|
await expect(async () => getAssignmentHandler(req as Request, res as Response)).rejects.toThrow(NotFoundException);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return an assignment', async () => {
|
||||||
|
const assignment = getAssignment01();
|
||||||
|
req = createRequestObject(assignment.within.classId as string, (assignment.id ?? 1).toString());
|
||||||
|
|
||||||
|
await getAssignmentHandler(req as Request, res as Response);
|
||||||
|
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ assignment: expect.anything() }));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return a list of assignments', async () => {
|
||||||
|
req = createRequestObject(getClass01().classId as string, 'irrelevant');
|
||||||
|
|
||||||
|
await getAllAssignmentsHandler(req as Request, res as Response);
|
||||||
|
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ assignments: expect.anything() }));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return a list of submissions for an assignment', async () => {
|
||||||
|
const assignment = getAssignment01();
|
||||||
|
req = createRequestObject(assignment.within.classId as string, (assignment.id ?? 1).toString());
|
||||||
|
|
||||||
|
await getAssignmentsSubmissionsHandler(req as Request, res as Response);
|
||||||
|
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ submissions: expect.anything() }));
|
||||||
|
});
|
||||||
|
});
|
|
@ -1,8 +1,17 @@
|
||||||
import { setupTestApp } from '../setup-tests.js';
|
import { setupTestApp } from '../setup-tests.js';
|
||||||
import { describe, it, expect, beforeAll, beforeEach, vi, Mock } from 'vitest';
|
import { describe, it, expect, beforeAll, beforeEach, vi, Mock } from 'vitest';
|
||||||
|
import {
|
||||||
|
createClassHandler,
|
||||||
|
deleteClassHandler,
|
||||||
|
getAllClassesHandler,
|
||||||
|
getClassHandler,
|
||||||
|
getClassStudentsHandler,
|
||||||
|
getTeacherInvitationsHandler,
|
||||||
|
} from '../../src/controllers/classes.js';
|
||||||
import { Request, Response } from 'express';
|
import { Request, Response } from 'express';
|
||||||
import { createClassHandler, deleteClassHandler } from '../../src/controllers/classes';
|
import { NotFoundException } from '../../src/exceptions/not-found-exception';
|
||||||
|
import { BadRequestException } from '../../src/exceptions/bad-request-exception';
|
||||||
|
import { getClass01 } from '../test_assets/classes/classes.testdata';
|
||||||
describe('Class controllers', () => {
|
describe('Class controllers', () => {
|
||||||
let req: Partial<Request>;
|
let req: Partial<Request>;
|
||||||
let res: Partial<Response>;
|
let res: Partial<Response>;
|
||||||
|
@ -44,4 +53,71 @@ describe('Class controllers', () => {
|
||||||
|
|
||||||
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ class: expect.anything() }));
|
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ class: expect.anything() }));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('Error class not found', async () => {
|
||||||
|
req = {
|
||||||
|
params: { id: 'doesnotexist' },
|
||||||
|
};
|
||||||
|
|
||||||
|
await expect(async () => getClassHandler(req as Request, res as Response)).rejects.toThrow(NotFoundException);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Error create a class without name', async () => {
|
||||||
|
req = {
|
||||||
|
body: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
await expect(async () => createClassHandler(req as Request, res as Response)).rejects.toThrow(BadRequestException);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('return list of students', async () => {
|
||||||
|
req = {
|
||||||
|
params: { id: getClass01().classId as string },
|
||||||
|
query: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
await getClassStudentsHandler(req as Request, res as Response);
|
||||||
|
|
||||||
|
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ students: expect.anything() }));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Error students on a non-existent class', async () => {
|
||||||
|
req = {
|
||||||
|
params: { id: 'doesnotexist' },
|
||||||
|
query: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
await expect(async () => getClassStudentsHandler(req as Request, res as Response)).rejects.toThrow(NotFoundException);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return 200 and a list of teacher-invitations', async () => {
|
||||||
|
const classId = getClass01().classId as string;
|
||||||
|
req = {
|
||||||
|
params: { id: classId },
|
||||||
|
query: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
await getTeacherInvitationsHandler(req as Request, res as Response);
|
||||||
|
|
||||||
|
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ invitations: expect.anything() }));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Error teacher-invitations on a non-existent class', async () => {
|
||||||
|
req = {
|
||||||
|
params: { id: 'doesnotexist' },
|
||||||
|
query: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
await expect(async () => getTeacherInvitationsHandler(req as Request, res as Response)).rejects.toThrow(NotFoundException);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return a list of classes', async () => {
|
||||||
|
req = {
|
||||||
|
query: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
await getAllClassesHandler(req as Request, res as Response);
|
||||||
|
|
||||||
|
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ classes: expect.anything() }));
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
140
backend/tests/controllers/groups.test.ts
Normal file
140
backend/tests/controllers/groups.test.ts
Normal file
|
@ -0,0 +1,140 @@
|
||||||
|
import { setupTestApp } from '../setup-tests.js';
|
||||||
|
import { describe, it, expect, beforeAll, beforeEach, vi, Mock } from 'vitest';
|
||||||
|
import { Request, Response } from 'express';
|
||||||
|
import {
|
||||||
|
createGroupHandler,
|
||||||
|
deleteGroupHandler,
|
||||||
|
getAllGroupsHandler,
|
||||||
|
getGroupHandler,
|
||||||
|
getGroupSubmissionsHandler,
|
||||||
|
} from '../../src/controllers/groups.js';
|
||||||
|
import { NotFoundException } from '../../src/exceptions/not-found-exception';
|
||||||
|
import { getClass01 } from '../test_assets/classes/classes.testdata';
|
||||||
|
import { getAssignment01, getAssignment02 } from '../test_assets/assignments/assignments.testdata';
|
||||||
|
import { getTestGroup01 } from '../test_assets/assignments/groups.testdata';
|
||||||
|
|
||||||
|
function createRequestObject(
|
||||||
|
classid: string,
|
||||||
|
assignmentid: string,
|
||||||
|
groupNumber: string
|
||||||
|
): {
|
||||||
|
query: { full: string };
|
||||||
|
params: { classid: string; groupid: string; assignmentid: string };
|
||||||
|
} {
|
||||||
|
return {
|
||||||
|
params: {
|
||||||
|
classid: classid,
|
||||||
|
assignmentid: assignmentid,
|
||||||
|
groupid: groupNumber,
|
||||||
|
},
|
||||||
|
query: {
|
||||||
|
full: 'true',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('Group controllers', () => {
|
||||||
|
let req: Partial<Request>;
|
||||||
|
let res: Partial<Response>;
|
||||||
|
|
||||||
|
let jsonMock: Mock;
|
||||||
|
let statusMock: Mock;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
await setupTestApp();
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
jsonMock = vi.fn();
|
||||||
|
statusMock = vi.fn().mockReturnThis();
|
||||||
|
|
||||||
|
res = {
|
||||||
|
json: jsonMock,
|
||||||
|
status: statusMock,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Error not found on a non-existing group', async () => {
|
||||||
|
req = {
|
||||||
|
params: {
|
||||||
|
classid: 'id01',
|
||||||
|
assignmentid: '1',
|
||||||
|
groupid: '154981', // Should not exist
|
||||||
|
},
|
||||||
|
query: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
await expect(async () => getGroupHandler(req as Request, res as Response)).rejects.toThrow(NotFoundException);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return 404 not found on a non-existing assignment', async () => {
|
||||||
|
req = {
|
||||||
|
params: {
|
||||||
|
classid: 'id01',
|
||||||
|
assignmentid: '1000', // Should not exist
|
||||||
|
groupid: '42000', // Should not exist
|
||||||
|
},
|
||||||
|
query: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
await expect(async () => getGroupHandler(req as Request, res as Response)).rejects.toThrow(NotFoundException);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return 404 not found ont a non-existing class', async () => {
|
||||||
|
req = {
|
||||||
|
params: {
|
||||||
|
classid: 'doesnotexist', // Should not exist
|
||||||
|
assignmentid: '1000', // Should not exist
|
||||||
|
groupid: '42000', // Should not exist
|
||||||
|
},
|
||||||
|
query: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
await expect(async () => getGroupHandler(req as Request, res as Response)).rejects.toThrow(NotFoundException);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return an existing group', async () => {
|
||||||
|
const group = getTestGroup01();
|
||||||
|
const classId = getClass01().classId as string;
|
||||||
|
req = createRequestObject(classId, (group.assignment.id ?? 1).toString(), (group.groupNumber ?? 1).toString());
|
||||||
|
|
||||||
|
await getGroupHandler(req as Request, res as Response);
|
||||||
|
|
||||||
|
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ group: expect.anything() }));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Create and delete', async () => {
|
||||||
|
const assignment = getAssignment02();
|
||||||
|
const classId = assignment.within.classId as string;
|
||||||
|
req = createRequestObject(classId, (assignment.id ?? 1).toString(), '1');
|
||||||
|
req.body = {
|
||||||
|
members: ['Noordkaap', 'DireStraits'],
|
||||||
|
};
|
||||||
|
|
||||||
|
await createGroupHandler(req as Request, res as Response);
|
||||||
|
|
||||||
|
await deleteGroupHandler(req as Request, res as Response);
|
||||||
|
|
||||||
|
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ group: expect.anything() }));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return the submissions for a group', async () => {
|
||||||
|
const group = getTestGroup01();
|
||||||
|
const classId = getClass01().classId as string;
|
||||||
|
req = createRequestObject(classId, (group.assignment.id ?? 1).toString(), (group.groupNumber ?? 1).toString());
|
||||||
|
|
||||||
|
await getGroupSubmissionsHandler(req as Request, res as Response);
|
||||||
|
|
||||||
|
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ submissions: expect.anything() }));
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return a list of groups for an assignment', async () => {
|
||||||
|
const assignment = getAssignment01();
|
||||||
|
const classId = assignment.within.classId as string;
|
||||||
|
req = createRequestObject(classId, (assignment.id ?? 1).toString(), '1');
|
||||||
|
|
||||||
|
await getAllGroupsHandler(req as Request, res as Response);
|
||||||
|
|
||||||
|
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ groups: expect.anything() }));
|
||||||
|
});
|
||||||
|
});
|
61
backend/tests/controllers/submissions.test.ts
Normal file
61
backend/tests/controllers/submissions.test.ts
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
import { setupTestApp } from '../setup-tests.js';
|
||||||
|
import { describe, it, expect, beforeAll, beforeEach, vi, Mock } from 'vitest';
|
||||||
|
import { getSubmissionHandler, getAllSubmissionsHandler } from '../../src/controllers/submissions.js';
|
||||||
|
import { Request, Response } from 'express';
|
||||||
|
import { NotFoundException } from '../../src/exceptions/not-found-exception';
|
||||||
|
import { getClass02 } from '../test_assets/classes/classes.testdata';
|
||||||
|
|
||||||
|
function createRequestObject(
|
||||||
|
hruid: string,
|
||||||
|
submissionNumber: string
|
||||||
|
): {
|
||||||
|
query: { language: string; version: string };
|
||||||
|
params: { hruid: string; id: string };
|
||||||
|
} {
|
||||||
|
return {
|
||||||
|
params: {
|
||||||
|
hruid: hruid,
|
||||||
|
id: submissionNumber,
|
||||||
|
},
|
||||||
|
query: {
|
||||||
|
language: 'en',
|
||||||
|
version: '1',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('Submission controllers', () => {
|
||||||
|
let req: Partial<Request>;
|
||||||
|
let res: Partial<Response>;
|
||||||
|
|
||||||
|
let jsonMock: Mock;
|
||||||
|
let statusMock: Mock;
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
await setupTestApp();
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
jsonMock = vi.fn();
|
||||||
|
statusMock = vi.fn().mockReturnThis();
|
||||||
|
|
||||||
|
res = {
|
||||||
|
json: jsonMock,
|
||||||
|
status: statusMock,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
it('error submission is not found', async () => {
|
||||||
|
req = createRequestObject('id01', '1000000');
|
||||||
|
|
||||||
|
await expect(async () => getSubmissionHandler(req as Request, res as Response)).rejects.toThrow(NotFoundException);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return a list of submissions for a learning object', async () => {
|
||||||
|
req = createRequestObject(getClass02().classId as string, 'irrelevant');
|
||||||
|
|
||||||
|
await getAllSubmissionsHandler(req as Request, res as Response);
|
||||||
|
|
||||||
|
expect(jsonMock).toHaveBeenCalledWith(expect.objectContaining({ submissions: expect.anything() }));
|
||||||
|
});
|
||||||
|
});
|
Loading…
Add table
Add a link
Reference in a new issue