diff --git a/backend/README.md b/backend/README.md index 0a7be481fc14c8d8854e4b662fe053653f137dd0..3ec9552367bcd61a6c51276719cce4b86eddc0b7 100644 --- a/backend/README.md +++ b/backend/README.md @@ -1,36 +1,38 @@ # Backend -This backend handles connection with the database and ongoing gameStates. +The backend handles ongoing lobbies and games. Also communicates with the firebase database for persistance. +The backend uses Node and is written in TypeScript with Express for routing. -The backend uses the express framework to handle requests. +## Documentation -Implemation in Typescript. +The API documentation is available at the root of the backend (http://localhost:4999/). The documentation is generated with Swagger. -Package handler is Yarn. +<img src="./assets/apidoc.png" width="400" alt="API-docs"> ## Setup -You must add a file in the following directory: `backend/keys/fb-key.json` +To access database you must add firebase-key to the following directory: `backend/keys/fb-key.json` -The key is login details to a Firebase service account, needed to access the database. - -## How to run +## How to run the backend Run the following commands in your terminal: -- `yarn` last ned avhengigheter -- `yarn start:local` start serveren i lokal modus -- `yarn start:prod` start serveren i produksjonsmodus - -## Nyttige kommandoer +- `yarn` download dependencies +- `yarn start:local` start server in development mode on localhost:4999 +- `yarn start:prod` start server in production mode on {machine_ip}:80 -Formatere koden: `yarn prettier` -Kjøre tester: `yarn test` +## CD: Deployment to virtual machine -## VM Deployment +The backend is automatically deployed to a virtual machine when a commit is pushed to the main branch. The deployment is done by a Gitlab CI server that connects to the VM with SSH and runs the `vm.sh` script. -The backend runs on NTNU VM. +The VM is hosted on NTNU (requires VPN) and can be accessed on the following address: IP: 10.212.26.72 Port: 80 -`vm.sh` defines the code that is needed to boot the vm (after CI connectes to the VM with SSH) +The VM utilizes Nodemon to automatically restart the server if a crash occurs. + +## CI: Testing + +The backend is automatically tested by a Gitlab CI server for each commit. +Formatter: `yarn prettier` +e2e tests: `yarn test` diff --git a/backend/assets/apidoc.png b/backend/assets/apidoc.png new file mode 100644 index 0000000000000000000000000000000000000000..92f457d45445d43524568577220667bd8c5df0dc Binary files /dev/null and b/backend/assets/apidoc.png differ diff --git a/backend/documentation.md b/backend/documentation.md deleted file mode 100644 index a5f4039d6237a14cc80548813fb7ae6da3139ec7..0000000000000000000000000000000000000000 --- a/backend/documentation.md +++ /dev/null @@ -1,183 +0,0 @@ -# API Documentation - -This is an API documentation for an Express server that handles users, lobbies, and game state. The server uses Firebase Firestore for database storage. - -### Base URL - -> http://localhost:3000 - -### GET /user/ - -Description: Returns a list of all user IDs -Response: -Status: 200 OK -Body: Array of user IDs - -> - 204 No users found (DB is empty) -> - 200 List of user-id's - -### GET /user/{id: string} - -> - 404 User not found -> - 200 User - -### GET /user/{idOrUsername: string} - -Returns data for the user with the specified ID or username. If the parameter provided matches an ID, the response will contain data for that user. If the parameter matches a username, the response will contain data for the first user with that username found in the database. - -```json -Status: 200 OK -{ - "username": "user1", - "highscore": 50, - "games": 10, - "wins": 5, - "losses": 5 -} -``` - -> - 404 User not found -> - 200 User - -### POST /user/create/{username: string} - -Creates a new user with the specified username. The username must be unique. If the username is already taken, the response will contain an error message. - -> - 409 User already exists -> - 201 User created - -### GET /highscores - -Returns an array of the top 10 users with the highest scores, sorted in descending order. - -> - 204 No highscores found -> - 200 List of users with highest score - -```json -Status: 200 OK -{ - [ - { - "username": "user1", - "highscore": 50, - "games": 10, - "wins": 5, - "losses": 5 - }, - { - "username": "user2", - "highscore": 40, - "games": 8, - "wins": 4, - "losses": 4 - }, - ... - ] -} -``` - -### POST /lobby/:id - -Joins a lobby with the specified ID. If the lobby does not exist, a new lobby is created. - -```json -POST /lobby/lobby1 -Body: { "userId": "user1" } -``` - -> - 201 Created - -### POST /lobby/:id/leave - -Leaves a lobby with the specified ID. - -Example request: - -```json -POST /lobby/lobby1/leave -Body: { "userId": "user1" } -``` - -### GET /game/:gameid/currentTurn - -Checks if it is the specified user's turn in the specified game. Returns the current game state if it is the user's turn. - -Example request: - -```json -GET /game/game1/currentTurn -Body: { "userName": "user1" } -``` - -Example response: - -```json -Status: 200 OK -{ - "gameId": "12345678", - "gameStatus": false, - "currentTurn": 0, - "users": { - [ - "User": { - "id": string; - "username": string; - "wins": number; - "losses": number; - "highscore": number; - }, - "IStat": { - "position": [0,0]; - "turretAngle": 0.5; - "isMirrored": true; - "health": 100; - "ammunition": 100; - "tankDirection": "left"; - "tankType": "M107"; - "score": 100; - }, - ... - ] - }, -} -``` - -### POST /game/:gameid/move - -Updates the game state for the specified game with the new game state provided in the request body. - -Example request: - -```json -POST /game/123456/move -Body: -{ "newGameState": - { - "gameId": "12345678", - "gameStatus": false, - "currentTurn": 0, - "users": { - [ - "User": { - "id": string; - "username": string; - "wins": number; - "losses": number; - "highscore": number; - }, - "IStat": { - "position": [0,0]; - "turretAngle": 0.5; - "isMirrored": true; - "health": 100; - "ammunition": 100; - "tankDirection": "left"; - "tankType": "M107"; - "score": 100; - }, - ... - ] - }, - } -} -``` diff --git a/backend/package.json b/backend/package.json index 5abc2784c8ad7677dbfab06e4f46bcd730ea26c8..d39e78f1700e8b0b1aef12af761db84fcb718bd5 100644 --- a/backend/package.json +++ b/backend/package.json @@ -16,6 +16,7 @@ "@types/cypress": "^1.1.3", "@types/express": "^4.17.17", "@types/jest": "^29.5.0", + "@types/swagger-jsdoc": "^6.0.1", "concurrently": "^8.0.1", "cross-env": "^7.0.3", "cypress": "^12.9.0", @@ -26,7 +27,10 @@ }, "dependencies": { "firebase-admin": "^11.5.0", + "node-cache": "^5.1.2", "prettier": "^2.8.4", - "simplex-noise": "^4.0.1" + "simplex-noise": "^4.0.1", + "swagger-jsdoc": "^6.2.8", + "swagger-ui-express": "^4.6.2" } } diff --git a/backend/src/controllers/gameController.ts b/backend/src/controllers/gameController.ts index 15499ec6e6db975a1eadff7d944e8b4fa4904520..85aa6d1a5de72c3dd709ec199dbf460b419a2a97 100644 --- a/backend/src/controllers/gameController.ts +++ b/backend/src/controllers/gameController.ts @@ -37,7 +37,7 @@ export const move = async (req: Request, res: Response): Promise<void> => { export const currentTurn = async (req: Request, res: Response): Promise<void> => { const game = gameHandler.getGameById(req.params.gameid); - const user = req.body.userName; + const user = req.body.username || req.body.userName; if (game) { if (user === game.getCurrentTurnUser().username) { // send the gamestate to the client diff --git a/backend/src/controllers/highscoreController.ts b/backend/src/controllers/highscoreController.ts index 31099358fcde2951b84e7bd5f9d5bdfe0906b010..dd508aaea2fb0dc33724db236967c1b150c436ef 100644 --- a/backend/src/controllers/highscoreController.ts +++ b/backend/src/controllers/highscoreController.ts @@ -5,17 +5,17 @@ import { getUserById } from '../functions/getUserById'; import { User } from '../../types/User'; import admin from '../functions/firebaseAdmin'; import { IGame } from '../interfaces/IGame'; +import { getTopUsers } from '../functions/firebaseCache'; const gameHandler = GameHandler.getInstance(); export const top10 = async (req: Request, res: Response): Promise<void> => { - const usersRef = admin.firestore().collection('users'); - const querySnapshot = await usersRef.orderBy('highscore', 'desc').limit(10).get(); + const topUsers = await getTopUsers(); - if (querySnapshot.empty) { + if (topUsers === null) { res.status(204).send('No highscores found'); } else { - const users = querySnapshot.docs.map( + const users = topUsers.docs.map( (doc: any) => ({ id: doc.id, diff --git a/backend/src/controllers/userController.ts b/backend/src/controllers/userController.ts index d919b7b6c6c59ccf1cb4177b6a4afc0740a6edf0..13b7676f675f1fbd7105fe117ee4e5db7589620d 100644 --- a/backend/src/controllers/userController.ts +++ b/backend/src/controllers/userController.ts @@ -3,18 +3,18 @@ import { GameHandler } from '../gameHandler'; import { getUserById } from '../functions/getUserById'; import { User } from '../../types/User'; import admin from '../functions/firebaseAdmin'; +import { getUsers } from '../functions/firebaseCache'; +import { log } from '../functions/console'; const gameHandler = GameHandler.getInstance(); export const users = async (req: Request, res: Response): Promise<void> => { - // returns all the user-id's - const usersRef = admin.firestore().collection('users'); - const querySnapshot = await usersRef.get(); + const users = await getUsers(); - if (querySnapshot.empty) { + if (users == null) { res.status(204).send('No users found'); } else { - const userids = querySnapshot.docs.map((doc: { id: any }) => doc.id); + const userids = users.docs.map((doc: { id: any }) => doc.id); res.status(200).send(userids); } }; @@ -34,31 +34,22 @@ export const getUser = async (req: Request, res: Response): Promise<void> => { export const createUser = async (req: Request, res: Response): Promise<void> => { const usersRef = admin.firestore().collection('users'); const username = req.params.username; - const querySnapshot = await usersRef.where('username', '==', username).get(); + log('firestore: quering for user with username: ' + username); + const user = await getUserById(username); - if (!querySnapshot.empty) { - const user: User[] = querySnapshot.docs.map((doc: any) => { - if (doc.data().username === username) { - return { id: doc.id, ...doc.data() }; - } - }); - res.status(409).send(user[0]); + if (user) { + res.status(409).send(user); } else { - const newUserRef = await usersRef.add({ - username: req.params.username, - highscore: 0, - games: 0, - wins: 0, - losses: 0, - }); - res.status(201).send({ + const initialUserData = { username: req.params.username, highscore: 0, games: 0, wins: 0, losses: 0, - id: newUserRef.id, - } as User); + }; + log('firestore: creating new user with username: ' + username); + const newUserRef = await usersRef.add(initialUserData); + res.status(201).send({ id: newUserRef.id, ...initialUserData } as User); } }; @@ -66,15 +57,16 @@ export const createUser = async (req: Request, res: Response): Promise<void> => export const deleteUser = async (req: Request, res: Response): Promise<void> => { const usersRef = admin.firestore().collection('users'); const username = req.params.username; + log('firestore: quering for user with username: ' + username); const querySnapshot = await usersRef.where('username', '==', username).get(); if (!querySnapshot.empty) { // delete user - const user: User[] = querySnapshot.docs.map((doc: any) => { - if (doc.data().username === username) { - return { id: doc.id, ...doc.data() }; - } - }); - await usersRef.doc(user[0].id).delete(); + const userDoc = querySnapshot.docs[0]; + const user: User = { id: userDoc.id, ...userDoc.data() }; + log( + 'firestore: deleting user with id: ' + user.id + ' and username: ' + user.username + ); + await usersRef.doc(user.id).delete(); res.status(204).send('User deleted'); } else { res.status(404).send('User not found'); diff --git a/backend/src/functions/console.ts b/backend/src/functions/console.ts index d1b44c809153dd33789e346350b3a78b1796846e..d6c294e9c0e37c21dcd1fca09f3efc51b1ae219f 100644 --- a/backend/src/functions/console.ts +++ b/backend/src/functions/console.ts @@ -4,12 +4,18 @@ * @param message The message to log * @returns void */ -export function log(message: string): void { +export function log(message: string, status: string = 'info'): void { const time = new Date().toLocaleTimeString(); - const file = new Error().stack?.split('at ')[2].split(' ')[0]; - const maxLength = 20; // Set the desired maximum length for the file section - const paddedFile = file?.padEnd(maxLength, ' '); - console.log(time + ' | ' + paddedFile + '| ' + message); + const file = new Error().stack?.split('at ')[2].split('\n')[0].trim(); + const maxLength = 25; // Set the desired maximum length for the file section + const paddedFile = + file?.split('/').pop()?.padEnd(maxLength, ' ') ?? ''.padEnd(maxLength, ' '); + // if status is info, log in default/black color. If warning use yellow, if danger use red + const statusColor = + status === 'info' ? '\x1b[0m' : status === 'warning' ? '\x1b[33m' : '\x1b[31m'; + process.stdout.write( + statusColor + time + ' | ' + paddedFile + ' | ' + message + '\x1b[0m\n' + ); } /** diff --git a/backend/src/functions/expressLogger.ts b/backend/src/functions/expressLogger.ts deleted file mode 100644 index 28852de1c393e3ad26da6a586a94c28dfbd3fe93..0000000000000000000000000000000000000000 --- a/backend/src/functions/expressLogger.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { log } from './console'; - -/** - * Custom middleware to log express requests - * @param req - * @param res - * @param next - */ - -export function expressLogger(req: any, res: any, next: any) { - // Store the original `.send()` method - const originalSend = res.send; - - // Override the `.send()` method - res.send = function (body: any) { - // Log the response - log(`${req.method} ${req.url} - Response: ${body}`); - - // Call the original `.send()` method with the provided body - originalSend.call(res, body); - }; - - // Call the next middleware or route handler in the stack - next(); -} diff --git a/backend/src/functions/firebaseCache.ts b/backend/src/functions/firebaseCache.ts new file mode 100644 index 0000000000000000000000000000000000000000..2104507f953beefc6f9fa709ad8e1b4c9e587c3c --- /dev/null +++ b/backend/src/functions/firebaseCache.ts @@ -0,0 +1,75 @@ +import NodeCache from 'node-cache'; +import admin from './firebaseAdmin'; +import { log } from './console'; + +/** + * This is a cache and firebaseHandler for the firestore database + */ + +const queryCache = new NodeCache({ stdTTL: 60 }); // 60 seconds TTL + +export async function getUsers(): Promise<any> { + const usersRef = admin.firestore().collection('users'); + const cacheKey = 'userRef'; + return retrieveFromCache(usersRef, queryCache, cacheKey); +} + +export async function getTopUsers(): Promise<any> { + const usersRef = admin.firestore().collection('users'); + const querySnapshot = await usersRef.orderBy('highscore', 'desc').limit(10); + const cacheKey = 'topUsers'; + return retrieveFromCache(querySnapshot, queryCache, cacheKey); +} + +/** + * A function that retrieves data from the cache if it exists, otherwise it retrieves it from firestore + * + * @param ref Firestore reference + * @param cache The cache to retrieve data from + * @param cacheKey The key to use for the cache + * @returns cached or firestore result (null if error) + */ +async function retrieveFromCache( + ref: any, + cache: NodeCache, + cacheKey: string +): Promise<any | null> { + const cachedValue = cache.get(cacheKey); + + if (cachedValue) { + log('Cache-hit: retrieved from cache'); + return cachedValue; + } else { + log('Cache-miss: retrieving from firestore...'); + return await sendFirestoreRequest(ref, cache, cacheKey); + } +} + +/** + * Sends a request to firestore and returns the result + * + * @param ref + * @param cache + * @param cacheKey + * @returns firestore result + */ +async function sendFirestoreRequest( + ref: any, + cache: NodeCache, + cacheKey: string +): Promise<any | null> { + try { + log('firebase: sending request to firestore...'); + const snapshot = await ref.get(); + cache.set(cacheKey, snapshot); + log("Cache-set: set cache for key '" + cacheKey + "'"); + return snapshot; + } catch (err: any) { + if (err?.code === 8) { + log('error: firestore max-limit reached', 'danger'); + } else { + log('error: firestore error code ' + err?.code, 'danger'); + } + return null; + } +} diff --git a/backend/src/functions/getUserById.ts b/backend/src/functions/getUserById.ts index dfe6901a04e2c99b19ea6682b7dff35b6ef31798..5810764e22d416516ec2fd034ef2ba319d182aeb 100644 --- a/backend/src/functions/getUserById.ts +++ b/backend/src/functions/getUserById.ts @@ -4,22 +4,26 @@ import admin from './firebaseAdmin'; export async function getUserById(id: string): Promise<User | null> { const usersRef = admin.firestore().collection('users'); - const snapshot = await usersRef.get(); const searchParam = id; - const user = - snapshot.docs.find((doc: { id: string }) => doc.id === searchParam) || - ( - await Promise.all( - snapshot.docs.map(async (doc: { id: string }) => { - const userSnapshot = await usersRef.doc(doc.id).get(); - return userSnapshot.data().username === searchParam ? userSnapshot : null; - }) - ) - ).find(Boolean); + // Search for user by id + log('firebase: searching for user by id: ' + searchParam); + let userSnapshot = await usersRef.doc(searchParam).get(); - if (user) { - return user.data() as User; + // If the user is not found by id, search for user by username + if (!userSnapshot.exists) { + log('firebase: searching for user by username: ' + searchParam); + const userByUsernameSnapshot = await usersRef + .where('username', '==', searchParam) + .get(); + if (!userByUsernameSnapshot.empty) { + userSnapshot = userByUsernameSnapshot.docs[0]; + } + } + + if (userSnapshot) { + log('found user: ' + userSnapshot.id); + return userSnapshot.data() as User; } else { log('User not found'); return null; diff --git a/backend/src/index.ts b/backend/src/index.ts index 85eda66a602052c4520eae4b848b8687277ab991..8ba97fb84085792e528589257f3014f19a60d9e6 100644 --- a/backend/src/index.ts +++ b/backend/src/index.ts @@ -1,6 +1,6 @@ import express from 'express'; import { GameHandler } from './gameHandler'; -import { expressLogger } from './functions/expressLogger'; +import { expressLogger } from './middleware/expressLogger'; import lobbyRoutes from './routes/lobbyRoutes'; import userRoutes from './routes/userRoutes'; import gameRoutes from './routes/gameRoutes'; @@ -9,6 +9,9 @@ import serverRoutes from './routes/serverRoutes'; import { log } from './functions/console'; import { welcome } from './functions/welcomeScreen'; import { disposeInactiveGames } from './functions/disposeInactiveGames'; +import swaggerJsdoc from 'swagger-jsdoc'; +import path from 'path'; +import { firebaseLogger } from './middleware/firebaseLogger'; new GameHandler(); // singleton ;) @@ -18,7 +21,7 @@ const port = process.env.NODE_ENV === 'production' ? 80 : 4999; // middleware app.use(express.json()); // for parsing application/json -app.use(expressLogger); // for logging +app.use(expressLogger); // for request logging // routes app.use('/lobby', lobbyRoutes); @@ -27,13 +30,40 @@ app.use('/game', gameRoutes); app.use('/highscore', highscoreRoutes); app.use('/server', serverRoutes); -// in testing, we don't want to cache the results +// Cache-Control app.set('etag', false); app.use((req, res, next) => { res.setHeader('Cache-Control', 'no-store'); next(); }); +// API Documentation +const swaggerUi = require('swagger-ui-express'); +const swaggerJSDoc = require('swagger-jsdoc'); +const swaggerOptions: swaggerJsdoc.Options = { + definition: { + openapi: '3.0.0', + info: { + title: 'TankWars API Documentation', + version: '1.0.1', + description: + 'The API lets clients create lobbies and send gamestates between users', + }, + servers: [ + { + url: 'http://localhost:4999', + }, + { + url: 'http://10.212.26.72:80', + }, + ], + }, + apis: ['./src/routes/*.ts', './src/controllers/*.ts'], +}; + +const swaggerDocs = swaggerJsdoc(swaggerOptions); +app.use('/', swaggerUi.serve, swaggerUi.setup(swaggerDocs)); + setInterval(() => { disposeInactiveGames(); }, 60 * 1000); @@ -43,7 +73,7 @@ app.listen(port, () => { if (process.env.NODE_ENV === 'production') { log('PRODUCTION MODE'); } else { - log('DEVELOPMENT MODE'); + log('DEVELOPMENT MODE', 'warning'); } - log(`Server started on port ${port}`); + log(`Server started on port ${port}`, 'warning'); }); diff --git a/backend/src/middleware/expressLogger.ts b/backend/src/middleware/expressLogger.ts new file mode 100644 index 0000000000000000000000000000000000000000..559bbdb3b56b4172ce44c116a0952057867b08db --- /dev/null +++ b/backend/src/middleware/expressLogger.ts @@ -0,0 +1,31 @@ +import { log } from '../functions/console'; +import { Request, Response, NextFunction } from 'express'; + +/** + * Custom middleware to log express requests + * @param req + * @param res + * @param next + */ + +export function expressLogger(req: Request, res: Response, next: NextFunction) { + // Store the original `.send()` method + const originalSend = res.send; + + // Override the `.send()` method + res.send = function (body: any): Response { + const bodyString: string = body.toString(); + if (bodyString.includes('<!DOCTYPE html>') || bodyString.includes('window.onload')) { + // dont print html/js files + log(`${req.method} ${req.url} - Response: HTML / JS file`); + } else { + log(`${req.method} ${req.url} - Response: ${body}`); + } + + // Call the original `.send()` method with the provided body + return originalSend.call(res, body); + }; + + // Call the next middleware or route handler in the stack + next(); +} diff --git a/backend/src/middleware/firebaseLogger.ts b/backend/src/middleware/firebaseLogger.ts new file mode 100644 index 0000000000000000000000000000000000000000..8f942121e2d2e7ea2b1ca0aa6394098407a8b092 --- /dev/null +++ b/backend/src/middleware/firebaseLogger.ts @@ -0,0 +1,15 @@ +import { Request, Response, NextFunction } from 'express'; +import { log } from '../functions/console'; + +export function firebaseLogger(req: Request, res: Response, next: NextFunction) { + const startTime = Date.now(); + const path = req.path; + + res.on('finish', () => { + const endTime = Date.now(); + const duration = endTime - startTime; + log('Firebase read/request: ' + path + ', duration: ' + duration + 'ms'); + }); + + next(); +} diff --git a/backend/src/routes/gameRoutes.ts b/backend/src/routes/gameRoutes.ts index 73f105d33649856fabbf50c0e9d114e2badac10f..865cb52f62ed239f1dfa19b9f99c241acbd09b97 100644 --- a/backend/src/routes/gameRoutes.ts +++ b/backend/src/routes/gameRoutes.ts @@ -3,9 +3,194 @@ import * as gameController from '../controllers/gameController'; const router = express.Router(); +/** + * @swagger + * /game/{lobbyId}: + * get: + * tags: [Game] + * summary: Retrive the gameID of a lobby. + * description: Retrive the gameID of a lobby. + * responses: + * 200: + * description: A gameId. + * content: + * text/plain: + * schema: + * type: string + * example: "0.0d8441fqd" + * 404: + * description: Lobby does not have ongoing game. + */ router.get('/:lobbyId', gameController.gameId); + +/** + * @swagger + * /game/move: + * post: + * tags: [Game] + * summary: Send a move. + * description: Client uses this to send a gameState as JSON to the server. + * parameters: + * - name: gameId + * in: path + * required: true + * description: gameId. + * schema: + * type: string + * requestBody: + * required: true + * content: + * application/json: + * schema: + * type: object + * properties: + * gameId: + * type: string + * gameStatus: + * type: boolean + * description: True if game is ongoing, false if game is over. + * currentTurn: + * type: integer + * users: + * type: array + * items: + * type: array + * items: + * oneOf: + * - type: object + * properties: + * wins: + * type: integer + * highscore: + * type: integer + * games: + * type: integer + * losses: + * type: integer + * username: + * type: string + * - type: object + * properties: + * position: + * type: integer + * turretAngle: + * type: integer + * health: + * type: integer + * ammunition: + * type: integer + * score: + * type: integer + * isMirrored: + * type: boolean + * tankDirection: + * type: string + * tankType: + * type: string + * responses: + * 200: + * description: A move was made. + * content: + * text/plain: + * schema: + * type: string + * example: Move made + * 404: + * description: A move was made. + * content: + * text/plain: + * schema: + * type: string + * example: Game not found + */ router.post('/:gameid/move', gameController.move); + +/** + * @swagger + * /game/{id}/currentTurn: + * post: + * tags: [Game] + * summary: Check current turn. + * description: If it is clients turn, a gamestate JSON will be sent. + * parameters: + * - name: username + * in: query + * required: true + * description: Username of the user polling the game. + * schema: + * type: string + * responses: + * 200: + * description: A move was made. + * content: + * application/json: + * schema: + * type: object + * properties: + * gameId: + * type: string + * gameStatus: + * type: boolean + * description: True if game is ongoing, false if game is over. + * currentTurn: + * type: integer + * users: + * type: array + * items: + * type: array + * items: + * oneOf: + * - type: object + * properties: + * wins: + * type: integer + * highscore: + * type: integer + * games: + * type: integer + * losses: + * type: integer + * username: + * type: string + * - type: object + * properties: + * position: + * type: integer + * turretAngle: + * type: integer + * health: + * type: integer + * ammunition: + * type: integer + * score: + * type: integer + * isMirrored: + * type: boolean + * tankDirection: + * type: string + * tankType: + * type: string + */ router.get('/:gameid/currentTurn', gameController.currentTurn); + +/** + * @swagger + * /game/{gameId}/terrain: + * get: + * tags: [Game] + * summary: Retrive the terrain of a game. + * description: Retrive the terrain of a game. + * responses: + * 200: + * description: An array of numbers representing the terrain. + * content: + * text/plain: + * schema: + * type: array + * example: [1,2,2,1,1,2,3] + * 404: + * description: Game not found. + */ router.get('/:gameid/terrain', gameController.getTerrain); export default router; diff --git a/backend/src/routes/highscoreRoutes.ts b/backend/src/routes/highscoreRoutes.ts index fcf34ec0d6edae1e525dc9dd6a6c91b3956d94c4..205df0fd75f020d87e2725b3cec6f629e2da1ba1 100644 --- a/backend/src/routes/highscoreRoutes.ts +++ b/backend/src/routes/highscoreRoutes.ts @@ -3,6 +3,43 @@ import * as highscoreController from '../controllers/highscoreController'; const router = express.Router(); -router.get('/', highscoreController.top10); +/** + * @swagger + * /highscore/top10: + * get: + * tags: [Highscore] + * summary: Get the top 10 highest ranking users. + * responses: + * 200: + * description: Top 10 highest ranking users found. + * content: + * application/json: + * schema: + * type: array + * items: + * type: object + * properties: + * username: + * type: string + * description: The username of the user. + * highscore: + * type: number + * description: The highscore of the user. + * games: + * type: number + * description: The number of games the user has played. + * wins: + * type: number + * description: The number of games the user has won. + * losses: + * type: number + * description: The number of games the user has lost. + * id: + * type: string + * description: The ID of the user. + * 204: + * description: No highscores found. + */ +router.get('/top10', highscoreController.top10); export default router; diff --git a/backend/src/routes/lobbyRoutes.ts b/backend/src/routes/lobbyRoutes.ts index 2e006b95ea924354ff18741dfbad241eeefa8d0a..5318f5ee1b5c00da1a47f8b9c50ee11a07d44474 100644 --- a/backend/src/routes/lobbyRoutes.ts +++ b/backend/src/routes/lobbyRoutes.ts @@ -3,7 +3,82 @@ import * as lobbyController from '../controllers/lobbyController'; const router = express.Router(); +/** + * @swagger + * /lobby/{id}/leave: + * post: + * tags: [Lobby] + * summary: Leaves a lobby. + * description: The user leaves the lobby with the given id. If the lobby is empty it will be disposed automatically. + * parameters: + * - name: username + * in: path + * required: true + * description: Username of the user leaving the lobby. + * schema: + * type: string + * responses: + * 200: + * description: User left lobby. + * content: + * text/plain: + * schema: + * type: string + * example: "Lobby left" + */ router.post('/:id/leave', lobbyController.leaveLobby); + +/** + * @swagger + * /lobby/{id}/join: + * post: + * tags: [Lobby] + * summary: Joins a lobby. + * description: The user joins the lobby with the given id. If the lobby is full, a game will be started automatically. + * parameters: + * - name: username + * in: path + * required: true + * description: Username of the user joining the lobby. + * schema: + * type: string + * responses: + * 200: + * description: Successfully joined lobby. (opponent already joined, game will be started automatically) + * content: + * text/plain: + * schema: + * type: string + * example: "Joined lobby" + * 201: + * description: Successfully created lobby. (needs to wait for opponent to join) + * content: + * text/plain: + * schema: + * type: string + * example: "Created lobby" + * 404: + * description: Lobby not found. + * content: + * text/plain: + * schema: + * type: string + * example: "Lobby not found" + * 409: + * description: User already joined lobby. + * content: + * text/plain: + * schema: + * type: string + * example: "User already in lobby" + * 429: + * description: Lobby is full. + * content: + * text/plain: + * schema: + * type: string + * example: "Lobby is full" + */ router.post('/:id/join', lobbyController.joinLobby); export default router; diff --git a/backend/src/routes/serverRoutes.ts b/backend/src/routes/serverRoutes.ts index 29eebfd075cd02c03509958163a46529ce6a5549..89a5b6c2bd3e23ac889a5631024e74df2c71d257 100644 --- a/backend/src/routes/serverRoutes.ts +++ b/backend/src/routes/serverRoutes.ts @@ -3,6 +3,38 @@ import * as serverController from '../controllers/serverController'; const router = express.Router(); +/** + * @swagger + * tags: + * - name: User + * description: Endpoints for managing users. + * - name: Game + * description: Endpoints for interacting with games. + * - name: Server + * description: Endpoints for interacting with the server. + * - name: Lobby + * description: Endpoints for interacting with lobbies. + * - name: Highscore + * description: Endpoints for interacting with highscores. + * + */ + +/** + * @swagger + * /server/heartbeat: + * get: + * tags: [Server] + * summary: Retrieve a response from the server (heartbeat) + * description: If the server is alive, it will respond with a 200 status code and the message "Server is alive" + * responses: + * 200: + * description: A heartbeat response. + * content: + * text/plain: + * schema: + * type: string + * example: "Server is alive" + */ router.get('/heartbeat', serverController.heartbeat); export default router; diff --git a/backend/src/routes/userRoutes.ts b/backend/src/routes/userRoutes.ts index a781f711b464a98c28448bb5d010c67d6c1545da..72e6a91b8fb7557559f389695cd7810b77bd7b8d 100644 --- a/backend/src/routes/userRoutes.ts +++ b/backend/src/routes/userRoutes.ts @@ -3,9 +3,114 @@ import * as userController from '../controllers/userController'; const router = express.Router(); +/** + * @swagger + * /user: + * get: + * summary: Returns all user IDs. + * tags: [User] + * responses: + * 200: + * description: List of user IDs. + * content: + * application/json: + * schema: + * type: array + * items: + * type: string + * example: ["0t73mMGkAADD7zjJVWXU","1mGO1z9484Y8OoCvjsF4","1yBYv7JzZupLCfmdmqLh","32E9WGluUTYu4DXl6lxh"] + * + * 204: + * description: No users found. + */ router.get('/', userController.users); -router.get('/:idOrUsername', userController.getUser); -router.post('/create/:username', userController.createUser); -router.post('/delete/:username', userController.deleteUser); // should be router.delete + +/** + * @swagger + * /user/{username}: + * get: + * tags: [User] + * summary: Returns data for a specific user. + * parameters: + * - name: username + * in: path + * required: true + * description: Username of the user (can also be the unique user-id). + * schema: + * type: string + * responses: + * 200: + * description: User data. + * 404: + * description: User not found. + */ +router.get('/:username', userController.getUser); + +/** + * @swagger + * /user/{username}: + * post: + * tags: [User] + * summary: Creates a new user 1. + * parameters: + * - name: username + * in: path + * required: true + * description: Username of the user. + * schema: + * type: string + * responses: + * 201: + * description: User created. + * content: + * application/json: + * schema: + * type: array + * items: + * type: object + * properties: + * username: + * type: string + * description: The username of the user. + * highscore: + * type: number + * description: The highscore of the user. + * games: + * type: number + * description: The number of games the user has played. + * wins: + * type: number + * description: The number of games the user has won. + * losses: + * type: number + * description: The number of games the user has lost. + * id: + * type: string + * description: The ID of the user. + * 409: + * description: User already exists. + */ +router.post('/:username', userController.createUser); + +/** + * @swagger + * /user/{username}: + * delete: + * tags: [User] + * summary: Deletes a user. + * parameters: + * - name: username + * in: path + * required: true + * description: Username of the user. + * schema: + * type: string + * responses: + * 204: + * description: User deleted. + * 404: + * description: User not found. + */ +router.delete('/:username', userController.deleteUser); export default router; diff --git a/backend/tsconfig.json b/backend/tsconfig.json index 38825eee024a61655bd045554b860de08ccc05d3..8fe805977e791cb9a87e3ee23bfb61680e92a012 100644 --- a/backend/tsconfig.json +++ b/backend/tsconfig.json @@ -5,7 +5,12 @@ "esModuleInterop": true, "outDir": "dist", "strict": true, - "sourceMap": true + "sourceMap": true, + "removeComments": false, + "emitDecoratorMetadata": true, + "experimentalDecorators": true, + "allowJs": true, + "declaration": true }, - "include": ["src/**/*", "types/**/*", "keys/**/*", "src/models/Lobby.ts"] + "include": ["src/**/*", "types/**/*", "keys/**/*"] } diff --git a/backend/yarn.lock b/backend/yarn.lock index fc4c1664ee61e1da2ed877e7789f67fb567f31ec..97a88bf9473c82314a04b52279d479c843f8f524 100644 --- a/backend/yarn.lock +++ b/backend/yarn.lock @@ -2,6 +2,38 @@ # yarn lockfile v1 +"@apidevtools/json-schema-ref-parser@^9.0.6": + version "9.1.2" + resolved "https://registry.yarnpkg.com/@apidevtools/json-schema-ref-parser/-/json-schema-ref-parser-9.1.2.tgz#8ff5386b365d4c9faa7c8b566ff16a46a577d9b8" + integrity sha512-r1w81DpR+KyRWd3f+rk6TNqMgedmAxZP5v5KWlXQWlgMUUtyEJch0DKEci1SorPMiSeM8XPl7MZ3miJ60JIpQg== + dependencies: + "@jsdevtools/ono" "^7.1.3" + "@types/json-schema" "^7.0.6" + call-me-maybe "^1.0.1" + js-yaml "^4.1.0" + +"@apidevtools/openapi-schemas@^2.0.4": + version "2.1.0" + resolved "https://registry.yarnpkg.com/@apidevtools/openapi-schemas/-/openapi-schemas-2.1.0.tgz#9fa08017fb59d80538812f03fc7cac5992caaa17" + integrity sha512-Zc1AlqrJlX3SlpupFGpiLi2EbteyP7fXmUOGup6/DnkRgjP9bgMM/ag+n91rsv0U1Gpz0H3VILA/o3bW7Ua6BQ== + +"@apidevtools/swagger-methods@^3.0.2": + version "3.0.2" + resolved "https://registry.yarnpkg.com/@apidevtools/swagger-methods/-/swagger-methods-3.0.2.tgz#b789a362e055b0340d04712eafe7027ddc1ac267" + integrity sha512-QAkD5kK2b1WfjDS/UQn/qQkbwF31uqRjPTrsCs5ZG9BQGAkjwvqGFjjPqAuzac/IYzpPtRzjCP1WrTuAIjMrXg== + +"@apidevtools/swagger-parser@10.0.3": + version "10.0.3" + resolved "https://registry.yarnpkg.com/@apidevtools/swagger-parser/-/swagger-parser-10.0.3.tgz#32057ae99487872c4dd96b314a1ab4b95d89eaf5" + integrity sha512-sNiLY51vZOmSPFZA5TF35KZ2HbgYklQnTSDnkghamzLb3EkNtcQnrBQEj5AOCxHpTtXpqMCRM1CrmV2rG6nw4g== + dependencies: + "@apidevtools/json-schema-ref-parser" "^9.0.6" + "@apidevtools/openapi-schemas" "^2.0.4" + "@apidevtools/swagger-methods" "^3.0.2" + "@jsdevtools/ono" "^7.1.3" + call-me-maybe "^1.0.1" + z-schema "^5.0.1" + "@babel/code-frame@^7.12.13": version "7.21.4" resolved "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.21.4.tgz" @@ -257,6 +289,11 @@ "@jridgewell/resolve-uri" "^3.0.3" "@jridgewell/sourcemap-codec" "^1.4.10" +"@jsdevtools/ono@^7.1.3": + version "7.1.3" + resolved "https://registry.yarnpkg.com/@jsdevtools/ono/-/ono-7.1.3.tgz#9df03bbd7c696a5c58885c34aa06da41c8543796" + integrity sha512-4JQNk+3mVzK3xh2rqd6RB4J46qUR19azEHBneZyTZM+c456qOrbbM/5xcR8huNCCcbVt7+UmizG6GuUvPvKUYg== + "@jsdoc/salty@^0.2.1": version "0.2.4" resolved "https://registry.npmjs.org/@jsdoc/salty/-/salty-0.2.4.tgz" @@ -423,6 +460,11 @@ expect "^29.0.0" pretty-format "^29.0.0" +"@types/json-schema@^7.0.6": + version "7.0.11" + resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.11.tgz#d421b6c527a3037f7c84433fd2c4229e016863d3" + integrity sha512-wOuvG1SN4Us4rez+tylwwwCV1psiNVOkJeM3AUWUNWg/jDQY2+HE/444y5gc+jBmRqASOm2Oeh5c1axHobwRKQ== + "@types/jsonwebtoken@^9.0.0": version "9.0.1" resolved "https://registry.npmjs.org/@types/jsonwebtoken/-/jsonwebtoken-9.0.1.tgz" @@ -514,6 +556,11 @@ resolved "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.1.tgz" integrity sha512-Hl219/BT5fLAaz6NDkSuhzasy49dwQS/DSdu4MdggFB8zcXv7vflBI3xp7FEmkmdDkBUI2bPUNeMttp2knYdxw== +"@types/swagger-jsdoc@^6.0.1": + version "6.0.1" + resolved "https://registry.yarnpkg.com/@types/swagger-jsdoc/-/swagger-jsdoc-6.0.1.tgz#94a99aca0356cb64ad2a6eb903ed034703453801" + integrity sha512-+MUpcbyxD528dECUBCEVm6abNuORdbuGjbrUdHDeAQ+rkPuo2a+L4N02WJHF3bonSSE6SJ3dUJwF2V6+cHnf0w== + "@types/yargs-parser@*": version "21.0.0" resolved "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-21.0.0.tgz" @@ -814,6 +861,11 @@ call-bind@^1.0.0: function-bind "^1.1.1" get-intrinsic "^1.0.2" +call-me-maybe@^1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/call-me-maybe/-/call-me-maybe-1.0.2.tgz#03f964f19522ba643b1b0693acb9152fe2074baa" + integrity sha512-HpX65o1Hnr9HH25ojC1YGs7HCQLq0GCOibSaWER0eNpgJ/Z1MZv2mTc7+xh6WOPxbRVcmgbv4hGU+uSQ/2xFZQ== + caseless@~0.12.0: version "0.12.0" resolved "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz" @@ -915,6 +967,11 @@ cliui@^8.0.1: strip-ansi "^6.0.1" wrap-ansi "^7.0.0" +clone@2.x: + version "2.1.2" + resolved "https://registry.yarnpkg.com/clone/-/clone-2.1.2.tgz#1b7f4b9f591f1e8f83670401600345a02887435f" + integrity sha512-3Pe/CF1Nn94hyhIYpjtiLhdCoEoz0DqQ+988E9gmeEdQZlojxnOb74wctFyuwWQHzqyf9X7C7MG8juUpqBJT8w== + color-convert@^1.9.0: version "1.9.3" resolved "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz" @@ -951,6 +1008,16 @@ combined-stream@^1.0.6, combined-stream@~1.0.6: dependencies: delayed-stream "~1.0.0" +commander@6.2.0: + version "6.2.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-6.2.0.tgz#b990bfb8ac030aedc6d11bc04d1488ffef56db75" + integrity sha512-zP4jEKbe8SHzKJYQmq8Y9gYjtO/POJLgIdKgV7B9qNmABVFVc+ctqSX6iXh4mCpJfRBOabiZ2YKPg8ciDw6C+Q== + +commander@^10.0.0: + version "10.0.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-10.0.0.tgz#71797971162cd3cf65f0b9d24eb28f8d303acdf1" + integrity sha512-zS5PnTI22FIRM6ylNW8G4Ap0IEOyk62fhLSD0+uHRT9McRCLGpkVNvao4bjimpK/GShynyQkFFxHhwMcETmduA== + commander@^5.1.0: version "5.1.0" resolved "https://registry.npmjs.org/commander/-/commander-5.1.0.tgz" @@ -1152,6 +1219,13 @@ diff@^4.0.1: resolved "https://registry.yarnpkg.com/diff/-/diff-4.0.2.tgz#60f3aecb89d5fae520c11aa19efc2bb982aade7d" integrity sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A== +doctrine@3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-3.0.0.tgz#addebead72a6574db783639dc87a121773973961" + integrity sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w== + dependencies: + esutils "^2.0.2" + duplexify@^4.0.0: version "4.1.2" resolved "https://registry.npmjs.org/duplexify/-/duplexify-4.1.2.tgz" @@ -1580,6 +1654,18 @@ glob-parent@~5.1.2: dependencies: is-glob "^4.0.1" +glob@7.1.6: + version "7.1.6" + resolved "https://registry.yarnpkg.com/glob/-/glob-7.1.6.tgz#141f33b81a7c2492e125594307480c46679278a6" + integrity sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA== + dependencies: + fs.realpath "^1.0.0" + inflight "^1.0.4" + inherits "2" + minimatch "^3.0.4" + once "^1.3.0" + path-is-absolute "^1.0.0" + glob@^7.1.3: version "7.2.3" resolved "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz" @@ -1927,6 +2013,13 @@ js-tokens@^4.0.0: resolved "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz" integrity sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ== +js-yaml@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-4.1.0.tgz#c1fb65f8f5017901cdd2c951864ba18458a10602" + integrity sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA== + dependencies: + argparse "^2.0.1" + js2xmlparser@^4.0.2: version "4.0.2" resolved "https://registry.npmjs.org/js2xmlparser/-/js2xmlparser-4.0.2.tgz" @@ -2108,6 +2201,21 @@ lodash.clonedeep@^4.5.0: resolved "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz" integrity sha512-H5ZhCF25riFd9uB5UCkVKo61m3S/xZk1x4wA6yp/L3RFP6Z/eHH1ymQcGLo7J3GMPfm0V/7m1tryHuGVxpqEBQ== +lodash.get@^4.4.2: + version "4.4.2" + resolved "https://registry.yarnpkg.com/lodash.get/-/lodash.get-4.4.2.tgz#2d177f652fa31e939b4438d5341499dfa3825e99" + integrity sha512-z+Uw/vLuy6gQe8cfaFWD7p0wVv8fJl3mbzXh33RS+0oW2wvUqiRXiQ69gLWSLpgB5/6sU+r6BlQR0MBILadqTQ== + +lodash.isequal@^4.5.0: + version "4.5.0" + resolved "https://registry.yarnpkg.com/lodash.isequal/-/lodash.isequal-4.5.0.tgz#415c4478f2bcc30120c22ce10ed3226f7d3e18e0" + integrity sha512-pDo3lu8Jhfjqls6GkMgpahsF9kCyayhgykjyLMNFTKWrpVdAQtYyB4muAMWozBB4ig/dtWAmsMxLEI8wuz+DYQ== + +lodash.mergewith@^4.6.2: + version "4.6.2" + resolved "https://registry.yarnpkg.com/lodash.mergewith/-/lodash.mergewith-4.6.2.tgz#617121f89ac55f59047c7aec1ccd6654c6590f55" + integrity sha512-GK3g5RPZWTRSeLSpgP8Xhra+pnjBC56q9FZYe1d5RN3TJ35dbkGy3YqBSMbyCrlbi+CM9Z3Jk5yTL7RCsqboyQ== + lodash.once@^4.1.1: version "4.1.1" resolved "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz" @@ -2255,7 +2363,7 @@ mimic-fn@^2.1.0: resolved "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz" integrity sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg== -minimatch@^3.1.1, minimatch@^3.1.2: +minimatch@^3.0.4, minimatch@^3.1.1, minimatch@^3.1.2: version "3.1.2" resolved "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz" integrity sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw== @@ -2299,6 +2407,13 @@ negotiator@0.6.3: resolved "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz" integrity sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg== +node-cache@^5.1.2: + version "5.1.2" + resolved "https://registry.yarnpkg.com/node-cache/-/node-cache-5.1.2.tgz#f264dc2ccad0a780e76253a694e9fd0ed19c398d" + integrity sha512-t1QzWwnk4sjLWaQAS8CHgOJ+RAfmHpxFWmc36IWTiWHQfs0w5JDMBS1b1ZxQteo0vVVuWJvIUKHDkkeK7vIGCg== + dependencies: + clone "2.x" + node-fetch@^2.6.1, node-fetch@^2.6.7: version "2.6.9" resolved "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.9.tgz" @@ -2898,6 +3013,37 @@ supports-color@^8.1.1: dependencies: has-flag "^4.0.0" +swagger-jsdoc@^6.2.8: + version "6.2.8" + resolved "https://registry.yarnpkg.com/swagger-jsdoc/-/swagger-jsdoc-6.2.8.tgz#6d33d9fb07ff4a7c1564379c52c08989ec7d0256" + integrity sha512-VPvil1+JRpmJ55CgAtn8DIcpBs0bL5L3q5bVQvF4tAW/k/9JYSj7dCpaYCAv5rufe0vcCbBRQXGvzpkWjvLklQ== + dependencies: + commander "6.2.0" + doctrine "3.0.0" + glob "7.1.6" + lodash.mergewith "^4.6.2" + swagger-parser "^10.0.3" + yaml "2.0.0-1" + +swagger-parser@^10.0.3: + version "10.0.3" + resolved "https://registry.yarnpkg.com/swagger-parser/-/swagger-parser-10.0.3.tgz#04cb01c18c3ac192b41161c77f81e79309135d03" + integrity sha512-nF7oMeL4KypldrQhac8RyHerJeGPD1p2xDh900GPvc+Nk7nWP6jX2FcC7WmkinMoAmoO774+AFXcWsW8gMWEIg== + dependencies: + "@apidevtools/swagger-parser" "10.0.3" + +swagger-ui-dist@>=4.11.0: + version "4.18.2" + resolved "https://registry.yarnpkg.com/swagger-ui-dist/-/swagger-ui-dist-4.18.2.tgz#323308f1c1d87a7c22ce3e273c31835eb680a71b" + integrity sha512-oVBoBl9Dg+VJw8uRWDxlyUyHoNEDC0c1ysT6+Boy6CTgr2rUcLcfPon4RvxgS2/taNW6O0+US+Z/dlAsWFjOAQ== + +swagger-ui-express@^4.6.2: + version "4.6.2" + resolved "https://registry.yarnpkg.com/swagger-ui-express/-/swagger-ui-express-4.6.2.tgz#61b2cb9fd7932cdccff99e0efdf700a5459e493c" + integrity sha512-MHIOaq9JrTTB3ygUJD+08PbjM5Tt/q7x80yz9VTFIatw8j5uIWKcr90S0h5NLMzFEDC6+eVprtoeA5MDZXCUKQ== + dependencies: + swagger-ui-dist ">=4.11.0" + teeny-request@^8.0.0: version "8.0.3" resolved "https://registry.npmjs.org/teeny-request/-/teeny-request-8.0.3.tgz" @@ -3089,6 +3235,11 @@ v8-compile-cache-lib@^3.0.1: resolved "https://registry.yarnpkg.com/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz#6336e8d71965cb3d35a1bbb7868445a7c05264bf" integrity sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg== +validator@^13.7.0: + version "13.9.0" + resolved "https://registry.yarnpkg.com/validator/-/validator-13.9.0.tgz#33e7b85b604f3bbce9bb1a05d5c3e22e1c2ff855" + integrity sha512-B+dGG8U3fdtM0/aNK4/X8CXq/EcxU2WPrPEkJGslb47qyHsxmbggTWK0yEA4qnYVNF+nxNlN88o14hIcPmSIEA== + vary@~1.1.2: version "1.1.2" resolved "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz" @@ -3185,6 +3336,11 @@ yallist@^4.0.0: resolved "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz" integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A== +yaml@2.0.0-1: + version "2.0.0-1" + resolved "https://registry.yarnpkg.com/yaml/-/yaml-2.0.0-1.tgz#8c3029b3ee2028306d5bcf396980623115ff8d18" + integrity sha512-W7h5dEhywMKenDJh2iX/LABkbFnBxasD27oyXWDS/feDsxiw0dD5ncXdYXgkvAsXIY2MpW/ZKkr9IU30DBdMNQ== + yargs-parser@^20.2.2: version "20.2.9" resolved "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.9.tgz" @@ -3238,3 +3394,14 @@ yocto-queue@^0.1.0: version "0.1.0" resolved "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz" integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q== + +z-schema@^5.0.1: + version "5.0.6" + resolved "https://registry.yarnpkg.com/z-schema/-/z-schema-5.0.6.tgz#46d6a687b15e4a4369e18d6cb1c7b8618fc256c5" + integrity sha512-+XR1GhnWklYdfr8YaZv/iu+vY+ux7V5DS5zH1DQf6bO5ufrt/5cgNhVO5qyhsjFXvsqQb/f08DWE9b6uPscyAg== + dependencies: + lodash.get "^4.4.2" + lodash.isequal "^4.5.0" + validator "^13.7.0" + optionalDependencies: + commander "^10.0.0"