style: fix linting issues met Prettier
This commit is contained in:
		
							parent
							
								
									b8aae0ab1b
								
							
						
					
					
						commit
						f347ec247d
					
				
					 33 changed files with 90 additions and 361 deletions
				
			
		|  | @ -6,7 +6,5 @@ export const DWENGO_API_BASE: string = 'https://dwengo.org/backend/api'; | ||||||
| 
 | 
 | ||||||
| // Logging
 | // Logging
 | ||||||
| 
 | 
 | ||||||
| export const LOG_LEVEL: string = | export const LOG_LEVEL: string = 'development' === process.env.NODE_ENV ? 'debug' : 'info'; | ||||||
|     'development' === process.env.NODE_ENV ? 'debug' : 'info'; | export const LOKI_HOST: string = process.env.LOKI_HOST || 'http://localhost:3102'; | ||||||
| export const LOKI_HOST: string = |  | ||||||
|     process.env.LOKI_HOST || 'http://localhost:3102'; |  | ||||||
|  |  | ||||||
|  | @ -1,17 +1,10 @@ | ||||||
| import { Request, Response } from 'express'; | import { Request, Response } from 'express'; | ||||||
| import { | import { getLearningObjectById, getLearningObjectIdsFromPath, getLearningObjectsFromPath } from '../services/learningObjects.js'; | ||||||
|     getLearningObjectById, |  | ||||||
|     getLearningObjectIdsFromPath, |  | ||||||
|     getLearningObjectsFromPath, |  | ||||||
| } from '../services/learningObjects.js'; |  | ||||||
| import { FALLBACK_LANG } from '../config.js'; | import { FALLBACK_LANG } from '../config.js'; | ||||||
| import { FilteredLearningObject } from '../interfaces/learningPath.js'; | import { FilteredLearningObject } from '../interfaces/learningPath.js'; | ||||||
| import { getLogger } from '../logging/initalize.js'; | import { getLogger } from '../logging/initalize.js'; | ||||||
| 
 | 
 | ||||||
| export async function getAllLearningObjects( | export async function getAllLearningObjects(req: Request, res: Response): Promise<void> { | ||||||
|     req: Request, |  | ||||||
|     res: Response |  | ||||||
| ): Promise<void> { |  | ||||||
|     try { |     try { | ||||||
|         const hruid = req.query.hruid as string; |         const hruid = req.query.hruid as string; | ||||||
|         const full = req.query.full === 'true'; |         const full = req.query.full === 'true'; | ||||||
|  | @ -26,10 +19,7 @@ export async function getAllLearningObjects( | ||||||
|         if (full) { |         if (full) { | ||||||
|             learningObjects = await getLearningObjectsFromPath(hruid, language); |             learningObjects = await getLearningObjectsFromPath(hruid, language); | ||||||
|         } else { |         } else { | ||||||
|             learningObjects = await getLearningObjectIdsFromPath( |             learningObjects = await getLearningObjectIdsFromPath(hruid, language); | ||||||
|                 hruid, |  | ||||||
|                 language |  | ||||||
|             ); |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         res.json(learningObjects); |         res.json(learningObjects); | ||||||
|  | @ -39,10 +29,7 @@ export async function getAllLearningObjects( | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export async function getLearningObject( | export async function getLearningObject(req: Request, res: Response): Promise<void> { | ||||||
|     req: Request, |  | ||||||
|     res: Response |  | ||||||
| ): Promise<void> { |  | ||||||
|     try { |     try { | ||||||
|         const { hruid } = req.params; |         const { hruid } = req.params; | ||||||
|         const language = (req.query.language as string) || FALLBACK_LANG; |         const language = (req.query.language as string) || FALLBACK_LANG; | ||||||
|  |  | ||||||
|  | @ -1,18 +1,12 @@ | ||||||
| import { Request, Response } from 'express'; | import { Request, Response } from 'express'; | ||||||
| import { themes } from '../data/themes.js'; | import { themes } from '../data/themes.js'; | ||||||
| import { FALLBACK_LANG } from '../config.js'; | import { FALLBACK_LANG } from '../config.js'; | ||||||
| import { | import { fetchLearningPaths, searchLearningPaths } from '../services/learningPaths.js'; | ||||||
|     fetchLearningPaths, |  | ||||||
|     searchLearningPaths, |  | ||||||
| } from '../services/learningPaths.js'; |  | ||||||
| import { getLogger } from '../logging/initalize.js'; | import { getLogger } from '../logging/initalize.js'; | ||||||
| /** | /** | ||||||
|  * Fetch learning paths based on query parameters. |  * Fetch learning paths based on query parameters. | ||||||
|  */ |  */ | ||||||
| export async function getLearningPaths( | export async function getLearningPaths(req: Request, res: Response): Promise<void> { | ||||||
|     req: Request, |  | ||||||
|     res: Response |  | ||||||
| ): Promise<void> { |  | ||||||
|     try { |     try { | ||||||
|         const hruids = req.query.hruid; |         const hruids = req.query.hruid; | ||||||
|         const themeKey = req.query.theme as string; |         const themeKey = req.query.theme as string; | ||||||
|  | @ -22,9 +16,7 @@ export async function getLearningPaths( | ||||||
|         let hruidList; |         let hruidList; | ||||||
| 
 | 
 | ||||||
|         if (hruids) { |         if (hruids) { | ||||||
|             hruidList = Array.isArray(hruids) |             hruidList = Array.isArray(hruids) ? hruids.map(String) : [String(hruids)]; | ||||||
|                 ? hruids.map(String) |  | ||||||
|                 : [String(hruids)]; |  | ||||||
|         } else if (themeKey) { |         } else if (themeKey) { | ||||||
|             const theme = themes.find((t) => t.title === themeKey); |             const theme = themes.find((t) => t.title === themeKey); | ||||||
|             if (theme) { |             if (theme) { | ||||||
|  | @ -36,27 +28,17 @@ export async function getLearningPaths( | ||||||
|                 return; |                 return; | ||||||
|             } |             } | ||||||
|         } else if (searchQuery) { |         } else if (searchQuery) { | ||||||
|             const searchResults = await searchLearningPaths( |             const searchResults = await searchLearningPaths(searchQuery, language); | ||||||
|                 searchQuery, |  | ||||||
|                 language |  | ||||||
|             ); |  | ||||||
|             res.json(searchResults); |             res.json(searchResults); | ||||||
|             return; |             return; | ||||||
|         } else { |         } else { | ||||||
|             hruidList = themes.flatMap((theme) => theme.hruids); |             hruidList = themes.flatMap((theme) => theme.hruids); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         const learningPaths = await fetchLearningPaths( |         const learningPaths = await fetchLearningPaths(hruidList, language, `HRUIDs: ${hruidList.join(', ')}`); | ||||||
|             hruidList, |  | ||||||
|             language, |  | ||||||
|             `HRUIDs: ${hruidList.join(', ')}` |  | ||||||
|         ); |  | ||||||
|         res.json(learningPaths.data); |         res.json(learningPaths.data); | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|         getLogger().error( |         getLogger().error('❌ Unexpected error fetching learning paths:', error); | ||||||
|             '❌ Unexpected error fetching learning paths:', |  | ||||||
|             error |  | ||||||
|         ); |  | ||||||
|         res.status(500).json({ error: 'Internal server error' }); |         res.status(500).json({ error: 'Internal server error' }); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -12,12 +12,11 @@ export function getThemes(req: Request, res: Response) { | ||||||
|     const language = (req.query.language as string)?.toLowerCase() || 'nl'; |     const language = (req.query.language as string)?.toLowerCase() || 'nl'; | ||||||
|     const translations = loadTranslations<Translations>(language); |     const translations = loadTranslations<Translations>(language); | ||||||
|     const themeList = themes.map((theme) => ({ |     const themeList = themes.map((theme) => ({ | ||||||
|             key: theme.title, |         key: theme.title, | ||||||
|             title: |         title: translations.curricula_page[theme.title]?.title || theme.title, | ||||||
|                 translations.curricula_page[theme.title]?.title || theme.title, |         description: translations.curricula_page[theme.title]?.description, | ||||||
|             description: translations.curricula_page[theme.title]?.description, |         image: `https://dwengo.org/images/curricula/logo_${theme.title}.png`, | ||||||
|             image: `https://dwengo.org/images/curricula/logo_${theme.title}.png`, |     })); | ||||||
|         })); |  | ||||||
| 
 | 
 | ||||||
|     res.json(themeList); |     res.json(themeList); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -3,10 +3,7 @@ import { Assignment } from '../../entities/assignments/assignment.entity.js'; | ||||||
| import { Class } from '../../entities/classes/class.entity.js'; | import { Class } from '../../entities/classes/class.entity.js'; | ||||||
| 
 | 
 | ||||||
| export class AssignmentRepository extends DwengoEntityRepository<Assignment> { | export class AssignmentRepository extends DwengoEntityRepository<Assignment> { | ||||||
|     public findByClassAndId( |     public findByClassAndId(within: Class, id: number): Promise<Assignment | null> { | ||||||
|         within: Class, |  | ||||||
|         id: number |  | ||||||
|     ): Promise<Assignment | null> { |  | ||||||
|         return this.findOne({ within: within, id: id }); |         return this.findOne({ within: within, id: id }); | ||||||
|     } |     } | ||||||
|     public findAllAssignmentsInClass(within: Class): Promise<Assignment[]> { |     public findAllAssignmentsInClass(within: Class): Promise<Assignment[]> { | ||||||
|  |  | ||||||
|  | @ -3,24 +3,16 @@ import { Group } from '../../entities/assignments/group.entity.js'; | ||||||
| import { Assignment } from '../../entities/assignments/assignment.entity.js'; | import { Assignment } from '../../entities/assignments/assignment.entity.js'; | ||||||
| 
 | 
 | ||||||
| export class GroupRepository extends DwengoEntityRepository<Group> { | export class GroupRepository extends DwengoEntityRepository<Group> { | ||||||
|     public findByAssignmentAndGroupNumber( |     public findByAssignmentAndGroupNumber(assignment: Assignment, groupNumber: number): Promise<Group | null> { | ||||||
|         assignment: Assignment, |  | ||||||
|         groupNumber: number |  | ||||||
|     ): Promise<Group | null> { |  | ||||||
|         return this.findOne({ |         return this.findOne({ | ||||||
|             assignment: assignment, |             assignment: assignment, | ||||||
|             groupNumber: groupNumber, |             groupNumber: groupNumber, | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|     public findAllGroupsForAssignment( |     public findAllGroupsForAssignment(assignment: Assignment): Promise<Group[]> { | ||||||
|         assignment: Assignment |  | ||||||
|     ): Promise<Group[]> { |  | ||||||
|         return this.findAll({ where: { assignment: assignment } }); |         return this.findAll({ where: { assignment: assignment } }); | ||||||
|     } |     } | ||||||
|     public deleteByAssignmentAndGroupNumber( |     public deleteByAssignmentAndGroupNumber(assignment: Assignment, groupNumber: number) { | ||||||
|         assignment: Assignment, |  | ||||||
|         groupNumber: number |  | ||||||
|     ) { |  | ||||||
|         return this.deleteWhere({ |         return this.deleteWhere({ | ||||||
|             assignment: assignment, |             assignment: assignment, | ||||||
|             groupNumber: groupNumber, |             groupNumber: groupNumber, | ||||||
|  |  | ||||||
|  | @ -5,10 +5,7 @@ import { LearningObjectIdentifier } from '../../entities/content/learning-object | ||||||
| import { Student } from '../../entities/users/student.entity.js'; | import { Student } from '../../entities/users/student.entity.js'; | ||||||
| 
 | 
 | ||||||
| export class SubmissionRepository extends DwengoEntityRepository<Submission> { | export class SubmissionRepository extends DwengoEntityRepository<Submission> { | ||||||
|     public findSubmissionByLearningObjectAndSubmissionNumber( |     public findSubmissionByLearningObjectAndSubmissionNumber(loId: LearningObjectIdentifier, submissionNumber: number): Promise<Submission | null> { | ||||||
|         loId: LearningObjectIdentifier, |  | ||||||
|         submissionNumber: number |  | ||||||
|     ): Promise<Submission | null> { |  | ||||||
|         return this.findOne({ |         return this.findOne({ | ||||||
|             learningObjectHruid: loId.hruid, |             learningObjectHruid: loId.hruid, | ||||||
|             learningObjectLanguage: loId.language, |             learningObjectLanguage: loId.language, | ||||||
|  | @ -17,10 +14,7 @@ export class SubmissionRepository extends DwengoEntityRepository<Submission> { | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public findMostRecentSubmissionForStudent( |     public findMostRecentSubmissionForStudent(loId: LearningObjectIdentifier, submitter: Student): Promise<Submission | null> { | ||||||
|         loId: LearningObjectIdentifier, |  | ||||||
|         submitter: Student |  | ||||||
|     ): Promise<Submission | null> { |  | ||||||
|         return this.findOne( |         return this.findOne( | ||||||
|             { |             { | ||||||
|                 learningObjectHruid: loId.hruid, |                 learningObjectHruid: loId.hruid, | ||||||
|  | @ -32,10 +26,7 @@ export class SubmissionRepository extends DwengoEntityRepository<Submission> { | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public findMostRecentSubmissionForGroup( |     public findMostRecentSubmissionForGroup(loId: LearningObjectIdentifier, group: Group): Promise<Submission | null> { | ||||||
|         loId: LearningObjectIdentifier, |  | ||||||
|         group: Group |  | ||||||
|     ): Promise<Submission | null> { |  | ||||||
|         return this.findOne( |         return this.findOne( | ||||||
|             { |             { | ||||||
|                 learningObjectHruid: loId.hruid, |                 learningObjectHruid: loId.hruid, | ||||||
|  | @ -47,10 +38,7 @@ export class SubmissionRepository extends DwengoEntityRepository<Submission> { | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public deleteSubmissionByLearningObjectAndSubmissionNumber( |     public deleteSubmissionByLearningObjectAndSubmissionNumber(loId: LearningObjectIdentifier, submissionNumber: number): Promise<void> { | ||||||
|         loId: LearningObjectIdentifier, |  | ||||||
|         submissionNumber: number |  | ||||||
|     ): Promise<void> { |  | ||||||
|         return this.deleteWhere({ |         return this.deleteWhere({ | ||||||
|             learningObjectHruid: loId.hruid, |             learningObjectHruid: loId.hruid, | ||||||
|             learningObjectLanguage: loId.language, |             learningObjectLanguage: loId.language, | ||||||
|  |  | ||||||
|  | @ -4,24 +4,16 @@ import { TeacherInvitation } from '../../entities/classes/teacher-invitation.ent | ||||||
| import { Teacher } from '../../entities/users/teacher.entity.js'; | import { Teacher } from '../../entities/users/teacher.entity.js'; | ||||||
| 
 | 
 | ||||||
| export class TeacherInvitationRepository extends DwengoEntityRepository<TeacherInvitation> { | export class TeacherInvitationRepository extends DwengoEntityRepository<TeacherInvitation> { | ||||||
|     public findAllInvitationsForClass( |     public findAllInvitationsForClass(clazz: Class): Promise<TeacherInvitation[]> { | ||||||
|         clazz: Class |  | ||||||
|     ): Promise<TeacherInvitation[]> { |  | ||||||
|         return this.findAll({ where: { class: clazz } }); |         return this.findAll({ where: { class: clazz } }); | ||||||
|     } |     } | ||||||
|     public findAllInvitationsBy(sender: Teacher): Promise<TeacherInvitation[]> { |     public findAllInvitationsBy(sender: Teacher): Promise<TeacherInvitation[]> { | ||||||
|         return this.findAll({ where: { sender: sender } }); |         return this.findAll({ where: { sender: sender } }); | ||||||
|     } |     } | ||||||
|     public findAllInvitationsFor( |     public findAllInvitationsFor(receiver: Teacher): Promise<TeacherInvitation[]> { | ||||||
|         receiver: Teacher |  | ||||||
|     ): Promise<TeacherInvitation[]> { |  | ||||||
|         return this.findAll({ where: { receiver: receiver } }); |         return this.findAll({ where: { receiver: receiver } }); | ||||||
|     } |     } | ||||||
|     public deleteBy( |     public deleteBy(clazz: Class, sender: Teacher, receiver: Teacher): Promise<void> { | ||||||
|         clazz: Class, |  | ||||||
|         sender: Teacher, |  | ||||||
|         receiver: Teacher |  | ||||||
|     ): Promise<void> { |  | ||||||
|         return this.deleteWhere({ |         return this.deleteWhere({ | ||||||
|             sender: sender, |             sender: sender, | ||||||
|             receiver: receiver, |             receiver: receiver, | ||||||
|  |  | ||||||
|  | @ -3,10 +3,7 @@ import { Attachment } from '../../entities/content/attachment.entity.js'; | ||||||
| import { LearningObject } from '../../entities/content/learning-object.entity.js'; | import { LearningObject } from '../../entities/content/learning-object.entity.js'; | ||||||
| 
 | 
 | ||||||
| export class AttachmentRepository extends DwengoEntityRepository<Attachment> { | export class AttachmentRepository extends DwengoEntityRepository<Attachment> { | ||||||
|     public findByLearningObjectAndNumber( |     public findByLearningObjectAndNumber(learningObject: LearningObject, sequenceNumber: number) { | ||||||
|         learningObject: LearningObject, |  | ||||||
|         sequenceNumber: number |  | ||||||
|     ) { |  | ||||||
|         return this.findOne({ |         return this.findOne({ | ||||||
|             learningObject: learningObject, |             learningObject: learningObject, | ||||||
|             sequenceNumber: sequenceNumber, |             sequenceNumber: sequenceNumber, | ||||||
|  |  | ||||||
|  | @ -3,9 +3,7 @@ import { LearningObject } from '../../entities/content/learning-object.entity.js | ||||||
| import { LearningObjectIdentifier } from '../../entities/content/learning-object-identifier.js'; | import { LearningObjectIdentifier } from '../../entities/content/learning-object-identifier.js'; | ||||||
| 
 | 
 | ||||||
| export class LearningObjectRepository extends DwengoEntityRepository<LearningObject> { | export class LearningObjectRepository extends DwengoEntityRepository<LearningObject> { | ||||||
|     public findByIdentifier( |     public findByIdentifier(identifier: LearningObjectIdentifier): Promise<LearningObject | null> { | ||||||
|         identifier: LearningObjectIdentifier |  | ||||||
|     ): Promise<LearningObject | null> { |  | ||||||
|         return this.findOne({ |         return this.findOne({ | ||||||
|             hruid: identifier.hruid, |             hruid: identifier.hruid, | ||||||
|             language: identifier.language, |             language: identifier.language, | ||||||
|  |  | ||||||
|  | @ -3,10 +3,7 @@ import { LearningPath } from '../../entities/content/learning-path.entity.js'; | ||||||
| import { Language } from '../../entities/content/language.js'; | import { Language } from '../../entities/content/language.js'; | ||||||
| 
 | 
 | ||||||
| export class LearningPathRepository extends DwengoEntityRepository<LearningPath> { | export class LearningPathRepository extends DwengoEntityRepository<LearningPath> { | ||||||
|     public findByHruidAndLanguage( |     public findByHruidAndLanguage(hruid: string, language: Language): Promise<LearningPath | null> { | ||||||
|         hruid: string, |  | ||||||
|         language: Language |  | ||||||
|     ): Promise<LearningPath | null> { |  | ||||||
|         return this.findOne({ hruid: hruid, language: language }); |         return this.findOne({ hruid: hruid, language: language }); | ||||||
|     } |     } | ||||||
|     // This repository is read-only for now since creating own learning object is an extension feature.
 |     // This repository is read-only for now since creating own learning object is an extension feature.
 | ||||||
|  |  | ||||||
|  | @ -1,8 +1,6 @@ | ||||||
| import { EntityRepository, FilterQuery } from '@mikro-orm/core'; | import { EntityRepository, FilterQuery } from '@mikro-orm/core'; | ||||||
| 
 | 
 | ||||||
| export abstract class DwengoEntityRepository< | export abstract class DwengoEntityRepository<T extends object> extends EntityRepository<T> { | ||||||
|     T extends object, |  | ||||||
| > extends EntityRepository<T> { |  | ||||||
|     public async save(entity: T) { |     public async save(entity: T) { | ||||||
|         const em = this.getEntityManager(); |         const em = this.getEntityManager(); | ||||||
|         em.persist(entity); |         em.persist(entity); | ||||||
|  |  | ||||||
|  | @ -4,11 +4,7 @@ import { Question } from '../../entities/questions/question.entity.js'; | ||||||
| import { Teacher } from '../../entities/users/teacher.entity.js'; | import { Teacher } from '../../entities/users/teacher.entity.js'; | ||||||
| 
 | 
 | ||||||
| export class AnswerRepository extends DwengoEntityRepository<Answer> { | export class AnswerRepository extends DwengoEntityRepository<Answer> { | ||||||
|     public createAnswer(answer: { |     public createAnswer(answer: { toQuestion: Question; author: Teacher; content: string }): Promise<Answer> { | ||||||
|         toQuestion: Question; |  | ||||||
|         author: Teacher; |  | ||||||
|         content: string; |  | ||||||
|     }): Promise<Answer> { |  | ||||||
|         const answerEntity = new Answer(); |         const answerEntity = new Answer(); | ||||||
|         answerEntity.toQuestion = answer.toQuestion; |         answerEntity.toQuestion = answer.toQuestion; | ||||||
|         answerEntity.author = answer.author; |         answerEntity.author = answer.author; | ||||||
|  | @ -21,10 +17,7 @@ export class AnswerRepository extends DwengoEntityRepository<Answer> { | ||||||
|             orderBy: { sequenceNumber: 'ASC' }, |             orderBy: { sequenceNumber: 'ASC' }, | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|     public removeAnswerByQuestionAndSequenceNumber( |     public removeAnswerByQuestionAndSequenceNumber(question: Question, sequenceNumber: number): Promise<void> { | ||||||
|         question: Question, |  | ||||||
|         sequenceNumber: number |  | ||||||
|     ): Promise<void> { |  | ||||||
|         return this.deleteWhere({ |         return this.deleteWhere({ | ||||||
|             toQuestion: question, |             toQuestion: question, | ||||||
|             sequenceNumber: sequenceNumber, |             sequenceNumber: sequenceNumber, | ||||||
|  |  | ||||||
|  | @ -4,11 +4,7 @@ import { LearningObjectIdentifier } from '../../entities/content/learning-object | ||||||
| import { Student } from '../../entities/users/student.entity.js'; | import { Student } from '../../entities/users/student.entity.js'; | ||||||
| 
 | 
 | ||||||
| export class QuestionRepository extends DwengoEntityRepository<Question> { | export class QuestionRepository extends DwengoEntityRepository<Question> { | ||||||
|     public createQuestion(question: { |     public createQuestion(question: { loId: LearningObjectIdentifier; author: Student; content: string }): Promise<Question> { | ||||||
|         loId: LearningObjectIdentifier; |  | ||||||
|         author: Student; |  | ||||||
|         content: string; |  | ||||||
|     }): Promise<Question> { |  | ||||||
|         const questionEntity = new Question(); |         const questionEntity = new Question(); | ||||||
|         questionEntity.learningObjectHruid = question.loId.hruid; |         questionEntity.learningObjectHruid = question.loId.hruid; | ||||||
|         questionEntity.learningObjectLanguage = question.loId.language; |         questionEntity.learningObjectLanguage = question.loId.language; | ||||||
|  | @ -17,9 +13,7 @@ export class QuestionRepository extends DwengoEntityRepository<Question> { | ||||||
|         questionEntity.content = question.content; |         questionEntity.content = question.content; | ||||||
|         return this.insert(questionEntity); |         return this.insert(questionEntity); | ||||||
|     } |     } | ||||||
|     public findAllQuestionsAboutLearningObject( |     public findAllQuestionsAboutLearningObject(loId: LearningObjectIdentifier): Promise<Question[]> { | ||||||
|         loId: LearningObjectIdentifier |  | ||||||
|     ): Promise<Question[]> { |  | ||||||
|         return this.findAll({ |         return this.findAll({ | ||||||
|             where: { |             where: { | ||||||
|                 learningObjectHruid: loId.hruid, |                 learningObjectHruid: loId.hruid, | ||||||
|  | @ -31,10 +25,7 @@ export class QuestionRepository extends DwengoEntityRepository<Question> { | ||||||
|             }, |             }, | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|     public removeQuestionByLearningObjectAndSequenceNumber( |     public removeQuestionByLearningObjectAndSequenceNumber(loId: LearningObjectIdentifier, sequenceNumber: number): Promise<void> { | ||||||
|         loId: LearningObjectIdentifier, |  | ||||||
|         sequenceNumber: number |  | ||||||
|     ): Promise<void> { |  | ||||||
|         return this.deleteWhere({ |         return this.deleteWhere({ | ||||||
|             learningObjectHruid: loId.hruid, |             learningObjectHruid: loId.hruid, | ||||||
|             learningObjectLanguage: loId.language, |             learningObjectLanguage: loId.language, | ||||||
|  |  | ||||||
|  | @ -1,9 +1,4 @@ | ||||||
| import { | import { AnyEntity, EntityManager, EntityName, EntityRepository } from '@mikro-orm/core'; | ||||||
|     AnyEntity, |  | ||||||
|     EntityManager, |  | ||||||
|     EntityName, |  | ||||||
|     EntityRepository, |  | ||||||
| } from '@mikro-orm/core'; |  | ||||||
| import { forkEntityManager } from '../orm.js'; | import { forkEntityManager } from '../orm.js'; | ||||||
| import { StudentRepository } from './users/student-repository.js'; | import { StudentRepository } from './users/student-repository.js'; | ||||||
| import { Student } from '../entities/users/student.entity.js'; | import { Student } from '../entities/users/student.entity.js'; | ||||||
|  | @ -43,9 +38,7 @@ export function transactional<T>(f: () => Promise<T>) { | ||||||
|     entityManager?.transactional(f); |     entityManager?.transactional(f); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function repositoryGetter<T extends AnyEntity, R extends EntityRepository<T>>( | function repositoryGetter<T extends AnyEntity, R extends EntityRepository<T>>(entity: EntityName<T>): () => R { | ||||||
|     entity: EntityName<T>, |  | ||||||
| ): () => R { |  | ||||||
|     let cachedRepo: R | undefined; |     let cachedRepo: R | undefined; | ||||||
|     return (): R => { |     return (): R => { | ||||||
|         if (!cachedRepo) { |         if (!cachedRepo) { | ||||||
|  |  | ||||||
|  | @ -23,13 +23,7 @@ export const themes: Theme[] = [ | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|         title: 'art', |         title: 'art', | ||||||
|         hruids: [ |         hruids: ['pn_werking', 'un_artificiele_intelligentie', 'art1', 'art2', 'art3'], | ||||||
|             'pn_werking', |  | ||||||
|             'un_artificiele_intelligentie', |  | ||||||
|             'art1', |  | ||||||
|             'art2', |  | ||||||
|             'art3', |  | ||||||
|         ], |  | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|         title: 'socialrobot', |         title: 'socialrobot', | ||||||
|  | @ -37,12 +31,7 @@ export const themes: Theme[] = [ | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|         title: 'agriculture', |         title: 'agriculture', | ||||||
|         hruids: [ |         hruids: ['pn_werking', 'un_artificiele_intelligentie', 'agri_landbouw', 'agri_lopendeband'], | ||||||
|             'pn_werking', |  | ||||||
|             'un_artificiele_intelligentie', |  | ||||||
|             'agri_landbouw', |  | ||||||
|             'agri_lopendeband', |  | ||||||
|         ], |  | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|         title: 'wegostem', |         title: 'wegostem', | ||||||
|  | @ -83,16 +72,7 @@ export const themes: Theme[] = [ | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|         title: 'python_programming', |         title: 'python_programming', | ||||||
|         hruids: [ |         hruids: ['pn_werking', 'pn_datatypes', 'pn_operatoren', 'pn_structuren', 'pn_functies', 'art2', 'stem_insectbooks', 'un_algoenprog'], | ||||||
|             'pn_werking', |  | ||||||
|             'pn_datatypes', |  | ||||||
|             'pn_operatoren', |  | ||||||
|             'pn_structuren', |  | ||||||
|             'pn_functies', |  | ||||||
|             'art2', |  | ||||||
|             'stem_insectbooks', |  | ||||||
|             'un_algoenprog', |  | ||||||
|         ], |  | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|         title: 'stem', |         title: 'stem', | ||||||
|  | @ -110,15 +90,7 @@ export const themes: Theme[] = [ | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|         title: 'care', |         title: 'care', | ||||||
|         hruids: [ |         hruids: ['pn_werking', 'un_artificiele_intelligentie', 'aiz1_zorg', 'aiz2_grafen', 'aiz3_unplugged', 'aiz4_eindtermen', 'aiz5_triage'], | ||||||
|             'pn_werking', |  | ||||||
|             'un_artificiele_intelligentie', |  | ||||||
|             'aiz1_zorg', |  | ||||||
|             'aiz2_grafen', |  | ||||||
|             'aiz3_unplugged', |  | ||||||
|             'aiz4_eindtermen', |  | ||||||
|             'aiz5_triage', |  | ||||||
|         ], |  | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|         title: 'chatbot', |         title: 'chatbot', | ||||||
|  |  | ||||||
|  | @ -1,11 +1,4 @@ | ||||||
| import { | import { Entity, Enum, ManyToOne, OneToMany, PrimaryKey, Property } from '@mikro-orm/core'; | ||||||
|     Entity, |  | ||||||
|     Enum, |  | ||||||
|     ManyToOne, |  | ||||||
|     OneToMany, |  | ||||||
|     PrimaryKey, |  | ||||||
|     Property, |  | ||||||
| } from '@mikro-orm/core'; |  | ||||||
| import { Class } from '../classes/class.entity.js'; | import { Class } from '../classes/class.entity.js'; | ||||||
| import { Group } from './group.entity.js'; | import { Group } from './group.entity.js'; | ||||||
| import { Language } from '../content/language.js'; | import { Language } from '../content/language.js'; | ||||||
|  |  | ||||||
|  | @ -1,10 +1,4 @@ | ||||||
| import { | import { Collection, Entity, ManyToMany, PrimaryKey, Property } from '@mikro-orm/core'; | ||||||
|     Collection, |  | ||||||
|     Entity, |  | ||||||
|     ManyToMany, |  | ||||||
|     PrimaryKey, |  | ||||||
|     Property, |  | ||||||
| } from '@mikro-orm/core'; |  | ||||||
| import { v4 } from 'uuid'; | import { v4 } from 'uuid'; | ||||||
| import { Teacher } from '../users/teacher.entity.js'; | import { Teacher } from '../users/teacher.entity.js'; | ||||||
| import { Student } from '../users/student.entity.js'; | import { Student } from '../users/student.entity.js'; | ||||||
|  |  | ||||||
|  | @ -1,13 +1,4 @@ | ||||||
| import { | import { Embeddable, Embedded, Entity, Enum, ManyToMany, OneToMany, PrimaryKey, Property } from '@mikro-orm/core'; | ||||||
|     Embeddable, |  | ||||||
|     Embedded, |  | ||||||
|     Entity, |  | ||||||
|     Enum, |  | ||||||
|     ManyToMany, |  | ||||||
|     OneToMany, |  | ||||||
|     PrimaryKey, |  | ||||||
|     Property, |  | ||||||
| } from '@mikro-orm/core'; |  | ||||||
| import { Language } from './language.js'; | import { Language } from './language.js'; | ||||||
| import { Attachment } from './attachment.entity.js'; | import { Attachment } from './attachment.entity.js'; | ||||||
| import { Teacher } from '../users/teacher.entity.js'; | import { Teacher } from '../users/teacher.entity.js'; | ||||||
|  |  | ||||||
|  | @ -1,13 +1,4 @@ | ||||||
| import { | import { Embeddable, Embedded, Entity, Enum, ManyToMany, OneToOne, PrimaryKey, Property } from '@mikro-orm/core'; | ||||||
|     Embeddable, |  | ||||||
|     Embedded, |  | ||||||
|     Entity, |  | ||||||
|     Enum, |  | ||||||
|     ManyToMany, |  | ||||||
|     OneToOne, |  | ||||||
|     PrimaryKey, |  | ||||||
|     Property, |  | ||||||
| } from '@mikro-orm/core'; |  | ||||||
| import { Language } from './language.js'; | import { Language } from './language.js'; | ||||||
| import { Teacher } from '../users/teacher.entity.js'; | import { Teacher } from '../users/teacher.entity.js'; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,9 +1,4 @@ | ||||||
| import { | import { createLogger, format, Logger as WinstonLogger, transports } from 'winston'; | ||||||
|     createLogger, |  | ||||||
|     format, |  | ||||||
|     Logger as WinstonLogger, |  | ||||||
|     transports, |  | ||||||
| } from 'winston'; |  | ||||||
| import LokiTransport from 'winston-loki'; | import LokiTransport from 'winston-loki'; | ||||||
| import { LokiLabels } from 'loki-logger-ts'; | import { LokiLabels } from 'loki-logger-ts'; | ||||||
| import { LOG_LEVEL, LOKI_HOST } from '../config.js'; | import { LOG_LEVEL, LOKI_HOST } from '../config.js'; | ||||||
|  | @ -48,9 +43,7 @@ function initializeLogger(): Logger { | ||||||
|         transports: [lokiTransport, consoleTransport], |         transports: [lokiTransport, consoleTransport], | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     logger.debug( |     logger.debug(`Logger initialized with level ${LOG_LEVEL}, Loki host ${LOKI_HOST}`); | ||||||
|         `Logger initialized with level ${LOG_LEVEL}, Loki host ${LOKI_HOST}` |  | ||||||
|     ); |  | ||||||
|     return logger; |     return logger; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -12,42 +12,28 @@ export class MikroOrmLogger extends DefaultLogger { | ||||||
| 
 | 
 | ||||||
|         switch (namespace) { |         switch (namespace) { | ||||||
|             case 'query': |             case 'query': | ||||||
|                 this.logger.debug( |                 this.logger.debug(this.createMessage(namespace, message, context)); | ||||||
|                     this.createMessage(namespace, message, context) |  | ||||||
|                 ); |  | ||||||
|                 break; |                 break; | ||||||
|             case 'query-params': |             case 'query-params': | ||||||
|                 // TODO Which log level should this be?
 |                 // TODO Which log level should this be?
 | ||||||
|                 this.logger.info( |                 this.logger.info(this.createMessage(namespace, message, context)); | ||||||
|                     this.createMessage(namespace, message, context) |  | ||||||
|                 ); |  | ||||||
|                 break; |                 break; | ||||||
|             case 'schema': |             case 'schema': | ||||||
|                 this.logger.info( |                 this.logger.info(this.createMessage(namespace, message, context)); | ||||||
|                     this.createMessage(namespace, message, context) |  | ||||||
|                 ); |  | ||||||
|                 break; |                 break; | ||||||
|             case 'discovery': |             case 'discovery': | ||||||
|                 this.logger.debug( |                 this.logger.debug(this.createMessage(namespace, message, context)); | ||||||
|                     this.createMessage(namespace, message, context) |  | ||||||
|                 ); |  | ||||||
|                 break; |                 break; | ||||||
|             case 'info': |             case 'info': | ||||||
|                 this.logger.info( |                 this.logger.info(this.createMessage(namespace, message, context)); | ||||||
|                     this.createMessage(namespace, message, context) |  | ||||||
|                 ); |  | ||||||
|                 break; |                 break; | ||||||
|             case 'deprecated': |             case 'deprecated': | ||||||
|                 this.logger.warn( |                 this.logger.warn(this.createMessage(namespace, message, context)); | ||||||
|                     this.createMessage(namespace, message, context) |  | ||||||
|                 ); |  | ||||||
|                 break; |                 break; | ||||||
|             default: |             default: | ||||||
|                 switch (context?.level) { |                 switch (context?.level) { | ||||||
|                     case 'info': |                     case 'info': | ||||||
|                         this.logger.info( |                         this.logger.info(this.createMessage(namespace, message, context)); | ||||||
|                             this.createMessage(namespace, message, context) |  | ||||||
|                         ); |  | ||||||
|                         break; |                         break; | ||||||
|                     case 'warning': |                     case 'warning': | ||||||
|                         this.logger.warn(message); |                         this.logger.warn(message); | ||||||
|  | @ -62,11 +48,7 @@ export class MikroOrmLogger extends DefaultLogger { | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private createMessage( |     private createMessage(namespace: LoggerNamespace, messageArg: string, context?: LogContext) { | ||||||
|         namespace: LoggerNamespace, |  | ||||||
|         messageArg: string, |  | ||||||
|         context?: LogContext |  | ||||||
|     ) { |  | ||||||
|         const labels: LokiLabels = { |         const labels: LokiLabels = { | ||||||
|             service: 'ORM', |             service: 'ORM', | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|  | @ -28,9 +28,7 @@ export async function initORM(testingMode: boolean = false) { | ||||||
| } | } | ||||||
| export function forkEntityManager(): EntityManager { | export function forkEntityManager(): EntityManager { | ||||||
|     if (!orm) { |     if (!orm) { | ||||||
|         throw Error( |         throw Error('Accessing the Entity Manager before the ORM is fully initialized.'); | ||||||
|             'Accessing the Entity Manager before the ORM is fully initialized.' |  | ||||||
|         ); |  | ||||||
|     } |     } | ||||||
|     return orm.em.fork(); |     return orm.em.fork(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,8 +1,5 @@ | ||||||
| import express from 'express'; | import express from 'express'; | ||||||
| import { | import { getAllLearningObjects, getLearningObject } from '../controllers/learningObjects.js'; | ||||||
|     getAllLearningObjects, |  | ||||||
|     getLearningObject, |  | ||||||
| } from '../controllers/learningObjects.js'; |  | ||||||
| 
 | 
 | ||||||
| const router = express.Router(); | const router = express.Router(); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -15,8 +15,7 @@ router.get('/:id', (req, res) => { | ||||||
|         student: '0', |         student: '0', | ||||||
|         group: '0', |         group: '0', | ||||||
|         time: new Date(2025, 1, 1), |         time: new Date(2025, 1, 1), | ||||||
|         content: |         content: 'Zijn alle gehele getallen groter dan 2 gelijk aan de som van 2 priemgetallen????', | ||||||
|             'Zijn alle gehele getallen groter dan 2 gelijk aan de som van 2 priemgetallen????', |  | ||||||
|         learningObject: '0', |         learningObject: '0', | ||||||
|         links: { |         links: { | ||||||
|             self: `${req.baseUrl}/${req.params.id}`, |             self: `${req.baseUrl}/${req.params.id}`, | ||||||
|  |  | ||||||
|  | @ -1,20 +1,12 @@ | ||||||
| import { DWENGO_API_BASE } from '../config.js'; | import { DWENGO_API_BASE } from '../config.js'; | ||||||
| import { fetchWithLogging } from '../util/apiHelper.js'; | import { fetchWithLogging } from '../util/apiHelper.js'; | ||||||
| import { | import { FilteredLearningObject, LearningObjectMetadata, LearningObjectNode, LearningPathResponse } from '../interfaces/learningPath.js'; | ||||||
|     FilteredLearningObject, |  | ||||||
|     LearningObjectMetadata, |  | ||||||
|     LearningObjectNode, |  | ||||||
|     LearningPathResponse, |  | ||||||
| } from '../interfaces/learningPath.js'; |  | ||||||
| import { fetchLearningPaths } from './learningPaths.js'; | import { fetchLearningPaths } from './learningPaths.js'; | ||||||
| import { getLogger, Logger } from '../logging/initalize.js'; | import { getLogger, Logger } from '../logging/initalize.js'; | ||||||
| 
 | 
 | ||||||
| const logger: Logger = getLogger(); | const logger: Logger = getLogger(); | ||||||
| 
 | 
 | ||||||
| function filterData( | function filterData(data: LearningObjectMetadata, htmlUrl: string): FilteredLearningObject { | ||||||
|     data: LearningObjectMetadata, |  | ||||||
|     htmlUrl: string |  | ||||||
| ): FilteredLearningObject { |  | ||||||
|     return { |     return { | ||||||
|         key: data.hruid, // Hruid learningObject (not path)
 |         key: data.hruid, // Hruid learningObject (not path)
 | ||||||
|         _id: data._id, |         _id: data._id, | ||||||
|  | @ -41,10 +33,7 @@ function filterData( | ||||||
| /** | /** | ||||||
|  * Fetches a single learning object by its HRUID |  * Fetches a single learning object by its HRUID | ||||||
|  */ |  */ | ||||||
| export async function getLearningObjectById( | export async function getLearningObjectById(hruid: string, language: string): Promise<FilteredLearningObject | null> { | ||||||
|     hruid: string, |  | ||||||
|     language: string |  | ||||||
| ): Promise<FilteredLearningObject | null> { |  | ||||||
|     const metadataUrl = `${DWENGO_API_BASE}/learningObject/getMetadata?hruid=${hruid}&language=${language}`; |     const metadataUrl = `${DWENGO_API_BASE}/learningObject/getMetadata?hruid=${hruid}&language=${language}`; | ||||||
|     const metadata = await fetchWithLogging<LearningObjectMetadata>( |     const metadata = await fetchWithLogging<LearningObjectMetadata>( | ||||||
|         metadataUrl, |         metadataUrl, | ||||||
|  | @ -63,26 +52,12 @@ export async function getLearningObjectById( | ||||||
| /** | /** | ||||||
|  * Generic function to fetch learning objects (full data or just HRUIDs) |  * Generic function to fetch learning objects (full data or just HRUIDs) | ||||||
|  */ |  */ | ||||||
| async function fetchLearningObjects( | async function fetchLearningObjects(hruid: string, full: boolean, language: string): Promise<FilteredLearningObject[] | string[]> { | ||||||
|     hruid: string, |  | ||||||
|     full: boolean, |  | ||||||
|     language: string |  | ||||||
| ): Promise<FilteredLearningObject[] | string[]> { |  | ||||||
|     try { |     try { | ||||||
|         const learningPathResponse: LearningPathResponse = |         const learningPathResponse: LearningPathResponse = await fetchLearningPaths([hruid], language, `Learning path for HRUID "${hruid}"`); | ||||||
|             await fetchLearningPaths( |  | ||||||
|                 [hruid], |  | ||||||
|                 language, |  | ||||||
|                 `Learning path for HRUID "${hruid}"` |  | ||||||
|             ); |  | ||||||
| 
 | 
 | ||||||
|         if ( |         if (!learningPathResponse.success || !learningPathResponse.data?.length) { | ||||||
|             !learningPathResponse.success || |             logger.warn(`⚠️ WARNING: Learning path "${hruid}" exists but contains no learning objects.`); | ||||||
|             !learningPathResponse.data?.length |  | ||||||
|         ) { |  | ||||||
|             logger.warn( |  | ||||||
|                 `⚠️ WARNING: Learning path "${hruid}" exists but contains no learning objects.` |  | ||||||
|             ); |  | ||||||
|             return []; |             return []; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -92,12 +67,9 @@ async function fetchLearningObjects( | ||||||
|             return nodes.map((node) => node.learningobject_hruid); |             return nodes.map((node) => node.learningobject_hruid); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         return await Promise.all( |         return await Promise.all(nodes.map(async (node) => getLearningObjectById(node.learningobject_hruid, language))).then((objects) => | ||||||
|             nodes.map(async (node) => getLearningObjectById( |             objects.filter((obj): obj is FilteredLearningObject => obj !== null) | ||||||
|                     node.learningobject_hruid, |         ); | ||||||
|                     language |  | ||||||
|                 )) |  | ||||||
|         ).then((objects) => objects.filter((obj): obj is FilteredLearningObject => obj !== null)); |  | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|         logger.error('❌ Error fetching learning objects:', error); |         logger.error('❌ Error fetching learning objects:', error); | ||||||
|         return []; |         return []; | ||||||
|  | @ -107,23 +79,13 @@ async function fetchLearningObjects( | ||||||
| /** | /** | ||||||
|  * Fetch full learning object data (metadata) |  * Fetch full learning object data (metadata) | ||||||
|  */ |  */ | ||||||
| export async function getLearningObjectsFromPath( | export async function getLearningObjectsFromPath(hruid: string, language: string): Promise<FilteredLearningObject[]> { | ||||||
|     hruid: string, |     return (await fetchLearningObjects(hruid, true, language)) as FilteredLearningObject[]; | ||||||
|     language: string |  | ||||||
| ): Promise<FilteredLearningObject[]> { |  | ||||||
|     return (await fetchLearningObjects( |  | ||||||
|         hruid, |  | ||||||
|         true, |  | ||||||
|         language |  | ||||||
|     )) as FilteredLearningObject[]; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Fetch only learning object HRUIDs |  * Fetch only learning object HRUIDs | ||||||
|  */ |  */ | ||||||
| export async function getLearningObjectIdsFromPath( | export async function getLearningObjectIdsFromPath(hruid: string, language: string): Promise<string[]> { | ||||||
|     hruid: string, |  | ||||||
|     language: string |  | ||||||
| ): Promise<string[]> { |  | ||||||
|     return (await fetchLearningObjects(hruid, false, language)) as string[]; |     return (await fetchLearningObjects(hruid, false, language)) as string[]; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,18 +1,11 @@ | ||||||
| import { fetchWithLogging } from '../util/apiHelper.js'; | import { fetchWithLogging } from '../util/apiHelper.js'; | ||||||
| import { DWENGO_API_BASE } from '../config.js'; | import { DWENGO_API_BASE } from '../config.js'; | ||||||
| import { | import { LearningPath, LearningPathResponse } from '../interfaces/learningPath.js'; | ||||||
|     LearningPath, |  | ||||||
|     LearningPathResponse, |  | ||||||
| } from '../interfaces/learningPath.js'; |  | ||||||
| import { getLogger, Logger } from '../logging/initalize.js'; | import { getLogger, Logger } from '../logging/initalize.js'; | ||||||
| 
 | 
 | ||||||
| const logger: Logger = getLogger(); | const logger: Logger = getLogger(); | ||||||
| 
 | 
 | ||||||
| export async function fetchLearningPaths( | export async function fetchLearningPaths(hruids: string[], language: string, source: string): Promise<LearningPathResponse> { | ||||||
|     hruids: string[], |  | ||||||
|     language: string, |  | ||||||
|     source: string |  | ||||||
| ): Promise<LearningPathResponse> { |  | ||||||
|     if (hruids.length === 0) { |     if (hruids.length === 0) { | ||||||
|         return { |         return { | ||||||
|             success: false, |             success: false, | ||||||
|  | @ -25,11 +18,7 @@ export async function fetchLearningPaths( | ||||||
|     const apiUrl = `${DWENGO_API_BASE}/learningPath/getPathsFromIdList`; |     const apiUrl = `${DWENGO_API_BASE}/learningPath/getPathsFromIdList`; | ||||||
|     const params = { pathIdList: JSON.stringify({ hruids }), language }; |     const params = { pathIdList: JSON.stringify({ hruids }), language }; | ||||||
| 
 | 
 | ||||||
|     const learningPaths = await fetchWithLogging<LearningPath[]>( |     const learningPaths = await fetchWithLogging<LearningPath[]>(apiUrl, `Learning paths for ${source}`, params); | ||||||
|         apiUrl, |  | ||||||
|         `Learning paths for ${source}`, |  | ||||||
|         params |  | ||||||
|     ); |  | ||||||
| 
 | 
 | ||||||
|     if (!learningPaths || learningPaths.length === 0) { |     if (!learningPaths || learningPaths.length === 0) { | ||||||
|         logger.warn(`⚠️ WARNING: No learning paths found for ${source}.`); |         logger.warn(`⚠️ WARNING: No learning paths found for ${source}.`); | ||||||
|  | @ -48,17 +37,10 @@ export async function fetchLearningPaths( | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export async function searchLearningPaths( | export async function searchLearningPaths(query: string, language: string): Promise<LearningPath[]> { | ||||||
|     query: string, |  | ||||||
|     language: string |  | ||||||
| ): 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[]>( |     const searchResults = await fetchWithLogging<LearningPath[]>(apiUrl, `Search learning paths with query "${query}"`, params); | ||||||
|         apiUrl, |  | ||||||
|         `Search learning paths with query "${query}"`, |  | ||||||
|         params |  | ||||||
|     ); |  | ||||||
|     return searchResults ?? []; |     return searchResults ?? []; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -12,11 +12,7 @@ const logger: Logger = getLogger(); | ||||||
|  * @param params |  * @param params | ||||||
|  * @returns The response data if successful, or null if an error occurs. |  * @returns The response data if successful, or null if an error occurs. | ||||||
|  */ |  */ | ||||||
| export async function fetchWithLogging<T>( | export async function fetchWithLogging<T>(url: string, description: string, params?: Record<string, any>): Promise<T | null> { | ||||||
|     url: string, |  | ||||||
|     description: string, |  | ||||||
|     params?: Record<string, any> |  | ||||||
| ): Promise<T | null> { |  | ||||||
|     try { |     try { | ||||||
|         const config: AxiosRequestConfig = params ? { params } : {}; |         const config: AxiosRequestConfig = params ? { params } : {}; | ||||||
| 
 | 
 | ||||||
|  | @ -25,19 +21,14 @@ export async function fetchWithLogging<T>( | ||||||
|     } catch (error: any) { |     } catch (error: any) { | ||||||
|         if (error.response) { |         if (error.response) { | ||||||
|             if (error.response.status === 404) { |             if (error.response.status === 404) { | ||||||
|                 logger.debug( |                 logger.debug(`❌ ERROR: ${description} not found (404) at "${url}".`); | ||||||
|                     `❌ ERROR: ${description} not found (404) at "${url}".` |  | ||||||
|                 ); |  | ||||||
|             } else { |             } else { | ||||||
|                 logger.debug( |                 logger.debug( | ||||||
|                     `❌ ERROR: Failed to fetch ${description}. Status: ${error.response.status} - ${error.response.statusText} (URL: "${url}")` |                     `❌ ERROR: Failed to fetch ${description}. Status: ${error.response.status} - ${error.response.statusText} (URL: "${url}")` | ||||||
|                 ); |                 ); | ||||||
|             } |             } | ||||||
|         } else { |         } else { | ||||||
|             logger.debug( |             logger.debug(`❌ ERROR: Network or unexpected error when fetching ${description}:`, error.message); | ||||||
|                 `❌ ERROR: Network or unexpected error when fetching ${description}:`, |  | ||||||
|                 error.message |  | ||||||
|             ); |  | ||||||
|         } |         } | ||||||
|         return null; |         return null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -36,9 +36,7 @@ export function getNumericEnvVar(envVar: EnvVar): number { | ||||||
|     const valueString = getEnvVar(envVar); |     const valueString = getEnvVar(envVar); | ||||||
|     const value = parseInt(valueString); |     const value = parseInt(valueString); | ||||||
|     if (isNaN(value)) { |     if (isNaN(value)) { | ||||||
|         throw new Error( |         throw new Error(`Invalid value for environment variable ${envVar.key}: ${valueString}. Expected a number.`); | ||||||
|             `Invalid value for environment variable ${envVar.key}: ${valueString}. Expected a number.` |  | ||||||
|         ); |  | ||||||
|     } else { |     } else { | ||||||
|         return value; |         return value; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -12,15 +12,8 @@ export function loadTranslations<T>(language: string): T { | ||||||
|         const yamlFile = fs.readFileSync(filePath, 'utf8'); |         const yamlFile = fs.readFileSync(filePath, 'utf8'); | ||||||
|         return yaml.load(yamlFile) as T; |         return yaml.load(yamlFile) as T; | ||||||
|     } catch (error) { |     } catch (error) { | ||||||
|         logger.warn( |         logger.warn(`Cannot load translation for ${language}, fallen back to dutch`, error); | ||||||
|             `Cannot load translation for ${language}, fallen back to dutch`, |         const fallbackPath = path.join(process.cwd(), '_i18n', `${FALLBACK_LANG}.yml`); | ||||||
|             error |  | ||||||
|         ); |  | ||||||
|         const fallbackPath = path.join( |  | ||||||
|             process.cwd(), |  | ||||||
|             '_i18n', |  | ||||||
|             `${FALLBACK_LANG}.yml` |  | ||||||
|         ); |  | ||||||
|         return yaml.load(fs.readFileSync(fallbackPath, 'utf8')) as T; |         return yaml.load(fs.readFileSync(fallbackPath, 'utf8')) as T; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -16,12 +16,9 @@ describe('StudentRepository', () => { | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     it('should return the queried student after he was added', async () => { |     it('should return the queried student after he was added', async () => { | ||||||
|         await studentRepository.insert( |         await studentRepository.insert(new Student(username, firstName, lastName)); | ||||||
|             new Student(username, firstName, lastName) |  | ||||||
|         ); |  | ||||||
| 
 | 
 | ||||||
|         const retrievedStudent = |         const retrievedStudent = await studentRepository.findByUsername(username); | ||||||
|             await studentRepository.findByUsername(username); |  | ||||||
|         expect(retrievedStudent).toBeTruthy(); |         expect(retrievedStudent).toBeTruthy(); | ||||||
|         expect(retrievedStudent?.firstName).toBe(firstName); |         expect(retrievedStudent?.firstName).toBe(firstName); | ||||||
|         expect(retrievedStudent?.lastName).toBe(lastName); |         expect(retrievedStudent?.lastName).toBe(lastName); | ||||||
|  | @ -30,8 +27,7 @@ describe('StudentRepository', () => { | ||||||
|     it('should no longer return the queried student after he was removed again', async () => { |     it('should no longer return the queried student after he was removed again', async () => { | ||||||
|         await studentRepository.deleteByUsername(username); |         await studentRepository.deleteByUsername(username); | ||||||
| 
 | 
 | ||||||
|         const retrievedStudent = |         const retrievedStudent = await studentRepository.findByUsername(username); | ||||||
|             await studentRepository.findByUsername(username); |  | ||||||
|         expect(retrievedStudent).toBeNull(); |         expect(retrievedStudent).toBeNull(); | ||||||
|     }); |     }); | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  | @ -16,12 +16,7 @@ export default [ | ||||||
|     prettierConfig, |     prettierConfig, | ||||||
|     includeIgnoreFile(gitignorePath), |     includeIgnoreFile(gitignorePath), | ||||||
|     { |     { | ||||||
|         ignores: [ |         ignores: ['**/dist/**', '**/.node_modules/**', '**/coverage/**', '**/.github/**'], | ||||||
|             '**/dist/**', |  | ||||||
|             '**/.node_modules/**', |  | ||||||
|             '**/coverage/**', |  | ||||||
|             '**/.github/**', |  | ||||||
|         ], |  | ||||||
|         files: ['**/*.ts', '**/*.cts', '**.*.mts', '**/*.ts'], |         files: ['**/*.ts', '**/*.cts', '**.*.mts', '**/*.ts'], | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|  |  | ||||||
|  | @ -15,8 +15,8 @@ const vueConfig = defineConfigWithVueTs( | ||||||
|         name: "app/files-to-lint", |         name: "app/files-to-lint", | ||||||
|         files: ["**/*.{ts,mts,tsx,vue}"], |         files: ["**/*.{ts,mts,tsx,vue}"], | ||||||
|         rules: { |         rules: { | ||||||
|             'no-useless-assignment': 'off' // Depend on `no-unused-vars` to catch this
 |             "no-useless-assignment": "off", // Depend on `no-unused-vars` to catch this
 | ||||||
|         } |         }, | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     { |     { | ||||||
|  |  | ||||||
		Reference in a new issue
	
	 Lint Action
						Lint Action