Switched to apollo-server-express. Added authentification middleware and login mutation. Refactored queries, added query
This commit is contained in:
parent
898b17510b
commit
09662877ea
@ -6,7 +6,8 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@prisma/client": "^2.7.1",
|
"@prisma/client": "^2.7.1",
|
||||||
"@types/jsonwebtoken": "^8.5.0",
|
"@types/jsonwebtoken": "^8.5.0",
|
||||||
"apollo-server": "^2.18.2",
|
"apollo-server-express": "^2.18.2",
|
||||||
|
"express-jwt": "^6.0.0",
|
||||||
"graphql": "^15.3.0",
|
"graphql": "^15.3.0",
|
||||||
"jsonwebtoken": "^8.5.1",
|
"jsonwebtoken": "^8.5.1",
|
||||||
"jwks-rsa": "^1.10.1",
|
"jwks-rsa": "^1.10.1",
|
||||||
|
53
src/controllers/index.ts
Normal file
53
src/controllers/index.ts
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
import { PrismaClient } from "@prisma/client"
|
||||||
|
import { getDBForm, getDBFormByUser } from "../db"
|
||||||
|
import { FullForm } from "../db/types"
|
||||||
|
|
||||||
|
import { Form as GraphqlForm, FormSubmission } from "../typeDefs/typeDefs.gen"
|
||||||
|
|
||||||
|
const getForm = async (
|
||||||
|
db: PrismaClient,
|
||||||
|
id?: number
|
||||||
|
): Promise<GraphqlForm | null> => {
|
||||||
|
const dbForm: FullForm = await getDBForm(db, id)
|
||||||
|
|
||||||
|
if (dbForm == null) throw new Error("Not found")
|
||||||
|
|
||||||
|
const form: GraphqlForm = {
|
||||||
|
id: dbForm.id,
|
||||||
|
title: dbForm.title,
|
||||||
|
questions: [...dbForm.choisesQuestions, ...dbForm.inputQuestions],
|
||||||
|
dateCreated: dbForm.dateCreated.toString(),
|
||||||
|
submissions: dbForm.submissions.map<FormSubmission>((submission) => ({
|
||||||
|
answers: submission.answers,
|
||||||
|
date: submission.date.toString(),
|
||||||
|
id: submission.id,
|
||||||
|
})),
|
||||||
|
}
|
||||||
|
|
||||||
|
return form
|
||||||
|
}
|
||||||
|
|
||||||
|
const getForms = async (
|
||||||
|
db: PrismaClient,
|
||||||
|
userId: number
|
||||||
|
): Promise<GraphqlForm[]> => {
|
||||||
|
const dbForms = await getDBFormByUser(db, userId)
|
||||||
|
|
||||||
|
const forms = [
|
||||||
|
...dbForms.map((form) => ({
|
||||||
|
id: form.id,
|
||||||
|
title: form.title,
|
||||||
|
questions: [...form.choisesQuestions, ...form.inputQuestions],
|
||||||
|
dateCreated: form.dateCreated.toString(),
|
||||||
|
submissions: form.submissions.map<FormSubmission>((submission) => ({
|
||||||
|
answers: submission.answers,
|
||||||
|
date: submission.date.toString(),
|
||||||
|
id: submission.id,
|
||||||
|
})),
|
||||||
|
})),
|
||||||
|
]
|
||||||
|
|
||||||
|
return forms
|
||||||
|
}
|
||||||
|
|
||||||
|
export { getForm, getForms }
|
@ -1,9 +1,9 @@
|
|||||||
import { PrismaClient } from "@prisma/client"
|
import { PrismaClient } from "@prisma/client"
|
||||||
|
|
||||||
const getForm = async (db: PrismaClient, id: number) =>
|
const getDBForm = async (db: PrismaClient, id?: number) => {
|
||||||
db.form.findOne({
|
return await db.form.findOne({
|
||||||
where: {
|
where: {
|
||||||
id: id,
|
id: id ? id : undefined,
|
||||||
},
|
},
|
||||||
include: {
|
include: {
|
||||||
author: true,
|
author: true,
|
||||||
@ -20,5 +20,29 @@ const getForm = async (db: PrismaClient, id: number) =>
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
}
|
||||||
|
|
||||||
export { getForm }
|
const getDBFormByUser = async (db: PrismaClient, id: number) => {
|
||||||
|
return await db.form.findMany({
|
||||||
|
where: {
|
||||||
|
author: {
|
||||||
|
id,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
choisesQuestions: {
|
||||||
|
include: {
|
||||||
|
variants: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
inputQuestions: true,
|
||||||
|
submissions: {
|
||||||
|
include: {
|
||||||
|
answers: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export { getDBForm, getDBFormByUser }
|
||||||
|
6
src/db/types.ts
Normal file
6
src/db/types.ts
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
import { PromiseReturnType } from "@prisma/client"
|
||||||
|
import { getDBForm } from "../db"
|
||||||
|
|
||||||
|
type FullForm = PromiseReturnType<typeof getDBForm>
|
||||||
|
|
||||||
|
export { FullForm }
|
27
src/index.ts
27
src/index.ts
@ -1,18 +1,35 @@
|
|||||||
import { ApolloServer } from "apollo-server"
|
import { ApolloServer } from "apollo-server-express"
|
||||||
|
import express from "express"
|
||||||
|
import expressJwt from "express-jwt"
|
||||||
|
import jwt from 'jsonwebtoken'
|
||||||
|
import { PrismaClient } from "@prisma/client"
|
||||||
|
|
||||||
import typeDefs from "./typeDefs"
|
import typeDefs from "./typeDefs"
|
||||||
import resolvers from "./resolvers"
|
import resolvers from "./resolvers"
|
||||||
import { PrismaClient } from "@prisma/client"
|
import { ApolloContextType } from "./types"
|
||||||
|
|
||||||
|
const app = express()
|
||||||
|
|
||||||
|
app.use(
|
||||||
|
expressJwt({
|
||||||
|
secret: "SuperSecret",
|
||||||
|
credentialsRequired: false,
|
||||||
|
algorithms: ["HS256"],
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
const server = new ApolloServer({
|
const server = new ApolloServer({
|
||||||
typeDefs,
|
typeDefs,
|
||||||
resolvers,
|
resolvers,
|
||||||
context: async () => {
|
context: async ({ req }): Promise<ApolloContextType> => {
|
||||||
const db = new PrismaClient()
|
const db = new PrismaClient()
|
||||||
|
|
||||||
return { db }
|
return { db }
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
server.listen().then(({ url }) => {
|
server.applyMiddleware({ app })
|
||||||
console.log(`Server ready at ${url}`)
|
|
||||||
|
app.listen(4000, () => {
|
||||||
|
console.log("Server ready at http://localhost:4000")
|
||||||
})
|
})
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
import { getForm } from "../db"
|
import { getForm, getForms } from "../controllers"
|
||||||
import {
|
import {
|
||||||
Form,
|
Form,
|
||||||
FormSubmission,
|
|
||||||
QueryFormArgs,
|
QueryFormArgs,
|
||||||
QuestionResolvers,
|
QuestionResolvers,
|
||||||
Resolver,
|
Resolver,
|
||||||
@ -15,32 +14,26 @@ const formQuery: Resolver<Form, {}, ApolloContextType, QueryFormArgs> = async (
|
|||||||
{ db }
|
{ db }
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
const dbForm = await getForm(db, id)
|
return await getForm(db, id)
|
||||||
|
} catch (err) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (dbForm == null) throw new Error("Not found")
|
const formsQuery: Resolver<Form[], {}, ApolloContextType> = async (
|
||||||
|
_,
|
||||||
const form: Form = {
|
__,
|
||||||
id: dbForm.id,
|
{ db }
|
||||||
title: dbForm.title,
|
) => {
|
||||||
questions: [...dbForm.choisesQuestions, ...dbForm.inputQuestions],
|
try {
|
||||||
dateCreated: dbForm.dateCreated.toString(),
|
return await getForms(db, 1)
|
||||||
submissions: dbForm.submissions.map<FormSubmission>((submission) => {
|
|
||||||
return {
|
|
||||||
answers: submission.answers,
|
|
||||||
date: submission.date.toString(),
|
|
||||||
id: submission.id,
|
|
||||||
}
|
|
||||||
}),
|
|
||||||
}
|
|
||||||
|
|
||||||
return form
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const QuestionResolver: QuestionResolvers = {
|
const QuestionResolver: QuestionResolvers = {
|
||||||
__resolveType(obj: any, context, info) {
|
__resolveType(obj: any) {
|
||||||
if (obj.type) {
|
if (obj.type) {
|
||||||
return "ChoisesQuestion"
|
return "ChoisesQuestion"
|
||||||
}
|
}
|
||||||
@ -49,7 +42,7 @@ const QuestionResolver: QuestionResolvers = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const AnswerResolver: AnswerResolvers = {
|
const AnswerResolver: AnswerResolvers = {
|
||||||
__resolveType(obj, context, info) {
|
__resolveType(obj) {
|
||||||
if (obj.type == "CHOISE") return "ChoiseAnswer"
|
if (obj.type == "CHOISE") return "ChoiseAnswer"
|
||||||
if (obj.type == "INPUT") return "InputAnswer"
|
if (obj.type == "INPUT") return "InputAnswer"
|
||||||
|
|
||||||
@ -57,4 +50,4 @@ const AnswerResolver: AnswerResolvers = {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
export { formQuery, QuestionResolver, AnswerResolver }
|
export { formQuery, formsQuery, QuestionResolver, AnswerResolver }
|
||||||
|
@ -0,0 +1,32 @@
|
|||||||
|
import jwt from "jsonwebtoken"
|
||||||
|
import { MutationLoginArgs, Resolver, User } from "../typeDefs/typeDefs.gen"
|
||||||
|
import { ApolloContextType, JwtPayload } from "../types"
|
||||||
|
|
||||||
|
const loginResolver: Resolver<
|
||||||
|
User,
|
||||||
|
{},
|
||||||
|
ApolloContextType,
|
||||||
|
MutationLoginArgs
|
||||||
|
> = async (_, { id, admin }, { db }) => {
|
||||||
|
try {
|
||||||
|
const payload: JwtPayload = {
|
||||||
|
id,
|
||||||
|
admin,
|
||||||
|
}
|
||||||
|
const token = jwt.sign(payload, "SuperSecret")
|
||||||
|
const user = await db.user.findOne({
|
||||||
|
where: {
|
||||||
|
id,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
return {
|
||||||
|
...user,
|
||||||
|
token: token,
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export { loginResolver }
|
@ -4,11 +4,17 @@ import {
|
|||||||
formQuery as form,
|
formQuery as form,
|
||||||
QuestionResolver as Question,
|
QuestionResolver as Question,
|
||||||
AnswerResolver as Answer,
|
AnswerResolver as Answer,
|
||||||
|
formsQuery as forms,
|
||||||
} from "./Form"
|
} from "./Form"
|
||||||
|
import { loginResolver as login } from "./User"
|
||||||
|
|
||||||
const resolvers: Resolvers<ApolloContextType> = {
|
const resolvers: Resolvers<ApolloContextType> = {
|
||||||
Query: {
|
Query: {
|
||||||
form,
|
form,
|
||||||
|
forms,
|
||||||
|
},
|
||||||
|
Mutation: {
|
||||||
|
login,
|
||||||
},
|
},
|
||||||
Question,
|
Question,
|
||||||
Answer,
|
Answer,
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { gql } from "apollo-server"
|
import { gql } from "apollo-server-express"
|
||||||
import fs from "fs"
|
import fs from "fs"
|
||||||
|
|
||||||
const typeDefs = gql(
|
const typeDefs = gql(
|
||||||
|
@ -3,6 +3,10 @@ type Query {
|
|||||||
form(id: Int!): Form
|
form(id: Int!): Form
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Mutation {
|
||||||
|
login(id: Int!, admin: Boolean!): User
|
||||||
|
}
|
||||||
|
|
||||||
type Form {
|
type Form {
|
||||||
id: Int!
|
id: Int!
|
||||||
title: String!
|
title: String!
|
||||||
@ -10,6 +14,7 @@ type Form {
|
|||||||
submissions: [FormSubmission!]!
|
submissions: [FormSubmission!]!
|
||||||
dateCreated: String!
|
dateCreated: String!
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Question {
|
interface Question {
|
||||||
title: String!
|
title: String!
|
||||||
number: Int!
|
number: Int!
|
||||||
@ -64,3 +69,10 @@ enum AnswerType {
|
|||||||
INPUT
|
INPUT
|
||||||
CHOISE
|
CHOISE
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type User {
|
||||||
|
name: String!
|
||||||
|
id: Int!
|
||||||
|
forms: [Form!]!
|
||||||
|
token: String
|
||||||
|
}
|
||||||
|
@ -1,5 +1,11 @@
|
|||||||
import { PrismaClient } from "@prisma/client"
|
import { PrismaClient } from "@prisma/client"
|
||||||
|
import {} from 'express-jwt'
|
||||||
|
|
||||||
export type ApolloContextType = {
|
export type ApolloContextType = {
|
||||||
db: PrismaClient
|
db: PrismaClient
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type JwtPayload = {
|
||||||
|
id: number,
|
||||||
|
admin: boolean
|
||||||
|
}
|
@ -1,69 +1,14 @@
|
|||||||
{
|
{
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
/* Visit https://aka.ms/tsconfig.json to read more about this file */
|
"target": "es6" /* Specify ECMAScript target version: 'ES3' (default), 'ES5', 'ES2015', 'ES2016', 'ES2017', 'ES2018', 'ES2019', 'ES2020', or 'ESNEXT'. */,
|
||||||
|
"module": "CommonJS" /* Specify module code generation: 'none', 'commonjs', 'amd', 'system', 'umd', 'es2015', 'es2020', or 'ESNext'. */,
|
||||||
/* Basic Options */
|
"strict": true /* Enable all strict type-checking options. */,
|
||||||
// "incremental": true, /* Enable incremental compilation */
|
"outDir": "dist",
|
||||||
"target": "es5", /* Specify ECMAScript target version: 'ES3' (default), 'ES5', 'ES2015', 'ES2016', 'ES2017', 'ES2018', 'ES2019', 'ES2020', or 'ESNEXT'. */
|
"moduleResolution": "Node",
|
||||||
"module": "commonjs", /* Specify module code generation: 'none', 'commonjs', 'amd', 'system', 'umd', 'es2015', 'es2020', or 'ESNext'. */
|
"incremental": true,
|
||||||
// "lib": [], /* Specify library files to be included in the compilation. */
|
"esModuleInterop": true /* Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'. */,
|
||||||
// "allowJs": true, /* Allow javascript files to be compiled. */
|
"skipLibCheck": true /* Skip type checking of declaration files. */,
|
||||||
// "checkJs": true, /* Report errors in .js files. */
|
"forceConsistentCasingInFileNames": true /* Disallow inconsistently-cased references to the same file. */
|
||||||
// "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */
|
},
|
||||||
// "declaration": true, /* Generates corresponding '.d.ts' file. */
|
"include": ["src"]
|
||||||
// "declarationMap": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */
|
|
||||||
// "sourceMap": true, /* Generates corresponding '.map' file. */
|
|
||||||
// "outFile": "./", /* Concatenate and emit output to single file. */
|
|
||||||
// "outDir": "./", /* Redirect output structure to the directory. */
|
|
||||||
// "rootDir": "./", /* Specify the root directory of input files. Use to control the output directory structure with --outDir. */
|
|
||||||
// "composite": true, /* Enable project compilation */
|
|
||||||
// "tsBuildInfoFile": "./", /* Specify file to store incremental compilation information */
|
|
||||||
// "removeComments": true, /* Do not emit comments to output. */
|
|
||||||
// "noEmit": true, /* Do not emit outputs. */
|
|
||||||
// "importHelpers": true, /* Import emit helpers from 'tslib'. */
|
|
||||||
// "downlevelIteration": true, /* Provide full support for iterables in 'for-of', spread, and destructuring when targeting 'ES5' or 'ES3'. */
|
|
||||||
// "isolatedModules": true, /* Transpile each file as a separate module (similar to 'ts.transpileModule'). */
|
|
||||||
|
|
||||||
/* Strict Type-Checking Options */
|
|
||||||
"strict": true, /* Enable all strict type-checking options. */
|
|
||||||
// "noImplicitAny": true, /* Raise error on expressions and declarations with an implied 'any' type. */
|
|
||||||
// "strictNullChecks": true, /* Enable strict null checks. */
|
|
||||||
// "strictFunctionTypes": true, /* Enable strict checking of function types. */
|
|
||||||
// "strictBindCallApply": true, /* Enable strict 'bind', 'call', and 'apply' methods on functions. */
|
|
||||||
// "strictPropertyInitialization": true, /* Enable strict checking of property initialization in classes. */
|
|
||||||
// "noImplicitThis": true, /* Raise error on 'this' expressions with an implied 'any' type. */
|
|
||||||
// "alwaysStrict": true, /* Parse in strict mode and emit "use strict" for each source file. */
|
|
||||||
|
|
||||||
/* Additional Checks */
|
|
||||||
// "noUnusedLocals": true, /* Report errors on unused locals. */
|
|
||||||
// "noUnusedParameters": true, /* Report errors on unused parameters. */
|
|
||||||
// "noImplicitReturns": true, /* Report error when not all code paths in function return a value. */
|
|
||||||
// "noFallthroughCasesInSwitch": true, /* Report errors for fallthrough cases in switch statement. */
|
|
||||||
|
|
||||||
/* Module Resolution Options */
|
|
||||||
// "moduleResolution": "node", /* Specify module resolution strategy: 'node' (Node.js) or 'classic' (TypeScript pre-1.6). */
|
|
||||||
// "baseUrl": "./", /* Base directory to resolve non-absolute module names. */
|
|
||||||
// "paths": {}, /* A series of entries which re-map imports to lookup locations relative to the 'baseUrl'. */
|
|
||||||
// "rootDirs": [], /* List of root folders whose combined content represents the structure of the project at runtime. */
|
|
||||||
// "typeRoots": [], /* List of folders to include type definitions from. */
|
|
||||||
// "types": [], /* Type declaration files to be included in compilation. */
|
|
||||||
// "allowSyntheticDefaultImports": true, /* Allow default imports from modules with no default export. This does not affect code emit, just typechecking. */
|
|
||||||
"esModuleInterop": true, /* Enables emit interoperability between CommonJS and ES Modules via creation of namespace objects for all imports. Implies 'allowSyntheticDefaultImports'. */
|
|
||||||
// "preserveSymlinks": true, /* Do not resolve the real path of symlinks. */
|
|
||||||
// "allowUmdGlobalAccess": true, /* Allow accessing UMD globals from modules. */
|
|
||||||
|
|
||||||
/* Source Map Options */
|
|
||||||
// "sourceRoot": "", /* Specify the location where debugger should locate TypeScript files instead of source locations. */
|
|
||||||
// "mapRoot": "", /* Specify the location where debugger should locate map files instead of generated locations. */
|
|
||||||
// "inlineSourceMap": true, /* Emit a single file with source maps instead of having a separate file. */
|
|
||||||
// "inlineSources": true, /* Emit the source alongside the sourcemaps within a single file; requires '--inlineSourceMap' or '--sourceMap' to be set. */
|
|
||||||
|
|
||||||
/* Experimental Options */
|
|
||||||
// "experimentalDecorators": true, /* Enables experimental support for ES7 decorators. */
|
|
||||||
// "emitDecoratorMetadata": true, /* Enables experimental support for emitting type metadata for decorators. */
|
|
||||||
|
|
||||||
/* Advanced Options */
|
|
||||||
"skipLibCheck": true, /* Skip type checking of declaration files. */
|
|
||||||
"forceConsistentCasingInFileNames": true /* Disallow inconsistently-cased references to the same file. */
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user