From 0b46e065f50d0c0a9b12ca173b576ab499a17fad Mon Sep 17 00:00:00 2001 From: Mister-Mario Date: Wed, 24 Apr 2024 12:03:40 +0200 Subject: [PATCH 01/18] Removed duplicated file --- webapp/openapi.yaml | 313 -------------------------------------------- 1 file changed, 313 deletions(-) delete mode 100644 webapp/openapi.yaml diff --git a/webapp/openapi.yaml b/webapp/openapi.yaml deleted file mode 100644 index 98968bbf..00000000 --- a/webapp/openapi.yaml +++ /dev/null @@ -1,313 +0,0 @@ -openapi: 3.0.0 -info: - title: wiq_en1b API - description: Our project's OpenAPI specification. - version: 0.1.2 - contact: - name: Project github - url: https://github.com/Arquisoft/wiq_en1b -servers: - - url: http://localhost:8000 - description: Development server - - url: http://SOMEIP:8000 - description: Production server -paths: - /adduser: - post: - summary: Add a new user to the database. - operationId: addUser - requestBody: - required: true - content: - application/json: - schema: - type: object - properties: - username: - type: string - description: User ID. - example: student - password: - type: string - description: User password. - example: pass - responses: - '200': - description: User added successfully. - content: - application/json: - schema: - type: object - properties: - username: - type: string - description: User ID - '400': - description: Failed to add user - content: - application/json: - schema: - type: object - properties: - error: - type: string - description: Error information. - example: Username already in use - '500': - description: Internal server error. - content: - application/json: - schema: - type: object - properties: - error: - type: string - description: Error information. - example: Internal Server Error - /health: - get: - summary: Check the health status of the service. - operationId: checkHealth - responses: - '200': - description: Service is healthy. - content: - application/json: - schema: - type: object - properties: - status: - type: string - description: Health status. - example: OK - /login: - post: - summary: Log in to the system. - operationId: loginUser - requestBody: - required: true - content: - application/json: - schema: - type: object - properties: - username: - type: string - description: User ID. - example: student - password: - type: string - description: User password. - example: pass - responses: - '200': - description: Login successful. Returns user token, and username. - content: - application/json: - schema: - type: object - properties: - token: - type: string - description: User token. - example: eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1c2VySWQiOiI2NWY3NTZkYjNmYTIyZDIyN2E0YjdjN2QiLCJpYXQiOjE3MTA3MDg3NDUsImV4cCI6MTcxMDcxMjM0NX0.VMG_5DOyQ4GYlJQRcu1I6ICG1IGzuo2Xuei093ONHxw - username: - type: string - description: Username. - example: student - '400': - description: Invalid credentials. - content: - application/json: - schema: - type: object - properties: - error: - type: string - description: Shows the error info.. - example: Invalid credentials - '500': - description: Internal server error. - content: - application/json: - schema: - type: object - properties: - error: - type: string - description: Error information. - example: Internal Server Error - /questions: - get: - summary: Get 5 random questions from the system in any language - responses: - '200': - description: OK - content: - application/json: - schema: - $ref: '#/components/schemas/QuestionResponse' - /questions/{lang}: - get: - summary: Get 5 random questions from the system in the lang specified - parameters: - - name: lang - in: path - required: true - schema: - type: string - description: Language code for filtering questions must be es or en - responses: - '200': - description: OK - content: - application/json: - schema: - $ref: '#/components/schemas/QuestionResponse' - /record: - post: - summary: Add a new record to the database - requestBody: - required: true - content: - application/json: - schema: - $ref: '#/components/schemas/Record' - responses: - '200': - description: OK - content: - application/json: - schema: - type: object - properties: - user: - type: string - /record/{user}: - get: - summary: Gets the records from a given user - parameters: - - name: user - in: path - required: true - schema: - type: string - description: Username for filtering records - responses: - '200': - description: OK - content: - application/json: - schema: - $ref: '#/components/schemas/RecordResponse' -components: - schemas: - LoginRequest: - type: object - properties: - username: - type: string - password: - type: string - required: - - username - - password - LoginResponse: - type: object - properties: - token: - type: string - User: - type: object - properties: - username: - type: string - email: - type: string - required: - - username - - email - UserResponse: - type: object - properties: - message: - type: string - QuestionResponse: - type: array - items: - type: object - properties: - question: - type: string - answers: - type: array - items: - type: string - correct_answer: - type: string - Record: - type: object - properties: - user: - type: string - game: - type: object - properties: - questions: - type: array - items: - type: object - properties: - question: - type: string - example: What is the capital of Oviedo? - answers: - type: array - items: - type: string - answerGiven: - type: string - correctAnswer: - type: string - points: - type: number - date: - type: string - required: - - user - - game - RecordResponse: - type: object - properties: - user: - type: string - games: - type: array - items: - type: object - properties: - questions: - type: array - items: - type: object - properties: - question: - type: string - example: What is the capital of Oviedo? - answers: - type: array - items: - type: string - answerGiven: - type: string - correctAnswer: - type: string - points: - type: number - date: - type: string - Error: - type: object - properties: - error: - type: string \ No newline at end of file From 610de2bb1571065b0a06d67882ba92fde4c1b99e Mon Sep 17 00:00:00 2001 From: ErdemYabaci Date: Wed, 24 Apr 2024 12:56:48 +0200 Subject: [PATCH 02/18] Modified user endpoints documentation --- gatewayservice/openapi.yaml | 61 ++++++++++++++++--------------------- 1 file changed, 26 insertions(+), 35 deletions(-) diff --git a/gatewayservice/openapi.yaml b/gatewayservice/openapi.yaml index 47615d2a..3b44ddec 100644 --- a/gatewayservice/openapi.yaml +++ b/gatewayservice/openapi.yaml @@ -9,9 +9,25 @@ info: servers: - url: http://localhost:8000 description: Development server - - url: http://wiqen1b.serveminecraft.net:8000 + - url: http://wiqen1b.serveminecraft.net:8000 description: Production server paths: + /health: + get: + summary: Check the health status of the service. + operationId: checkHealth + responses: + '200': + description: Service is healthy. + content: + application/json: + schema: + type: object + properties: + status: + type: string + description: Health status. + example: OK /adduser: post: summary: Add a new user to the database. @@ -37,11 +53,7 @@ paths: content: application/json: schema: - type: object - properties: - username: - type: string - description: User ID + $ref: '#/components/shcemas/LoginResponse' '400': description: Failed to add user content: @@ -64,22 +76,6 @@ paths: type: string description: Error information. example: Internal Server Error - /health: - get: - summary: Check the health status of the service. - operationId: checkHealth - responses: - '200': - description: Service is healthy. - content: - application/json: - schema: - type: object - properties: - status: - type: string - description: Health status. - example: OK /login: post: summary: Log in to the system. @@ -93,28 +89,19 @@ paths: properties: username: type: string - description: User ID. + description: UserID. example: student password: type: string - description: User password. + description: Userpassword. example: pass responses: '200': - description: Login successful. Returns user token, and username. + description: Login successful. Returns user token, username and email. content: application/json: schema: - type: object - properties: - token: - type: string - description: User token. - example: eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1c2VySWQiOiI2NWY3NTZkYjNmYTIyZDIyN2E0YjdjN2QiLCJpYXQiOjE3MTA3MDg3NDUsImV4cCI6MTcxMDcxMjM0NX0.VMG_5DOyQ4GYlJQRcu1I6ICG1IGzuo2Xuei093ONHxw - username: - type: string - description: Username. - example: student + $ref: '#/components/shcemas/LoginResponse' '400': description: Invalid credentials. content: @@ -217,6 +204,10 @@ components: properties: token: type: string + username: + type: string + email: + type: string User: type: object properties: From 0f83f6a88c9a75ac8141df15e6560ecbd2b5d603 Mon Sep 17 00:00:00 2001 From: ErdemYabaci Date: Wed, 24 Apr 2024 13:53:30 +0200 Subject: [PATCH 03/18] Modified questions endpoints documentation --- gatewayservice/openapi.yaml | 135 +++++++++++++++++++++++++++++++----- 1 file changed, 119 insertions(+), 16 deletions(-) diff --git a/gatewayservice/openapi.yaml b/gatewayservice/openapi.yaml index 3b44ddec..1e9df50f 100644 --- a/gatewayservice/openapi.yaml +++ b/gatewayservice/openapi.yaml @@ -107,23 +107,13 @@ paths: content: application/json: schema: - type: object - properties: - error: - type: string - description: Shows the error info.. - example: Invalid credentials + $ref: '#/components/shcemas/Error' '500': description: Internal server error. content: application/json: schema: - type: object - properties: - error: - type: string - description: Error information. - example: Internal Server Error + $ref: '#/components/shcemas/Error' /questions: get: summary: Get 5 random questions from the system in any language @@ -133,7 +123,13 @@ paths: content: application/json: schema: - $ref: '#/components/schemas/QuestionResponse' + $ref: '#/components/schemas/QuestionResponse' + '500': + description: Internal server error. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' /questions/{lang}: get: summary: Get 5 random questions from the system in the lang specified @@ -143,7 +139,97 @@ paths: required: true schema: type: string - description: Language code for filtering questions must be es or en + enum: [es, en, tr] + description: Language code for filtering questions + security: + - BearerAuth: [] + responses: + '200': + description: OK + content: + application/json: + schema: + $ref: '#/components/schemas/QuestionResponse' + '400': + description: Wrong values given. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' + '500': + description: Internal server error. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' + /questions/{lang}/{amount}: + get: + summary: Get an amount of random questions from the system in the lang specified + parameters: + - name: lang + in: path + required: true + schema: + type: string + enum: [es, en, tr] + description: Language code for filtering questions + - name: amount + in: path + required: true + schema: + type: integer + minimun: 1 + maximun: 20 + description: Amount of questions retrieved + security: + - BearerAuth: [] + responses: + '200': + description: OK + content: + application/json: + schema: + $ref: '#/components/schemas/QuestionResponse' + '400': + description: Wrong values given. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' + '500': + description: Internal server error. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' + /questions/{lang}/{amount}/{type}: + get: + summary: Get an amount of questions of the given type from the system in the lang specified + parameters: + - name: lang + in: path + required: true + schema: + type: string + enum: [es, en, tr] + description: Language code for filtering questions + - name: amount + in: path + required: true + schema: + type: integer + minimun: 1 + maximun: 20 + description: Amount of questions retrieved + - name: type + in: path + required: true + schema: + type: string + enum: [POPULATION, CAPITAL, LANGUAGE, SIZE] + description: Type for filtering questions + security: + - BearerAuth: [] responses: '200': description: OK @@ -151,6 +237,18 @@ paths: application/json: schema: $ref: '#/components/schemas/QuestionResponse' + '400': + description: Wrong values given. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' + '500': + description: Internal server error. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' /record: post: summary: Add a new record to the database @@ -188,6 +286,11 @@ paths: schema: $ref: '#/components/schemas/RecordResponse' components: + securitySchemes: + BearerAuth: + type: http + scheme: bearer + bearerFormat: JWT schemas: LoginRequest: type: object @@ -261,7 +364,7 @@ components: correctAnswer: type: string points: - type: number + type: integer date: type: string required: @@ -294,7 +397,7 @@ components: correctAnswer: type: string points: - type: number + type: integer date: type: string Error: From 46fa6c8e5eb93b558a8efeedeb725d76f6a0eb30 Mon Sep 17 00:00:00 2001 From: ErdemYabaci Date: Wed, 24 Apr 2024 15:12:59 +0200 Subject: [PATCH 04/18] Modified record endpoints documentation --- gatewayservice/openapi.yaml | 88 ++++++++++++++++++++++++++++++++++++- 1 file changed, 87 insertions(+), 1 deletion(-) diff --git a/gatewayservice/openapi.yaml b/gatewayservice/openapi.yaml index 1e9df50f..d721ac42 100644 --- a/gatewayservice/openapi.yaml +++ b/gatewayservice/openapi.yaml @@ -258,6 +258,8 @@ paths: application/json: schema: $ref: '#/components/schemas/Record' + security: + - BearerAuth: [] responses: '200': description: OK @@ -268,6 +270,12 @@ paths: properties: user: type: string + '500': + description: Internal server error. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' /record/{user}: get: summary: Gets the records from a given user @@ -278,6 +286,8 @@ paths: schema: type: string description: Username for filtering records + security: + - BearerAuth: [] responses: '200': description: OK @@ -285,6 +295,67 @@ paths: application/json: schema: $ref: '#/components/schemas/RecordResponse' + '400': + description: Wrong values given. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' + '500': + description: Internal server error. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' + /record/ranking/top10: + get: + summary: Gets the top 10 users on the system + security: + - BearerAuth: [] + responses: + '200': + description: OK + content: + application/json: + schema: + $ref: '#/components/schemas/Top10Ranking' + '500': + description: Internal server error. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' + /record/ranking/{user}: + get: + summary: Gets the ranking information of the given user + parameters: + - name: user + in: path + required: true + schema: + type: string + description: username of the user + security: + - BearerAuth: [] + responses: + '200': + description: OK + content: + application/json: + schema: + $ref: '#/components/schemas/UserRanking' + '400': + description: Wrong values given. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' + '500': + description: Internal server error. + content: + application/json: + schema: + $ref: '#/components/shcemas/Error' components: securitySchemes: BearerAuth: @@ -404,4 +475,19 @@ components: type: object properties: error: - type: string \ No newline at end of file + type: string + UserRanking: + type: object + properties: + _id: + type: string + totalPoints: + type: integer + totalCompetitiveGames: + type: integer + position: + type: integer + Top10Ranking: + type: array + items: + $ref: '#/components/schemas/UserRanking' \ No newline at end of file From 0dc1658d63e12a35c545ff2d84e16fa4e0d1c4d6 Mon Sep 17 00:00:00 2001 From: Mister-Mario Date: Wed, 24 Apr 2024 18:56:08 +0200 Subject: [PATCH 05/18] Tested that it works against swaggerEditor online --- gatewayservice/gateway-service.js | 24 ++-- gatewayservice/openapi.yaml | 177 ++++++++++++++++-------------- 2 files changed, 104 insertions(+), 97 deletions(-) diff --git a/gatewayservice/gateway-service.js b/gatewayservice/gateway-service.js index c08d0e37..f592f9e0 100644 --- a/gatewayservice/gateway-service.js +++ b/gatewayservice/gateway-service.js @@ -33,7 +33,7 @@ app.post('/login', async (req, res) => { const authResponse = await axios.post(authServiceUrl+'/login', req.body); res.json(authResponse.data); } catch (error) { - manageError(error) + manageError(res, error) } }); @@ -43,7 +43,7 @@ app.post('/adduser', async (req, res) => { const userResponse = await axios.post(userServiceUrl+'/adduser', req.body); res.json(userResponse.data); } catch (error) { - manageError(error); + manageError(res, error); } }); @@ -55,7 +55,7 @@ app.get('/questions', verifyToken, async (req, res) => { const questionResponse = await axios.get(questionServiceUrl+'/questions'); res.json(questionResponse.data); } catch (error) { - manageError(error) + manageError(res, error) } }); @@ -78,7 +78,7 @@ app.get('/questions/:lang/:amount/:type', verifyToken, async (req, res) => { } } catch (error) { - manageError(error) + manageError(res, error) } }); @@ -97,7 +97,7 @@ app.get('/questions/:lang/:amount', verifyToken, async (req, res) => { res.json(questionResponse.data); } } catch (error) { - manageError(error) + manageError(res, error) } }); @@ -115,7 +115,7 @@ app.get('/questions/:lang', verifyToken, async (req, res) => { } catch (error) { - manageError(error) + manageError(res, error) } }); @@ -126,7 +126,7 @@ app.post('/record', verifyToken, async(req, res) => { const recordResponse = await axios.post(recordServiceUrl+'/record', req.body); res.json(recordResponse.data); } catch (error) { - manageError(error) + manageError(res, error) } }); @@ -136,7 +136,7 @@ app.get('/record/ranking/top10', verifyToken, async(req, res)=>{ const recordResponse = await axios.get(recordServiceUrl + '/record/ranking/top10'); res.json(recordResponse.data); } catch (error) { - manageError(error) + manageError(res, error) } }); @@ -151,7 +151,7 @@ app.get('/record/ranking/:user', verifyToken, async(req, res)=>{ res.json(recordResponse.data); } } catch (error) { - manageError(error) + manageError(res, error) } }); @@ -166,7 +166,7 @@ app.get('/record/:user', verifyToken, async(req, res)=>{ res.json(recordResponse.data); } } catch (error) { - manageError(error) + manageError(res, error) } }); @@ -206,7 +206,7 @@ function verifyToken(req, res, next) { } function validateLang(lang){ - return ['en', 'es', 'tk'].includes(lang); + return ['en', 'es', 'tr'].includes(lang); } function validateAmount(amount) { @@ -223,7 +223,7 @@ function validateUser(user){ return !(/\s/.test(user)) //True if there are no spaces } -function manageError(error){ +function manageError(res, error){ if(error.response) //Some microservice responded with an error res.status(error.response.status).json({ error: error.response.data.error }); else //Some other error diff --git a/gatewayservice/openapi.yaml b/gatewayservice/openapi.yaml index d721ac42..6755daa9 100644 --- a/gatewayservice/openapi.yaml +++ b/gatewayservice/openapi.yaml @@ -1,16 +1,16 @@ openapi: 3.0.0 info: - title: wiq_en1b API - description: Our project's OpenAPI specification. - version: 0.1.2 + title: WIQ_en1b API + description: WIQ_en1b OpenAPI specification. + version: "0.2" contact: name: Project github url: https://github.com/Arquisoft/wiq_en1b servers: + - url: http://wiqen1b.serveminecraft.net:8000 + description: Production server - url: http://localhost:8000 description: Development server - - url: http://wiqen1b.serveminecraft.net:8000 - description: Production server paths: /health: get: @@ -39,21 +39,29 @@ paths: schema: type: object properties: + email: + type: string + description: User email + example: user@example.com username: type: string - description: User ID. + description: User name example: student password: type: string - description: User password. - example: pass + description: User password + example: password123 + repeatPassword: + type: string + description: Userpassword + example: password123 responses: '200': description: User added successfully. content: application/json: schema: - $ref: '#/components/shcemas/LoginResponse' + $ref: '#/components/schemas/LoginResponse' '400': description: Failed to add user content: @@ -101,22 +109,27 @@ paths: content: application/json: schema: - $ref: '#/components/shcemas/LoginResponse' + $ref: '#/components/schemas/LoginResponse' '400': description: Invalid credentials. content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' '500': description: Internal server error. content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' /questions: get: summary: Get 5 random questions from the system in any language + parameters: + - name: token + in: header + schema: + type: string responses: '200': description: OK @@ -129,20 +142,22 @@ paths: content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' /questions/{lang}: get: summary: Get 5 random questions from the system in the lang specified parameters: + - name: token + in: header + schema: + type: string - name: lang in: path required: true schema: type: string - enum: [es, en, tr] - description: Language code for filtering questions - security: - - BearerAuth: [] + enum: [es, en, tk] + description: Language code for filtering questions responses: '200': description: OK @@ -155,34 +170,36 @@ paths: content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' '500': description: Internal server error. content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' /questions/{lang}/{amount}: get: summary: Get an amount of random questions from the system in the lang specified parameters: + - name: token + in: header + schema: + type: string - name: lang in: path required: true schema: type: string - enum: [es, en, tr] + enum: [es, en, tk] description: Language code for filtering questions - name: amount in: path required: true schema: type: integer - minimun: 1 - maximun: 20 - description: Amount of questions retrieved - security: - - BearerAuth: [] + minimum: 1 + maximum: 20 + description: Amount of questions retrieved must be between 1 and 20 responses: '200': description: OK @@ -195,41 +212,43 @@ paths: content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' '500': description: Internal server error. content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' /questions/{lang}/{amount}/{type}: get: summary: Get an amount of questions of the given type from the system in the lang specified parameters: + - name: token + in: header + schema: + type: string - name: lang in: path required: true schema: type: string - enum: [es, en, tr] + enum: ["es", "en", "tk"] description: Language code for filtering questions - name: amount in: path required: true schema: type: integer - minimun: 1 - maximun: 20 - description: Amount of questions retrieved + minimum: 1 + maximum: 20 + description: Amount of questions retrieved must be between 1 and 20 - name: type in: path required: true schema: type: string - enum: [POPULATION, CAPITAL, LANGUAGE, SIZE] - description: Type for filtering questions - security: - - BearerAuth: [] + enum: ["POPULATION", "CAPITAL", "LANGUAGE", "SIZE"] + description: Type for filtering questions responses: '200': description: OK @@ -242,24 +261,27 @@ paths: content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' '500': description: Internal server error. content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' /record: post: summary: Add a new record to the database + parameters: + - name: token + in: header + schema: + type: string requestBody: required: true content: application/json: schema: $ref: '#/components/schemas/Record' - security: - - BearerAuth: [] responses: '200': description: OK @@ -275,19 +297,21 @@ paths: content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' /record/{user}: get: summary: Gets the records from a given user parameters: + - name: token + in: header + schema: + type: string - name: user in: path required: true schema: type: string description: Username for filtering records - security: - - BearerAuth: [] responses: '200': description: OK @@ -300,18 +324,21 @@ paths: content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' '500': description: Internal server error. content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' /record/ranking/top10: get: summary: Gets the top 10 users on the system - security: - - BearerAuth: [] + parameters: + - name: token + in: header + schema: + type: string responses: '200': description: OK @@ -324,19 +351,21 @@ paths: content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' /record/ranking/{user}: get: summary: Gets the ranking information of the given user parameters: + - name: token + in: header + schema: + type: string - name: user in: path required: true schema: type: string description: username of the user - security: - - BearerAuth: [] responses: '200': description: OK @@ -349,30 +378,15 @@ paths: content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' '500': description: Internal server error. content: application/json: schema: - $ref: '#/components/shcemas/Error' + $ref: '#/components/schemas/Error' components: - securitySchemes: - BearerAuth: - type: http - scheme: bearer - bearerFormat: JWT schemas: - LoginRequest: - type: object - properties: - username: - type: string - password: - type: string - required: - - username - - password LoginResponse: type: object properties: @@ -382,21 +396,6 @@ components: type: string email: type: string - User: - type: object - properties: - username: - type: string - email: - type: string - required: - - username - - email - UserResponse: - type: object - properties: - message: - type: string QuestionResponse: type: array items: @@ -476,7 +475,7 @@ components: properties: error: type: string - UserRanking: + Ranking: type: object properties: _id: @@ -487,7 +486,15 @@ components: type: integer position: type: integer + UserRanking: + type: object + properties: + userCompetitiveStats: + $ref: '#/components/schemas/Ranking' Top10Ranking: - type: array - items: - $ref: '#/components/schemas/UserRanking' \ No newline at end of file + type: object + properties: + usersCompetitiveStats: + type: array + items: + $ref: '#/components/schemas/Ranking' \ No newline at end of file From 4625dd620c28a4b9fe07e0fc728409e83f07e37a Mon Sep 17 00:00:00 2001 From: Mister-Mario Date: Wed, 24 Apr 2024 20:16:46 +0200 Subject: [PATCH 06/18] Opened questions endpoint so some dont need token --- gatewayservice/gateway-service.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/gatewayservice/gateway-service.js b/gatewayservice/gateway-service.js index c08d0e37..34ce5eaa 100644 --- a/gatewayservice/gateway-service.js +++ b/gatewayservice/gateway-service.js @@ -48,7 +48,7 @@ app.post('/adduser', async (req, res) => { } }); -app.get('/questions', verifyToken, async (req, res) => { +app.get('/questions', async (req, res) => { try { // Forward the question request to the quetion service @@ -101,7 +101,7 @@ app.get('/questions/:lang/:amount', verifyToken, async (req, res) => { } }); -app.get('/questions/:lang', verifyToken, async (req, res) => { +app.get('/questions/:lang', async (req, res) => { try { if(!validateLang(req.params.lang.toString())) res.status(400).json({ error: 'Wrong values given' }); From 59a2b4a4975f482adddeb79888825a119e447eaf Mon Sep 17 00:00:00 2001 From: uo289267 Date: Wed, 24 Apr 2024 20:33:51 +0200 Subject: [PATCH 07/18] Added Play as a Guest Functionality --- webapp/public/game.png | Bin 0 -> 22717 bytes webapp/src/App.js | 2 +- webapp/src/components/Home/Home.js | 4 ++-- .../src/components/questionView/QuestionView.js | 11 ++++++++--- webapp/src/translations/en/global.json | 3 ++- webapp/src/translations/es/global.json | 3 ++- 6 files changed, 15 insertions(+), 8 deletions(-) create mode 100644 webapp/public/game.png diff --git a/webapp/public/game.png b/webapp/public/game.png new file mode 100644 index 0000000000000000000000000000000000000000..a547de5b7918184bff9d4dd67b59f87a0eaf0a41 GIT binary patch literal 22717 zcmeFZXIE2U(*_D52nbRY1nC42=^(vJ4OLLOH0ix}2u(x<1BBjHnn-U6Jwa4@@4bVR z(0hO!e4g{ZYn`8P)_Onf{o&qoUo(4V?YU>J*&knOt3J3-cOM4_=K)wvNe>4HaCZy9 zA-s2YISClMy1U?d>#4rLsTiQ&!ogw20V^rI4!}jX5;X8AH^Oh|sqV=OetC-`e9x9H zz-$S4$ae4b!DB(o35%B$qq=>}s>(`n>3He_Z=sJVNYe4_fDe^PEH7>$aG6;vv~g{d z{Z)CzcH2&qWqf*N@^4v-lKmrx!D@PD`eXNUiDg8%;ngXK%0D9pzg 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 t{ErOz-;n&P5C1m~|8L`P@ha#bP8^_P!} /> : } /> : } /> - : } /> + } /> : } /> : }/> : } /> diff --git a/webapp/src/components/Home/Home.js b/webapp/src/components/Home/Home.js index d2da0752..4af21adb 100644 --- a/webapp/src/components/Home/Home.js +++ b/webapp/src/components/Home/Home.js @@ -30,8 +30,8 @@ function Home() { W - - Instructions + + Game diff --git a/webapp/src/components/questionView/QuestionView.js b/webapp/src/components/questionView/QuestionView.js index 7b757576..61047729 100644 --- a/webapp/src/components/questionView/QuestionView.js +++ b/webapp/src/components/questionView/QuestionView.js @@ -25,8 +25,12 @@ function QuestionView({type= "COMPETITIVE", amount=5}){ const generateQuestions = async (numQuestion) => { if (numQuestion < 0) { try { - - var generatedQuestions = await questionGenerator.generateQuestions(i18n.language, type, amount, cookie.token); + var generatedQuestions; + if(cookie){ + generatedQuestions = await questionGenerator.generateQuestions(i18n.language, type, amount, cookie.token); + }else{ + generateQuestions = await questionGenerator.generateQuestions(i18n.language, type, amount, null);//playing as guest + } setQuestions(generatedQuestions); setnumQuestion(0); } catch (error) { @@ -110,7 +114,8 @@ function QuestionView({type= "COMPETITIVE", amount=5}){ creationHistoricalRecord.setCompetitive(type === 'COMPETITIVE'); creationHistoricalRecord.setDate(Date.now()); creationHistoricalRecord.setPoints(points); - creationHistoricalRecord.sendRecord(cookie.username, cookie.token); + if(cookie) + creationHistoricalRecord.sendRecord(cookie.username, cookie.token); } }, 1000); diff --git a/webapp/src/translations/en/global.json b/webapp/src/translations/en/global.json index 54f087cf..1bcf3aa0 100644 --- a/webapp/src/translations/en/global.json +++ b/webapp/src/translations/en/global.json @@ -8,7 +8,8 @@ "msg1": "Welcome to the knowledge challenge! Get ready to test your mind and demonstrate your skills. In this exciting game, you will face a series of questions that will test your knowledge on a variety of topics.", "msg2": "So go ahead, prove that you are the master of questions! Are you ready to accept the challenge?", "clickOpen": "Click to continue", - "clickClose": "Click to close" + "clickClose": "Click to close", + "game":"Play as a guest" }, "navBar": { diff --git a/webapp/src/translations/es/global.json b/webapp/src/translations/es/global.json index 092ecc74..8906341c 100644 --- a/webapp/src/translations/es/global.json +++ b/webapp/src/translations/es/global.json @@ -8,7 +8,8 @@ "msg1": "Bienvenido al desafío de conocimientos! Prepárate para poner a prueba tu mente y demostrar tus habilidades. En este emocionante juego, te enfrentarás a una serie de preguntas que pondrán a prueba tu conocimiento en una variedad de temas.", "msg2": "¡Así que adelante, demuestra que eres el maestro de las preguntas! ¿Estás listo para aceptar el desafío?", "clickOpen": "Haz click para continuar", - "clickClose": "Haz click para cerrar" + "clickClose": "Haz click para cerrar", + "game":"Juega como invitado" }, From eecfb76a5b58d0f995c11640020e1ee3e1d23450 Mon Sep 17 00:00:00 2001 From: Mister-Mario Date: Wed, 24 Apr 2024 20:49:18 +0200 Subject: [PATCH 08/18] Tested and fixed --- webapp/src/components/questionView/QuestionView.js | 13 +++++-------- 1 file changed, 5 insertions(+), 8 deletions(-) diff --git a/webapp/src/components/questionView/QuestionView.js b/webapp/src/components/questionView/QuestionView.js index 61047729..4a90bc4c 100644 --- a/webapp/src/components/questionView/QuestionView.js +++ b/webapp/src/components/questionView/QuestionView.js @@ -14,23 +14,20 @@ import BackButtonToGameMenu from '../fragments/BackButtonToGameMenu'; const creationHistoricalRecord = new CreationHistoricalRecord(); const questionGenerator = new QuestionGenerator(); var points = 0; +var playAsGuestUsername = "Play as Guest" //Wont be ever used as it has spaces +var playAsGuestToken = " " function QuestionView({type= "COMPETITIVE", amount=5}){ const [numQuestion, setnumQuestion] = useState(-1); const [questions, setQuestions] = useState(null); const[t, i18n] = useTranslation("global"); - const cookie = JSON.parse(Cookies.get('user')) + const cookie = JSON.parse(Cookies.get('user')??JSON.stringify({username : playAsGuestUsername, token : playAsGuestToken})) const [audio] = useState(new Audio('/tictac.mp3')); const generateQuestions = async (numQuestion) => { if (numQuestion < 0) { try { - var generatedQuestions; - if(cookie){ - generatedQuestions = await questionGenerator.generateQuestions(i18n.language, type, amount, cookie.token); - }else{ - generateQuestions = await questionGenerator.generateQuestions(i18n.language, type, amount, null);//playing as guest - } + var generatedQuestions = await questionGenerator.generateQuestions(i18n.language, type, amount, cookie.token); setQuestions(generatedQuestions); setnumQuestion(0); } catch (error) { @@ -114,7 +111,7 @@ function QuestionView({type= "COMPETITIVE", amount=5}){ creationHistoricalRecord.setCompetitive(type === 'COMPETITIVE'); creationHistoricalRecord.setDate(Date.now()); creationHistoricalRecord.setPoints(points); - if(cookie) + if(cookie.username !== playAsGuestUsername) creationHistoricalRecord.sendRecord(cookie.username, cookie.token); } }, 1000); From 8479e47853f00ea187848265c1c019fefe8ec98c Mon Sep 17 00:00:00 2001 From: Mister-Mario Date: Wed, 24 Apr 2024 20:57:18 +0200 Subject: [PATCH 09/18] Fixed tests to test with new home --- webapp/src/components/Home/Home.test.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/webapp/src/components/Home/Home.test.js b/webapp/src/components/Home/Home.test.js index 0d16902b..78b36338 100644 --- a/webapp/src/components/Home/Home.test.js +++ b/webapp/src/components/Home/Home.test.js @@ -22,8 +22,8 @@ describe('Home component', () => { // Click toggle open button fireEvent.click(screen.getByTitle("home.clickOpen")); - // Icons of instructions, login and signup must be shown - expect(screen.getByAltText('Instructions')).toBeVisible(); + // Icons of play as guest, login and signup must be shown + expect(screen.getByAltText('Game')).toBeVisible(); expect(screen.getByAltText('Login')).toBeVisible(); expect(screen.getByAltText('Add user')).toBeVisible(); From e49e22e0992a91169cff2c76be497e426c92964a Mon Sep 17 00:00:00 2001 From: Mister-Mario Date: Thu, 25 Apr 2024 12:23:34 +0200 Subject: [PATCH 10/18] Updated backend --- users/authservice/auth-service.js | 2 +- .../auth-model.js => user-model.js} | 0 users/userservice/user-model.js | 24 ------------------- users/userservice/user-service.js | 2 +- 4 files changed, 2 insertions(+), 26 deletions(-) rename users/{authservice/auth-model.js => user-model.js} (100%) delete mode 100644 users/userservice/user-model.js diff --git a/users/authservice/auth-service.js b/users/authservice/auth-service.js index 924dc512..4b051af3 100644 --- a/users/authservice/auth-service.js +++ b/users/authservice/auth-service.js @@ -2,7 +2,7 @@ const express = require('express'); const mongoose = require('mongoose'); const bcrypt = require('bcrypt'); const jwt = require('jsonwebtoken'); -const User = require('./auth-model') +const User = require('../user-model') const app = express(); const port = 8002; diff --git a/users/authservice/auth-model.js b/users/user-model.js similarity index 100% rename from users/authservice/auth-model.js rename to users/user-model.js diff --git a/users/userservice/user-model.js b/users/userservice/user-model.js deleted file mode 100644 index e6643ff2..00000000 --- a/users/userservice/user-model.js +++ /dev/null @@ -1,24 +0,0 @@ -const mongoose = require('mongoose'); - -const userSchema = new mongoose.Schema({ - email: { - type: String, - required: true, - }, - username: { - type: String, - required: true, - }, - password: { - type: String, - required: true, - }, - createdAt: { - type: Date, - default: Date.now, - }, -}); - -const User = mongoose.model('User', userSchema); - -module.exports = User \ No newline at end of file diff --git a/users/userservice/user-service.js b/users/userservice/user-service.js index a6ea8a09..d0a3f455 100644 --- a/users/userservice/user-service.js +++ b/users/userservice/user-service.js @@ -4,7 +4,7 @@ const mongoose = require('mongoose'); const bcrypt = require('bcrypt'); const bodyParser = require('body-parser'); const jwt = require('jsonwebtoken'); -const User = require('./user-model') +const User = require('../user-model') const app = express(); const port = 8001; From e9843d4f1023304ce2d037ef4d48bc0e01262731 Mon Sep 17 00:00:00 2001 From: Mister-Mario Date: Thu, 25 Apr 2024 12:25:59 +0200 Subject: [PATCH 11/18] Fixed import --- users/authservice/auth-service.test.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/users/authservice/auth-service.test.js b/users/authservice/auth-service.test.js index 7f258f5c..bcd98b40 100644 --- a/users/authservice/auth-service.test.js +++ b/users/authservice/auth-service.test.js @@ -1,7 +1,7 @@ const request = require('supertest'); const { MongoMemoryServer } = require('mongodb-memory-server'); const bcrypt = require('bcrypt'); -const User = require('./auth-model'); +const User = require('../user-model'); let mongoServer; let app; From ceac612a0b4edcd980dd93682c0d79c09e893694 Mon Sep 17 00:00:00 2001 From: Mister-Mario Date: Thu, 25 Apr 2024 12:35:11 +0200 Subject: [PATCH 12/18] Fixed --- users/authservice/auth-service.js | 2 +- users/authservice/auth-service.test.js | 5 +++-- users/{ => authservice}/user-model.js | 0 users/userservice/user-model.js | 12 ++++++++++++ users/userservice/user-service.js | 2 +- 5 files changed, 17 insertions(+), 4 deletions(-) rename users/{ => authservice}/user-model.js (100%) create mode 100644 users/userservice/user-model.js diff --git a/users/authservice/auth-service.js b/users/authservice/auth-service.js index 4b051af3..a233e849 100644 --- a/users/authservice/auth-service.js +++ b/users/authservice/auth-service.js @@ -2,7 +2,7 @@ const express = require('express'); const mongoose = require('mongoose'); const bcrypt = require('bcrypt'); const jwt = require('jsonwebtoken'); -const User = require('../user-model') +const User = require('./user-model') const app = express(); const port = 8002; diff --git a/users/authservice/auth-service.test.js b/users/authservice/auth-service.test.js index bcd98b40..e1518db7 100644 --- a/users/authservice/auth-service.test.js +++ b/users/authservice/auth-service.test.js @@ -1,13 +1,14 @@ const request = require('supertest'); const { MongoMemoryServer } = require('mongodb-memory-server'); const bcrypt = require('bcrypt'); -const User = require('../user-model'); +const User = require('./user-model'); let mongoServer; let app; //test user let user = { + email: "user@gmail.com", username: 'testuser', password: 'testpassword', }; @@ -15,7 +16,7 @@ let user = { async function addUser(user){ const hashedPassword = await bcrypt.hash(user.password, 10); const newUser = new User({ - email: "user@gmail.com", + email: user.email, username: user.username, password: hashedPassword, createdAt: new Date() diff --git a/users/user-model.js b/users/authservice/user-model.js similarity index 100% rename from users/user-model.js rename to users/authservice/user-model.js diff --git a/users/userservice/user-model.js b/users/userservice/user-model.js new file mode 100644 index 00000000..5cdbaf69 --- /dev/null +++ b/users/userservice/user-model.js @@ -0,0 +1,12 @@ +const mongoose = require('mongoose'); + +const userSchema = new mongoose.Schema({ + email: String, + username: String, + password: String, + createdAt: Date, +}); + +const User = mongoose.model('User', userSchema); + +module.exports = User \ No newline at end of file diff --git a/users/userservice/user-service.js b/users/userservice/user-service.js index d0a3f455..a6ea8a09 100644 --- a/users/userservice/user-service.js +++ b/users/userservice/user-service.js @@ -4,7 +4,7 @@ const mongoose = require('mongoose'); const bcrypt = require('bcrypt'); const bodyParser = require('body-parser'); const jwt = require('jsonwebtoken'); -const User = require('../user-model') +const User = require('./user-model') const app = express(); const port = 8001; From 283543586ee5c80c0ac5d33190fc74bc61489a38 Mon Sep 17 00:00:00 2001 From: Mister-Mario Date: Thu, 25 Apr 2024 12:37:16 +0200 Subject: [PATCH 13/18] Added a conditional to not load the api-doc in case the .yaml does not exist --- gatewayservice/gateway-service.js | 18 +++++++++++------- 1 file changed, 11 insertions(+), 7 deletions(-) diff --git a/gatewayservice/gateway-service.js b/gatewayservice/gateway-service.js index f592f9e0..62b3dc83 100644 --- a/gatewayservice/gateway-service.js +++ b/gatewayservice/gateway-service.js @@ -171,15 +171,19 @@ app.get('/record/:user', verifyToken, async(req, res)=>{ }); // Read the OpenAPI YAML file synchronously -const file = fs.readFileSync('./openapi.yaml', 'utf8'); +const openapiPath = './openapi.yaml' +if(fs.existsSync(openapiPath)){ + const file = fs.readFileSync(openapiPath, 'utf8'); -// Parse the YAML content into a JavaScript object representing the Swagger document -const swaggerDocument = YAML.parse(file); + // Parse the YAML content into a JavaScript object representing the Swagger document + const swaggerDocument = YAML.parse(file); + + // Serve the Swagger UI documentation at the '/api-doc' endpoint + // This middleware serves the Swagger UI files and sets up the Swagger UI page + // It takes the parsed Swagger document as input + app.use('/api-doc', swaggerUi.serve, swaggerUi.setup(swaggerDocument)); +} -// Serve the Swagger UI documentation at the '/api-doc' endpoint -// This middleware serves the Swagger UI files and sets up the Swagger UI page -// It takes the parsed Swagger document as input -app.use('/api-doc', swaggerUi.serve, swaggerUi.setup(swaggerDocument)); // Start the gateway service const server = app.listen(port, () => { From bc59b7b7da5a73768cfa9090f8dca9853bd08a6d Mon Sep 17 00:00:00 2001 From: uo289267 Date: Thu, 25 Apr 2024 12:43:03 +0200 Subject: [PATCH 14/18] Added new steps --- webapp/e2e/features/competitiveGame.feature | 5 ++ webapp/e2e/steps/competitiveGame.steps.js | 50 +++++++++++++++++++ webapp/e2e/test-environment-setup.js | 4 +- .../GameConfigurator/GameConfigurator.js | 2 +- 4 files changed, 58 insertions(+), 3 deletions(-) create mode 100644 webapp/e2e/features/competitiveGame.feature create mode 100644 webapp/e2e/steps/competitiveGame.steps.js diff --git a/webapp/e2e/features/competitiveGame.feature b/webapp/e2e/features/competitiveGame.feature new file mode 100644 index 00000000..073860be --- /dev/null +++ b/webapp/e2e/features/competitiveGame.feature @@ -0,0 +1,5 @@ +Feature: Game Configurator and Competitive Game functionality + Scenario: Create Competitive Game should go to /questions + Given I am on the game configurator + When I click on new competitive game + Then I am in /questions \ No newline at end of file diff --git a/webapp/e2e/steps/competitiveGame.steps.js b/webapp/e2e/steps/competitiveGame.steps.js new file mode 100644 index 00000000..037bd8b7 --- /dev/null +++ b/webapp/e2e/steps/competitiveGame.steps.js @@ -0,0 +1,50 @@ +const puppeteer = require('puppeteer'); +const { defineFeature, loadFeature } = require('jest-cucumber'); +const setDefaultOptions = require('expect-puppeteer').setDefaultOptions; + +const feature = loadFeature('./features/gameMenu.feature'); + +const { register, login, logout } = require("../utils"); + +let page; +let browser; + +const email = "testUser@example.com"; +const username = "testUser" +const password = "testUserPassword" + +defineFeature(feature, test => { + + beforeAll(async () => { + browser = await puppeteer.launch({ + headless: "new", + slowMo: 40, + defaultViewport: { width: 1920, height: 1080 }, + args: ['--window-size=1920,1080'] + }); + + page = await browser.newPage(); + setDefaultOptions({ timeout: 30000 }); + + await register(page, email, username, password); + }); + + beforeEach(async () => { + await logout(page); + await login(page, username, password); + }) + + test('Create Competitive Game should go to /questions', ({ given, then }) => { + given('I am on the game configurator', async () => { + await page.goto('http://localhost:3000/configurator'); + await page.waitForSelector('.GameConfiguratorDiv'); + }); + when('I click on new competitive game', async () => { + await page.click('#competitive'); + }); + then('I am in /questions', async () => { + await expect(page).toMatchElement('.questionContainer'); + }); + }); + +}); diff --git a/webapp/e2e/test-environment-setup.js b/webapp/e2e/test-environment-setup.js index efd12b6e..349be901 100644 --- a/webapp/e2e/test-environment-setup.js +++ b/webapp/e2e/test-environment-setup.js @@ -12,14 +12,14 @@ async function startServer() { mongoserver = await MongoMemoryServer.create(); //Populate db - await loadQuestions(); + //await loadQuestions(); const mongoUri = mongoserver.getUri(); process.env.MONGODB_URI = mongoUri; process.env.MONGODB_URI_QUESTIONS = mongoUri; questionservice = await require("../../questionservice/question-service") recordservice = await require("../../users/recordservice/record-service"); - userservice = await require("../../users/userservice/user-service"); + //userservice = await require("../../users/userservice/user-service"); authservice = await require("../../users/authservice/auth-service"); gatewayservice = await require("../../gatewayservice/gateway-service"); } diff --git a/webapp/src/components/GameConfigurator/GameConfigurator.js b/webapp/src/components/GameConfigurator/GameConfigurator.js index 4a536729..5ab12b27 100644 --- a/webapp/src/components/GameConfigurator/GameConfigurator.js +++ b/webapp/src/components/GameConfigurator/GameConfigurator.js @@ -79,7 +79,7 @@ function ButtonCustomized({t,handleClick}) { function ButtonCompetitive({t}){ return ( - +

{t("gameConfigurator.play_competi")}

); From baa22ada34077cfa698c06a32592bb49d9f700f4 Mon Sep 17 00:00:00 2001 From: uo289267 Date: Thu, 25 Apr 2024 13:00:40 +0200 Subject: [PATCH 15/18] Fixed e2e --- webapp/e2e/steps/competitiveGame.steps.js | 6 +++--- webapp/e2e/test-environment-setup.js | 13 ++++++++++--- 2 files changed, 13 insertions(+), 6 deletions(-) diff --git a/webapp/e2e/steps/competitiveGame.steps.js b/webapp/e2e/steps/competitiveGame.steps.js index 037bd8b7..082c0cf9 100644 --- a/webapp/e2e/steps/competitiveGame.steps.js +++ b/webapp/e2e/steps/competitiveGame.steps.js @@ -2,7 +2,7 @@ const puppeteer = require('puppeteer'); const { defineFeature, loadFeature } = require('jest-cucumber'); const setDefaultOptions = require('expect-puppeteer').setDefaultOptions; -const feature = loadFeature('./features/gameMenu.feature'); +const feature = loadFeature('./features/competitiveGame.feature'); const { register, login, logout } = require("../utils"); @@ -17,7 +17,7 @@ defineFeature(feature, test => { beforeAll(async () => { browser = await puppeteer.launch({ - headless: "new", + headless: false, slowMo: 40, defaultViewport: { width: 1920, height: 1080 }, args: ['--window-size=1920,1080'] @@ -34,7 +34,7 @@ defineFeature(feature, test => { await login(page, username, password); }) - test('Create Competitive Game should go to /questions', ({ given, then }) => { + test('Create Competitive Game should go to /questions', ({ given,when, then }) => { given('I am on the game configurator', async () => { await page.goto('http://localhost:3000/configurator'); await page.waitForSelector('.GameConfiguratorDiv'); diff --git a/webapp/e2e/test-environment-setup.js b/webapp/e2e/test-environment-setup.js index 349be901..44e73ddb 100644 --- a/webapp/e2e/test-environment-setup.js +++ b/webapp/e2e/test-environment-setup.js @@ -12,14 +12,14 @@ async function startServer() { mongoserver = await MongoMemoryServer.create(); //Populate db - //await loadQuestions(); + await loadQuestions(); const mongoUri = mongoserver.getUri(); process.env.MONGODB_URI = mongoUri; process.env.MONGODB_URI_QUESTIONS = mongoUri; questionservice = await require("../../questionservice/question-service") recordservice = await require("../../users/recordservice/record-service"); - //userservice = await require("../../users/userservice/user-service"); + userservice = await require("../../users/userservice/user-service"); authservice = await require("../../users/authservice/auth-service"); gatewayservice = await require("../../gatewayservice/gateway-service"); } @@ -294,7 +294,14 @@ async function loadQuestions() { }] //No need of loading questions for these tests - //await Question.bulkSave(questions); + await questions.forEach(async question =>{ + let dbQuestion = new Question(); + dbQuestion.question=question.question; + dbQuestion.answers=question.answers; + dbQuestion.language=question.language; + dbQuestion.type=question.type; + await dbQuestion.save(); + }); } startServer(); From ec4ddb63b4db7f9c48768b3474750aea4fcb761f Mon Sep 17 00:00:00 2001 From: uo289267 Date: Thu, 25 Apr 2024 13:07:12 +0200 Subject: [PATCH 16/18] dont forget the new for headless --- webapp/e2e/steps/competitiveGame.steps.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/webapp/e2e/steps/competitiveGame.steps.js b/webapp/e2e/steps/competitiveGame.steps.js index 082c0cf9..95f41dad 100644 --- a/webapp/e2e/steps/competitiveGame.steps.js +++ b/webapp/e2e/steps/competitiveGame.steps.js @@ -17,7 +17,7 @@ defineFeature(feature, test => { beforeAll(async () => { browser = await puppeteer.launch({ - headless: false, + headless: "new", slowMo: 40, defaultViewport: { width: 1920, height: 1080 }, args: ['--window-size=1920,1080'] From 4709fc8e7578d31f43b59a8d54cfa5261bfb9d7b Mon Sep 17 00:00:00 2001 From: uo289267 Date: Thu, 25 Apr 2024 13:14:31 +0200 Subject: [PATCH 17/18] Fixed All for question configurator --- webapp/src/components/questionView/QuestionGenerator.js | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/webapp/src/components/questionView/QuestionGenerator.js b/webapp/src/components/questionView/QuestionGenerator.js index ec7a1b6d..5ede8c08 100644 --- a/webapp/src/components/questionView/QuestionGenerator.js +++ b/webapp/src/components/questionView/QuestionGenerator.js @@ -41,7 +41,10 @@ class QuestionGenerator{ let response; if(type==="COMPETITIVE"){ response = await axios.get(this.apiUrl + '/' + lang, {headers : {'token':token}}); - }else{ + }else if(type==="ALL"){ + response = await axios.get(this.apiUrl + '/' + lang + '/' +amount, {headers : {'token':token}}); + } + else{ response = await axios.get(this.apiUrl + '/' + lang + '/' +amount + '/' + type, {headers : {'token':token}}); } console.log(response) From d184f3436877f828ef51ecbab79e1549ab12472d Mon Sep 17 00:00:00 2001 From: uo289267 Date: Thu, 25 Apr 2024 16:07:50 +0200 Subject: [PATCH 18/18] fixed flaky --- webapp/e2e/features/competitiveGame.feature | 4 ++++ webapp/e2e/steps/competitiveGame.steps.js | 17 +++++++++++++++-- .../GameConfigurator/GameConfigurator.js | 2 +- 3 files changed, 20 insertions(+), 3 deletions(-) diff --git a/webapp/e2e/features/competitiveGame.feature b/webapp/e2e/features/competitiveGame.feature index 073860be..30620078 100644 --- a/webapp/e2e/features/competitiveGame.feature +++ b/webapp/e2e/features/competitiveGame.feature @@ -2,4 +2,8 @@ Feature: Game Configurator and Competitive Game functionality Scenario: Create Competitive Game should go to /questions Given I am on the game configurator When I click on new competitive game + Then I am in /questions + Scenario: Create Customized Game should go to /questions + Given I am on the game configurator + When I click on new customized game Then I am in /questions \ No newline at end of file diff --git a/webapp/e2e/steps/competitiveGame.steps.js b/webapp/e2e/steps/competitiveGame.steps.js index 95f41dad..8bd7682d 100644 --- a/webapp/e2e/steps/competitiveGame.steps.js +++ b/webapp/e2e/steps/competitiveGame.steps.js @@ -9,8 +9,8 @@ const { register, login, logout } = require("../utils"); let page; let browser; -const email = "testUser@example.com"; -const username = "testUser" +const email = "testUser1@example.com"; +const username = "testUser1" const password = "testUserPassword" defineFeature(feature, test => { @@ -46,5 +46,18 @@ defineFeature(feature, test => { await expect(page).toMatchElement('.questionContainer'); }); }); + + test('Create Customized Game should go to /questions', ({ given,when, then }) => { + given('I am on the game configurator', async () => { + await page.goto('http://localhost:3000/configurator'); + await page.waitForSelector('.GameConfiguratorDiv'); + }); + when('I click on new customized game', async () => { + await page.click('.linkButton'); + }); + then('I am in /questions', async () => { + await expect(page).toMatchElement('.questionContainer'); + }); + }); }); diff --git a/webapp/src/components/GameConfigurator/GameConfigurator.js b/webapp/src/components/GameConfigurator/GameConfigurator.js index 5ab12b27..899a8528 100644 --- a/webapp/src/components/GameConfigurator/GameConfigurator.js +++ b/webapp/src/components/GameConfigurator/GameConfigurator.js @@ -71,7 +71,7 @@ function ButtonRandomizeCustom({t,handleClick}){ function ButtonCustomized({t,handleClick}) { return ( - + ); }