diff --git a/.gitignore b/.gitignore index 9db3cfb0b35c9ac31228ab865721650dd0e9512f..a31d6a524eab9912473012be6b823a3416fbbf94 100644 --- a/.gitignore +++ b/.gitignore @@ -36,3 +36,7 @@ lerna-debug.log* # db connection .env *.providers.ts + +# uploads +images/* +!images/default.jpeg diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index e197579511b72d74b03e46cad79a6c3369014b06..f5a573ea003752d767736d4a37a1b7f99f9dbdb7 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -34,7 +34,7 @@ e2e-testing: - docker-compose build --no-cache - docker-compose up -d after_script: - - sleep 8 + - sleep 3 - echo "create database ehasa;" | docker exec -i postgis psql -U postgres - echo "create user ehasa;" | docker exec -i postgis psql -U postgres - echo "alter user ehasa with encrypted password 'salasana';" | docker exec -i postgis psql -U postgres diff --git a/images/default.jpeg b/images/default.jpeg new file mode 100644 index 0000000000000000000000000000000000000000..7e5e656834bfa61b0fc135c94d1fb9089596f2f6 Binary files /dev/null and b/images/default.jpeg differ diff --git a/package-lock.json b/package-lock.json index be2f7c8964168262bdeaafcfad35ef0656a2656e..01d2eb7272357854133cd7fe3325d9234a86c4cb 100644 --- a/package-lock.json +++ b/package-lock.json @@ -426,11 +426,11 @@ } }, "@nestjs/common": { - "version": "6.2.4", - "resolved": "https://registry.npmjs.org/@nestjs/common/-/common-6.2.4.tgz", - "integrity": "sha512-YZvJ6/S7yVQZK+9rupCzMCg4tpbc9DyVvLoTx0NBDqExTCUNcNEcCtn0AZrO/hLqbeYODnJwGE2NxkH1R/qw+w==", + "version": "6.5.2", + "resolved": "https://registry.npmjs.org/@nestjs/common/-/common-6.5.2.tgz", + "integrity": "sha512-vkB6JLPPckjS35usLMV3Q6vljkgzhN3jgQ+U1VY6cKriyjnkIcUKo37tNQSPYkaAGe1pOLK4IkmwMTSyhNieyg==", "requires": { - "axios": "0.18.0", + "axios": "0.19.0", "cli-color": "1.4.0", "uuid": "3.3.2" } @@ -1272,12 +1272,19 @@ "dev": true }, "axios": { - "version": "0.18.0", - "resolved": "https://registry.npmjs.org/axios/-/axios-0.18.0.tgz", - "integrity": "sha1-MtU+SFHv3AoRmTts0AB4nXDAUQI=", + "version": "0.19.0", + "resolved": "https://registry.npmjs.org/axios/-/axios-0.19.0.tgz", + "integrity": "sha512-1uvKqKQta3KBxIz14F2v06AEHZ/dIoeKfbTRkK1E5oqjDnuEerLmYTgJB5AiQZHJcljpg1TuRzdjDR06qNk0DQ==", "requires": { - "follow-redirects": "^1.3.0", - "is-buffer": "^1.1.5" + "follow-redirects": "1.5.10", + "is-buffer": "^2.0.2" + }, + "dependencies": { + "is-buffer": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.3.tgz", + "integrity": "sha512-U15Q7MXTuZlrbymiz95PJpZxu8IlipAp4dtS3wOdgPXx3mqBnslrWU14kxfHB+Py/+2PVKSr37dMAgM2A4uArw==" + } } }, "babel-jest": { @@ -2172,11 +2179,12 @@ } }, "d": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/d/-/d-1.0.0.tgz", - "integrity": "sha1-dUu1v+VUUdpppYuU1F9MWwRi1Y8=", + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/d/-/d-1.0.1.tgz", + "integrity": "sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==", "requires": { - "es5-ext": "^0.10.9" + "es5-ext": "^0.10.50", + "type": "^1.0.1" } }, "dashdash": { @@ -2222,6 +2230,7 @@ "version": "3.2.6", "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", + "dev": true, "requires": { "ms": "^2.1.1" } @@ -2598,13 +2607,13 @@ } }, "es6-weak-map": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/es6-weak-map/-/es6-weak-map-2.0.2.tgz", - "integrity": "sha1-XjqzIlH/0VOKH45f+hNXdy+S2W8=", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/es6-weak-map/-/es6-weak-map-2.0.3.tgz", + "integrity": "sha512-p5um32HOTO1kP+w7PRnB+5lQ43Z6muuMuIMffvDN8ZB4GcnjLBV6zGStpbASIMk4DCAvEaamhe2zhyCb/QXXsA==", "requires": { "d": "1", - "es5-ext": "^0.10.14", - "es6-iterator": "^2.0.1", + "es5-ext": "^0.10.46", + "es6-iterator": "^2.0.3", "es6-symbol": "^3.1.1" } }, @@ -2819,11 +2828,6 @@ } } }, - "express-rate-limit": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/express-rate-limit/-/express-rate-limit-4.0.4.tgz", - "integrity": "sha512-DLRj2vMO7Xgai8qWKU9O6ZztF2bdDmfFNFi9k3G9BPzJ+7MG7eWaaBikbe0eBpNGSxU8JziwW0PQKG78aNWa6g==" - }, "extend": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", @@ -3026,11 +3030,26 @@ } }, "follow-redirects": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.7.0.tgz", - "integrity": "sha512-m/pZQy4Gj287eNy94nivy5wchN3Kp+Q5WgUPNy5lJSZ3sgkVKSYV/ZChMAQVIgx1SqfZ2zBZtPA2YlXIWxxJOQ==", + "version": "1.5.10", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.5.10.tgz", + "integrity": "sha512-0V5l4Cizzvqt5D44aTXbFZz+FtyXV1vrDN6qrelxtfYQKW0KO0W2T/hkE8xvGa/540LkZlkaUjO4ailYTFtHVQ==", "requires": { - "debug": "^3.2.6" + "debug": "=3.1.0" + }, + "dependencies": { + "debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + } } }, "for-in": { @@ -4048,7 +4067,8 @@ "is-buffer": { "version": "1.1.6", "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", - "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==" + "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", + "dev": true }, "is-callable": { "version": "1.1.4", @@ -5095,9 +5115,9 @@ } }, "lodash": { - "version": "4.17.11", - "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.11.tgz", - "integrity": "sha512-cQKh8igo5QUhZ7lg38DYWAxMvjSAKG0A8wGSVimP07SIUEK2UO+arSRKbRZWtelMtN5V0Hkwh5ryOto/SshYIg==", + "version": "4.17.14", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.14.tgz", + "integrity": "sha512-mmKYbW3GLuJeX+iGP+Y7Gp1AiGHGbXHCOh/jZmrawMmsE7MS4znI3RL2FsjbqOyMayHInjOeykW7PEajUk1/xw==", "dev": true }, "lodash.includes": { @@ -5368,9 +5388,9 @@ "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=" }, "mixin-deep": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.1.tgz", - "integrity": "sha512-8ZItLHeEgaqEvd5lYBXfm4EZSFCX29Jb9K+lAHhDKzReKBQKj3R+7NOF6tjqYi9t4oI8VUfaWITJQm86wnXGNQ==", + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.2.tgz", + "integrity": "sha512-WRoDn//mXBiJ1H40rqa3vH0toePwSsGb45iInWlTySa+Uu4k3tYUSxa2v1KqAiLtvlrSzaExqS1gtk96A9zvEA==", "dev": true, "requires": { "for-in": "^1.0.2", @@ -6638,9 +6658,9 @@ "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=" }, "set-value": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.0.tgz", - "integrity": "sha512-hw0yxk9GT/Hr5yJEYnHNKYXkIA8mVJgd9ditYZCe16ZczcaELYYcfvaXesNACk2O8O0nTiPQcQhGUQj8JLzeeg==", + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.1.tgz", + "integrity": "sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw==", "dev": true, "requires": { "extend-shallow": "^2.0.1", @@ -7627,6 +7647,11 @@ "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=", "dev": true }, + "type": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/type/-/type-1.0.1.tgz", + "integrity": "sha512-MAM5dBMJCJNKs9E7JXo4CXRAansRfG0nlJxW7Wf6GZzSOvH31zClSaHdIMWLehe/EGMBkqeC55rrkaOr5Oo7Nw==" + }, "type-check": { "version": "0.3.2", "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", @@ -7824,38 +7849,15 @@ "integrity": "sha512-5/4etnCkd9c8gwgowi5/om/mYO5ajCaOgdzj/oW+0eQV9WxKBDZw5+ycmKmeaTXjInS/W0BzpGLo2xR2aBwZdg==" }, "union-value": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.0.tgz", - "integrity": "sha1-XHHDTLW61dzr4+oM0IIHulqhrqQ=", + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.1.tgz", + "integrity": "sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==", "dev": true, "requires": { "arr-union": "^3.1.0", "get-value": "^2.0.6", "is-extendable": "^0.1.1", - "set-value": "^0.4.3" - }, - "dependencies": { - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - }, - "set-value": { - "version": "0.4.3", - "resolved": "https://registry.npmjs.org/set-value/-/set-value-0.4.3.tgz", - "integrity": "sha1-fbCPnT0i3H945Trzw79GZuzfzPE=", - "dev": true, - "requires": { - "extend-shallow": "^2.0.1", - "is-extendable": "^0.1.1", - "is-plain-object": "^2.0.1", - "to-object-path": "^0.3.0" - } - } + "set-value": "^2.0.1" } }, "unique-string": { diff --git a/package.json b/package.json index 794db7d353397154ef795d59135b5771df8300ff..3677b2a8db9be496cadc813cab1446501627bbc3 100644 --- a/package.json +++ b/package.json @@ -20,7 +20,7 @@ "test:e2e": "jest --config ./test/jest-e2e.json" }, "dependencies": { - "@nestjs/common": "^6.0.0", + "@nestjs/common": "^6.5.2", "@nestjs/core": "^6.0.0", "@nestjs/jwt": "^6.1.1", "@nestjs/platform-express": "^6.0.0", @@ -30,7 +30,6 @@ "bcryptjs": "^2.4.3", "class-transformer": "^0.2.3", "class-validator": "^0.9.1", - "express-rate-limit": "^4.0.4", "jsonwebtoken": "^8.5.1", "passport-jwt": "^4.0.0", "pg": "^7.11.0", diff --git a/src/app.controller.ts b/src/app.controller.ts index cce879ee622146012901c9adb47ef40c0fd3a555..045e5b1ae398cd994ada21cd7f9512b475af30ec 100644 --- a/src/app.controller.ts +++ b/src/app.controller.ts @@ -1,4 +1,4 @@ -import { Controller, Get } from '@nestjs/common'; +import { Controller, Get, Param, Res } from '@nestjs/common'; import { AppService } from './app.service'; @Controller() @@ -9,4 +9,9 @@ export class AppController { getHello(): string { return this.appService.getHello(); } + + @Get('images/:id') + returnImage(@Param('id') id, @Res() res) { + return; + } } diff --git a/src/app.module.ts b/src/app.module.ts index f2c7b7765304436488bbadb3fdfce4fbdc1fbd9a..05685cd88382fd6e6e29caadfbab83b0c658d953 100644 --- a/src/app.module.ts +++ b/src/app.module.ts @@ -2,15 +2,12 @@ import { Module } from '@nestjs/common'; import { APP_FILTER, APP_INTERCEPTOR, APP_GUARD } from '@nestjs/core'; import { TypeOrmModule } from '@nestjs/typeorm'; import { Connection } from 'typeorm'; - import { AppController } from './app.controller'; import { AppService } from './app.service'; - import { RolesGuard } from './shared/roles.guard'; -import { LoggingInterceptor } from './shared/logging.interceptor'; +//import { LoggingInterceptor } from './shared/logging.interceptor'; import { StatesGuard } from './shared/states.guard'; import { HttpErrorFilter } from './shared/http-error.filter'; - import { NotificationModule } from './notifications/notifications.module'; import { TaskModule } from './task/task.module'; import { TrackingModule } from './tracking/tracking.module'; @@ -19,6 +16,28 @@ import { DrawModule } from './draw/draw.module'; import { FactionModule } from './faction/faction.module'; import { GameModule } from './game/game.module'; import { ScoreModule } from './score/score.module'; +import { ReplayModule } from './replay/replay.module'; +import { TickModule } from './tick/tick.module'; + + + + +/////////////////////////////////////////////////////////////////////////////////////////////////////////// +/// Core of the server, /// +/// Every module is being imported and combined here. /// +/// /// +/// AppController needs to be kept in for SSL verification to work (root needs to return something) /// +/// /// +/// TypeOrmModule checks ormconfig.json for database connection. /// +/// /// +/// More information on global decorators can be found from shared folder. /// +/// /// +/// Providers can be found from shared folder /// +/// - HttpErrorFilter /// +/// - LoggingInterceptor /// +/// - RolesGuard Decorator /// +/// - StatesGuard Decorator /// +/////////////////////////////////////////////////////////////////////////////////////////////////////////// @Module({ imports: [ @@ -31,6 +50,8 @@ import { ScoreModule } from './score/score.module'; FactionModule, TrackingModule, ScoreModule, + ReplayModule, + TickModule, ], controllers: [AppController], providers: [ @@ -39,10 +60,7 @@ import { ScoreModule } from './score/score.module'; provide: APP_FILTER, useClass: HttpErrorFilter, }, - { - provide: APP_INTERCEPTOR, - useClass: LoggingInterceptor, - }, + { provide: APP_GUARD, useClass: RolesGuard, diff --git a/src/draw/coordinate.entity.ts b/src/draw/coordinate.entity.ts index ef5447f61cfd1dd07d1f81a8ee1fce8e8256da2a..da5e940a74b31a21479527b02b847aaeda9d950e 100644 --- a/src/draw/coordinate.entity.ts +++ b/src/draw/coordinate.entity.ts @@ -1,22 +1,22 @@ -import { - Entity, - Column, - PrimaryGeneratedColumn, - ManyToOne, - Timestamp, -} from 'typeorm'; +import { Entity, Column, PrimaryGeneratedColumn, ManyToOne } from 'typeorm'; -import { Game_PersonEntity, GameEntity } from '../game/game.entity'; +import { GameEntity } from '../game/game.entity'; import { FactionEntity } from '../faction/faction.entity'; +/////////////////////////////////////////////////////////////////////////// +/// MapDrawingEntity & MapDrawingHistoryEntity reflect database tables. /// +/// /// +/// MapDrawing ownershipCheck checks users rights to MapDrawing /// +/////////////////////////////////////////////////////////////////////////// + @Entity('MapDrawing') export class MapDrawingEntity { @PrimaryGeneratedColumn('uuid') mapDrawingId: string; @Column({ type: 'bool', nullable: true }) drawingIsActive: boolean; - @Column({ type: 'time', nullable: true }) drawingValidTill: string; - @Column({ type: 'json', nullable: true }) data: JSON; + // When Faction or game that has the drawing in question is deleted from + // the database, the drawing is also deleted @ManyToOne(type => FactionEntity, faction => faction.mapDrawings, { onDelete: 'CASCADE', }) @@ -25,27 +25,28 @@ export class MapDrawingEntity { onDelete: 'CASCADE', }) gameId: GameEntity; + + async ownershipCheck(factionEntity, role) { + if (role === 'admin') { + return factionEntity == this.faction; + } else { + return this.faction && factionEntity.factionId === this.faction.factionId + ? true + : false; + } + } } -@Entity('Game_Person_MapDrawing') -export class Game_Person_MapDrawingEntity { - @PrimaryGeneratedColumn('uuid') GPmapDrawingId: string; - @Column({ type: 'timestamp' }) GPCTimeStamp: Timestamp; +@Entity('MapDrawingHistory') +export class MapDrawingHistoryEntity { + @PrimaryGeneratedColumn('uuid') mapDrawingHistoryId: string; + @Column('float') timestamp: number; + @Column('bool') drawingIsActive: boolean; + @Column('json') data: JSON; - @ManyToOne( - type => Game_PersonEntity, - game_person => game_person.gamepersonId, - { - onDelete: 'CASCADE', - }, - ) - game_person: Game_PersonEntity; - @ManyToOne( - type => MapDrawingEntity, - map_drawing => map_drawing.mapDrawingId, - { - onDelete: 'CASCADE', - }, - ) - map_drawing: MapDrawingEntity; + // If drawing is deleted, it's histories are deleted also + @ManyToOne(() => MapDrawingEntity, mapDrawing => mapDrawing.mapDrawingId, { + onDelete: 'CASCADE', + }) + mapdrawing: string; } diff --git a/src/draw/draw.controller.ts b/src/draw/draw.controller.ts index 26f6cf8c56142ef190ad3bcbfd7ce9d32caa371e..d66bebe1ffe091fcda211a64f2b4b49601dd75b1 100644 --- a/src/draw/draw.controller.ts +++ b/src/draw/draw.controller.ts @@ -1,42 +1,50 @@ import { Controller, Put, - UseGuards, Get, Param, UsePipes, ValidationPipe, Body, + UseInterceptors, + ClassSerializerInterceptor, } from '@nestjs/common'; -import { AuthGuard } from '../shared/auth.guard'; import { DrawService } from './draw.service'; import { Roles, GameStates } from '../shared/guard.decorator'; -import { MapDrawingDTO, ReturnDrawingsDTO } from './mapdrawing.dto'; +import { MapDrawingDTO } from './mapdrawing.dto'; +import { GamePerson } from '../game/gameperson.decorator'; + +////////////////////////////////////////////////////////////////////////// +/// DrawController /// +/// /// +/// Functions either insert or return MapDrawing data, /// +/// Insert functions require user to have proper role (either gm /// +/// or commander) in the game to be able store the data: /// +/// MapDrawingDTO data to database. /// +/// Data return functions require atleast spectator role. /// +////////////////////////////////////////////////////////////////////////// -/* - DrawController - - Functions either insert or return MapDrawing data, - Insert functions require user to have proper role (either gm or commander) in the game to be able store the ddata: MapDrawingDTOata to database. - Data return functions require atleast spectator role. - */ @Controller('draw') export class DrawController { constructor(private drawService: DrawService) {} @Put('mapdrawing/:id') @Roles('admin', 'factionleader') - @GameStates('CREATED', 'STARTED') + @GameStates('CREATED', 'STARTED', 'PAUSED') @UsePipes(new ValidationPipe()) - async draw(@Param('id') gameId, @Body() data: MapDrawingDTO) { - return this.drawService.draw(gameId, data); + async draw( + @GamePerson() gameperson, + @Param('id') gameId, + @Body() data: MapDrawingDTO, + ) { + return this.drawService.draw(gameperson, gameId, data); } @Get('map/:id') - @UseGuards(new AuthGuard()) - @UsePipes(new ValidationPipe()) - async drawMap(@Param('id') id, @Body() data: ReturnDrawingsDTO) { - return this.drawService.drawMap(id, data); + @Roles('admin', 'factionleader', 'soldier', 'groupleader') + @UseInterceptors(ClassSerializerInterceptor) + async drawMap(@GamePerson() gameperson, @Param('id') gameId) { + return this.drawService.drawMap(gameperson, gameId); } } diff --git a/src/draw/draw.module.ts b/src/draw/draw.module.ts index a56cbbbb7fb49b476842177a4a0a9e7e0b592b3c..f03fc3acf1657620259eab31621c81f72ded7d46 100644 --- a/src/draw/draw.module.ts +++ b/src/draw/draw.module.ts @@ -3,20 +3,28 @@ import { TypeOrmModule } from '@nestjs/typeorm'; import { DrawController } from './draw.controller'; import { DrawService } from './draw.service'; -import { MapDrawingEntity } from '../draw/coordinate.entity'; +import { + MapDrawingEntity, + MapDrawingHistoryEntity, +} from '../draw/coordinate.entity'; import { FactionEntity } from '../faction/faction.entity'; import { Game_PersonEntity } from '../game/game.entity'; -/* -Draw -- contains everything to do with mapdrawing data. -*/ +import { NotificationModule } from 'src/notifications/notifications.module'; + +///////////////////////////////////////////////////////////////////// +/// Draw /// +/// - contains everything to do with mapdrawing data. /// +///////////////////////////////////////////////////////////////////// + @Module({ imports: [ TypeOrmModule.forFeature([ MapDrawingEntity, + MapDrawingHistoryEntity, FactionEntity, Game_PersonEntity, ]), + NotificationModule, ], controllers: [DrawController], providers: [DrawService], diff --git a/src/draw/draw.service.ts b/src/draw/draw.service.ts index d5ddf23e9574771bc769e2b6d58b59a4e95ad10d..da4b239dd5ef98c1572b5e64ff52525cd9f16e38 100644 --- a/src/draw/draw.service.ts +++ b/src/draw/draw.service.ts @@ -1,46 +1,99 @@ -import { Injectable } from '@nestjs/common'; +import { Injectable, HttpException, HttpStatus } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; -import { MapDrawingEntity } from '../draw/coordinate.entity'; -import { MapDrawingDTO, ReturnDrawingsDTO } from './mapdrawing.dto'; +import { + MapDrawingEntity, + MapDrawingHistoryEntity, +} from '../draw/coordinate.entity'; +import { MapDrawingDTO } from './mapdrawing.dto'; +import { NotificationGateway } from '../notifications/notifications.gateway'; + +/////////////////////////////////////////////////////////////////////////// +/// DrawService contains handling of MapDrawings and MapDrawinghistory /// +/////////////////////////////////////////////////////////////////////////// @Injectable() export class DrawService { constructor( @InjectRepository(MapDrawingEntity) private mapDrawingRepository: Repository<MapDrawingEntity>, + @InjectRepository(MapDrawingHistoryEntity) + private mapDrawHistoryRepository: Repository<MapDrawingHistoryEntity>, + private notificationGateway: NotificationGateway, ) {} - async draw(gameId, data: MapDrawingDTO) { + async draw(gameperson, gameId, data: MapDrawingDTO) { data['gameId'] = gameId; const drawing = await this.mapDrawingRepository.create(data); + this.notificationGateway.server.emit(gameId, { + type: 'drawing-update', + }); + // create new instance if id is null if (data.mapDrawingId == null || data.mapDrawingId == '') { - // luo uuden instanssin. + drawing.faction = gameperson.faction; const mapDrawing = await this.mapDrawingRepository.insert(drawing); + // create a history entity and insert it + await this.createHistory(data, gameperson, mapDrawing); return mapDrawing.identifiers; - } else { - //päivittää mapDrawingin + } + // get ref from db + const draw = await this.mapDrawingRepository.findOne({ + where: { mapDrawingId: data.mapDrawingId }, + relations: ['faction'], + }); + if (await draw.ownershipCheck(gameperson.faction, gameperson.role)) { + // else update the existing instance + await this.createHistory(data, gameperson); return await this.mapDrawingRepository.save(drawing); } + + throw new HttpException( + 'Drawing is not from your faction!', + HttpStatus.BAD_REQUEST, + ); } - // draw map based on game and - async drawMap(id, data: ReturnDrawingsDTO) { + // used to create mapDrawing history entity entry + private async createHistory(data, gameperson, mapDrawing?) { + // create a history entity and insert it + const history = await this.mapDrawHistoryRepository.create({ + data: data.data, + drawingIsActive: data.drawingIsActive, + mapdrawing: + data.mapDrawingId || mapDrawing.identifiers[0]['mapDrawingId'], + timestamp: Date.now(), + }); + history.data['faction'] = gameperson.faction + ? gameperson.faction.factionName + : 'admin'; + await this.mapDrawHistoryRepository.insert(history); + } + + // draw map based on game and gameperson faction + async drawMap(gameperson, gameId) { + // return all active drawings if admin + if (gameperson.role === 'admin') { + return await this.mapDrawingRepository.find({ + where: { gameId: gameId, drawingIsActive: true }, + relations: ['faction'], + }); + } // return mapdrawings with given faction and gameid return await this.mapDrawingRepository.find({ where: [ { - gameId: id, - faction: data.factionId, + gameId: gameId, + faction: gameperson.faction, drawingIsActive: true, }, { - gameId: id, + gameId: gameId, faction: null, drawingIsActive: true, }, ], + relations: ['faction'], }); } } diff --git a/src/draw/mapdrawing.dto.ts b/src/draw/mapdrawing.dto.ts index f7e7c781a0cbd308edf5f8c14f068b449f43a05e..f342dd2a924fb8718fce8c3f8097c7fc43fea26b 100644 --- a/src/draw/mapdrawing.dto.ts +++ b/src/draw/mapdrawing.dto.ts @@ -1,6 +1,5 @@ import { IsUUID, IsOptional, IsBoolean, Allow } from 'class-validator'; -import { FactionEntity } from '../faction/faction.entity'; import { GameEntity } from '../game/game.entity'; export class MapDrawingDTO { @@ -12,16 +11,6 @@ export class MapDrawingDTO { @IsOptional() @IsUUID('4') gameId: GameEntity; - @IsOptional() - @IsUUID('4') - faction?: FactionEntity; @IsBoolean() drawingIsActive?: boolean; - drawingValidTill?: string; -} - -export class ReturnDrawingsDTO { - @IsOptional() - @IsUUID('4') - factionId: FactionEntity; } diff --git a/src/faction/faction.controller.ts b/src/faction/faction.controller.ts index da7e5950626bbcd5385d80755d208e59be589b48..5eb905cb17b45b451720b0a021bc2ed877ece4ac 100644 --- a/src/faction/faction.controller.ts +++ b/src/faction/faction.controller.ts @@ -7,8 +7,7 @@ import { Body, Get, Put, - UseInterceptors, - ClassSerializerInterceptor, + Delete, } from '@nestjs/common'; import { AuthGuard } from '../shared/auth.guard'; @@ -19,9 +18,28 @@ import { PromotePlayerDTO, JoinFactionDTO, JoinGameGroupDTO, + FactionDTO, } from './faction.dto'; import { FactionService } from './faction.service'; import { Roles, GameStates } from '../shared/guard.decorator'; +import { GamePerson } from '../game/gameperson.decorator'; + +///////////////////////////////////////////////////////////////////////////////// +/// FactionController is being used for routing: /// +/// /// +/// Group /// +/// - create group when game status is CREATED /// +/// - getting groups with faction id(this is used mainly for listing players) /// +/// - joining group when game status is CREATED /// +/// /// +/// Faction /// +/// - checking users faction /// +/// - joining faction /// +/// - leaving faction /// +/// - changing faction multiplier (not implemented) /// +/// /// +/// See shared folder files for more information on decorators. /// +///////////////////////////////////////////////////////////////////////////////// @Controller('faction') export class FactionController { @@ -37,9 +55,7 @@ export class FactionController { @Param('id') id: string, @Body() data: GameGroupDTO, ) { - try { - return this.factionservice.createGroup(person, id, data); - } catch (error) {} + return this.factionservice.createGroup(person, id, data); } // id is faction ID @@ -53,17 +69,11 @@ export class FactionController { @Roles('soldier') @GameStates('CREATED') async joinGroup( - @User('id') person, - @Param('id') id, + @GamePerson() gameperson, + @Param('id') gameId, @Body() data: JoinGameGroupDTO, ) { - return this.factionservice.joinGroup(person, id, data); - } - - @UseInterceptors(ClassSerializerInterceptor) - @Get('get-faction-members/:id') - async getFactionMembers(@Param('id') factionId) { - return this.factionservice.listFactionMembers(factionId); + return this.factionservice.joinGroup(gameperson, data); } // param game ID is passed to @Roles @@ -79,7 +89,7 @@ export class FactionController { // :id is the id of the game, and is needed for GameStates to check the state of the game @Put('join-faction/:id') @UseGuards(new AuthGuard()) - @GameStates('CREATED', 'STARTED') + @GameStates('CREATED', 'STARTED', 'PAUSED') @UsePipes(new ValidationPipe()) joinFaction( @User('id') person, @@ -89,6 +99,24 @@ export class FactionController { return this.factionservice.joinFaction(person, data); } + // used to leave a faction + // :id is the if of the game + @Delete('leave/:id') + @Roles('soldier', 'factionleader') + @GameStates('CREATED') + leaveFaction(@GamePerson('gamepersonId') gamepersonId) { + return this.factionservice.leaveFaction(gamepersonId); + } + + // used to change factions multiplier + // not implemented in frontend uncomment this and services equivalent when needed + // @Put('faction-multiplier/:id') + // @Roles('admin') + // @GameStates('STARTED') + // factionMultiplier(@Param('id') game, @Body() body: FactionDTO) { + // return this.factionservice.changeFactionMultiplier(body); + // } + // check if person belongs to a faction in a game @Get('check-faction/:id') @UseGuards(new AuthGuard()) diff --git a/src/faction/faction.dto.ts b/src/faction/faction.dto.ts index 0b65c47007d6424584d8fdb35ac9d9ee53eaa924..511fce12d078c3c126641e62658540cbde7ae71b 100644 --- a/src/faction/faction.dto.ts +++ b/src/faction/faction.dto.ts @@ -8,13 +8,21 @@ import { Min, Max, IsOptional, + IsHexColor, + IsIn, } from 'class-validator'; import { GameEntity } from '../game/game.entity'; -import { RoleValidation, Uuid } from '../shared/custom-validation'; import { GameDTO } from '../game/game.dto'; import { FactionEntity, GameGroupEntity } from './faction.entity'; +///////////////////////////////////////////////////////////////////////////////////////////////////////////// +/// Contains Validation for FactionDTO, JoinFactionDTO, PromotePlayerDTO, GameGroupDTO, JoinGameGroupDTO /// +/// /// +/// uses class-validator built in validations /// +/// see https://github.com/typestack/class-validator /// +///////////////////////////////////////////////////////////////////////////////////////////////////////////// + export class FactionDTO { @IsOptional() @IsUUID('4') @@ -23,6 +31,8 @@ export class FactionDTO { @IsNotEmpty() @Length(2, 31) factionName: string; + @IsHexColor() + colour: string; @IsString() @IsNotEmpty() @Length(3, 15) @@ -46,7 +56,7 @@ export class JoinFactionDTO { export class PromotePlayerDTO { @IsUUID('4') player: string; - @Validate(RoleValidation) + @IsIn(['admin', 'soldier', 'factionleader']) role: string; } @@ -54,6 +64,8 @@ export class GameGroupDTO { @IsString() @Length(3, 31) name: string; + @IsIn(['infantry', 'recon', 'mechanized']) + class: string; @IsUUID('4') faction: FactionEntity; } diff --git a/src/faction/faction.entity.ts b/src/faction/faction.entity.ts index 3aa74cf615d77deb1421d52d46d4159bf7e0d241..20b4a0a28b1d6eba7bd24cb0e0bad3fdd518bdcc 100644 --- a/src/faction/faction.entity.ts +++ b/src/faction/faction.entity.ts @@ -5,7 +5,6 @@ import { OneToMany, ManyToOne, OneToOne, - Timestamp, JoinColumn, } from 'typeorm'; @@ -13,80 +12,49 @@ import { GameEntity } from '../game/game.entity'; import { Game_PersonEntity } from '../game/game.entity'; import { MapDrawingEntity } from '../draw/coordinate.entity'; import { Exclude } from 'class-transformer'; +import { ScoreEntity } from '../score/score.entity'; -//Faction, PowerUp, Faction_powerUp, FP_History, Score +/////////////////////////////////////////////////////////////////// +/// FactionEntity & GameGroupEntity reflect database tables. /// +/////////////////////////////////////////////////////////////////// @Entity('Faction') export class FactionEntity { @PrimaryGeneratedColumn('uuid') factionId: string; @Column('text') factionName: string; @Column({ type: 'float' }) multiplier: number; + @Column('text') colour: string; + // Faction's password won't be included when FactionEntity is transformed @Exclude() @Column({ type: 'text' }) factionPassword: string; @OneToMany(type => Game_PersonEntity, game_persons => game_persons.faction) game_persons: Game_PersonEntity[]; + // When Game where a Faction is in is deleted, the Factions in that game are also deleted @ManyToOne(type => GameEntity, game => game.factions, { onDelete: 'CASCADE', }) game: GameEntity; @OneToMany(type => MapDrawingEntity, mapDrawings => mapDrawings.faction) mapDrawings: MapDrawingEntity[]; - - factionObject() { - const { factionId, factionName, game } = this; - return { factionId, factionName, game }; - } + @OneToMany(type => ScoreEntity, score => score.scoreId, { + onDelete: 'NO ACTION', + }) + scores: ScoreEntity[]; passwordCheck(pass: string) { return pass == this.factionPassword ? true : false; } } -/* @Entity('PowerUp') -export class PowerUpEntity { - @PrimaryGeneratedColumn('uuid') powerUpId: string; - @Column({ type: 'text' }) powerUpName: string; - @Column({ type: 'text' }) powerUpDescription: string; - @Column({ type: 'int' }) amount: number; - @Column({ type: 'time' }) cooldown: string; - - @OneToMany(type => FactionEntity, factions => factions.factionId, { - onDelete: 'CASCADE', - }) - factions: Faction_PowerUpEntity[]; -} - -@Entity('Faction_PowerUp') -export class Faction_PowerUpEntity { - @PrimaryGeneratedColumn('uuid') faction_powerUpId: string; - - @ManyToOne(type => FactionEntity, faction => faction.factionId) - faction: FactionEntity; - @ManyToOne(type => PowerUpEntity, powerUp => powerUp.factions) - powerUp: PowerUpEntity; - @OneToMany(type => FP_HistoryEntity, histories => histories.faction_PowerUp) - histories: FP_HistoryEntity[]; -} - -@Entity('FP_History') -export class FP_HistoryEntity { - @PrimaryGeneratedColumn('uuid') historyId: string; - @Column({ type: 'timestamp' }) historyTimeStamp: Timestamp; - - @ManyToOne( - type => Faction_PowerUpEntity, - faction_PowerUp => faction_PowerUp.histories, - ) - faction_PowerUp: Faction_PowerUpEntity; -} */ - @Entity('GameGroup') export class GameGroupEntity { @PrimaryGeneratedColumn('uuid') id: string; @Column('text') name: string; + @Column('text') class: string; + // When Groups leader, players or Faction is deleted, the Group(s) go also @OneToOne(type => Game_PersonEntity, person => person.leaderGroup, { onDelete: 'CASCADE', }) diff --git a/src/faction/faction.module.ts b/src/faction/faction.module.ts index dcb399aa1119851781270477c448ecec7362c804..3a3026da3695e79a7e375ea963cceee3360e048f 100644 --- a/src/faction/faction.module.ts +++ b/src/faction/faction.module.ts @@ -6,6 +6,19 @@ import { FactionService } from './faction.service'; import { GameGroupEntity, FactionEntity } from './faction.entity'; import { Game_PersonEntity } from '../game/game.entity'; +///////////////////////////// +/// Entities /// +/// - FactionEntity /// +/// - Game_PersonEntity /// +/// - GameGroupEntity /// +/// /// +/// Controllers /// +/// - FactionController /// +/// /// +/// Provider /// +/// - FactionService /// +///////////////////////////// + @Module({ imports: [ TypeOrmModule.forFeature([ diff --git a/src/faction/faction.service.ts b/src/faction/faction.service.ts index 67edef1db5d071e1936aba03ded5c5051c1ae713..9447c36777b01762a697bf75b38c36d14b34b4b8 100644 --- a/src/faction/faction.service.ts +++ b/src/faction/faction.service.ts @@ -3,9 +3,29 @@ import { InjectRepository } from '@nestjs/typeorm'; import { Repository, Not } from 'typeorm'; import { FactionEntity, GameGroupEntity } from './faction.entity'; -import { JoinFactionDTO, GameGroupDTO, JoinGameGroupDTO } from './faction.dto'; +import { + JoinFactionDTO, + GameGroupDTO, + JoinGameGroupDTO, + FactionDTO, +} from './faction.dto'; import { Game_PersonEntity } from '../game/game.entity'; +/////////////////////////////////////////////////////// +/// FactionService contains functions for /// +/// - Joining faction /// +/// - Leaving faction /// +/// - Change faction multiplier (not implemented) /// +/// /// +/// Group /// +/// - Creating group /// +/// - List faction players in groups /// +/// /// +/// Player /// +/// - Promote player /// +/// - verifying user /// +/////////////////////////////////////////////////////// + @Injectable() export class FactionService { constructor( @@ -53,6 +73,23 @@ export class FactionService { } } + // removes entry from gameperson + async leaveFaction(gamepersonId) { + await this.game_PersonRepository.delete({ gamepersonId }); + return { + message: 'leaved faction', + }; + } + + // changes factions multiplier to the value given to FactionDTO + // async changeFactionMultiplier(body: FactionDTO) { + // const faction = await this.factionRepository.findOne({ + // where: { factionId: body.factionId }, + // }); + // faction.multiplier = body.multiplier; + // return await this.factionRepository.save(faction); + // } + async promotePlayer(body) { const gamepersonId = body.player; // get playerdata @@ -60,9 +97,15 @@ export class FactionService { where: { gamepersonId }, }); if (gameperson) { - const factionleader = await this.game_PersonRepository.create(gameperson); - factionleader.role = body.role; - return await this.game_PersonRepository.save(factionleader); + const promotedPlayer = await this.game_PersonRepository.create( + gameperson, + ); + promotedPlayer.role = body.role; + if (body.role === 'admin') { + promotedPlayer.faction = null; + promotedPlayer.group = null; + } + return await this.game_PersonRepository.save(promotedPlayer); } throw new HttpException('player does not exist', HttpStatus.BAD_REQUEST); } @@ -75,12 +118,11 @@ export class FactionService { game: gameId, }); // check if the authenticated person already belongs to a group - if ( - await this.game_PersonRepository.findOne({ - group: Not(null), - person: person, - }) - ) { + let check = await this.game_PersonRepository.findOne({ + where: { person: person, game: gameId }, + relations: ['group'], + }); + if (check.group) { throw new HttpException( 'You already belong to a group!', HttpStatus.BAD_REQUEST, @@ -103,47 +145,92 @@ export class FactionService { }; } + // get the groups in the given Faction async showGroups(factionId) { - return await this.game_GroupRepository.find({ - relations: ['leader', 'players'], + let players = await this.game_PersonRepository.find({ where: { faction: factionId }, + relations: ['person', 'group'], }); - } - async joinGroup(person, gameId, data: JoinGameGroupDTO) { - const gamePerson = await this.game_PersonRepository.findOne({ - person: person, - game: gameId, + players.sort(function(a, b) { + return a.person.name.localeCompare(b.person.name); }); - gamePerson.group = data.groupId; - await this.game_PersonRepository.save(gamePerson); - return { - message: 'Joined group', - }; - } - async listFactionMembers(faction) { - const members = await this.game_PersonRepository.find({ - where: { faction }, - relations: ['person'], + let groups = await this.game_GroupRepository.find({ + where: { faction: factionId }, + relations: ['leader', 'leader.person'], }); - members.sort(function(a, b) { - return a['person']['name'].localeCompare(b['person']['name']); + + let resObj = await Promise.all( + groups.map(async group => { + return await { + id: group.id, + name: group.name, + class: group.class, + leader: group.leader.person.name, + players: [], + }; + }), + ); + + resObj.push({ + id: 'empty-group-id', + name: 'No group', + class: 'infantry', + leader: '', + players: [], }); - return members; + + await Promise.all( + players.map(async player => { + for (let i = 0; i < resObj.length; i++) { + if (player.group == null) { + resObj[resObj.length - 1].players.push({ + gamepersonId: player.gamepersonId, + role: player.role, + name: player.person.name, + }); + break; + } + if (resObj[i].name == player.group.name) { + resObj[i].players.push({ + gamepersonId: player.gamepersonId, + role: player.role, + name: player.person.name, + }); + break; + } + } + }), + ); + + return resObj; } + // puts a non admin or faction leader player into a specified group + async joinGroup(gameperson, data: JoinGameGroupDTO) { + gameperson.group = data.groupId; + await this.game_PersonRepository.save(gameperson); + return { + message: 'Joined group', + }; + } + + // checks if player is in a faction and what role the player is in async verifyUser(person, game) { const gameperson = await this.game_PersonRepository.findOne({ where: { person, game }, - relations: ['faction'], + relations: ['faction', 'group'], }); - if (gameperson) { + if (gameperson && gameperson.faction) { return { - message: gameperson, + factionId: gameperson.faction.factionId, + factionName: gameperson.faction.factionName, + role: gameperson.role, + group: gameperson.group ? true : false, }; } else { - throw new HttpException('No faction was found', HttpStatus.BAD_REQUEST); + return gameperson ? { role: gameperson.role } : { role: '' }; } } } diff --git a/src/game/game.controller.ts b/src/game/game.controller.ts index 8d7494ba4dfa4a77e22963600e45a7579fca5d5d..93068318edb8a99755c8cd44adaa795573888981 100644 --- a/src/game/game.controller.ts +++ b/src/game/game.controller.ts @@ -10,27 +10,43 @@ import { UseInterceptors, ClassSerializerInterceptor, Delete, + UploadedFile, + Res, } from '@nestjs/common'; +import { FileInterceptor } from '@nestjs/platform-express'; +import { diskStorage } from 'multer'; +import { extname } from 'path'; import { GameService } from './game.service'; import { AuthGuard } from '../shared/auth.guard'; import { User } from '../user/user.decorator'; -import { GameDTO, FlagboxEventDTO, GameStateDTO } from './game.dto'; +import { GameDTO, FlagboxEventDTO, GameStateDTO, newGameDTO } from './game.dto'; import { ValidationPipe } from '../shared/validation.pipe'; import { Roles, GameStates } from '../shared/guard.decorator'; import { GameEntity } from './game.entity'; +///////////////////////////////////////////////////////////////////////// +/// GameController /// +/// /// +/// Functions for creating, editing, deleting and listing games. /// +/// Also there are functions to get objective point info and list /// +/// of Factions in game /// +/// /// +///////////////////////////////////////////////////////////////////////// + @Controller('game') export class GameController { constructor(private gameservice: GameService) {} + //new game @Post('new') @UseGuards(new AuthGuard()) @UsePipes(new ValidationPipe()) - async newGame(@User('id') person, @Body() body: GameDTO) { + async newGame(@User('id') person, @Body() body: newGameDTO) { return this.gameservice.createNewGame(person, body); } + // edit game @Put('edit/:id') @Roles('admin') @GameStates('CREATED') @@ -40,6 +56,7 @@ export class GameController { return this.gameservice.editGame(id, body); } + // delete game @Delete('delete/:id') @Roles('admin') @GameStates('CREATED') @@ -47,6 +64,7 @@ export class GameController { return this.gameservice.deleteGame(id); } + // change game state @Put('edit-state/:id') @Roles('admin') @UsePipes(new ValidationPipe()) @@ -54,9 +72,16 @@ export class GameController { return this.gameservice.updateGameStatus(body); } + // list all games @Get('listgames') - async listGames() { - return this.gameservice.listGames(); + async listGames(state) { + return this.gameservice.listGames(state); + } + + // list games based on parameter + @Get('listgames/:state') + async listGamesState(@Param('state') state: string) { + return this.gameservice.listGames(state); } // ClassSerializerInterceptor removes excluded columns set in Entities @@ -66,20 +91,57 @@ export class GameController { return this.gameservice.returnGameInfo(id); } + //get all factions @Get('get-factions/:id') @Roles('admin') async returnGameFactions(@Param('id') id: GameEntity) { return this.gameservice.listFactions(id); } + // get flagbox events + @Get('flag-events/:id') + async returnFlagboxInfo(@Param('id') id: GameEntity) { + return this.gameservice.returnObjectivePointInfo(id); + } + + // initial settings for flagbox @Get('flag/:id') async flagboxQuery(@Param('id') id: string) { return this.gameservice.flagboxQuery(id); } + // flagbox event @Post('flag/:id') @GameStates('STARTED') async flagboxEvent(@Param('id') id: string, @Body() data: FlagboxEventDTO) { return this.gameservice.flagboxEvent(id, data); } + + // image upload + @Post('upload') + @UseInterceptors( + FileInterceptor('image', { + storage: diskStorage({ + destination: './images', + filename: (req, file, cb) => { + // Generating a 32 random chars long string + const randomName = Array(32) + .fill(null) + .map(() => Math.round(Math.random() * 16).toString(16)) + .join(''); + //Calling the callback passing the random name generated with the original extension name + cb(null, `${randomName}${extname(file.originalname)}`); + }, + }), + }), + ) + uploadImage(@UploadedFile() image) { + return image; + } + + // get images + @Get('images/:img') + returnImage(@Param('img') image, @Res() res) { + return res.sendFile(image, { root: 'images' }); + } } diff --git a/src/game/game.dto.ts b/src/game/game.dto.ts index 266a8b1564e5b11fb42da755359c79d37f4b170b..a5bb525ddc6c34a17da91787a4bf5027405099f2 100644 --- a/src/game/game.dto.ts +++ b/src/game/game.dto.ts @@ -4,7 +4,6 @@ import { Length, IsDateString, IsNumber, - Validate, Min, Max, ValidateNested, @@ -15,7 +14,6 @@ import { } from 'class-validator'; import { ObjectivePointEntity } from './game.entity'; -import { CenterJSON } from '../shared/custom-validation'; import { FactionDTO } from '../faction/faction.dto'; import { CenterDTO, NodeSettingsDTO } from './game.json.dto'; import { Type } from 'class-transformer'; @@ -62,21 +60,25 @@ export class newGameDTO { @IsNotEmpty() @Length(1, 255) desc: string; - @IsNotEmpty() - @Validate(CenterJSON) - center: JSON; + @ValidateNested() + @Type(() => CenterDTO) + center: CenterDTO; @IsDateString() @IsNotEmpty() startdate: string; @IsDateString() @IsNotEmpty() enddate: string; + @Length(0, 65) + image: string; + @Allow() + map?: JSON; } export class GameStateDTO { @IsUUID('4') id: string; - @IsIn(['CREATED', 'STARTED', 'PAUSED', 'ENDED']) + @IsIn(['CREATED', 'STARTED', 'PAUSED', 'ENDED', 'ONGOING']) state: string; } @@ -86,16 +88,18 @@ export class FlagboxDTO { objectivePointId: string; @IsString() @IsNotEmpty() - @Length(7) + @Length(7, 7) objectivePointDescription: string; @IsNumber() objectivePointMultiplier: number; + @IsOptional() + data: JSON; } export class FlagboxEventDTO { @IsString() @IsNotEmpty() - @Length(7) + @Length(7, 7) node_id: string; @IsNumber() @Min(0) @@ -109,6 +113,6 @@ export class FlagboxEventDTO { @Min(0) @Max(3) capture: number; // which faction is capturing, same logic as in owner with numbers - oP_HistoryTimestamp?: string; + oP_HistoryTimestamp?: number; objective_point?: ObjectivePointEntity; } diff --git a/src/game/game.entity.ts b/src/game/game.entity.ts index 7ae6e29955d02a2c1368b5f720200f009bd92ee1..56f3f3f6e2ffe341e6677e574bebf4b98b2b7be7 100644 --- a/src/game/game.entity.ts +++ b/src/game/game.entity.ts @@ -28,6 +28,7 @@ export class GameEntity { @Column('text') state: string; @Column('timestamp') startdate: Timestamp; @Column('timestamp') enddate: Timestamp; + @Column('text') image: string; @OneToMany(type => FactionEntity, factions => factions.game) factions: FactionEntity[]; @@ -56,6 +57,7 @@ export class GameEntity { export class Game_PersonEntity { @PrimaryGeneratedColumn('uuid') gamepersonId: string; @Column({ type: 'text', nullable: true }) role: string; + // If a Faction or Game where the GamePerson was in is deleted, the GamePerson is also deleted @ManyToOne(type => FactionEntity, faction => faction.game_persons, { onDelete: 'CASCADE', }) @@ -68,6 +70,8 @@ export class Game_PersonEntity { person: PersonEntity; @OneToOne(type => GameGroupEntity, group => group.leader) leaderGroup: GameGroupEntity; + + // When a Group where GamePerson is is deleted, nothing happens to the GamePerson @ManyToOne(type => GameGroupEntity, group => group.players, { onDelete: 'NO ACTION', }) @@ -80,22 +84,31 @@ export class ObjectivePointEntity { @PrimaryGeneratedColumn('uuid') objectivePointId: string; @Column({ type: 'text' }) objectivePointDescription: string; @Column({ type: 'float' }) objectivePointMultiplier: number; + @Column({ type: 'json' }) data: JSON; - @ManyToOne(type => MapDrawingEntity, coordinate => coordinate.data, { - onDelete: 'CASCADE', - }) - coordinate: MapDrawingEntity; + // If the Game where the ObjectivePoint was in is deleted, the ObjectivePoint is also deleted @ManyToOne(type => GameEntity, game => game.objective_points, { onDelete: 'CASCADE', }) game: GameEntity; + @OneToMany( + () => ObjectivePoint_HistoryEntity, + history => history.objective_point, + { + onDelete: 'NO ACTION', + }, + ) + history: ObjectivePoint_HistoryEntity[]; } @Entity('ObjectivePoint_History') export class ObjectivePoint_HistoryEntity { @PrimaryGeneratedColumn('uuid') oP_HistoryId: string; - @Column({ type: 'timestamp' }) oP_HistoryTimestamp: Timestamp; + @Column({ type: 'float' }) oP_HistoryTimestamp: number; @Column('float') action: number; + + // If the owner Faction, capturer Faction or ObjectivePoint, that has, is trying to have or is the point where + // ObjectivePointHistory points to is deleted, the ObjectivePointHistory is also deleted @ManyToOne(type => FactionEntity, factionEntity => factionEntity.factionId, { onDelete: 'CASCADE', }) diff --git a/src/game/game.module.ts b/src/game/game.module.ts index 1c9e86faf59645bf537e76a8720bb6ab741fe6bd..b30424ced696027a22d255e8618c90ad40209f29 100644 --- a/src/game/game.module.ts +++ b/src/game/game.module.ts @@ -13,6 +13,14 @@ import { PersonEntity } from '../user/user.entity'; import { GameGroupEntity } from '../faction/faction.entity'; import { FactionEntity } from '../faction/faction.entity'; import { NotificationModule } from '../notifications/notifications.module'; +import { ScoreService } from '../score/score.service'; +import { ScoreEntity } from '../score/score.entity'; +import { MulterModule } from '@nestjs/platform-express'; + +///////////////////////////////////////////////////////////////////// +/// Game /// +/// - contains everything to do with Game data /// +///////////////////////////////////////////////////////////////////// @Module({ imports: [ @@ -24,10 +32,14 @@ import { NotificationModule } from '../notifications/notifications.module'; GameGroupEntity, ObjectivePointEntity, ObjectivePoint_HistoryEntity, + ScoreEntity, ]), NotificationModule, + MulterModule.register({ + dest: './images', + }), ], controllers: [GameController], - providers: [GameService], + providers: [GameService, ScoreService], }) export class GameModule {} diff --git a/src/game/game.service.ts b/src/game/game.service.ts index 3f6506b649f7dbe09936e283b4658f9765be8ec4..1567d30a1c5aa751de2ef9b0dff4d18af7af94d4 100644 --- a/src/game/game.service.ts +++ b/src/game/game.service.ts @@ -8,7 +8,7 @@ import { ObjectivePointEntity, ObjectivePoint_HistoryEntity, } from './game.entity'; -import { GameDTO, FlagboxEventDTO, GameStateDTO } from './game.dto'; +import { GameDTO, FlagboxEventDTO, GameStateDTO, newGameDTO } from './game.dto'; import { PersonEntity } from '../user/user.entity'; import { FactionEntity } from '../faction/faction.entity'; import { NotificationGateway } from '../notifications/notifications.gateway'; @@ -30,9 +30,8 @@ export class GameService { >, private notificationGateway: NotificationGateway, ) {} - // create a new game - async createNewGame(personId: PersonEntity, gameData: GameDTO) { + async createNewGame(personId: PersonEntity, gameData: newGameDTO) { // checks if a game with the same name exists already if (await this.gameRepository.findOne({ name: gameData.name })) { throw new HttpException('Game already exists', HttpStatus.BAD_REQUEST); @@ -81,6 +80,14 @@ export class GameService { HttpStatus.BAD_REQUEST, ); } + // check that there's location data for each added objective point + gameData.objective_points.forEach(obj => { + if (!obj['data']) + throw new HttpException( + 'Objective Point error. Add location for each Objective Point.', + HttpStatus.BAD_REQUEST, + ); + }); // get factions that have been added previously let factions = await this.factionRepository.find({ game: id }); @@ -119,8 +126,10 @@ export class GameService { ({ objectivePointId }) => objectivePointId, ); flagboxes.map(async flagbox => { - if (!flagboxIds.includes(flagbox.objectivePointDescription)) { - await this.objectivePointRepository.delete(flagbox); + if (!flagboxIds.includes(flagbox.objectivePointId)) { + await this.objectivePointRepository.delete({ + objectivePointId: flagbox.objectivePointId, + }); } }); gameData.objective_points.map(async flagbox => { @@ -129,13 +138,18 @@ export class GameService { game: gameId, }); await this.objectivePointRepository.save(newFlagbox); + // create base status for flagbox + this.flagboxEvent(gameId, { + node_id: flagbox.objectivePointDescription, + owner: 0, + action: 0, + capture: 0, + }); }); } else { await this.objectivePointRepository.delete({ game: id }); } - // TO DO: ADD FLAGBOX LOCATION TO MAPDRAWING ENTITY - return { message: 'Game updated', }; @@ -146,6 +160,7 @@ export class GameService { if (updatedGame) { updatedGame.state = game.state; await this.gameRepository.save(updatedGame); + // notify players about game state change this.notificationGateway.server.emit(game.id, { type: 'gamestate-update', @@ -162,7 +177,7 @@ export class GameService { } async deleteGame(id) { - // TODO: Delete factions from Faction table associated with the deleted game + // Delete factions from Faction table associated with the deleted game await this.gameRepository.delete({ id }); return { message: 'Game deleted', @@ -170,15 +185,35 @@ export class GameService { } // returns name and id of each game - async listGames() { - const games = await this.gameRepository.find(); - return games.map(game => { - return game.gameObject(); - }); + async listGames(state) { + if (state == null) { + const games = await this.gameRepository.find(); + return games.map(game => { + return game.gameObject(); + }); + } else if (state == 'ONGOING') { + const games = await this.gameRepository.find({ + where: [ + { state: 'CREATED' }, + { state: 'STARTED' }, + { state: 'PAUSED' }, + ], + }); + return games.map(game => { + return game.gameObject(); + }); + } else { + const games = await this.gameRepository.find({ + where: { state: state }, + }); + return games.map(game => { + return game.gameObject(); + }); + } } // returns information about a game identified by id - async returnGameInfo(id: string) { + async returnGameInfo(id) { const game = await this.gameRepository.findOne({ where: { id: id }, relations: ['factions', 'objective_points'], @@ -190,6 +225,53 @@ export class GameService { return game; } + // returns information about game's flagboxes and their most recent event + async returnObjectivePointInfo(gameId) { + const info = await this.objectivePointRepository.find({ + where: { game: gameId }, + relations: ['history', 'history.owner', 'history.capture'], + }); + let response = await Promise.all( + info.map(async obj => { + let history = obj.history.pop(); + return await { + objectivePointId: obj.objectivePointId, + objectivePointDescription: obj.objectivePointDescription, + objectivePointMultiplier: obj.objectivePointMultiplier, + action: { + status: history.action, + message: { + 0: 'No capture ongoing', + 1: `Captured by ${ + history.owner ? history.owner.factionName : 'neutral' + }`, + 2: `Being captured by ${ + history.capture ? history.capture.factionName : 'neutral' + }`, + }[history.action], + }, + owner: await this.infoHelper(history.owner), + capture: await this.infoHelper(history.capture), + data: obj.data, + }; + }), + ); + return response; + } + + //returns flagbox colour and faction + private async infoHelper(obj) { + return (await obj) + ? { + factionName: obj.factionName, + colour: obj.colour, + } + : { + factionName: 'neutral', + colour: '#000000', + }; + } + // returns flagbox settings async flagboxQuery(gameId) { const game = await this.gameRepository.findOne({ id: gameId }); @@ -204,7 +286,7 @@ export class GameService { const objectiveRef = await this.objectivePointRepository.findOne({ where: { objectivePointDescription: data.node_id, game: gameId }, }); - data.oP_HistoryTimestamp = new Date(Date.now()).toLocaleString(); + data.oP_HistoryTimestamp = Date.now(); const eventUpdate = await this.objectivePoint_HistoryRepository.create({ oP_HistoryTimestamp: data.oP_HistoryTimestamp, action: data.action, diff --git a/src/game/gameperson.decorator.ts b/src/game/gameperson.decorator.ts new file mode 100644 index 0000000000000000000000000000000000000000..3e2a232880332fbd62b54f76bbb37bf91ef714e7 --- /dev/null +++ b/src/game/gameperson.decorator.ts @@ -0,0 +1,16 @@ +import { createParamDecorator } from '@nestjs/common'; + +/////////////////////////////////////////////////////////////////////////// +/// gives service access to the gameperson object /// +/// Game_PersonEntity { /// +/// gamepersonId /// +/// role /// +/// faction /// +/// } /// +/// /// +/// See more information from decorators in shared folder files. /// +/////////////////////////////////////////////////////////////////////////// + +export const GamePerson = createParamDecorator((data, req) => { + return data ? req.gameperson[data] : req.gameperson; +}); diff --git a/src/main.ts b/src/main.ts index 68871be82d89a37a410b93b252fb6f1e712ad143..20e08b20631984132638f542266bd34a38ef74ba 100644 --- a/src/main.ts +++ b/src/main.ts @@ -1,26 +1,21 @@ import { NestFactory } from '@nestjs/core'; -import * as rateLimit from 'express-rate-limit'; import { AppModule } from './app.module'; -/* - Main.ts starts the server. -*/ - -// due to a bug with newest release of express-rate-limit, call for rateLimit is broken -// (rateLimit as any) works as a workaround for now -// see https://github.com/nfriedly/express-rate-limit/issues/138 -const limiter = (rateLimit as any)({ - windowMs: 60 * 1000, // one minute - max: 100, // limit each IP to 100 requests per windowMs -}); +/////////////////////////////////////////////////////////////////////////// +/// Main.ts starts the server. /// +/// /// +/// .env.PORT is not defined, port 5000 will be listened by default /// +/////////////////////////////////////////////////////////////////////////// async function bootstrap() { + // port opened + const port = 5000; + const app = await NestFactory.create(AppModule); // Cors is needed for application/json POST app.enableCors(); - // apply limiter to all routes - app.use(limiter); - await app.listen(5000); + // Server will listen on port + await app.listen(process.env.PORT || port); } bootstrap(); diff --git a/src/notifications/notification.entity.ts b/src/notifications/notification.entity.ts index f9d5dca9290c30da406a63adf56ee06fe716a7b5..5e1f841cf031767574c75d54af65ca209fc7e86b 100644 --- a/src/notifications/notification.entity.ts +++ b/src/notifications/notification.entity.ts @@ -16,6 +16,7 @@ export class NotificationEntity { @Column({ type: 'text' }) message: string; @CreateDateColumn() issued: Date; + // Notifications are deleted if the game is deleted @ManyToOne(type => GameEntity, game => game.id, { onDelete: 'CASCADE', }) diff --git a/src/notifications/notifications.controller.ts b/src/notifications/notifications.controller.ts new file mode 100644 index 0000000000000000000000000000000000000000..83101be3d6cb57b58f5147ddbf0347a3291bb5db --- /dev/null +++ b/src/notifications/notifications.controller.ts @@ -0,0 +1,16 @@ +import { Controller, Get, Param } from '@nestjs/common'; +import { NotificationsService } from './notifications.service'; +import { Roles } from 'src/shared/guard.decorator'; + +@Controller('notifications') +export class NotificationsController { + constructor(private notificationService: NotificationsService) {} + + // get all sent notifications for game + // :id is the id of the game + @Get(':id') + @Roles('admin', 'factionleader', 'soldier', 'groupleader') + async(@Param('id') gameId) { + return this.notificationService.getNotifications(gameId); + } +} diff --git a/src/notifications/notifications.gateway.ts b/src/notifications/notifications.gateway.ts index 5e952d6c78c6f35977f362a7459c0ebf65d1ab5e..79fcc0e5f287858f1cf62862dde722e372ba8ef9 100644 --- a/src/notifications/notifications.gateway.ts +++ b/src/notifications/notifications.gateway.ts @@ -16,6 +16,10 @@ import { GameEntity } from '../game/game.entity'; import { NotificationdDTO } from './notification.dto'; import { ValidationPipe } from '../shared/validation.pipe'; +/////////////////////////////////////////////////////////////////////////////////// +/// NotificationGateway contains websocket server and listener for game-info /// +/////////////////////////////////////////////////////////////////////////////////// + @WebSocketGateway() export class NotificationGateway implements OnGatewayInit, OnGatewayConnection, OnGatewayDisconnect { @@ -50,12 +54,25 @@ export class NotificationGateway async handleMessage(client: Socket, data: NotificationdDTO) { // check if the game exists and is either started or paused const game = await this.gameRepository.findOne({ id: data.game }); - if (game && ['STARTED', 'PAUSED'].includes(game.state)) { + if (!game) { + // inform user about error + this.server.to(client.id).emit(data.game, { + type: 'error', + message: 'Game was not found', + }); + } + if (['STARTED', 'PAUSED'].includes(game.state)) { // send the message to all clients listening to gameId branch this.server.emit(data.game, data); // create entry for notification in db const message = await this.notificationRepository.create(data); await this.notificationRepository.insert(message); + } else { + // inform user about error + this.server.to(client.id).emit(data.game, { + type: 'error', + message: 'Notifications can be sent only in STARTED and PAUSED state', + }); } } } diff --git a/src/notifications/notifications.module.ts b/src/notifications/notifications.module.ts index 41e2fcaa36d1928dff9a60bd2c6853e240baf16f..5ca243fcb3c39c6d49298beff12de51634575e19 100644 --- a/src/notifications/notifications.module.ts +++ b/src/notifications/notifications.module.ts @@ -4,10 +4,18 @@ import { TypeOrmModule } from '@nestjs/typeorm'; import { NotificationGateway } from './notifications.gateway'; import { NotificationEntity } from './notification.entity'; import { GameEntity } from '../game/game.entity'; +import { NotificationsController } from './notifications.controller'; +import { NotificationsService } from './notifications.service'; + +///////////////////////////////////////////////////////////////////// +/// Notification /// +/// - contains everything to do with Notification data. /// +///////////////////////////////////////////////////////////////////// @Module({ imports: [TypeOrmModule.forFeature([NotificationEntity, GameEntity])], - providers: [NotificationGateway], + providers: [NotificationGateway, NotificationsService], exports: [NotificationGateway], + controllers: [NotificationsController], }) export class NotificationModule {} diff --git a/src/notifications/notifications.service.ts b/src/notifications/notifications.service.ts new file mode 100644 index 0000000000000000000000000000000000000000..0bbe016ecba9bde516b99222bbe388b3269bba6c --- /dev/null +++ b/src/notifications/notifications.service.ts @@ -0,0 +1,17 @@ +import { Injectable } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; +import { NotificationEntity } from './notification.entity'; +import { Repository } from 'typeorm'; + +@Injectable() +export class NotificationsService { + constructor( + @InjectRepository(NotificationEntity) + private notificationRepository: Repository<NotificationEntity>, + ) {} + + // finds all notifications for specified game + async getNotifications(game: string) { + return this.notificationRepository.find({ game }); + } +} diff --git a/src/replay/replay.controller.ts b/src/replay/replay.controller.ts new file mode 100644 index 0000000000000000000000000000000000000000..e39e99d8ba1e5be763e21fc1e00b6496e6e95648 --- /dev/null +++ b/src/replay/replay.controller.ts @@ -0,0 +1,33 @@ +import { + Controller, + Get, + Param, + Post, + UseInterceptors, + ClassSerializerInterceptor, +} from '@nestjs/common'; +import { ReplayService } from './replay.service'; +import { Roles } from 'src/shared/guard.decorator'; + +///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// +/// POST mockdata is mainly used for development, it can be removed from the code when needed, remember to remove service & test relations. /// +///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + +@Controller('replay') +export class ReplayController { + constructor(private replayservice: ReplayService) {} + + // gets replay data for specified Game + @Get(':id') + @UseInterceptors(ClassSerializerInterceptor) + async replayInfo(@Param('id') gameId) { + return this.replayservice.replayData(gameId); + } + + // gets mockdata for specified Game + @Post('mockdata/:id') + @Roles('admin') + async mockData(@Param('id') gameId) { + return this.replayservice.mockdata(gameId); + } +} diff --git a/src/replay/replay.module.ts b/src/replay/replay.module.ts new file mode 100644 index 0000000000000000000000000000000000000000..961e599e4cb9aca8a27dac015b83edb386e41e64 --- /dev/null +++ b/src/replay/replay.module.ts @@ -0,0 +1,59 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { ReplayController } from './replay.controller'; +import { ReplayService } from './replay.service'; +import { + GameEntity, + Game_PersonEntity, + ObjectivePointEntity, + ObjectivePoint_HistoryEntity, +} from '../game/game.entity'; +import { FactionEntity, GameGroupEntity } from '../faction/faction.entity'; +import { UserService } from '../user/user.service'; +import { FactionService } from '../faction/faction.service'; +import { TrackingService } from '../tracking/tracking.service'; +import { TrackingEntity } from '../tracking/tracking.entity'; +import { PersonEntity } from '../user/user.entity'; +import { + MapDrawingEntity, + MapDrawingHistoryEntity, +} from '../draw/coordinate.entity'; +import { ScoreService } from '../score/score.service'; +import { ScoreEntity } from '../score/score.entity'; +import { NotificationModule } from '../notifications/notifications.module'; +import { GameService } from '../game/game.service'; + +///////////////////////////////////////////////////////////////////// +/// Replay /// +/// - contains everything to do with Replay data. /// +///////////////////////////////////////////////////////////////////// + +@Module({ + imports: [ + TypeOrmModule.forFeature([ + PersonEntity, + GameEntity, + FactionEntity, + TrackingEntity, + GameGroupEntity, + Game_PersonEntity, + MapDrawingEntity, + MapDrawingHistoryEntity, + ScoreEntity, + ObjectivePointEntity, + ObjectivePoint_HistoryEntity, + ]), + NotificationModule, + ], + controllers: [ReplayController], + providers: [ + ReplayService, + UserService, + FactionService, + TrackingService, + ScoreService, + GameService, + ], +}) +export class ReplayModule {} diff --git a/src/replay/replay.service.ts b/src/replay/replay.service.ts new file mode 100644 index 0000000000000000000000000000000000000000..be423495a09073676a7c7b81916876b23f5ecf38 --- /dev/null +++ b/src/replay/replay.service.ts @@ -0,0 +1,323 @@ +import { Injectable } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; +import { Repository } from 'typeorm'; +import * as jwt from 'jsonwebtoken'; + +import { FactionEntity } from '../faction/faction.entity'; +import { + GameEntity, + ObjectivePointEntity, + ObjectivePoint_HistoryEntity, +} from '../game/game.entity'; +import { TrackingService } from '../tracking/tracking.service'; +import { UserService } from '../user/user.service'; +import { FactionService } from '../faction/faction.service'; +import { TrackingEntity } from '../tracking/tracking.entity'; +import { + MapDrawingEntity, + MapDrawingHistoryEntity, +} from '../draw/coordinate.entity'; +import { ScoreService } from '../score/score.service'; +import { ScoreEntity } from '../score/score.entity'; +import { GameService } from '../game/game.service'; + +@Injectable() +export class ReplayService { + constructor( + @InjectRepository(FactionEntity) + private factionRepository: Repository<FactionEntity>, + @InjectRepository(GameEntity) + private gameRepository: Repository<GameEntity>, + @InjectRepository(TrackingEntity) + private trackingRepository: Repository<TrackingEntity>, + @InjectRepository(MapDrawingEntity) + private mapdrawingRepository: Repository<MapDrawingEntity>, + @InjectRepository(MapDrawingHistoryEntity) + private mapHistoryRepository: Repository<MapDrawingHistoryEntity>, + @InjectRepository(ScoreEntity) + private scoreRepository: Repository<ScoreEntity>, + @InjectRepository(ObjectivePointEntity) + private objectivepointRepository: Repository<ObjectivePointEntity>, + private trackingService: TrackingService, + private userService: UserService, + private factionService: FactionService, + private scoreService: ScoreService, + private gameService: GameService, + ) {} + + async replayData(gameId) { + // this block returns game's initial location + let gameObj = await this.gameRepository.findOne({ + where: { id: gameId }, + select: ['center'], + }); + let gamelocation = [gameObj.center.lat, gameObj.center.lng]; + // this block returns all player data from the game + let playerdata = await this.trackingRepository.find({ + where: { game: gameId }, + relations: ['faction', 'gamepersonId', 'gamepersonId.person'], + }); + // parse data + const currentdata = await Promise.all( + playerdata.map(async player => { + player.data.map(async data => { + data['info'] = [ + { key: 'icon', value: player.icon }, + { key: 'colour', value: player.faction.colour }, + { key: 'name', value: player.gamepersonId.person.name }, + { key: 'role', value: player.gamepersonId.role }, + ]; + }); + return player['data']; + }), + ); + + // this block returns all faction data from the game + + let factions = await this.factionRepository.find({ game: gameId }); + let currentFactions = factions.map(faction => { + return { + name: faction.factionName, + colour: faction.colour, + // all factions will be rendered on the map on default + active: true, + }; + }); + + // this block returns all score data from the game + let currentScore = []; + await Promise.all( + factions.map(async faction => { + let scores = await this.scoreRepository.find({ + where: { faction: faction }, + relations: ['faction'], + }); + currentScore.push( + scores.map(score => { + return { + score: score.score, + timestamp: score.scoreTimeStamp, + faction: score.faction['factionName'], + }; + }), + ); + }), + ); + + // this block returns all map drawings from the game + + let refs = await this.mapdrawingRepository.find({ + where: { gameId: gameId }, + select: ['mapDrawingId'], + }); + + let drawData = await Promise.all( + refs.map(async ref => { + return await this.mapHistoryRepository.find({ + where: { mapdrawing: ref.mapDrawingId }, + }); + }), + ); + + // this function returns all flagbox-events from the game + + let objectivepoints = await this.returnObjectivePointInfo(gameId); + + return { + location: gamelocation, + players: currentdata, + factions: currentFactions, + scores: currentScore, + drawings: drawData, + objectivepoints: objectivepoints, + }; + } + + // returns information about game's flagboxes and all of their events + async returnObjectivePointInfo(gameId) { + const info = await this.objectivepointRepository.find({ + where: { game: gameId }, + relations: ['history', 'history.owner', 'history.capture'], + }); + let response = await Promise.all( + info.map(async obj => { + return await { + objectivePointId: obj.objectivePointId, + objectivePointDescription: obj.objectivePointDescription, + objectivePointMultiplier: obj.objectivePointMultiplier, + data: obj.data, + history: await this.parseHistory(obj.history), + }; + }), + ); + return response; + } + + // loops all events in history array and returns an array of objects formatted for replay + private async parseHistory(history: ObjectivePoint_HistoryEntity[]) { + return await Promise.all( + history.map(async event => { + return { + timestamp: event.oP_HistoryTimestamp, + action: { + status: event.action, + message: { + 0: 'No capture ongoing', + 1: `Captured by ${ + event.owner ? event.owner.factionName : 'neutral' + }`, + 2: `Being captured by ${ + event.capture ? event.capture.factionName : 'neutral' + }`, + }[event.action], + }, + owner: await this.infoHelper(event.owner), + capture: await this.infoHelper(event.capture), + }; + }), + ); + } + + // small helper function that formats data for replay + private async infoHelper(obj) { + return (await obj) + ? { + factionName: obj.factionName, + colour: obj.colour, + } + : { + factionName: 'neutral', + colour: '#000000', + }; + } + // generate mockdata for a 3 day game + // create x amount of players + // assign them to two separate factions + // assign them to three separate groups + // insert x amount of locations for each players around some lat lng area + // insert x amont of flagbox events + // insert x amount of score ticks for score mockdata + // use the game's initial geojson to draw game area + + async mockdata(gameId) { + // initial settings for mockdata + // set the x amount of users to be created + const USER_AMOUNT = 100; + // set the x amount of locations to be created + const USER_LOCATIONS = 10; + // set the LAT and LNG for initial location + const LAT = 62.24147; + const LNG = 25.72088; + // set the x amount of flagbox events + // not used at the moment + const FLAGBOX_EVENTS = 8; + // set the score tick amount + // not used at the moment + const SCORE_TICKS = 10; + // setup the arrays for users, gamepersons and groups + const users = []; + const gamepersons = []; + const groups = []; + // characters for generating random usernames and the length of the username + const chars = + 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789'; + const N = 15; + // used to equally divide the players to factions + let f = 0; + + // get game info and factions associated with it + const game = await this.gameRepository.findOne({ + where: { id: gameId }, + relations: ['factions'], + }); + // get groups for all factions + await game.factions.forEach(async faction => { + groups.push(await this.factionService.showGroups(faction.factionId)); + }); + // get all objective point refs + const objectivepoints = await this.objectivepointRepository.find({ + game: gameId, + }); + // create x amount of users for the mock game with random username + for (let i = 0; i < USER_AMOUNT; i++) { + let res = await this.userService.register({ + name: Array(N) + .join() + .split(',') + .map(function() { + return chars.charAt(Math.floor(Math.random() * chars.length)); + }) + .join(''), + password: 'asd', + }); + // get user information by verifying the token returned by register + let user = await jwt.verify(res.token, process.env.SECRET); + // push the created user to users array + users.push(user); + + // reset index if it's out of range + if (f === game.factions.length) f = 0; + // join faction with the created user + let gameperson = await this.factionService.joinFaction(user['id'], { + factionId: game.factions[f].factionId, + factionPassword: game.factions[f].factionPassword, + game: gameId, + }); + // join a group randomly + await this.factionService.joinGroup(gameperson, { + groupId: groups[f][Math.floor(Math.random() * 3)], + }); + // push the gameperson ref to gamepersons array + gamepersons.push(gameperson); + f++; + } + // push x amount of user locations + for (let i = 1; i < USER_LOCATIONS + 1; i++) { + let date: number = Date.now(); + let x = 1; + // score ticks with players to sync them + await this.scoreService.scoreTick(gameId); + // flagbox events with players to sync them + // use helper function to generate a random event + let event = await this.createEvent(objectivepoints); + await this.gameService.flagboxEvent(gameId, event); + // add location entry for each gameperson + await Promise.all( + gamepersons.map(async gameperson => { + // format player locations with x-modifier that same factions are closer to each other + if (x > game.factions.length) x = 1; + let dataObject = { + lat: LAT + ((i + Math.random()) * 5) / 2000, + lng: LNG + x / 100 + (Math.random() * 5) / 2000, + time: date, + }; + x++; + // push the tracking data + await this.trackingService.trackLocation( + gameperson, + gameId, + dataObject, + ); + }), + ); + } + + return { + message: 'all done', + }; + } + + // creates randomized events for randomly chosen flagbox + // may result in impossible scenarios where owner is capturing their own flagbox + // use only for testing flagbox replay functionalities + private async createEvent(objectivepoints: ObjectivePointEntity[]) { + let point = + objectivepoints[Math.floor(Math.random() * objectivepoints.length)]; + return await { + node_id: point.objectivePointDescription, + owner: Math.round(Math.random()) + 1, + action: Math.round(Math.random()) ? 2 : 0, + capture: Math.round(Math.random()) + 1, + }; + } +} diff --git a/src/score/score.controller.ts b/src/score/score.controller.ts index 73fda22e699f5702b1616360bd7e95f0ac8aeb10..344c33c02a84f1e49b202090dffedb0d42d23ce8 100644 --- a/src/score/score.controller.ts +++ b/src/score/score.controller.ts @@ -1,4 +1,13 @@ -import { Controller, Post, UsePipes, Body, Param, Get } from '@nestjs/common'; +import { + Controller, + Post, + UsePipes, + Body, + Param, + Get, + UseInterceptors, + ClassSerializerInterceptor, +} from '@nestjs/common'; import { ValidationPipe } from '../shared/validation.pipe'; import { ScoreService } from './score.service'; @@ -20,16 +29,9 @@ export class ScoreController { return this.scoreService.addScore(data, gameId); } - // temporary scoreTick path, :id is gameId - @Get('tick-score/:id') - @GameStates('STARTED') - async scoreTick(@Param('id') gameId: GameEntity) { - return this.scoreService.scoreTick(gameId); - } - - // shows scores, :id is gameId + // shows scores, :id is gameId @Get('get-score/:id') - @GameStates('STARTED') + @UseInterceptors(ClassSerializerInterceptor) async getScores(@Param('id') gameId: GameEntity) { return this.scoreService.getScores(gameId); } diff --git a/src/score/score.dto.ts b/src/score/score.dto.ts index 99955f56eff34a41ce46111d7fd0cb9cd597386f..2feecbfd8aecd189d9fb1f8239abf2793fe6051b 100644 --- a/src/score/score.dto.ts +++ b/src/score/score.dto.ts @@ -3,7 +3,7 @@ import { IsNumber, Min, Max, IsUUID } from 'class-validator'; export class ScoreDTO { @IsNumber() @Min(1) - @Max(99) + @Max(999) score: number; @IsUUID('4') faction: string; diff --git a/src/score/score.entity.ts b/src/score/score.entity.ts index 6ca9ab77b968060a6dc88c860425119914cec030..01fb60bf11ad23461f9bba6492b6ebea68c61c45 100644 --- a/src/score/score.entity.ts +++ b/src/score/score.entity.ts @@ -1,19 +1,13 @@ -import { - Entity, - PrimaryGeneratedColumn, - Column, - ManyToOne, - Timestamp, - CreateDateColumn, -} from 'typeorm'; +import { Entity, PrimaryGeneratedColumn, Column, ManyToOne } from 'typeorm'; import { FactionEntity } from '../faction/faction.entity'; @Entity('Score') export class ScoreEntity { @PrimaryGeneratedColumn('uuid') scoreId: string; @Column({ type: 'float' }) score: number; - @CreateDateColumn({ type: 'timestamp' }) scoreTimeStamp: Timestamp; + @Column({ type: 'float' }) scoreTimeStamp: number; + // when Faction is deleted, it's Score is also deleted @ManyToOne(type => FactionEntity, factionName => factionName.factionId, { onDelete: 'CASCADE', }) diff --git a/src/score/score.module.ts b/src/score/score.module.ts index c04486fc676fca0d7c6a38c74374e03e23a1008c..9ce2f41c5c8ab0cbdbf4c11d1709f305323ba870 100644 --- a/src/score/score.module.ts +++ b/src/score/score.module.ts @@ -11,6 +11,11 @@ import { import { ScoreEntity } from './score.entity'; import { NotificationModule } from '../notifications/notifications.module'; +///////////////////////////////////////////////////////////////////// +/// Score /// +/// - contains everything to do with Score data. /// +///////////////////////////////////////////////////////////////////// + @Module({ imports: [ TypeOrmModule.forFeature([ diff --git a/src/score/score.service.ts b/src/score/score.service.ts index dba8e6e79206322a4dffae937b5af6b2bdd89041..ab6d522a805a00dc4456c5a64f550d30ff34fffd 100644 --- a/src/score/score.service.ts +++ b/src/score/score.service.ts @@ -26,30 +26,15 @@ export class ScoreService { private notificationGateway: NotificationGateway, ) {} - async addScore(scoreData: ScoreDTO, gameId: GameEntity) { - // check if faction exists - const faction = await this.factionRepository.findOne({ - factionId: scoreData.faction, - }); - if (!faction) { - throw new HttpException('Faction was not found', HttpStatus.BAD_REQUEST); - } - // get the previous score and add it, if it exists - let lastScore = await this.scoreRepository.findOne({ - where: { faction: scoreData.faction }, - order: { scoreTimeStamp: 'DESC' }, - }); - if (lastScore) { - scoreData.score += lastScore.score; - } - // add the score for Faction - const newScore = await this.scoreRepository.create(scoreData); - await this.scoreRepository.insert(newScore); + async addScore(scoreData: ScoreDTO, gameId) { + await this.pushScore(scoreData); + this.notificationGateway.server.emit(gameId, { type: 'score-update' }); return { message: 'Score updated!', }; } + // function to run on timer tick async scoreTick(gameId) { // get game's flagboxes const flagboxes = await this.flagRepository.find({ game: gameId }); @@ -69,7 +54,8 @@ export class ScoreService { i => i.faction === current.owner.factionId, ); index !== -1 - ? await (scoreData[index]['score'] += box.objectivePointMultiplier) + ? await (scoreData[index]['score'] += + box.objectivePointMultiplier * current.owner.multiplier) : await scoreData.push({ score: box.objectivePointMultiplier, faction: current.owner.factionId, @@ -78,7 +64,7 @@ export class ScoreService { }), ); scoreData.map(async data => { - await this.addScore(data, gameId); + await this.pushScore(data); }); this.notificationGateway.server.emit(gameId, { type: 'score-update' }); return { @@ -86,34 +72,48 @@ export class ScoreService { }; } + private async pushScore(scoreData: ScoreDTO) { + // check if faction exists + const faction = await this.factionRepository.findOne({ + factionId: scoreData.faction, + }); + if (!faction) { + throw new HttpException('Faction was not found', HttpStatus.BAD_REQUEST); + } + // get the previous score and add it, if it exists + let lastScore = await this.scoreRepository.findOne({ + where: { faction: scoreData.faction }, + order: { scoreTimeStamp: 'DESC' }, + }); + if (lastScore) { + scoreData.score += lastScore.score * faction.multiplier; + } + // add the score for Faction + const newScore = await this.scoreRepository.create(scoreData); + newScore.scoreTimeStamp = Date.now(); + await this.scoreRepository.insert(newScore); + } + async getScores(gameId: GameEntity) { // find games factions const factions = await this.factionRepository.find({ - where: {game: gameId,}, + where: { game: gameId }, relations: ['game'], }); let scores = []; await Promise.all( factions.map(async factionNow => { let score = await this.scoreRepository.findOne({ - where: {faction: factionNow}, + where: { faction: factionNow }, relations: ['faction'], - order: {scoreTimeStamp: 'DESC'}, + order: { scoreTimeStamp: 'DESC' }, }); //if score was found, put info to scores array if (score.faction) { - let index = await scores.findIndex( - i => i.faction === score.faction, - ); scores.push(score); } - })) - return scores; + }), + ); + return scores; } -} // - -// Hae kaikki Objective pointit -// aja map funktio pelin objective pointteihin -// jokaisella objective point ID:llä hae historystä -// relaatio, missä uusin timestamp -// katso uusimmista history entrystä omistaja +} diff --git a/src/shared/auth.guard.ts b/src/shared/auth.guard.ts index ff2bdf406d5badffc467cdf22d0f61935aae0af8..acde301a9dca4202d4971a3b34e6d251dc07a7c1 100644 --- a/src/shared/auth.guard.ts +++ b/src/shared/auth.guard.ts @@ -7,6 +7,14 @@ import { } from '@nestjs/common'; import * as jwt from 'jsonwebtoken'; +///////////////////////////////////////////////////////// +/// https://docs.nestjs.com/guards /// +/// AuthGuard verifies the user's token /// +/// It adds user information to request.user /// +/// which can be used by UserDecorator in services /// +/// return 403 if token validation fails /// +///////////////////////////////////////////////////////// + @Injectable() export class AuthGuard implements CanActivate { // check for logged in user diff --git a/src/shared/custom-validation.ts b/src/shared/custom-validation.ts deleted file mode 100644 index 13b96372e0b76a2897364baa15562edf383a2d81..0000000000000000000000000000000000000000 --- a/src/shared/custom-validation.ts +++ /dev/null @@ -1,52 +0,0 @@ -import { - ValidatorConstraint, - ValidatorConstraintInterface, - ValidationArguments, - Validator, -} from 'class-validator'; - -// check if input is null or valid uuid -@ValidatorConstraint({ name: 'uuid', async: true }) -export class Uuid implements ValidatorConstraintInterface { - validate(uuid: string, args: ValidationArguments) { - const validator = new Validator(); - return validator.isUUID(uuid, '4') || uuid == null; // for async validations you must return a Promise<boolean> here - } - - defaultMessage(args: ValidationArguments) { - return 'Not valid uuid'; - } -} - -// checks if role is valid -@ValidatorConstraint({ name: 'roleValidation', async: true }) -export class RoleValidation implements ValidatorConstraintInterface { - validate(role: string, args: ValidationArguments) { - const validRoles = ['admin', 'soldier', 'factionleader']; - return validRoles.includes(role); - } - - defaultMessage(args: ValidationArguments) { - return 'Not valid uuid'; - } -} - -// checks for valid JSON for center -@ValidatorConstraint({ name: 'centerJSON', async: true }) -export class CenterJSON implements ValidatorConstraintInterface { - validate(center: JSON, args: ValidationArguments) { - const validator = new Validator(); - return ( - validator.isNumber(center['lat']) && - validator.isNumber(center['lng']) && - validator.min(center['lat'], -90) && - validator.max(center['lat'], 90) && - validator.min(center['lng'], -180) && - validator.max(center['lng'], 180) - ); - } - - defaultMessage(args: ValidationArguments) { - return 'Error with center JSON'; - } -} diff --git a/src/shared/guard.decorator.ts b/src/shared/guard.decorator.ts index b299d6de913f54dd496c344f6dcdafd2942b2c9c..a313b3a6e52321b70b2f9768b5131252cb7816f7 100644 --- a/src/shared/guard.decorator.ts +++ b/src/shared/guard.decorator.ts @@ -1,5 +1,12 @@ import { SetMetadata } from '@nestjs/common'; +///////////////////////////////////////////////////////// +/// pass information from controllers to guards /// +/// for example @Roles("admin") passes it to /// +/// roles.guard, which compares user's role /// +/// to the values return by SetMetadata /// +///////////////////////////////////////////////////////// + export const Roles = (...roles: string[]) => SetMetadata('roles', roles); export const GameStates = (...states: string[]) => diff --git a/src/shared/http-error.filter.ts b/src/shared/http-error.filter.ts index 8a65068c4275c4a8d576da54b0340c17b6ac8cc4..bf7316fa20d8c5e3107a10b0972a1d1d9e18e940 100644 --- a/src/shared/http-error.filter.ts +++ b/src/shared/http-error.filter.ts @@ -7,6 +7,12 @@ import { HttpStatus, } from '@nestjs/common'; +///////////////////////////////////////////////////////// +/// Global tryCatch for catching errors in services /// +/// Returns error message for end-users /// +/// Also logs the error in console /// +///////////////////////////////////////////////////////// + @Catch() export class HttpErrorFilter implements ExceptionFilter { catch(exception: HttpException, host: ArgumentsHost) { diff --git a/src/shared/logging.interceptor.ts b/src/shared/logging.interceptor.ts index 55083faabd0a781ace687fc2cb9c2977f0befaa9..c29717474c87516cc3e1492bc24a5f13f848c89a 100644 --- a/src/shared/logging.interceptor.ts +++ b/src/shared/logging.interceptor.ts @@ -1,7 +1,13 @@ -import { Injectable, NestInterceptor, ExecutionContext, CallHandler, Logger } from '@nestjs/common'; +import { + Injectable, + NestInterceptor, + ExecutionContext, + CallHandler, + Logger, +} from '@nestjs/common'; import { Observable } from 'rxjs'; import { tap } from 'rxjs/operators'; - +/* @Injectable() export class LoggingInterceptor implements NestInterceptor { intercept( @@ -19,4 +25,4 @@ export class LoggingInterceptor implements NestInterceptor { )) ) } -} \ No newline at end of file +}*/ diff --git a/src/shared/roles.controller.ts b/src/shared/roles.controller.ts deleted file mode 100644 index 0d91e061c7fea732af225819e87f8529ff326eab..0000000000000000000000000000000000000000 --- a/src/shared/roles.controller.ts +++ /dev/null @@ -1,88 +0,0 @@ -const AccessControl = require('accesscontrol'); - -const grants = { - admin: { - mapmarker: { - 'create:any': [], - 'delete:any': [], - 'read:any': [], - 'update:any': [], - }, - powerup: { - 'create:any': [], - 'delete:any': [], - 'read:any': [], - 'update:any': [], - }, - faction: { - 'create:any': [], - 'delete:any': [], - 'read:any': [], - 'update:any': [], - }, - players: { - 'create:any': [], - 'delete:any': [], - 'read:any': [], - 'update:any': [], - }, - }, - faction_leader: { - mapmarker: { - 'create:own': [], - 'delete:own': [], - 'read:own': [], - }, - powerup: { - 'read:own': [], - }, - faction: { - 'read:own': [], - 'update:own': [], - }, - players: { - 'read:own': [], - 'update:own': [], - }, - }, - //player & spectator -}; - -const ac = new AccessControl(grants); - -/*const express = require ('express'); -const router express.Router; - -const ac = new AccessControl(); -ac.grant('faction_leader') // define new or modify existing role. also takes an array. - .createOwn('mapmarker') // equivalent to .createOwn('video', ['*']) - .deleteOwn('mapmarker') - .readOwn('mapmarker') - .grant('admin') // switch to another role without breaking the chain - .extend('user') // inherit role capabilities. also takes an array - .updateAny('mapmarker', ['title']) // explicitly defined attributes - .deleteAny('mapmarker') - .readAny('mapmarker'); - -//const -let permission = ac.can('user').createOwn('mapmarker'); -console.log(permission.granted); // —> true -console.log(permission.attributes); // —> ['*'] (all attributes) - -permission = ac.can('admin').updateAny('mapmarker'); -console.log(permission.granted); // —> true -console.log(permission.attributes); // —> ['title'] - -router.get('/videos/:title', function (req, res, next) { - const permission = ac.can(req.user.role).readAny('video'); - if (permission.granted) { - Video.find(req.params.title, function (err, data) { - if (err || !data) return res.status(404).end(); - // filter data by permission attributes and send. - res.json(permission.filter(data)); - }); - } else { - // resource is forbidden for this user/role - res.status(403).end(); - } -});*/ diff --git a/src/shared/roles.guard.ts b/src/shared/roles.guard.ts index 73693f9e1551da1b1b34056a665eb59c1211b3ef..5e97482e408df52075c4e97553707e32ca70a447 100644 --- a/src/shared/roles.guard.ts +++ b/src/shared/roles.guard.ts @@ -9,9 +9,19 @@ import { Reflector } from '@nestjs/core'; import * as jwt from 'jsonwebtoken'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; +import { Validator } from 'class-validator'; import { Game_PersonEntity } from '../game/game.entity'; +///////////////////////////////////////////////////////// +/// https://docs.nestjs.com/guards /// +/// RolesGuard verifies the user's token and role /// +/// It adds user information to request.user /// +/// which can be used by GamePerson /// +/// decorator in services /// +/// return 403 if token/role validation fails /// +///////////////////////////////////////////////////////// + @Injectable() export class RolesGuard implements CanActivate { constructor( @@ -32,11 +42,20 @@ export class RolesGuard implements CanActivate { return false; } const gameId = request.params.id; + // create a valifator + const validator = new Validator(); + // verify UUID + if (!validator.isUUID(gameId)) { + throw new HttpException('Game not found', HttpStatus.BAD_REQUEST); + } request.user = await this.getUserObject(request.headers.authorization); const role = await this.game_PersonRepository.findOne({ - person: request.user['id'], - game: gameId, + where: { person: request.user['id'], game: gameId }, + relations: ['faction', 'group'], }); + // add gameperson role to the request + // @GamePerson decorator can access it and pass it to service + request.gameperson = role; // check that the role matches the criteria and that token is valid for this game return role && roles.includes(role['role']); } diff --git a/src/shared/states.guard.ts b/src/shared/states.guard.ts index 9e85e94e40f92ed6b016e451c592a3025a684cad..544918d56e4653f6d52950a06ee8abd3e0eea4f1 100644 --- a/src/shared/states.guard.ts +++ b/src/shared/states.guard.ts @@ -8,9 +8,17 @@ import { import { Reflector } from '@nestjs/core'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; +import { Validator } from 'class-validator'; import { GameEntity } from '../game/game.entity'; +////////////////////////////////////////////////////////// +/// https://docs.nestjs.com/guards /// +/// StatesGuard verifies the game's state /// +/// Guard needs gameId as 'id' in request parameters /// +/// return 400 if state if state validation fails /// +////////////////////////////////////////////////////////// + @Injectable() export class StatesGuard implements CanActivate { constructor( @@ -29,6 +37,12 @@ export class StatesGuard implements CanActivate { } const request = context.switchToHttp().getRequest(); const gameId = request.params.id; + // create a valifator + const validator = new Validator(); + // verify UUID + if (!validator.isUUID(gameId)) { + throw new HttpException('Game not found', HttpStatus.BAD_REQUEST); + } const gameRef = await this.gameRepository.findOne({ id: gameId, }); diff --git a/src/shared/validation.pipe.ts b/src/shared/validation.pipe.ts index cff51f1b068b67886c29dd85587760a1fa016f1e..7e4bb8796f7b036c20238729b54f9c3d6093505c 100644 --- a/src/shared/validation.pipe.ts +++ b/src/shared/validation.pipe.ts @@ -7,7 +7,15 @@ import { } from '@nestjs/common'; import { validate } from 'class-validator'; import { plainToClass } from 'class-transformer'; -import { AdvancedConsoleLogger } from 'typeorm'; + +/////////////////////////////////////////////////////////// +/// https://docs.nestjs.com/techniques/validation /// +/// ValidationPipe for validating DTO's /// +/// DTO's use ClassValidator which are /// +/// validated by ValidationPipes /// +/// return 400 if pipe validation fails with /// +/// errorMessage stating reason for validation fail /// +/////////////////////////////////////////////////////////// @Injectable() export class ValidationPipe implements PipeTransform<any> { diff --git a/src/task/task.controller.ts b/src/task/task.controller.ts index 8283ba5bd2c4d7da4eb668ac6268d5165930e3f4..ba9f6cc9e9be2e6f8408c337ed270d0a587d2ba9 100644 --- a/src/task/task.controller.ts +++ b/src/task/task.controller.ts @@ -12,7 +12,7 @@ import { TaskService } from './task.service'; import { CreateTaskDTO, EditTaskDTO, DeleteTaskDTO } from './task.dto'; import { Roles } from '../shared/guard.decorator'; import { ValidationPipe } from '../shared/validation.pipe'; -import { User } from '../user/user.decorator'; +import { GamePerson } from 'src/game/gameperson.decorator'; @Controller('task') export class TaskController { @@ -42,14 +42,14 @@ export class TaskController { @Roles('admin') @UsePipes(new ValidationPipe()) async deleteTask(@Param('id') id: string, @Body() data: DeleteTaskDTO) { - return this.taskService.deleteTask(data); + return this.taskService.deleteTask(data, id); } // lists all the tasks for the game if the user has game_person entry // :id is the id of the game @Get('get-tasks/:id') @Roles('soldier', 'factionleader', 'admin') - async fetchTasks(@User('id') person, @Param('id') id: string) { - return this.taskService.fetchTasks(person, id); + async fetchTasks(@GamePerson() gameperson, @Param('id') id: string) { + return this.taskService.fetchTasks(gameperson, id); } } diff --git a/src/task/task.dto.ts b/src/task/task.dto.ts index a1576e7d12c9d9a6263b445fdba293fe94e8964a..02653c39d5762784215303956652aaec78c60f81 100644 --- a/src/task/task.dto.ts +++ b/src/task/task.dto.ts @@ -5,10 +5,10 @@ import { Validate, IsUUID, Equals, + IsOptional, } from 'class-validator'; import { FactionEntity } from '../faction/faction.entity'; import { GameEntity } from '../game/game.entity'; -import { Uuid } from '../shared/custom-validation'; export class CreateTaskDTO { @IsString() @@ -19,7 +19,8 @@ export class CreateTaskDTO { taskDescription: string; @IsBoolean() taskIsActive: boolean; - @Validate(Uuid) + @IsOptional() + @IsUUID('4') faction: FactionEntity; @Equals(null) taskWinner: FactionEntity; diff --git a/src/task/task.entity.ts b/src/task/task.entity.ts index f1c1cf103fa4338a7f10d0305888d13c097d7a09..762dbc038992f394b1bd773b9fd4e2965fe9a145 100644 --- a/src/task/task.entity.ts +++ b/src/task/task.entity.ts @@ -16,6 +16,7 @@ export class TaskEntity { @Column({ type: 'text' }) taskDescription: string; @Column({ type: 'bool' }) taskIsActive: boolean; + // when a Faction or Game is deleted, affiliated Tasks are also deleted @ManyToOne(type => FactionEntity, faction => faction.factionId, { onDelete: 'CASCADE', }) diff --git a/src/task/task.module.ts b/src/task/task.module.ts index 66d5ac5fce93d8447317343283a84b5002755ab6..9b009fc9480aab4e6708a9d8a986447393b76634 100644 --- a/src/task/task.module.ts +++ b/src/task/task.module.ts @@ -5,12 +5,16 @@ import { TaskService } from './task.service'; import { TaskController } from './task.controller'; import { TaskEntity } from './task.entity'; import { FactionEntity } from '../faction/faction.entity'; -import { Game_PersonEntity } from '../game/game.entity'; import { NotificationModule } from '../notifications/notifications.module'; +///////////////////////////////////////////////////////////////////// +/// Task /// +/// - contains everything to do with Task data. /// +///////////////////////////////////////////////////////////////////// + @Module({ imports: [ - TypeOrmModule.forFeature([TaskEntity, FactionEntity, Game_PersonEntity]), + TypeOrmModule.forFeature([TaskEntity, FactionEntity]), NotificationModule, ], controllers: [TaskController], diff --git a/src/task/task.service.ts b/src/task/task.service.ts index d79338e0d21e9ce741c67d4bf94b89ba64e4c9c9..0461d310de44f882ccdf18c4b2ba29f01cb5d8a7 100644 --- a/src/task/task.service.ts +++ b/src/task/task.service.ts @@ -5,7 +5,6 @@ import { InjectRepository } from '@nestjs/typeorm'; import { TaskEntity } from './task.entity'; import { CreateTaskDTO, EditTaskDTO, DeleteTaskDTO } from './task.dto'; import { FactionEntity } from '../faction/faction.entity'; -import { Game_PersonEntity } from '../game/game.entity'; import { NotificationGateway } from '../notifications/notifications.gateway'; @Injectable() @@ -15,8 +14,6 @@ export class TaskService { private taskRepository: Repository<TaskEntity>, @InjectRepository(FactionEntity) private factionRepository: Repository<FactionEntity>, - @InjectRepository(Game_PersonEntity) - private gamePersonRepository: Repository<Game_PersonEntity>, private notificationGateway: NotificationGateway, ) {} @@ -35,10 +32,7 @@ export class TaskService { await this.taskRepository.insert(createdTask); // notify subscribers about a new task // if faction was set it notifies only faction members, else everyone - this.notificationGateway.server.emit( - task.faction != null ? task.faction.toString() : task.taskGame.toString(), - { type: 'task-update' }, - ); + this.triggerTasks(task, task.taskGame); return { message: 'Task added', }; @@ -62,15 +56,17 @@ export class TaskService { task.taskWinner = data.taskWinner; task.taskIsActive = false; await this.taskRepository.save(task); + this.triggerTasks(task, data.taskGame); return { message: 'Task updated and closed', }; } - async deleteTask(data: DeleteTaskDTO) { + async deleteTask(data: DeleteTaskDTO, gameId) { const task = await this.taskRepository.findOne({ taskId: data.taskId }); if (task) { await this.taskRepository.delete({ taskId: task.taskId }); + this.triggerTasks(task, gameId); return { message: 'Task deleted', }; @@ -78,15 +74,17 @@ export class TaskService { throw new HttpException('Task not found', HttpStatus.BAD_REQUEST); } - async fetchTasks(user, taskGame) { - const gamePerson = await this.gamePersonRepository.findOne({ - where: { - person: user, - game: taskGame, - }, - relations: ['faction'], + private triggerTasks(task, gameId) { + this.notificationGateway.server.emit(gameId, { + type: 'task-update', + message: task.faction != null ? task.faction.toString() : '', }); - if (gamePerson.role == 'admin') { + } + + // finds all tasks if admin and if non-admin player it finds the playres own + // tasks and general tasks + async fetchTasks(gameperson, taskGame) { + if (gameperson.role == 'admin') { return await this.taskRepository.find({ where: { taskGame: taskGame }, relations: ['faction', 'taskWinner'], @@ -97,7 +95,7 @@ export class TaskService { where: [ { taskGame: taskGame, - faction: gamePerson.faction.factionId, + faction: gameperson.faction.factionId, }, { taskGame: taskGame, diff --git a/src/tick/tick.module.ts b/src/tick/tick.module.ts new file mode 100644 index 0000000000000000000000000000000000000000..b07fa5fa1eca4c4357d6e551983edf1b5c3aebe6 --- /dev/null +++ b/src/tick/tick.module.ts @@ -0,0 +1,32 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { + GameEntity, + ObjectivePointEntity, + ObjectivePoint_HistoryEntity, +} from '../game/game.entity'; +import { TickService } from './tick.service'; +import { ScoreService } from '../score/score.service'; +import { FactionEntity } from '../faction/faction.entity'; +import { ScoreEntity } from '../score/score.entity'; +import { NotificationModule } from '../notifications/notifications.module'; + +@Module({ + imports: [ + TypeOrmModule.forFeature([ + GameEntity, + ScoreEntity, + ObjectivePointEntity, + ObjectivePoint_HistoryEntity, + FactionEntity, + ]), + NotificationModule, + ], + providers: [TickService, ScoreService], +}) +export class TickModule { + constructor(private tickService: TickService) { + this.tickService.startTimer(); + } +} diff --git a/src/tick/tick.service.ts b/src/tick/tick.service.ts new file mode 100644 index 0000000000000000000000000000000000000000..3a5f7cc1dac03303eaf0febcba0cd603fdeaf45a --- /dev/null +++ b/src/tick/tick.service.ts @@ -0,0 +1,50 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { ScoreService } from '../score/score.service'; +import { InjectRepository } from '@nestjs/typeorm'; +import { GameEntity } from 'src/game/game.entity'; +import { Repository } from 'typeorm'; + +@Injectable() +export class TickService { + constructor( + private scoreService: ScoreService, + @InjectRepository(GameEntity) + private gameRepository: Repository<GameEntity>, + ) { + // whenever Tickservice is called, it will start ticktimer + /* + WARNING: multiple calls start multiple timers, + if you need to use this service somewhere else remember to call startTimer method from somewhere else + */ + } + + private logger: Logger = new Logger('TickLogger'); + // tickinterval in milliseconds (10 minutes = 600 000) + private readonly tickInterval: number = 60000; + + // initializing timer + async startTimer() { + this.logger.log('Started timer'); + setInterval(this.Tick, this.tickInterval); + } + + // returns name and id of each game + async listGames() { + const games = await this.gameRepository.find({ + where: { state: 'STARTED' }, + }); + return games.map(game => { + return game.gameObject(); + }); + } + + // tick score for games with STARTED-status + Tick = async _ => { + this.logger.log('Ticking'); + let games = await this.listGames(); + games.map(game => { + this.logger.log(game); + this.scoreService.scoreTick(game.id); + }); + }; +} diff --git a/src/tracking/geo.dto.ts b/src/tracking/geo.dto.ts index 352faf704db72912a548734185d277253d67c789..681e1b6d1f5c4cc30d66e113b5d1fb0dbf653dd9 100644 --- a/src/tracking/geo.dto.ts +++ b/src/tracking/geo.dto.ts @@ -1,5 +1,7 @@ import { IsNumber, Min, Max, Allow } from 'class-validator'; +// latitude and longitude accepts degrees from worldmap + export class GeoDTO { @IsNumber() @Min(-90) diff --git a/src/tracking/tracking.controller.ts b/src/tracking/tracking.controller.ts index ca0e79ee5195503f61d827dbc170474e66fd1bad..a98c12bfb399cc797c9dd642de2663ae5d0a5d76 100644 --- a/src/tracking/tracking.controller.ts +++ b/src/tracking/tracking.controller.ts @@ -2,18 +2,19 @@ import { Controller, Post, Param, - UseGuards, UsePipes, Body, Get, + UseInterceptors, + ClassSerializerInterceptor, } from '@nestjs/common'; import { TrackingService } from './tracking.service'; -import { TrackingDTO } from './tracking.dto'; import { User } from '../user/user.decorator'; import { Roles, GameStates } from '../shared/guard.decorator'; import { ValidationPipe } from '../shared/validation.pipe'; import { GeoDTO } from './geo.dto'; +import { GamePerson } from '../game/gameperson.decorator'; @Controller('tracking') export class TrackingController { @@ -26,17 +27,19 @@ export class TrackingController { @GameStates('STARTED') @UsePipes(new ValidationPipe()) async trackLocation( - @User('id') userId, + @GamePerson() gameperson, @Param('id') id, @Body() trackdata: GeoDTO, ) { - return this.trackingservice.trackLocation(userId, id, trackdata); + return this.trackingservice.trackLocation(gameperson, id, trackdata); } + // finds certain player's location + // :id is the id of the game @Get('players/:id') @Roles('admin', 'factionleader') @GameStates('STARTED', 'PAUSED') - async getPlayerLocations(@User('id') userId, @Param('id') gameId) { - return this.trackingservice.getPlayers(userId, gameId); + async getPlayerLocations(@GamePerson() gameperson, @Param('id') gameId) { + return this.trackingservice.getPlayers(gameperson, gameId); } } diff --git a/src/tracking/tracking.dto.ts b/src/tracking/tracking.dto.ts index 81cc71597fa93910cb195b1fd4e30f38c62c1f05..181430ba1f631fda4202d9d5115247a27c82a99c 100644 --- a/src/tracking/tracking.dto.ts +++ b/src/tracking/tracking.dto.ts @@ -1,5 +1,4 @@ -import { Game_PersonEntity } from '../game/game.entity'; -import { Allow, ValidateNested } from 'class-validator'; +import { ValidateNested } from 'class-validator'; import { GeoDTO } from './geo.dto'; import { Type } from 'class-transformer'; diff --git a/src/tracking/tracking.entity.ts b/src/tracking/tracking.entity.ts index c2699a09388ce8ad47c7119b46f4614fa52b34ec..dc398ac8d9398aa475de128fd2bc6bfb87074013 100644 --- a/src/tracking/tracking.entity.ts +++ b/src/tracking/tracking.entity.ts @@ -1,13 +1,15 @@ import { Entity, PrimaryGeneratedColumn, Column, ManyToOne } from 'typeorm'; import { Game_PersonEntity, GameEntity } from '../game/game.entity'; -import { FactionEntity } from 'src/faction/faction.entity'; +import { FactionEntity } from '../faction/faction.entity'; import { GeoDTO } from './geo.dto'; @Entity('Tracking') export class TrackingEntity { @PrimaryGeneratedColumn('uuid') id: string; @Column({ type: 'json', nullable: true }) data: GeoDTO[]; + @Column('text') icon: string; + // when the GamePerson is deleted it's tracking data is also deleted @ManyToOne(type => Game_PersonEntity, person => person.gamepersonId, { onDelete: 'CASCADE', }) diff --git a/src/tracking/tracking.module.ts b/src/tracking/tracking.module.ts index 45b167fc526ab41addbb87575518c9acae884482..de4afda4d1450da298902942709df1f039adc024 100644 --- a/src/tracking/tracking.module.ts +++ b/src/tracking/tracking.module.ts @@ -5,9 +5,23 @@ import { TrackingController } from './tracking.controller'; import { TrackingService } from './tracking.service'; import { TrackingEntity } from './tracking.entity'; import { Game_PersonEntity } from '../game/game.entity'; +import { PersonEntity } from '../user/user.entity'; +import { FactionEntity } from '../faction/faction.entity'; + +///////////////////////////////////////////////////////////////////// +/// Tracking /// +/// - contains everything to do with Tracking data. /// +///////////////////////////////////////////////////////////////////// @Module({ - imports: [TypeOrmModule.forFeature([TrackingEntity, Game_PersonEntity])], + imports: [ + TypeOrmModule.forFeature([ + TrackingEntity, + Game_PersonEntity, + PersonEntity, + FactionEntity, + ]), + ], controllers: [TrackingController], providers: [TrackingService], }) diff --git a/src/tracking/tracking.service.ts b/src/tracking/tracking.service.ts index 9941256dde52240d7c90b45d61b7b32ef4e1c242..175c3eaaeb548507dc103c484b0f908edba9f1e1 100644 --- a/src/tracking/tracking.service.ts +++ b/src/tracking/tracking.service.ts @@ -1,13 +1,11 @@ -import { Injectable, HttpStatus, HttpException } from '@nestjs/common'; +import { Injectable, HttpException, HttpStatus } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; import { Game_PersonEntity } from '../game/game.entity'; import { TrackingEntity } from './tracking.entity'; -import { TrackingDTO } from './tracking.dto'; -import { FactionEntity } from '../faction/faction.entity'; -import { NotificationGateway } from 'src/notifications/notifications.gateway'; import { GeoDTO } from './geo.dto'; +import { FactionEntity } from 'src/faction/faction.entity'; @Injectable() export class TrackingService { @@ -16,87 +14,111 @@ export class TrackingService { private trackingrepository: Repository<TrackingEntity>, @InjectRepository(Game_PersonEntity) private gamepersonrepository: Repository<Game_PersonEntity>, + @InjectRepository(FactionEntity) + private factionRepository: Repository<FactionEntity>, ) {} - async trackLocation(personId, gameId, trackdata: GeoDTO) { - // find player - let gameperson = await this.gamepersonrepository.findOne({ - where: { game: gameId, person: personId }, - relations: ['faction'], - }); - if (!gameperson) { - throw new HttpException( - 'You have not joined this game', - HttpStatus.BAD_REQUEST, - ); - } + private icons = { + infantry: 'infantry.svg', + recon: 'recon.svg', + mechanized: 'mechanized.svg', + }; + + async trackLocation( + gameperson: Game_PersonEntity, + gameId, + trackdata: GeoDTO, + ) { + // find ref to gameperson's tracking data let trackedperson = await this.trackingrepository.findOne({ gamepersonId: gameperson, }); - // if player has pushed tracking data, update entry if (trackedperson) { trackdata['time'] = Date.now(); //add coordinates trackedperson.data.push(trackdata); //add timestamp await this.trackingrepository.save(trackedperson); - return { code: 201, message: 'Location updated!' }; + return { message: 'Location updated!' }; } else { - // first entry will be empty trackdata['time'] = Date.now(); // initialize data trackedperson = await this.trackingrepository.create(trackedperson); + // if group exists, add icon based on that, else add default icon + trackedperson.icon = gameperson.group + ? this.icons[gameperson.group.class] + : 'infantry.svg'; trackedperson.data = [trackdata]; trackedperson.faction = gameperson.faction; trackedperson.game = gameId; trackedperson.gamepersonId = gameperson; await this.trackingrepository.save(trackedperson); - return { code: 201, message: 'Entry Created!' }; + return { message: 'Entry Created!' }; } } // get player data while game is running - async getPlayers(userId, gameId) { - // get gameperson - const gameperson = await this.gamepersonrepository.findOne({ - where: { person: userId, game: gameId }, - relations: ['faction'], - }); - - let playerdata; + async getPlayers(gameperson, gameId) { + let playerdata = []; // get playerdata if (gameperson.faction) { - playerdata = await this.trackingrepository.find({ - where: { faction: gameperson.faction }, - relations: ['faction', 'gamepersonId'], - }); + // create an array of the response as frontend maps the response + // to create different clusters for factions + playerdata.push( + await this.trackingrepository.find({ + where: { faction: gameperson.faction }, + relations: ['faction', 'gamepersonId', 'gamepersonId.person'], + }), + ); } else { - playerdata = await this.trackingrepository.find({ - where: { game: gameId }, - relations: ['faction', 'gamepersonId'], - }); + let factions = await this.factionRepository.find({ game: gameId }); + playerdata = await Promise.all( + factions.map(async faction => { + let rawdata = await this.trackingrepository.find({ + where: { faction: faction.factionId }, + relations: ['faction', 'gamepersonId', 'gamepersonId.person'], + }); + let groups = { + 'infantry.svg': [], + 'recon.svg': [], + 'mechanized.svg': [], + }; + rawdata.forEach(async player => { + groups[player.icon].push(player); + }); + return groups; + }), + ); } - // parse data + // create an array for each faction + // inside faction create an array for each icon type + // insisde icon arrays include all players with same icon const currentdata = await Promise.all( - playerdata.map(async player => { - return { - gamepersonId: player['gamepersonId']['gamepersonId'], - gamepersonRole: player['gamepersonId']['role'], - factionId: player['faction']['factionId'], - coordinates: player['data'].pop(), - }; + await playerdata.map(async faction => { + let data = []; + for (let group in faction) { + data.push( + await Promise.all( + faction[group].map(async player => { + return await { + username: player['gamepersonId']['person']['name'], + gamepersonId: player['gamepersonId']['gamepersonId'], + gamepersonRole: player['gamepersonId']['role'], + factionId: player['faction']['factionId'], + factionColour: player['faction']['colour'], + icon: player['icon'], + coordinates: player['data'].pop(), + }; + }), + ), + ); + } + return data; }), ); - return currentdata; } - - private async mapFunction(data): Promise<Number> { - return await data.map(type => { - return type; - }); - } } diff --git a/src/user/user.controller.ts b/src/user/user.controller.ts index 4d4158d3f037850c8c5230975ef5f46ae5a75ede..8e7635767424c627f8372d8b312cc2b7229e673a 100644 --- a/src/user/user.controller.ts +++ b/src/user/user.controller.ts @@ -12,6 +12,16 @@ import { UserDTO } from './user.dto'; import { AuthGuard } from '../shared/auth.guard'; import { ValidationPipe } from '../shared/validation.pipe'; +///////////////////////////////////////////////////////////////////// +/// UserController is being used for routing: /// +/// - Login /// +/// - Register /// +/// /// +/// - Verify is checking for logged in user /// +/// /// +/// See shared folder files for more information on decorators. /// +///////////////////////////////////////////////////////////////////// + @Controller('user') export class UserController { constructor(private userService: UserService) {} diff --git a/src/user/user.decorator.ts b/src/user/user.decorator.ts index 2c499d904927d0502567fc2131270b81170327f4..4a6ec4788ceb7698d57d6c7986dd98d11e1dae31 100644 --- a/src/user/user.decorator.ts +++ b/src/user/user.decorator.ts @@ -1,6 +1,14 @@ -import { createParamDecorator } from "@nestjs/common"; +import { createParamDecorator } from '@nestjs/common'; + +/////////////////////////////////////////////////////////////////////////////// +/// UserDecorator /// +/// /// +/// See auth.guard.ts in shared folder for more information /// +/// /// +/// Returns user id and user name, this is mainly used to return user id /// +/////////////////////////////////////////////////////////////////////////////// // used to pass user information to controllers export const User = createParamDecorator((data, req) => { - return data ? req.user[data] : req.user; -}) \ No newline at end of file + return data ? req.user[data] : req.user; +}); diff --git a/src/user/user.dto.ts b/src/user/user.dto.ts index c5bfa70cc3551645e1773e9aa7c4fb898b3f9887..243dae1faa53acc3c9a6a051db32b1a20b1728d3 100644 --- a/src/user/user.dto.ts +++ b/src/user/user.dto.ts @@ -1,10 +1,18 @@ import { IsString, IsNotEmpty, Length } from 'class-validator'; +/////////////////////////////////////////////////////////// +/// Contains Validation for UserDTO /// +/// uses class-validator built in validations /// +/// see https://github.com/typestack/class-validator /// +/////////////////////////////////////////////////////////// + export class UserDTO { - // uses class-validator built in validations - // see https://github.com/typestack/class-validator - @IsString() @IsNotEmpty() @Length(3, 31) - name: string; - @IsString() @IsNotEmpty() @Length(3, 255) - password: string; -} \ No newline at end of file + @IsString() + @IsNotEmpty() + @Length(3, 31) + name: string; + @IsString() + @IsNotEmpty() + @Length(3, 255) + password: string; +} diff --git a/src/user/user.entity.ts b/src/user/user.entity.ts index 716020ae49d675641863a1711e424b85dbd67608..1c4c8d9eb7a38f6fb1f8003a6c5cd7a02d44d339 100644 --- a/src/user/user.entity.ts +++ b/src/user/user.entity.ts @@ -11,6 +11,15 @@ import * as jwt from 'jsonwebtoken'; import { Game_PersonEntity } from '../game/game.entity'; import { Exclude } from 'class-transformer'; +///////////////////////////////////////////////////////////////////////////// +/// UserEntity reflects database table. /// +/// /// +/// Before handling password to database we encrypt it with bcrypt. /// +/// password field will be excluded unless we call repository relation. /// +/// /// +/// token will be created when user registers or logs in to the system. /// +///////////////////////////////////////////////////////////////////////////// + @Entity('Person') export class PersonEntity { @PrimaryGeneratedColumn('uuid') id: string; @@ -39,6 +48,7 @@ export class PersonEntity { return await bcrypt.compareSync(attempt, this.password); } + // creates token from id and name, it will be created through jsonwebtoken and .env SECRET field private get token() { const { id, name } = this; return jwt.sign( diff --git a/src/user/user.module.ts b/src/user/user.module.ts index 6cb77a7b735df532d710febc33a4df53f264563a..85ec0ffd05dff0bd401da8d2807ed81ddbafdd57 100644 --- a/src/user/user.module.ts +++ b/src/user/user.module.ts @@ -3,11 +3,21 @@ import { TypeOrmModule } from '@nestjs/typeorm'; import { UserController } from './user.controller'; import { UserService } from './user.service'; -import { PersonEntity} from './user.entity'; -import { GameEntity } from '../game/game.entity'; +import { PersonEntity } from './user.entity'; + +/////////////////////////// +/// Entities /// +/// - PersonEntity /// +/// /// +/// Controllers /// +/// - UserController /// +/// /// +/// Provider /// +/// - UserService /// +/////////////////////////// @Module({ - imports: [TypeOrmModule.forFeature([PersonEntity, GameEntity])], + imports: [TypeOrmModule.forFeature([PersonEntity])], controllers: [UserController], providers: [UserService], }) diff --git a/src/user/user.service.ts b/src/user/user.service.ts index aa511e27802548a17449e7912144598e7eda3732..5e05634bb81e51d45d3241f3cb4d1fdfc26ee1b9 100644 --- a/src/user/user.service.ts +++ b/src/user/user.service.ts @@ -4,16 +4,23 @@ import { InjectRepository } from '@nestjs/typeorm'; import { PersonEntity } from './user.entity'; import { UserDTO } from './user.dto'; -import { GameEntity } from '../game/game.entity'; -import { GameDTO } from '../game/game.dto'; + +/////////////////////////////////////////////////////////// +/// UserService contains functions for /// +/// - Login /// +/// - Register /// +/// /// +/// Both functions return logged in users tokenObject /// +/// See more info in UserEntity. /// +/// /// +/// See more info on DTO in it's respective file /// +/////////////////////////////////////////////////////////// @Injectable() export class UserService { constructor( @InjectRepository(PersonEntity) private userRepository: Repository<PersonEntity>, - @InjectRepository(GameEntity) - private gameRepository: Repository<GameEntity>, ) {} async register(data: UserDTO) { @@ -38,25 +45,4 @@ export class UserService { } return user.tokenObject(); } - - // liitytään peliin - async joinGame(game: GameDTO, data: UserDTO) { - try { - // etsi peli valinnan mukaan ja otetaan käyttäjän rooli kyseisestä pelistä talteen - const role = await this.gameRepository.findOne(); - return role; - } catch (error) { - return error.message; - } - } - - // liitytään factionii - async joinFaction(game: GameDTO, data: UserDTO): Promise<boolean> { - try { - // tarkistetaan factionin salasana - return true; - } catch (error) { - return error; - } - } }