feat: answer routes en put question
This commit is contained in:
		
							parent
							
								
									6a1adb0ee3
								
							
						
					
					
						commit
						e8e2466b76
					
				
					 12 changed files with 288 additions and 62 deletions
				
			
		
							
								
								
									
										101
									
								
								backend/src/controllers/answers.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								backend/src/controllers/answers.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,101 @@ | ||||||
|  | import {Request, Response} from "express"; | ||||||
|  | import {requireFields} from "./error-helper"; | ||||||
|  | import {getLearningObjectId, getQuestionId} from "./questions"; | ||||||
|  | import {createAnswer, deleteAnswer, getAnswer, getAnswersByQuestion, updateAnswer} from "../services/answers"; | ||||||
|  | import {FALLBACK_SEQ_NUM} from "../config"; | ||||||
|  | import {AnswerData} from "@dwengo-1/common/interfaces/answer"; | ||||||
|  | 
 | ||||||
|  | export async function getAnswersHandler(req: Request, res: Response): Promise<void> { | ||||||
|  |     const hruid = req.params.hruid; | ||||||
|  |     const version = req.params.version; | ||||||
|  |     const language = req.query.lang as string; | ||||||
|  |     const seq = req.params.seq; | ||||||
|  |     const full = req.query.full === 'true'; | ||||||
|  |     requireFields({ hruid }) | ||||||
|  | 
 | ||||||
|  |     const learningObjectId = getLearningObjectId(hruid, version, language); | ||||||
|  |     const questionId = getQuestionId(learningObjectId, seq); | ||||||
|  | 
 | ||||||
|  |     const answers = await getAnswersByQuestion(questionId, full); | ||||||
|  | 
 | ||||||
|  |     res.json({ answers }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function getAnswerHandler(req: Request, res: Response): Promise<void>{ | ||||||
|  |     const hruid = req.params.hruid; | ||||||
|  |     const version = req.params.version; | ||||||
|  |     const language = req.query.lang as string; | ||||||
|  |     const seq = req.params.seq; | ||||||
|  |     const seqAnswer = req.params.seqAnswer; | ||||||
|  |     requireFields({ hruid }) | ||||||
|  | 
 | ||||||
|  |     const learningObjectId = getLearningObjectId(hruid, version, language); | ||||||
|  |     const questionId = getQuestionId(learningObjectId, seq); | ||||||
|  | 
 | ||||||
|  |     const sequenceNumber = Number(seqAnswer) || FALLBACK_SEQ_NUM; | ||||||
|  |     const answer = await getAnswer(questionId, sequenceNumber); | ||||||
|  | 
 | ||||||
|  |     res.json({ answer }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function createAnswerHandler(req: Request, res: Response): Promise<void> { | ||||||
|  |     const hruid = req.params.hruid; | ||||||
|  |     const version = req.params.version; | ||||||
|  |     const language = req.query.lang as string; | ||||||
|  |     const seq = req.params.seq; | ||||||
|  |     requireFields({ hruid }) | ||||||
|  | 
 | ||||||
|  |     const learningObjectId = getLearningObjectId(hruid, version, language); | ||||||
|  |     const questionId = getQuestionId(learningObjectId, seq); | ||||||
|  | 
 | ||||||
|  |     const author = req.body.author as string; | ||||||
|  |     const content = req.body.content as string; | ||||||
|  |     requireFields({ author, content }); | ||||||
|  | 
 | ||||||
|  |     const answerData = req.body as AnswerData; | ||||||
|  | 
 | ||||||
|  |     const answer = await createAnswer(questionId, answerData); | ||||||
|  | 
 | ||||||
|  |     res.json({ answer }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function deleteAnswerHandler(req: Request, res: Response): Promise<void> { | ||||||
|  |     const hruid = req.params.hruid; | ||||||
|  |     const version = req.params.version; | ||||||
|  |     const language = req.query.lang as string; | ||||||
|  |     const seq = req.params.seq; | ||||||
|  |     const seqAnswer = req.params.seqAnswer; | ||||||
|  |     requireFields({ hruid }) | ||||||
|  | 
 | ||||||
|  |     const learningObjectId = getLearningObjectId(hruid, version, language); | ||||||
|  |     const questionId = getQuestionId(learningObjectId, seq); | ||||||
|  | 
 | ||||||
|  |     const sequenceNumber = Number(seqAnswer) || FALLBACK_SEQ_NUM; | ||||||
|  |     const answer = await deleteAnswer(questionId, sequenceNumber); | ||||||
|  | 
 | ||||||
|  |     res.json({ answer }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | export async function updateAnswerHandler(req: Request, res: Response): Promise<void> { | ||||||
|  |     const hruid = req.params.hruid; | ||||||
|  |     const version = req.params.version; | ||||||
|  |     const language = req.query.lang as string; | ||||||
|  |     const seq = req.params.seq; | ||||||
|  |     const seqAnswer = req.params.seqAnswer; | ||||||
|  |     requireFields({ hruid }) | ||||||
|  | 
 | ||||||
|  |     const learningObjectId = getLearningObjectId(hruid, version, language); | ||||||
|  |     const questionId = getQuestionId(learningObjectId, seq); | ||||||
|  | 
 | ||||||
|  |     const content = req.body.content as string; | ||||||
|  |     requireFields({ content }); | ||||||
|  | 
 | ||||||
|  |     const answerData = req.body as AnswerData; | ||||||
|  | 
 | ||||||
|  |     const sequenceNumber = Number(seqAnswer) || FALLBACK_SEQ_NUM; | ||||||
|  |     const answer = await updateAnswer(questionId, sequenceNumber, answerData); | ||||||
|  | 
 | ||||||
|  |     res.json({ answer }); | ||||||
|  | } | ||||||
|  | @ -1,12 +1,17 @@ | ||||||
| import { Request, Response } from 'express'; | import { Request, Response } from 'express'; | ||||||
| import { createQuestion, deleteQuestion, getAllQuestions, getAnswersByQuestion, getQuestion } from '../services/questions.js'; | import { | ||||||
|  |     createQuestion, | ||||||
|  |     deleteQuestion, | ||||||
|  |     getAllQuestions, | ||||||
|  |     getQuestion, updateQuestion | ||||||
|  | } from '../services/questions.js'; | ||||||
| import {FALLBACK_LANG, FALLBACK_SEQ_NUM, FALLBACK_VERSION_NUM} from '../config.js'; | import {FALLBACK_LANG, FALLBACK_SEQ_NUM, FALLBACK_VERSION_NUM} from '../config.js'; | ||||||
| import { LearningObjectIdentifier } from '../entities/content/learning-object-identifier.js'; | import { LearningObjectIdentifier } from '../entities/content/learning-object-identifier.js'; | ||||||
| import { QuestionDTO, QuestionId } from '@dwengo-1/common/interfaces/question'; | import {QuestionData, QuestionId} from '@dwengo-1/common/interfaces/question'; | ||||||
| import { Language } from '@dwengo-1/common/util/language'; | import { Language } from '@dwengo-1/common/util/language'; | ||||||
| import {requireFields} from "./error-helper"; | import {requireFields} from "./error-helper"; | ||||||
| 
 | 
 | ||||||
| function getLearningObjectId(hruid: string, version: string, lang: string): LearningObjectIdentifier { | export function getLearningObjectId(hruid: string, version: string, lang: string): LearningObjectIdentifier { | ||||||
|     return { |     return { | ||||||
|         hruid, |         hruid, | ||||||
|         language: (lang || FALLBACK_LANG) as Language, |         language: (lang || FALLBACK_LANG) as Language, | ||||||
|  | @ -14,7 +19,7 @@ function getLearningObjectId(hruid: string, version: string, lang: string): Lear | ||||||
|     }; |     }; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function getQuestionId(learningObjectIdentifier: LearningObjectIdentifier, seq: string): QuestionId { | export function getQuestionId(learningObjectIdentifier: LearningObjectIdentifier, seq: string): QuestionId { | ||||||
|     return { |     return { | ||||||
|         learningObjectIdentifier, |         learningObjectIdentifier, | ||||||
|         sequenceNumber: seq ? Number(seq) : FALLBACK_SEQ_NUM, |         sequenceNumber: seq ? Number(seq) : FALLBACK_SEQ_NUM, | ||||||
|  | @ -51,31 +56,21 @@ export async function getQuestionHandler(req: Request, res: Response): Promise<v | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export async function getQuestionAnswersHandler(req: Request, res: Response): Promise<void> { | export async function createQuestionHandler(req: Request, res: Response): Promise<void> { | ||||||
|     const hruid = req.params.hruid; |     const hruid = req.params.hruid; | ||||||
|     const version = req.params.version; |     const version = req.params.version; | ||||||
|     const language = req.query.lang as string; |     const language = req.query.lang as string; | ||||||
|     const seq = req.params.seq; |  | ||||||
|     const full = req.query.full === 'true'; |  | ||||||
|     requireFields({ hruid }) |     requireFields({ hruid }) | ||||||
| 
 | 
 | ||||||
|     const learningObjectId = getLearningObjectId(hruid, version, language); |     const loId = getLearningObjectId(hruid, version, language); | ||||||
|     const questionId = getQuestionId(learningObjectId, seq); |  | ||||||
| 
 | 
 | ||||||
|     const answers = await getAnswersByQuestion(questionId, full); |     const author = req.body.author as string; | ||||||
|  |     const content = req.body.content as string; | ||||||
|  |     requireFields({ author, content }); | ||||||
| 
 | 
 | ||||||
|     res.json({ answers }); |     const questionData = req.body as QuestionData; | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| export async function createQuestionHandler(req: Request, res: Response): Promise<void> { |     const question = await createQuestion(loId, questionData); | ||||||
|     const learningObjectIdentifier = req.body.learningObjectIdentifier; |  | ||||||
|     const author = req.body.author; |  | ||||||
|     const content = req.body.content; |  | ||||||
|     requireFields({ learningObjectIdentifier, author, content }); |  | ||||||
| 
 |  | ||||||
|     const questionDTO = req.body as QuestionDTO; |  | ||||||
| 
 |  | ||||||
|     const question = await createQuestion(questionDTO); |  | ||||||
| 
 | 
 | ||||||
|     res.json({ question }); |     res.json({ question }); | ||||||
| 
 | 
 | ||||||
|  | @ -95,3 +90,23 @@ export async function deleteQuestionHandler(req: Request, res: Response): Promis | ||||||
| 
 | 
 | ||||||
|     res.json({ question }); |     res.json({ question }); | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | export async function updateQuestionHandler(req: Request, res: Response): Promise<void> { | ||||||
|  |     const hruid = req.params.hruid; | ||||||
|  |     const version = req.params.version; | ||||||
|  |     const language = req.query.lang as string; | ||||||
|  |     const seq = req.params.seq; | ||||||
|  |     requireFields({ hruid }) | ||||||
|  | 
 | ||||||
|  |     const learningObjectId = getLearningObjectId(hruid, version, language); | ||||||
|  |     const questionId = getQuestionId(learningObjectId, seq); | ||||||
|  | 
 | ||||||
|  |     const content = req.body.content as string; | ||||||
|  |     requireFields({ content }); | ||||||
|  | 
 | ||||||
|  |     const questionData = req.body as QuestionData; | ||||||
|  | 
 | ||||||
|  |     const question = await updateQuestion(questionId, questionData); | ||||||
|  | 
 | ||||||
|  |     res.json({ question }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @ -19,10 +19,20 @@ export class AnswerRepository extends DwengoEntityRepository<Answer> { | ||||||
|             orderBy: { sequenceNumber: 'ASC' }, |             orderBy: { sequenceNumber: 'ASC' }, | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |     public async findAnswer(question: Question, sequenceNumber: number) { | ||||||
|  |         return this.findOne({ | ||||||
|  |             toQuestion: question, sequenceNumber | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|     public async removeAnswerByQuestionAndSequenceNumber(question: Question, sequenceNumber: number): Promise<void> { |     public async removeAnswerByQuestionAndSequenceNumber(question: Question, sequenceNumber: number): Promise<void> { | ||||||
|         return this.deleteWhere({ |         return this.deleteWhere({ | ||||||
|             toQuestion: question, |             toQuestion: question, | ||||||
|             sequenceNumber: sequenceNumber, |             sequenceNumber: sequenceNumber, | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |     public async updateContent(answer: Answer, newContent: string){ | ||||||
|  |         answer.content = newContent; | ||||||
|  |         await this.save(answer); | ||||||
|  |         return answer; | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -70,4 +70,10 @@ export class QuestionRepository extends DwengoEntityRepository<Question> { | ||||||
|             sequenceNumber |             sequenceNumber | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     public async updateContent(question: Question, newContent: string): Promise<Question> { | ||||||
|  |         question.content = newContent; | ||||||
|  |         await this.save(question); | ||||||
|  |         return question; | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,14 +1,14 @@ | ||||||
| import { mapToUserDTO } from './user.js'; |  | ||||||
| import {mapToQuestionDTO, mapToQuestionDTOId} from './question.js'; | import {mapToQuestionDTO, mapToQuestionDTOId} from './question.js'; | ||||||
| import { Answer } from '../entities/questions/answer.entity.js'; | import { Answer } from '../entities/questions/answer.entity.js'; | ||||||
| import { AnswerDTO, AnswerId } from '@dwengo-1/common/interfaces/answer'; | import { AnswerDTO, AnswerId } from '@dwengo-1/common/interfaces/answer'; | ||||||
|  | import {mapToTeacherDTO} from "./teacher"; | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Convert a Question entity to a DTO format. |  * Convert a Question entity to a DTO format. | ||||||
|  */ |  */ | ||||||
| export function mapToAnswerDTO(answer: Answer): AnswerDTO { | export function mapToAnswerDTO(answer: Answer): AnswerDTO { | ||||||
|     return { |     return { | ||||||
|         author: mapToUserDTO(answer.author), |         author: mapToTeacherDTO(answer.author), | ||||||
|         toQuestion: mapToQuestionDTO(answer.toQuestion), |         toQuestion: mapToQuestionDTO(answer.toQuestion), | ||||||
|         sequenceNumber: answer.sequenceNumber!, |         sequenceNumber: answer.sequenceNumber!, | ||||||
|         timestamp: answer.timestamp.toISOString(), |         timestamp: answer.timestamp.toISOString(), | ||||||
|  |  | ||||||
							
								
								
									
										22
									
								
								backend/src/routes/answers.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								backend/src/routes/answers.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,22 @@ | ||||||
|  | import express from "express"; | ||||||
|  | import { | ||||||
|  |     createAnswerHandler, | ||||||
|  |     deleteAnswerHandler, | ||||||
|  |     getAnswerHandler, | ||||||
|  |     getAnswersHandler, | ||||||
|  |     updateAnswerHandler | ||||||
|  | } from "../controllers/answers"; | ||||||
|  | 
 | ||||||
|  | const router = express.Router({ mergeParams: true }); | ||||||
|  | 
 | ||||||
|  | router.get('/', getAnswersHandler); | ||||||
|  | 
 | ||||||
|  | router.post('/', createAnswerHandler) | ||||||
|  | 
 | ||||||
|  | router.get('/:seqAnswer', getAnswerHandler) | ||||||
|  | 
 | ||||||
|  | router.delete('/:seqAnswer', deleteAnswerHandler); | ||||||
|  | 
 | ||||||
|  | router.put('/:seqAnswer', updateAnswerHandler); | ||||||
|  | 
 | ||||||
|  | export default router; | ||||||
|  | @ -3,9 +3,10 @@ import { | ||||||
|     createQuestionHandler, |     createQuestionHandler, | ||||||
|     deleteQuestionHandler, |     deleteQuestionHandler, | ||||||
|     getAllQuestionsHandler, |     getAllQuestionsHandler, | ||||||
|     getQuestionAnswersHandler, |  | ||||||
|     getQuestionHandler, |     getQuestionHandler, | ||||||
| } from '../controllers/questions.js'; | } from '../controllers/questions.js'; | ||||||
|  | import answerRoutes from './answers.js'; | ||||||
|  | 
 | ||||||
| const router = express.Router({ mergeParams: true }); | const router = express.Router({ mergeParams: true }); | ||||||
| 
 | 
 | ||||||
| // Query language
 | // Query language
 | ||||||
|  | @ -20,6 +21,6 @@ router.delete('/:seq', deleteQuestionHandler); | ||||||
| // Information about a question with id
 | // Information about a question with id
 | ||||||
| router.get('/:seq', getQuestionHandler); | router.get('/:seq', getQuestionHandler); | ||||||
| 
 | 
 | ||||||
| router.get('/answers/:seq', getQuestionAnswersHandler); | router.use('/:seq/answer', answerRoutes); | ||||||
| 
 | 
 | ||||||
| export default router; | export default router; | ||||||
|  |  | ||||||
							
								
								
									
										68
									
								
								backend/src/services/answers.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								backend/src/services/answers.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,68 @@ | ||||||
|  | import {getAnswerRepository} from "../data/repositories"; | ||||||
|  | import {Answer} from "../entities/questions/answer.entity"; | ||||||
|  | import {mapToAnswerDTO, mapToAnswerDTOId} from "../interfaces/answer"; | ||||||
|  | import {fetchTeacher} from "./teachers"; | ||||||
|  | import {fetchQuestion} from "./questions"; | ||||||
|  | import {QuestionId} from "@dwengo-1/common/interfaces/question"; | ||||||
|  | import {AnswerData, AnswerDTO, AnswerId} from "@dwengo-1/common/interfaces/answer"; | ||||||
|  | import {NotFoundException} from "../exceptions/not-found-exception"; | ||||||
|  | 
 | ||||||
|  | export async function getAnswersByQuestion(questionId: QuestionId, full: boolean): Promise<AnswerDTO[] | AnswerId[]> { | ||||||
|  |     const answerRepository = getAnswerRepository(); | ||||||
|  |     const question = await fetchQuestion(questionId); | ||||||
|  | 
 | ||||||
|  |     const answers: Answer[] = await answerRepository.findAllAnswersToQuestion(question); | ||||||
|  | 
 | ||||||
|  |     if (full) { | ||||||
|  |         return answers.map(mapToAnswerDTO); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return answers.map(mapToAnswerDTOId); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function createAnswer(questionId: QuestionId, answerData: AnswerData): Promise<AnswerDTO> { | ||||||
|  |     const answerRepository = getAnswerRepository(); | ||||||
|  |     const toQuestion = await fetchQuestion(questionId); | ||||||
|  |     const author = await fetchTeacher(answerData.author); | ||||||
|  |     const content = answerData.content; | ||||||
|  | 
 | ||||||
|  |     const answer = await answerRepository.createAnswer({ | ||||||
|  |         toQuestion, author, content | ||||||
|  |     }); | ||||||
|  |     return mapToAnswerDTO(answer); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async function fetchAnswer(questionId: QuestionId, sequenceNumber: number): Promise<Answer> { | ||||||
|  |     const answerRepository = getAnswerRepository(); | ||||||
|  |     const question = await fetchQuestion(questionId); | ||||||
|  |     const answer = await answerRepository.findAnswer(question, sequenceNumber); | ||||||
|  | 
 | ||||||
|  |     if (!answer){ | ||||||
|  |         throw new NotFoundException('Answer with questionID and sequence number not found'); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return answer; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function getAnswer(questionId: QuestionId, sequenceNumber: number): Promise<AnswerDTO> { | ||||||
|  |     const answer = await fetchAnswer(questionId, sequenceNumber); | ||||||
|  |     return mapToAnswerDTO(answer); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function deleteAnswer(questionId: QuestionId, sequenceNumber: number): Promise<AnswerDTO> { | ||||||
|  |     const answerRepository = getAnswerRepository(); | ||||||
|  | 
 | ||||||
|  |     const question = await fetchQuestion(questionId); | ||||||
|  |     const answer = await fetchAnswer(questionId, sequenceNumber); | ||||||
|  | 
 | ||||||
|  |     await answerRepository.removeAnswerByQuestionAndSequenceNumber(question, sequenceNumber); | ||||||
|  |     return mapToAnswerDTO(answer); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export async function updateAnswer(questionId: QuestionId, sequenceNumber: number, answerData: AnswerData){ | ||||||
|  |     const answerRepository = getAnswerRepository(); | ||||||
|  |     const answer = await fetchAnswer(questionId, sequenceNumber); | ||||||
|  | 
 | ||||||
|  |     const newAnswer = await answerRepository.updateContent(answer, answerData.content); | ||||||
|  |     return mapToAnswerDTO(newAnswer); | ||||||
|  | } | ||||||
|  | @ -1,13 +1,11 @@ | ||||||
| import { getAnswerRepository, getQuestionRepository } from '../data/repositories.js'; | import { getQuestionRepository } from '../data/repositories.js'; | ||||||
| import {mapToLearningObjectID, mapToQuestionDTO, mapToQuestionDTOId} from '../interfaces/question.js'; | import {mapToLearningObjectID, mapToQuestionDTO, mapToQuestionDTOId} from '../interfaces/question.js'; | ||||||
| import { Question } from '../entities/questions/question.entity.js'; | import { Question } from '../entities/questions/question.entity.js'; | ||||||
| import { Answer } from '../entities/questions/answer.entity.js'; |  | ||||||
| import { mapToAnswerDTO, mapToAnswerDTOId } from '../interfaces/answer.js'; |  | ||||||
| import { QuestionRepository } from '../data/questions/question-repository.js'; | import { QuestionRepository } from '../data/questions/question-repository.js'; | ||||||
| import { LearningObjectIdentifier } from '../entities/content/learning-object-identifier.js'; | import { LearningObjectIdentifier } from '../entities/content/learning-object-identifier.js'; | ||||||
| import { QuestionDTO, QuestionId } from '@dwengo-1/common/interfaces/question'; | import {QuestionData, QuestionDTO, QuestionId} from '@dwengo-1/common/interfaces/question'; | ||||||
| import { AnswerDTO, AnswerId } from '@dwengo-1/common/interfaces/answer'; |  | ||||||
| import {NotFoundException} from "../exceptions/not-found-exception"; | import {NotFoundException} from "../exceptions/not-found-exception"; | ||||||
|  | import {FALLBACK_VERSION_NUM} from "../config"; | ||||||
| import {fetchStudent} from "./students"; | import {fetchStudent} from "./students"; | ||||||
| 
 | 
 | ||||||
| export async function getAllQuestions(id: LearningObjectIdentifier, full: boolean): Promise<QuestionDTO[] | QuestionId[]> { | export async function getAllQuestions(id: LearningObjectIdentifier, full: boolean): Promise<QuestionDTO[] | QuestionId[]> { | ||||||
|  | @ -21,7 +19,7 @@ export async function getAllQuestions(id: LearningObjectIdentifier, full: boolea | ||||||
|     return questions.map(mapToQuestionDTOId); |     return questions.map(mapToQuestionDTOId); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| async function fetchQuestion(questionId: QuestionId): Promise<Question> { | export async function fetchQuestion(questionId: QuestionId): Promise<Question> { | ||||||
|     const questionRepository = getQuestionRepository(); |     const questionRepository = getQuestionRepository(); | ||||||
|     const question = await questionRepository.findByLearningObjectAndSequenceNumber( |     const question = await questionRepository.findByLearningObjectAndSequenceNumber( | ||||||
|         mapToLearningObjectID(questionId.learningObjectIdentifier), |         mapToLearningObjectID(questionId.learningObjectIdentifier), | ||||||
|  | @ -40,42 +38,37 @@ export async function getQuestion(questionId: QuestionId): Promise<QuestionDTO> | ||||||
|     return mapToQuestionDTO(question); |     return mapToQuestionDTO(question); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export async function getAnswersByQuestion(questionId: QuestionId, full: boolean): Promise<AnswerDTO[] | AnswerId[]> { | export async function createQuestion(loId: LearningObjectIdentifier, questionData: QuestionData): Promise<QuestionDTO> { | ||||||
|     const answerRepository = getAnswerRepository(); |  | ||||||
|     const question = await fetchQuestion(questionId); |  | ||||||
| 
 |  | ||||||
|     const answers: Answer[] = await answerRepository.findAllAnswersToQuestion(question); |  | ||||||
| 
 |  | ||||||
|     if (full) { |  | ||||||
|         return answers.map(mapToAnswerDTO); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return answers.map(mapToAnswerDTOId); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| export async function createQuestion(questionDTO: QuestionDTO): Promise<QuestionDTO | null> { |  | ||||||
|     const questionRepository = getQuestionRepository(); |     const questionRepository = getQuestionRepository(); | ||||||
|     const author = await fetchStudent(questionDTO.author); |     const author = await fetchStudent(questionData.author!); | ||||||
|  |     const content = questionData.content; | ||||||
| 
 | 
 | ||||||
|     await questionRepository.createQuestion({ |     const question = await questionRepository.createQuestion({ | ||||||
|         loId: mapToLearningObjectID(questionDTO.learningObjectIdentifier), |         loId, author, content | ||||||
|         author, |  | ||||||
|         content: questionDTO.content, |  | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     return questionDTO; |     return mapToQuestionDTO(question); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export async function deleteQuestion(questionId: QuestionId): Promise<QuestionDTO | null> { | export async function deleteQuestion(questionId: QuestionId): Promise<QuestionDTO> { | ||||||
|     const questionRepository = getQuestionRepository(); |     const questionRepository = getQuestionRepository(); | ||||||
| 
 |     const question = await fetchQuestion(questionId); // throws error if not found
 | ||||||
|     const question = await fetchQuestion(questionId); |  | ||||||
| 
 | 
 | ||||||
|     const loId: LearningObjectIdentifier = { |     const loId: LearningObjectIdentifier = { | ||||||
|         ...questionId.learningObjectIdentifier, |         ...questionId.learningObjectIdentifier, | ||||||
|         version: questionId.learningObjectIdentifier.version ?? 1, |         version: questionId.learningObjectIdentifier.version ?? FALLBACK_VERSION_NUM, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     await questionRepository.removeQuestionByLearningObjectAndSequenceNumber(loId, questionId.sequenceNumber); |     await questionRepository.removeQuestionByLearningObjectAndSequenceNumber(loId, questionId.sequenceNumber); | ||||||
|     return mapToQuestionDTO(question); |     return mapToQuestionDTO(question); | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | export async function updateQuestion(questionId: QuestionId, questionData: QuestionData): Promise<QuestionDTO> { | ||||||
|  |     const questionRepository = getQuestionRepository(); | ||||||
|  |     const question = await fetchQuestion(questionId); | ||||||
|  | 
 | ||||||
|  |     const newQuestion = await questionRepository.updateContent(question, questionData.content); | ||||||
|  |     return mapToQuestionDTO(newQuestion); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |  | ||||||
|  | @ -1,14 +1,19 @@ | ||||||
| import { UserDTO } from './user'; |  | ||||||
| import { QuestionDTO, QuestionId } from './question'; | import { QuestionDTO, QuestionId } from './question'; | ||||||
|  | import {TeacherDTO} from "./teacher"; | ||||||
| 
 | 
 | ||||||
| export interface AnswerDTO { | export interface AnswerDTO { | ||||||
|     author: UserDTO; |     author: TeacherDTO; | ||||||
|     toQuestion: QuestionDTO; |     toQuestion: QuestionDTO; | ||||||
|     sequenceNumber: number; |     sequenceNumber: number; | ||||||
|     timestamp: string; |     timestamp: string; | ||||||
|     content: string; |     content: string; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | export interface AnswerData { | ||||||
|  |     author: string; | ||||||
|  |     content: string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| export interface AnswerId { | export interface AnswerId { | ||||||
|     author: string; |     author: string; | ||||||
|     toQuestion: QuestionId; |     toQuestion: QuestionId; | ||||||
|  |  | ||||||
|  | @ -4,8 +4,13 @@ import { StudentDTO } from './student'; | ||||||
| export interface QuestionDTO { | export interface QuestionDTO { | ||||||
|     learningObjectIdentifier: LearningObjectIdentifierDTO; |     learningObjectIdentifier: LearningObjectIdentifierDTO; | ||||||
|     sequenceNumber?: number; |     sequenceNumber?: number; | ||||||
|     author: string; |     author: StudentDTO; | ||||||
|     timestamp?: string; |     timestamp: string; | ||||||
|  |     content: string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export interface QuestionData { | ||||||
|  |     author?: string; | ||||||
|     content: string; |     content: string; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Reference in a new issue
	
	 Gabriellvl
						Gabriellvl