Added form creation and submission validation. Fixed some form submission showing errors
This commit is contained in:
parent
c640602e99
commit
3367930690
@ -23,7 +23,20 @@ import {
|
|||||||
getDBForm,
|
getDBForm,
|
||||||
getDBFormsByUser,
|
getDBFormsByUser,
|
||||||
submitDBAnswer,
|
submitDBAnswer,
|
||||||
|
getDBFormQuestions,
|
||||||
} from '../db'
|
} from '../db'
|
||||||
|
import {
|
||||||
|
validateCreateFormParameters,
|
||||||
|
validateSubmitAnswerParameters,
|
||||||
|
} from './validate'
|
||||||
|
|
||||||
|
const formatQuestions = (
|
||||||
|
choisesQuestions: (ChoisesQuestion & {
|
||||||
|
variants: Variant[]
|
||||||
|
})[],
|
||||||
|
inputQuestions: InputQuestion[]
|
||||||
|
) =>
|
||||||
|
[...choisesQuestions, ...inputQuestions].sort((a, b) => a.number - b.number)
|
||||||
|
|
||||||
const getForm = async (
|
const getForm = async (
|
||||||
db: PrismaClient,
|
db: PrismaClient,
|
||||||
@ -39,15 +52,19 @@ const getForm = async (
|
|||||||
author: dbForm.author,
|
author: dbForm.author,
|
||||||
dateCreated: dbForm.dateCreated.toString(),
|
dateCreated: dbForm.dateCreated.toString(),
|
||||||
id: dbForm.id,
|
id: dbForm.id,
|
||||||
questions: [...dbForm.choisesQuestions, ...dbForm.inputQuestions].sort(
|
questions: formatQuestions(
|
||||||
(a, b) => a.number - b.number
|
dbForm.choisesQuestions,
|
||||||
|
dbForm.inputQuestions
|
||||||
),
|
),
|
||||||
submissions: dbForm.submissions.map((submission) => ({
|
submissions:
|
||||||
user: submission.user,
|
dbForm.submissions.length == 0
|
||||||
answers: submission.answers,
|
? null
|
||||||
date: submission.date.toString(),
|
: dbForm.submissions.map((submission) => ({
|
||||||
id: submission.id,
|
user: submission.user,
|
||||||
})),
|
answers: submission.answers,
|
||||||
|
date: submission.date.toString(),
|
||||||
|
id: submission.id,
|
||||||
|
})),
|
||||||
title: dbForm.title,
|
title: dbForm.title,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -93,6 +110,8 @@ const createFormFrom = async (
|
|||||||
try {
|
try {
|
||||||
const parsedQuestions = <UploadedQuestion[]>JSON.parse(params.questions)
|
const parsedQuestions = <UploadedQuestion[]>JSON.parse(params.questions)
|
||||||
|
|
||||||
|
await validateCreateFormParameters(params.title, parsedQuestions)
|
||||||
|
|
||||||
const newForm: FormConstructor = {
|
const newForm: FormConstructor = {
|
||||||
choisesQuestions: {
|
choisesQuestions: {
|
||||||
create: parsedQuestions.flatMap<CreateChoises>(
|
create: parsedQuestions.flatMap<CreateChoises>(
|
||||||
@ -139,9 +158,22 @@ const submitAnswer = async (
|
|||||||
userId: number
|
userId: number
|
||||||
): Promise<ServerAnswer> => {
|
): Promise<ServerAnswer> => {
|
||||||
try {
|
try {
|
||||||
|
const form = await getDBFormQuestions(db, formId)
|
||||||
|
if (!form) throw new UserInputError("Can't submit form")
|
||||||
|
|
||||||
|
console.log(formatQuestions(form.choisesQuestions, form.inputQuestions))
|
||||||
|
|
||||||
|
form.submissions.forEach((submission) => {
|
||||||
|
if (submission.userId === userId)
|
||||||
|
throw new UserInputError("Can't submit same form more than once")
|
||||||
|
})
|
||||||
|
|
||||||
const parsedAnswers = <DbAnswer[]>JSON.parse(answers)
|
const parsedAnswers = <DbAnswer[]>JSON.parse(answers)
|
||||||
|
|
||||||
console.log(parsedAnswers)
|
await validateSubmitAnswerParameters(
|
||||||
|
parsedAnswers,
|
||||||
|
formatQuestions(form.choisesQuestions, form.inputQuestions)
|
||||||
|
)
|
||||||
|
|
||||||
const res = await submitDBAnswer(db, userId, formId, parsedAnswers)
|
const res = await submitDBAnswer(db, userId, formId, parsedAnswers)
|
||||||
|
|
||||||
@ -165,9 +197,7 @@ const formatForms = (
|
|||||||
forms.map<GraphqlForm>((form) => ({
|
forms.map<GraphqlForm>((form) => ({
|
||||||
dateCreated: form.dateCreated.toString(),
|
dateCreated: form.dateCreated.toString(),
|
||||||
id: form.id,
|
id: form.id,
|
||||||
questions: [...form.choisesQuestions, ...form.inputQuestions].sort(
|
questions: formatQuestions(form.choisesQuestions, form.inputQuestions),
|
||||||
(a, b) => a.number - b.number
|
|
||||||
),
|
|
||||||
submissions: form.submissions.map((submission) => ({
|
submissions: form.submissions.map((submission) => ({
|
||||||
answers: submission.answers,
|
answers: submission.answers,
|
||||||
date: submission.date.toString(),
|
date: submission.date.toString(),
|
||||||
|
127
src/controllers/validate.ts
Normal file
127
src/controllers/validate.ts
Normal file
@ -0,0 +1,127 @@
|
|||||||
|
'use strict'
|
||||||
|
import { UserInputError } from 'apollo-server-express'
|
||||||
|
import { Answer } from '@prisma/client'
|
||||||
|
import {
|
||||||
|
UploadedChoisesQuestion,
|
||||||
|
UploadedInputQuestion,
|
||||||
|
UploadedQuestion,
|
||||||
|
} from './types'
|
||||||
|
import { ChoisesQuestion, InputQuestion, Variant } from 'typeDefs/typeDefs.gen'
|
||||||
|
|
||||||
|
const choisesVariants = ['CHECK', 'CHOOSE', 'SELECT']
|
||||||
|
|
||||||
|
const validateCreateFormParameters = async (
|
||||||
|
title: string,
|
||||||
|
questions: UploadedQuestion[]
|
||||||
|
) => {
|
||||||
|
if (!title)
|
||||||
|
throw new UserInputError("Form title can't be empty", {
|
||||||
|
invalidArgs: ['title'],
|
||||||
|
})
|
||||||
|
|
||||||
|
questions.forEach(
|
||||||
|
(question: UploadedChoisesQuestion | UploadedInputQuestion) => {
|
||||||
|
if (!question.title)
|
||||||
|
throw new UserInputError("Question title can't be empty", {
|
||||||
|
invalidArgs: ['questions'],
|
||||||
|
})
|
||||||
|
|
||||||
|
if ('type' in question) {
|
||||||
|
if (!question.variants || question.variants.length < 1)
|
||||||
|
throw new UserInputError(
|
||||||
|
'Question with choises must have at least one answer variant',
|
||||||
|
{ invalidArgs: ['questions'] }
|
||||||
|
)
|
||||||
|
|
||||||
|
question.variants.forEach((variant) => {
|
||||||
|
if (!variant.text || variant.text.length < 1)
|
||||||
|
throw new UserInputError("Choises variant text can't be empty", {
|
||||||
|
invalidArgs: ['questions'],
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!choisesVariants.includes(question.type))
|
||||||
|
throw new UserInputError(
|
||||||
|
'Question with choises must be of one of supported types',
|
||||||
|
{ invalidArgs: ['questions'] }
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const validateSubmitAnswerParameters = async (
|
||||||
|
answers: Answer[],
|
||||||
|
questions: (
|
||||||
|
| (ChoisesQuestion & {
|
||||||
|
variants: Variant[]
|
||||||
|
})
|
||||||
|
| InputQuestion
|
||||||
|
)[]
|
||||||
|
) => {
|
||||||
|
questions.forEach((question, questionIndex) => {
|
||||||
|
const answer = answers[questionIndex]
|
||||||
|
|
||||||
|
if (!answer)
|
||||||
|
throw new UserInputError('Every required question must have answer', {
|
||||||
|
invalidArgs: ['answers'],
|
||||||
|
})
|
||||||
|
|
||||||
|
if (!answer.type)
|
||||||
|
throw new UserInputError('Type must be specified for answer', {
|
||||||
|
invalidArgs: ['answers'],
|
||||||
|
})
|
||||||
|
|
||||||
|
if (answer.type !== 'CHOISE' && answer.type !== 'INPUT')
|
||||||
|
throw new UserInputError('Answer must have supported type', {
|
||||||
|
invalidArgs: ['answers'],
|
||||||
|
})
|
||||||
|
|
||||||
|
if (answer.type === 'CHOISE' && !('type' in question))
|
||||||
|
throw new UserInputError(
|
||||||
|
`Answer ${questionIndex + 1} must be of 'INPUT' type`,
|
||||||
|
{
|
||||||
|
invalidArgs: ['answers'],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
if (answer.type === 'INPUT' && 'type' in question)
|
||||||
|
throw new UserInputError(
|
||||||
|
`Answer ${questionIndex + 1} must be of 'CHOISE' type`,
|
||||||
|
{
|
||||||
|
invalidArgs: ['answers'],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
if (answer.type === 'CHOISE' && answer.userChoise === null)
|
||||||
|
throw new UserInputError(
|
||||||
|
"Question of type 'CHOISE' must have choise number set",
|
||||||
|
{
|
||||||
|
invalidArgs: ['answers'],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
if (answer.type === 'INPUT' && answer.userInput === null)
|
||||||
|
throw new UserInputError(
|
||||||
|
"Question of type 'INPUT' must have input string",
|
||||||
|
{
|
||||||
|
invalidArgs: ['answers'],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
if (
|
||||||
|
answer.userChoise !== null &&
|
||||||
|
(question as ChoisesQuestion).variants &&
|
||||||
|
answer.userChoise > (question as ChoisesQuestion).variants.length - 1
|
||||||
|
)
|
||||||
|
throw new UserInputError(
|
||||||
|
"Can't have chosen number bigger than amount of variants: " +
|
||||||
|
(question as ChoisesQuestion).variants.length,
|
||||||
|
{
|
||||||
|
invalidArgs: ['answers'],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export { validateCreateFormParameters, validateSubmitAnswerParameters }
|
@ -176,6 +176,26 @@ const submitDBAnswer = (
|
|||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const getDBFormQuestions = async (db: PrismaClient, formId: number) =>
|
||||||
|
db.form.findOne({
|
||||||
|
where: {
|
||||||
|
id: formId,
|
||||||
|
},
|
||||||
|
select: {
|
||||||
|
choisesQuestions: {
|
||||||
|
include: {
|
||||||
|
variants: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
inputQuestions: true,
|
||||||
|
submissions: {
|
||||||
|
select: {
|
||||||
|
userId: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
export {
|
export {
|
||||||
createDBForm,
|
createDBForm,
|
||||||
createDBUser,
|
createDBUser,
|
||||||
@ -184,4 +204,5 @@ export {
|
|||||||
getDBFormAuthor,
|
getDBFormAuthor,
|
||||||
getDBFormsByUser,
|
getDBFormsByUser,
|
||||||
submitDBAnswer,
|
submitDBAnswer,
|
||||||
|
getDBFormQuestions,
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user