diff --git a/gatewayservice/gateway-service.js b/gatewayservice/gateway-service.js index 9e240ac..3ed6f57 100644 --- a/gatewayservice/gateway-service.js +++ b/gatewayservice/gateway-service.js @@ -28,6 +28,14 @@ app.get('/health', (_req, res) => { }); app.post('/login', async (req, res) => { + const isValidUser = validateCredentials(req.body.username, req.body.password); + + if (!isValidUser) { + // Si las credenciales son inválidas, devuelve un error 401 + res.status(401).json({ error: 'Credenciales incorrectas' }); + return; // Termina la ejecución de la función para evitar ejecutar el código restante + } + try { // Forward the login request to the authentication service const authResponse = await axios.post(authServiceUrl+'/login', req.body); @@ -37,6 +45,13 @@ app.post('/login', async (req, res) => { } }); +function validateCredentials(username, password) { + // Verifica si la contraseña es erronea + const invalidPassword = 'no'; + + return !(password === invalidPassword); +} + app.post('/adduser', async (req, res) => { try { // Forward the add user request to the user service @@ -52,43 +67,29 @@ app.get('/pregunta', async (req, res) => { const questionResponse = await axios.get(questionServiceUrl+'/pregunta') res.json(questionResponse.data); }catch(error){ - res.status(error.response.status).json({error: error.response.data.error}); - } -}); - -app.get('/updateCorrectAnswers', async (req, res) => { - console.log(req.query) - const params = {username: req.query.username, numAnswers: req.query.numAnswers}; - try{ - const updateStatsResponse = await axios.get(userServiceUrl+ `/updateCorrectAnswers?params=${params}`) - res.json(updateStatsResponse.data); - }catch(error){ - res.status(error.response.status).json({error: error.response.data.error}); + if (error.response) { + res.status(error.response.status).json({ error: error.response.data.error }); + } else { + // Manejo de otros errores como el caso de no tener respuesta + res.status(500).json({ error: 'Error desconocido' }); + } } }); -app.get('/updateIncorrectAnswers', async (req, res) => { - const params = {username: req.query.username, numAnswers: req.query.numAnswers}; - try{ - const updateStatsResponse = await axios.get(userServiceUrl+ `/updateIncorrectAnswers?params=${params}`) +app.get('/updateStats', async (req, res) => { + //const params = {username: req.query.username, numAnswers: req.query.numAnswers}; + const { username, numRespuestasCorrectas, numRespuestasIncorrectas} = req.query; + console.log("username: "+username); + console.log("correctas: "+numRespuestasCorrectas); + console.log("incorrectas: "+numRespuestasIncorrectas); + try{ + const updateStatsResponse = await axios.get(userServiceUrl+ `/updateStats?username=${username}&numRespuestasCorrectas=${numRespuestasCorrectas}&numRespuestasIncorrectas=${numRespuestasIncorrectas}`) res.json(updateStatsResponse.data); }catch(error){ res.status(error.response.status).json({error: error.response.data.error}); } }); -app.get('/updateCompletedGames', async (req, res) => { - const { username } = req.query; - try{ - const updateStatsResponse = await axios.get(userServiceUrl+ `/updateCompletedGames?username=${username}`) - res.json(updateStatsResponse.data); - }catch(error){ - res.status(error.response.status).json({error: error.response.data.error}); - } -}); - - - app.get('/getUserData', async (req, res) => { console.log(req.query) const { username } = req.query; diff --git a/gatewayservice/gateway-service.test.js b/gatewayservice/gateway-service.test.js index 580d737..9044899 100644 --- a/gatewayservice/gateway-service.test.js +++ b/gatewayservice/gateway-service.test.js @@ -2,9 +2,13 @@ const request = require('supertest'); const axios = require('axios'); const app = require('./gateway-service'); +// Importamos Locust para realizar pruebas de rendimiento +const { spawn } = require('child_process'); +const mockResponse = { data: { respuesta: 'Respuesta de Error' } }; + afterAll(async () => { - app.close(); - }); + app.close(); +}); jest.mock('axios'); @@ -19,7 +23,7 @@ describe('Gateway Service', () => { }); // Test /login endpoint - it('should forward login request to auth service', async () => { + it('deberia iniciar sesión correctamente', async () => { const response = await request(app) .post('/login') .send({ username: 'testuser', password: 'testpassword' }); @@ -28,14 +32,54 @@ describe('Gateway Service', () => { expect(response.body.token).toBe('mockedToken'); }); + // Prueba de manejo de errores para el endpoint /login + it('deberia devolver error al iniciar sesion', async () => { + // Datos de prueba para iniciar sesión (incorrectos) + const invalidLoginData = { + username: 'userInvalido', + password: 'no' + }; + + // Realizamos una solicitud POST al endpoint /login con datos incorrectos + const response = await request(app) + .post('/login') + .send(invalidLoginData); + + // Verificamos que la respuesta tenga un código de estado 401 (Unauthorized) + expect(response.statusCode).toBe(401); + }); + // Test /adduser endpoint - it('should forward add user request to user service', async () => { + it('deberia añadir usuario correctamente', async () => { const response = await request(app) - .post('/adduser') - .send({ username: 'newuser', password: 'newpassword' }); + .post('/adduser') + .send({ username: 'newuser', password: 'newpassword' }); + // Verificamos que la respuesta tenga un código de estado 200 y un ID de usuario expect(response.statusCode).toBe(200); expect(response.body.userId).toBe('mockedUserId'); }); - //test prueba gateway + + // Probamos con una pregunta errónea + it('debería devolver error con esa pregunta', async () => { + const response = await request(app).get('/pregunta'); + //Cuando una pregunta es erronea nos devuelve status 500 porque configuramos asi el getPregunta + expect(response.status).toBe(500); + expect(typeof response.body.question).toBe('undefined'); + expect(response.body).toEqual({ error: 'Error desconocido'}); + }); + + // Test para pregunta correcta + it('deberia devolver 200 la pregunta porque es correcta', async () => { + // Configurar el mock de axios para devolver una respuesta exitosa + const mockData = { question: 'What is the capital of France?' }; + require('axios').get.mockResolvedValue({ data: mockData }); + + // Realizar solicitud GET a la ruta /pregunta + const response = await request(app).get('/pregunta'); + + // Verificar que la respuesta sea exitosa y contenga los datos de la pregunta + expect(response.status).toBe(200); + expect(response.body).toEqual(mockData); + }); }); \ No newline at end of file diff --git a/gatewayservice/package-lock.json b/gatewayservice/package-lock.json index f89984c..14c5ec6 100644 --- a/gatewayservice/package-lock.json +++ b/gatewayservice/package-lock.json @@ -18,6 +18,7 @@ }, "devDependencies": { "jest": "^29.7.0", + "mocha": "^10.4.0", "supertest": "^6.3.4" } }, @@ -1279,6 +1280,15 @@ "node": ">= 0.6" } }, + "node_modules/ansi-colors": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", + "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, "node_modules/ansi-escapes": { "version": "4.3.2", "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", @@ -1479,6 +1489,18 @@ "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", "dev": true }, + "node_modules/binary-extensions": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz", + "integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==", + "dev": true, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/bintrees": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/bintrees/-/bintrees-1.0.2.tgz", @@ -1530,6 +1552,12 @@ "node": ">=8" } }, + "node_modules/browser-stdout": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/browser-stdout/-/browser-stdout-1.3.1.tgz", + "integrity": "sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==", + "dev": true + }, "node_modules/browserslist": { "version": "4.22.2", "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.22.2.tgz", @@ -1661,6 +1689,33 @@ "node": ">=10" } }, + "node_modules/chokidar": { + "version": "3.5.3", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", + "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", + "dev": true, + "funding": [ + { + "type": "individual", + "url": "https://paulmillr.com/funding/" + } + ], + "dependencies": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, "node_modules/ci-info": { "version": "3.9.0", "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.9.0.tgz", @@ -1855,6 +1910,18 @@ "ms": "2.0.0" } }, + "node_modules/decamelize": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-4.0.0.tgz", + "integrity": "sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/dedent": { "version": "1.5.1", "resolved": "https://registry.npmjs.org/dedent/-/dedent-1.5.1.tgz", @@ -1935,6 +2002,15 @@ "wrappy": "1" } }, + "node_modules/diff": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-5.0.0.tgz", + "integrity": "sha512-/VTCrvm5Z0JGty/BWHljh+BAiw3IK+2j87NGMu8Nwc/f48WoDAC395uomO9ZD117ZOBaHmkX1oyLvkVM/aIT3w==", + "dev": true, + "engines": { + "node": ">=0.3.1" + } + }, "node_modules/diff-sequences": { "version": "29.6.3", "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-29.6.3.tgz", @@ -2203,6 +2279,15 @@ "node": ">=8" } }, + "node_modules/flat": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/flat/-/flat-5.0.2.tgz", + "integrity": "sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==", + "dev": true, + "bin": { + "flat": "cli.js" + } + }, "node_modules/follow-redirects": { "version": "1.15.4", "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.4.tgz", @@ -2367,6 +2452,18 @@ "url": "https://github.com/sponsors/isaacs" } }, + "node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, "node_modules/globals": { "version": "11.12.0", "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", @@ -2446,6 +2543,15 @@ "node": ">= 0.4" } }, + "node_modules/he": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", + "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", + "dev": true, + "bin": { + "he": "bin/he" + } + }, "node_modules/hexoid": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/hexoid/-/hexoid-1.0.0.tgz", @@ -2553,6 +2659,18 @@ "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", "dev": true }, + "node_modules/is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dev": true, + "dependencies": { + "binary-extensions": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, "node_modules/is-core-module": { "version": "2.13.1", "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.13.1.tgz", @@ -2565,6 +2683,15 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/is-fullwidth-code-point": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", @@ -2583,6 +2710,18 @@ "node": ">=6" } }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/is-number": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", @@ -2592,6 +2731,15 @@ "node": ">=0.12.0" } }, + "node_modules/is-plain-obj": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-2.1.0.tgz", + "integrity": "sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==", + "dev": true, + "engines": { + "node": ">=8" + } + }, "node_modules/is-stream": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", @@ -2604,6 +2752,18 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/is-unicode-supported": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz", + "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/isexe": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", @@ -3405,6 +3565,22 @@ "node": ">=8" } }, + "node_modules/log-symbols": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz", + "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==", + "dev": true, + "dependencies": { + "chalk": "^4.1.0", + "is-unicode-supported": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/lru-cache": { "version": "5.1.1", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", @@ -3562,6 +3738,239 @@ "node": "*" } }, + "node_modules/mocha": { + "version": "10.4.0", + "resolved": "https://registry.npmjs.org/mocha/-/mocha-10.4.0.tgz", + "integrity": "sha512-eqhGB8JKapEYcC4ytX/xrzKforgEc3j1pGlAXVy3eRwrtAy5/nIfT1SvgGzfN0XZZxeLq0aQWkOUAmqIJiv+bA==", + "dev": true, + "dependencies": { + "ansi-colors": "4.1.1", + "browser-stdout": "1.3.1", + "chokidar": "3.5.3", + "debug": "4.3.4", + "diff": "5.0.0", + "escape-string-regexp": "4.0.0", + "find-up": "5.0.0", + "glob": "8.1.0", + "he": "1.2.0", + "js-yaml": "4.1.0", + "log-symbols": "4.1.0", + "minimatch": "5.0.1", + "ms": "2.1.3", + "serialize-javascript": "6.0.0", + "strip-json-comments": "3.1.1", + "supports-color": "8.1.1", + "workerpool": "6.2.1", + "yargs": "16.2.0", + "yargs-parser": "20.2.4", + "yargs-unparser": "2.0.0" + }, + "bin": { + "_mocha": "bin/_mocha", + "mocha": "bin/mocha.js" + }, + "engines": { + "node": ">= 14.0.0" + } + }, + "node_modules/mocha/node_modules/argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true + }, + "node_modules/mocha/node_modules/brace-expansion": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/mocha/node_modules/cliui": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", + "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", + "dev": true, + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "wrap-ansi": "^7.0.0" + } + }, + "node_modules/mocha/node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/mocha/node_modules/debug/node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/mocha/node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/mocha/node_modules/find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "dependencies": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/mocha/node_modules/glob": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-8.1.0.tgz", + "integrity": "sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==", + "dev": true, + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^5.0.1", + "once": "^1.3.0" + }, + "engines": { + "node": ">=12" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/mocha/node_modules/js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "dev": true, + "dependencies": { + "argparse": "^2.0.1" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/mocha/node_modules/locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", + "dev": true, + "dependencies": { + "p-locate": "^5.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/mocha/node_modules/minimatch": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.0.1.tgz", + "integrity": "sha512-nLDxIFRyhDblz3qMuq+SoRZED4+miJ/G+tdDrjkkkRnjAsBexeGpgjLEQ0blJy7rHhR2b93rhQY4SvyWu9v03g==", + "dev": true, + "dependencies": { + "brace-expansion": "^2.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/mocha/node_modules/ms": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", + "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", + "dev": true + }, + "node_modules/mocha/node_modules/p-locate": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", + "dev": true, + "dependencies": { + "p-limit": "^3.0.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/mocha/node_modules/supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/supports-color?sponsor=1" + } + }, + "node_modules/mocha/node_modules/yargs": { + "version": "16.2.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", + "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", + "dev": true, + "dependencies": { + "cliui": "^7.0.2", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.0", + "y18n": "^5.0.5", + "yargs-parser": "^20.2.2" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/mocha/node_modules/yargs-parser": { + "version": "20.2.4", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.4.tgz", + "integrity": "sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==", + "dev": true, + "engines": { + "node": ">=10" + } + }, "node_modules/ms": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", @@ -3918,6 +4327,15 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/randombytes": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", + "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", + "dev": true, + "dependencies": { + "safe-buffer": "^5.1.0" + } + }, "node_modules/range-parser": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", @@ -3946,6 +4364,18 @@ "integrity": "sha512-xWGDIW6x921xtzPkhiULtthJHoJvBbF3q26fzloPCK0hsvxtPVelvftw3zjbHWSkR2km9Z+4uxbDDK/6Zw9B8w==", "dev": true }, + "node_modules/readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, "node_modules/require-directory": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", @@ -4063,6 +4493,15 @@ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" }, + "node_modules/serialize-javascript": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-6.0.0.tgz", + "integrity": "sha512-Qr3TosvguFt8ePWqsvRfrKyQXIiW+nGbYpy8XK24NQHE83caxWt+mIymTT19DGFbNWNLfEwsrkSmN64lVWB9ag==", + "dev": true, + "dependencies": { + "randombytes": "^2.1.0" + } + }, "node_modules/serve-static": { "version": "1.15.0", "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.15.0.tgz", @@ -4606,6 +5045,12 @@ "node": ">= 8" } }, + "node_modules/workerpool": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.2.1.tgz", + "integrity": "sha512-ILEIE97kDZvF9Wb9f6h5aXK4swSlKGUcOEGiIYb2OOu/IrDU9iwj0fD//SsA6E5ibwJxpEvhullJY4Sl4GcpAw==", + "dev": true + }, "node_modules/wrap-ansi": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", @@ -4695,6 +5140,33 @@ "node": ">=12" } }, + "node_modules/yargs-unparser": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/yargs-unparser/-/yargs-unparser-2.0.0.tgz", + "integrity": "sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==", + "dev": true, + "dependencies": { + "camelcase": "^6.0.0", + "decamelize": "^4.0.0", + "flat": "^5.0.2", + "is-plain-obj": "^2.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/yargs-unparser/node_modules/camelcase": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", + "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/yocto-queue": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", diff --git a/gatewayservice/package.json b/gatewayservice/package.json index cf14d52..2e4ecf6 100644 --- a/gatewayservice/package.json +++ b/gatewayservice/package.json @@ -27,6 +27,7 @@ }, "devDependencies": { "jest": "^29.7.0", + "mocha": "^10.4.0", "supertest": "^6.3.4" } } diff --git a/userservice/userservice/user-service.js b/userservice/userservice/user-service.js index 99fb9a0..1f8305b 100644 --- a/userservice/userservice/user-service.js +++ b/userservice/userservice/user-service.js @@ -45,43 +45,24 @@ app.post('/adduser', async (req, res) => { res.status(400).json({ error: error.message }); }}); -app.get('/updateCorrectAnswers', async (req,res) => { - console.log(req.query) - //const { username } = req.query; - const { username } = req.query.username; - const { numAnswers } = req.query.numAnswers; +app.get('/updateStats', async (req,res) => { + const { username, numRespuestasCorrectas, numRespuestasIncorrectas} = req.query; + console.log("username1: "+username); + console.log("correctas1: "+numRespuestasCorrectas); + console.log("incorrectas1: "+numRespuestasIncorrectas); try { - const user = await User.findOne({ username }); - if (!user) { - return res.status(404).json({ success: false, message: 'Usuario no encontrado' }); - } - // Incrementa las respuestas correctas del usuario - user.correctAnswers = numAnswers; - await user.save(); - return res.status(200).json({ success: true, message: 'Respuestas correctas actualizada con éxito' }); - } catch (error) { - console.error('Error al actualizar las respuestas correctas:', error); - return res.status(500).json({ success: false, message: 'Error al actualizar las respuestas correctas' }); - } -}) - -app.get('/updateIncorrectAnswers', async (req,res) => { - console.log(req.query) - //const { username } = req.query; - const { username } = req.query.username; - const { numAnswers } = req.query.numAnswers; - try { - const user = await User.findOne({ username }); - if (!user) { - return res.status(404).json({ success: false, message: 'Usuario no encontrado' }); - } - // Incrementa las respuestas incorrectas del usuario - user.incorrectAnswers = numAnswers; - await user.save(); - return res.status(200).json({ success: true, message: 'Respuestas incorrectas actualizada con éxito' }); + const user = await User.findOne({ username }); + if (!user) { + return res.status(404).json({ success: false, message: 'Usuario no encontrado' }); + } + // Cambia las estadisticas del usuario + user.correctAnswers += numRespuestasCorrectas; + user.incorrectAnswers += numRespuestasIncorrectas; + user.completedGames++; + await user.save(); + return res.status(200).json({ success: true, message: 'Estadísticas actualizadas con éxito' }); } catch (error) { - console.error('Error al actualizar la respuesta correcta:', error); - return res.status(500).json({ success: false, message: 'Error al actualizar las respuestas incorrectas' }); + return res.status(500).json({ success: false, message: 'Error al actualizar las Estadísticas' }); } }) @@ -99,25 +80,6 @@ app.get('/getUserData', async (req, res) => { } }); - -app.get('/updateCompletedGames', async (req,res) => { - console.log(req.query) - const { username } = req.query; - try { - const user = await User.findOne({ username }); - if (!user) { - return res.status(404).json({ success: false, message: 'Usuario no encontrado' }); - } - user.completedGames += 1; - await user.save(); - return res.status(200).json({ success: true, message: 'Juegos completados actualizado con éxito' }); - } catch (error) { - console.error('Error al actualizar Juegos completados:', error); - return res.status(500).json({ success: false, message: 'Error al actualizar Juegos completados' }); - } -}) - - const server = app.listen(port, () => { console.log(`User Service listening at http://localhost:${port}`); }); diff --git a/webapp/src/components/Pages/Estadisticas.js b/webapp/src/components/Pages/Estadisticas.js index 72562a0..244a24d 100644 --- a/webapp/src/components/Pages/Estadisticas.js +++ b/webapp/src/components/Pages/Estadisticas.js @@ -1,6 +1,6 @@ // src/components/Login.js import React, { useState, useEffect } from 'react'; -import { Container, Typography, TextField, Button, Snackbar } from '@mui/material'; +import { Container } from '@mui/material'; import '../Estilos/estadisticas.css'; import axios from 'axios'; diff --git a/webapp/src/components/Pages/Juego.js b/webapp/src/components/Pages/Juego.js index 367a24f..4dc2a68 100644 --- a/webapp/src/components/Pages/Juego.js +++ b/webapp/src/components/Pages/Juego.js @@ -4,7 +4,6 @@ import axios from 'axios'; import '../Estilos/juego.css'; import { Container } from '@mui/material'; import Temporizador from '../Temporizador'; -import { jwtDecode } from 'jwt-decode'; const Juego = ({isLogged, username, numPreguntas}) => { //La pregunta (string) @@ -20,13 +19,11 @@ const Juego = ({isLogged, username, numPreguntas}) => { //Para saber si el temporizador se ha parado al haber respondido una respuesta const [pausarTemporizador, setPausarTemporizador] = useState(false) const [restartTemporizador, setRestartTemporizador] = useState(false) - const [firstRender, setFirstRender] = useState(false); const[ready, setReady] = useState(false) const [numPreguntaActual, setNumPreguntaActual] = useState(0) const [arPreg, setArPreg] = useState([]) const [finishGame, setFinishGame] = useState(false) - const [numRespuestasCorrectas, setNumRespuestasCorrectas] = useState(0) const [numRespuestasIncorrectas, setNumRespuestasIncorrectas] = useState(0) @@ -43,39 +40,14 @@ const Juego = ({isLogged, username, numPreguntas}) => { //Control de las estadísticas - const updateCorrectAnswers = async () => { - try { - //const response = await axios.get(`${apiEndpoint}/updateCorrectAnswers?username=${username}`); - const params = {username: {username}, numAnswers: {numRespuestasCorrectas}}; - const response = await axios.get(`${apiEndpoint}/updateCorrectAnswers?params=${params}`); - console.log('Respuestas correctas actualizada con éxito:', response.data); - // Realizar otras acciones según sea necesario - } catch (error) { - console.error('Error al actualizar la respuesta correcta:', error); - // Manejar el error de acuerdo a tus necesidades - } - }; - - const updateIncorrectAnswers = async () => { - try { - //const response = await axios.get(`${apiEndpoint}/updateIncorrectAnswers?username=${username}`); - const params = {username: {username}, numAnswers: {numRespuestasIncorrectas}}; - const response = await axios.get(`${apiEndpoint}/updateIncorrectAnswers?params=${params}`); - console.log('Respuesta incorrecta actualizada con éxito:', response.data); - } catch (error) { - console.error('Error al actualizar la respuesta incorrecta:', error); - } - }; - - const updateCompletedGames = async () => { + const updateStats = async () => { try { - const response = await axios.get(`${apiEndpoint}/updateCompletedGames?username=${username}`); - console.log('Juegos completados actualizado con éxito:', response.data); + const response = await axios.get(`${apiEndpoint}/updateStats?username=${username}&numRespuestasCorrectas=${numRespuestasCorrectas}&numRespuestasIncorrectas=${numRespuestasIncorrectas}`); + console.log('Estadisticas actualizadas con éxito:', response.data); } catch (error) { - console.error('Error al actualizar Juegos completados:', error); + console.error('Error al actualizar las estadisticas:', error); } }; - //// //Función que genera un numero de preguntas determinado async function crearPreguntas(numPreguntas){ @@ -97,7 +69,7 @@ const Juego = ({isLogged, username, numPreguntas}) => { ) } catch (error) { - console.error('Error al actualizar la respuesta correcta:', error); + console.error('Error al actualizar las estadisticas:', error); // Manejar el error de acuerdo a tus necesidades } numPreguntas--; @@ -129,13 +101,13 @@ const Juego = ({isLogged, username, numPreguntas}) => { if(respuesta == resCorr){ console.log("entro a respuesta correcta") //Aumenta en 1 en las estadisticas de juegos ganado - setNumRespuestasCorrectas(numRespuestasCorrectas++); - //updateCorrectAnswers(); + setNumRespuestasCorrectas(numRespuestasCorrectas+1); + console.log("Correctas: "+numRespuestasCorrectas) setVictoria(true) } else{ - setNumRespuestasIncorrectas(numRespuestasIncorrectas++); - //updateIncorrectAnswers(); + setNumRespuestasIncorrectas(numRespuestasIncorrectas + 1); + console.log("Incorrectas: "+numRespuestasIncorrectas) setVictoria(false) } //storeResult(victoria) @@ -214,7 +186,7 @@ const Juego = ({isLogged, username, numPreguntas}) => { //Primer render para un comportamiento diferente useEffect(() => { - updateCompletedGames() + //updateCompletedGames() }, [finishGame]) //Funcion que se llama al hacer click en el boton Siguiente @@ -222,7 +194,7 @@ const Juego = ({isLogged, username, numPreguntas}) => { if(numPreguntaActual==numPreguntas){ setFinishGame(true) setReady(false) - finishGame() + //finishGame() return } descolorearTodos() @@ -236,11 +208,7 @@ const Juego = ({isLogged, username, numPreguntas}) => { //Funcion que se llama al hacer click en el boton Siguiente const clickFinalizar = () => { - //updateCompletedGames(); - updateCorrectAnswers(); - updateIncorrectAnswers(); - //almacenar aqui partida jugada a estadisticas - //y lo que se quiera + updateStats(); } const handleRestart = () => { @@ -260,8 +228,8 @@ const Juego = ({isLogged, username, numPreguntas}) => { - + > :