diff --git a/src/score/score.controller.ts b/src/score/score.controller.ts new file mode 100644 index 0000000000000000000000000000000000000000..554e0617591574ce5c34b3dff8b5fdbaacab4ca4 --- /dev/null +++ b/src/score/score.controller.ts @@ -0,0 +1,29 @@ +import { Controller, Post, UsePipes, Body, Param, Get } from '@nestjs/common'; + +import { ValidationPipe } from '../shared/validation.pipe'; +import { ScoreService } from './score.service'; +import { ScoreDTO } from './score.dto'; +import { GameEntity } from '../game/game.entity'; +import { Roles, GameStates } from '../shared/guard.decorator'; + +@Controller('score') +export class ScoreController { + constructor(private scoreService: ScoreService) {} + + // adds score manually to Faction + // :id is gameId + @Post('add-score/:id') + @Roles('admin') + @GameStates('STARTED') + @UsePipes(new ValidationPipe()) + async addingScore(@Body() data: ScoreDTO, @Param('id') gameId: GameEntity) { + 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); + } +} diff --git a/src/score/score.dto.ts b/src/score/score.dto.ts new file mode 100644 index 0000000000000000000000000000000000000000..99955f56eff34a41ce46111d7fd0cb9cd597386f --- /dev/null +++ b/src/score/score.dto.ts @@ -0,0 +1,10 @@ +import { IsNumber, Min, Max, IsUUID } from 'class-validator'; + +export class ScoreDTO { + @IsNumber() + @Min(1) + @Max(99) + score: number; + @IsUUID('4') + faction: string; +} diff --git a/src/score/score.module.ts b/src/score/score.module.ts new file mode 100644 index 0000000000000000000000000000000000000000..7f7601d70521cc0b957e41c9e0ecf2c631777f77 --- /dev/null +++ b/src/score/score.module.ts @@ -0,0 +1,25 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { ScoreController } from './score.controller'; +import { ScoreService } from './score.service'; +import { FactionEntity } from '../faction/faction.entity'; +import { + ObjectivePointEntity, + ObjectivePoint_HistoryEntity, +} from '../game/game.entity'; +import { ScoreEntity } from './score.entity'; + +@Module({ + imports: [ + TypeOrmModule.forFeature([ + ScoreEntity, + ObjectivePointEntity, + ObjectivePoint_HistoryEntity, + FactionEntity, + ]), + ], + controllers: [ScoreController], + providers: [ScoreService], +}) +export class ScoreModule {} diff --git a/src/score/score.service.ts b/src/score/score.service.ts new file mode 100644 index 0000000000000000000000000000000000000000..0ecea91032c73a6745e0ea0bd6dd207a510e31b0 --- /dev/null +++ b/src/score/score.service.ts @@ -0,0 +1,94 @@ +import { Injectable, HttpException, HttpStatus } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; +import { Repository } from 'typeorm'; + +import { FactionEntity } from '../faction/faction.entity'; +import { ScoreDTO } from './score.dto'; +import { + ObjectivePoint_HistoryEntity, + ObjectivePointEntity, + GameEntity, +} from '../game/game.entity'; +import { ScoreEntity } from './score.entity'; +import { map } from 'rxjs/operators'; + +@Injectable() +export class ScoreService { + constructor( + @InjectRepository(ScoreEntity) + private scoreRepository: Repository<ScoreEntity>, + @InjectRepository(ObjectivePointEntity) + private flagRepository: Repository<ObjectivePointEntity>, + @InjectRepository(ObjectivePoint_HistoryEntity) + private flagHistoryRepository: Repository<ObjectivePoint_HistoryEntity>, + @InjectRepository(FactionEntity) + private factionRepository: Repository<FactionEntity>, + ) {} + + 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); + return { + code: 201, + message: 'Score updated!', + }; + } + + async scoreTick(gameId) { + // get game's flagboxes + const flagboxes = await this.flagRepository.find({ game: gameId }); + // create an array of DTOs for adding score + let scoreData = []; + await Promise.all( + flagboxes.map(async box => { + // get the newest entry in history + let current = await this.flagHistoryRepository.findOne({ + where: { objective_point: box.objectivePointId }, + relations: ['owner'], + order: { oP_HistoryTimestamp: 'DESC' }, + }); + // if result was found, add score to the owner + if (current.owner) { + let index = await scoreData.findIndex( + i => i.faction === current.owner.factionId, + ); + index !== -1 + ? await (scoreData[index]['score'] += box.objectivePointMultiplier) + : await scoreData.push({ + score: box.objectivePointMultiplier, + faction: current.owner.factionId, + }); + } + }), + ); + scoreData.map(async data => { + await this.addScore(data, gameId); + }); + return { + code: 200, + message: 'Scores added', + }; + } +} // + +// 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