From 0c60bb3351f326656944943bc433a86b94544df9 Mon Sep 17 00:00:00 2001 From: "Dario G. Mori" Date: Tue, 16 Apr 2024 13:17:50 +0200 Subject: [PATCH 1/5] feat: get current game when /play is called --- .../lab/en2b/quizapi/game/GameController.java | 15 ++++++++++++++ .../lab/en2b/quizapi/game/GameService.java | 20 ++++++++++++++++++- 2 files changed, 34 insertions(+), 1 deletion(-) diff --git a/api/src/main/java/lab/en2b/quizapi/game/GameController.java b/api/src/main/java/lab/en2b/quizapi/game/GameController.java index 08af1201..b4b2f89e 100644 --- a/api/src/main/java/lab/en2b/quizapi/game/GameController.java +++ b/api/src/main/java/lab/en2b/quizapi/game/GameController.java @@ -39,6 +39,21 @@ public ResponseEntity newGame(@RequestParam(required = false) S throw new IllegalArgumentException("Custom game mode requires a body"); return ResponseEntity.ok(gameService.newGame(lang,gamemode,customGameDto,authentication)); } + @Operation(summary = "Gets the current game", description = "Requests the API to get the current game") + @ApiResponses(value = { + @ApiResponse(responseCode = "200", description = "Successfully retrieved"), + @ApiResponse(responseCode = "403", description = "You are not logged in", content = @io.swagger.v3.oas.annotations.media.Content), + @ApiResponse(responseCode = "403", description = "You are not logged in", content = @io.swagger.v3.oas.annotations.media.Content), + }) + @Parameters({ + @Parameter(name = "lang", description = "The language of the game", example = "en"), + @Parameter(name = "gamemode", description = "The gamemode of the game", example = "KIWI_QUEST") + }) + @io.swagger.v3.oas.annotations.parameters.RequestBody(description = "The custom game dto, only required if the gamemode is CUSTOM") + @GetMapping("/play") + public ResponseEntity getGame(Authentication authentication){ + return ResponseEntity.ok(gameService.getGame(authentication)); + } @Operation(summary = "Starts a new round", description = "Starts the round (asks a question and its possible answers to the API and start the timer) for a given authentication (a player)") @ApiResponses(value = { diff --git a/api/src/main/java/lab/en2b/quizapi/game/GameService.java b/api/src/main/java/lab/en2b/quizapi/game/GameService.java index 5d33dbfa..d1c61101 100644 --- a/api/src/main/java/lab/en2b/quizapi/game/GameService.java +++ b/api/src/main/java/lab/en2b/quizapi/game/GameService.java @@ -39,7 +39,7 @@ public class GameService { @Transactional public GameResponseDto newGame(String lang, GameMode gamemode, CustomGameDto newGameDto, Authentication authentication) { // Check if there is an active game for the user - Optional game = gameRepository.findActiveGameForUser(userService.getUserByAuthentication(authentication).getId()); + Optional game = getCurrentGameForAuth(authentication); if (game.isPresent() && !wasGameMeantToBeOver(game.get())){ // If there is an active game and it should not be over, return it return gameResponseDtoMapper.apply(game.get()); @@ -179,4 +179,22 @@ public List getQuestionGameModes() { //new GameModeDto("Custom","Don't like our gamemodes? That's fine! (I only feel a bit offended)",GameMode.CUSTOM,"FaCog") ); } + + /** + * Gets the game + * @param authentication the authentication of the user + * @return the game that is currently active + */ + public GameResponseDto getGame(Authentication authentication) { + return gameResponseDtoMapper.apply(getCurrentGameForAuth(authentication).orElseThrow()); + } + + /** + * Gets the current game for the user + * @param authentication the authentication of the user + * @return the current game + */ + private Optional getCurrentGameForAuth(Authentication authentication){ + return gameRepository.findActiveGameForUser(userService.getUserByAuthentication(authentication).getId()); + } } From db6dad9bf5e647033d624547fc63e855aa11e3c6 Mon Sep 17 00:00:00 2001 From: "Dario G. Mori" Date: Tue, 16 Apr 2024 13:37:24 +0200 Subject: [PATCH 2/5] test: get current game when /play is called --- .../lab/en2b/quizapi/game/GameServiceTest.java | 16 ++++++++++++++++ 1 file changed, 16 insertions(+) diff --git a/api/src/test/java/lab/en2b/quizapi/game/GameServiceTest.java b/api/src/test/java/lab/en2b/quizapi/game/GameServiceTest.java index b9c4b9d4..ff9a948c 100644 --- a/api/src/test/java/lab/en2b/quizapi/game/GameServiceTest.java +++ b/api/src/test/java/lab/en2b/quizapi/game/GameServiceTest.java @@ -217,6 +217,22 @@ public void newGameCustomGame(){ assertEquals(defaultGameResponseDto, gameDto); } + // GET GAME + @Test + public void getGame(){ + Authentication authentication = mock(Authentication.class); + when(userService.getUserByAuthentication(authentication)).thenReturn(defaultUser); + when(gameRepository.findActiveGameForUser(any())).thenReturn(Optional.of(defaultGame)); + GameResponseDto gameDto = gameService.getGame(authentication); + gameDto.setId(null); + assertEquals(defaultGameResponseDto, gameDto); + } + @Test + public void getGameNotActive(){ + Authentication authentication = mock(Authentication.class); + when(userService.getUserByAuthentication(authentication)).thenReturn(defaultUser); + assertThrows(NoSuchElementException.class, () -> gameService.getGame(authentication)); + } // START ROUND TESTS @Test From 6f8cbc9bff42459b6980ccd6a5ed5b67e3920cf5 Mon Sep 17 00:00:00 2001 From: "Dario G. Mori" Date: Tue, 16 Apr 2024 13:40:04 +0200 Subject: [PATCH 3/5] feat: is game active endpoint --- .../lab/en2b/quizapi/game/GameController.java | 17 +++++++++++------ .../java/lab/en2b/quizapi/game/GameService.java | 9 +++++++++ .../game/dtos/GameActiveResponseDto.java | 16 ++++++++++++++++ 3 files changed, 36 insertions(+), 6 deletions(-) create mode 100644 api/src/main/java/lab/en2b/quizapi/game/dtos/GameActiveResponseDto.java diff --git a/api/src/main/java/lab/en2b/quizapi/game/GameController.java b/api/src/main/java/lab/en2b/quizapi/game/GameController.java index b4b2f89e..3192b857 100644 --- a/api/src/main/java/lab/en2b/quizapi/game/GameController.java +++ b/api/src/main/java/lab/en2b/quizapi/game/GameController.java @@ -43,18 +43,23 @@ public ResponseEntity newGame(@RequestParam(required = false) S @ApiResponses(value = { @ApiResponse(responseCode = "200", description = "Successfully retrieved"), @ApiResponse(responseCode = "403", description = "You are not logged in", content = @io.swagger.v3.oas.annotations.media.Content), - @ApiResponse(responseCode = "403", description = "You are not logged in", content = @io.swagger.v3.oas.annotations.media.Content), - }) - @Parameters({ - @Parameter(name = "lang", description = "The language of the game", example = "en"), - @Parameter(name = "gamemode", description = "The gamemode of the game", example = "KIWI_QUEST") + @ApiResponse(responseCode = "404", description = "No active game", content = @io.swagger.v3.oas.annotations.media.Content), }) - @io.swagger.v3.oas.annotations.parameters.RequestBody(description = "The custom game dto, only required if the gamemode is CUSTOM") @GetMapping("/play") public ResponseEntity getGame(Authentication authentication){ return ResponseEntity.ok(gameService.getGame(authentication)); } + @Operation(summary = "Checks if there is an active game", description = "Requests the API to check if there exists an active game for a given authentication (a player)") + @ApiResponses(value = { + @ApiResponse(responseCode = "200", description = "Successfully retrieved"), + @ApiResponse(responseCode = "403", description = "You are not logged in", content = @io.swagger.v3.oas.annotations.media.Content), + }) + @GetMapping("/is-active") + public ResponseEntity isActive(Authentication authentication){ + return ResponseEntity.ok(gameService.isActive(authentication)); + } + @Operation(summary = "Starts a new round", description = "Starts the round (asks a question and its possible answers to the API and start the timer) for a given authentication (a player)") @ApiResponses(value = { @ApiResponse(responseCode = "200", description = "Successfully retrieved"), diff --git a/api/src/main/java/lab/en2b/quizapi/game/GameService.java b/api/src/main/java/lab/en2b/quizapi/game/GameService.java index d1c61101..d75940ab 100644 --- a/api/src/main/java/lab/en2b/quizapi/game/GameService.java +++ b/api/src/main/java/lab/en2b/quizapi/game/GameService.java @@ -197,4 +197,13 @@ public GameResponseDto getGame(Authentication authentication) { private Optional getCurrentGameForAuth(Authentication authentication){ return gameRepository.findActiveGameForUser(userService.getUserByAuthentication(authentication).getId()); } + + /** + * Checks if the game is active + * @param authentication the authentication of the user + * @return the response of the check + */ + public GameActiveResponseDto isActive(Authentication authentication) { + return new GameActiveResponseDto(getCurrentGameForAuth(authentication).isPresent()); + } } diff --git a/api/src/main/java/lab/en2b/quizapi/game/dtos/GameActiveResponseDto.java b/api/src/main/java/lab/en2b/quizapi/game/dtos/GameActiveResponseDto.java new file mode 100644 index 00000000..e68219e8 --- /dev/null +++ b/api/src/main/java/lab/en2b/quizapi/game/dtos/GameActiveResponseDto.java @@ -0,0 +1,16 @@ +package lab.en2b.quizapi.game.dtos; + +import com.fasterxml.jackson.annotation.JsonProperty; +import io.swagger.v3.oas.annotations.media.Schema; +import lombok.AllArgsConstructor; +import lombok.Data; +import lombok.NoArgsConstructor; + +@AllArgsConstructor +@Data +@NoArgsConstructor +public class GameActiveResponseDto { + @JsonProperty("is_active") + @Schema(description = "Whether the game is active or not",example = "true") + private boolean isActive; +} From 8f1434483ff8a7122d880001e07dcb584d0079c5 Mon Sep 17 00:00:00 2001 From: "Dario G. Mori" Date: Tue, 16 Apr 2024 13:57:01 +0200 Subject: [PATCH 4/5] feat: get active game and is active --- .../en2b/quizapi/game/GameControllerTest.java | 36 ++++++++++++++++++- .../en2b/quizapi/game/GameServiceTest.java | 18 ++++++++++ 2 files changed, 53 insertions(+), 1 deletion(-) diff --git a/api/src/test/java/lab/en2b/quizapi/game/GameControllerTest.java b/api/src/test/java/lab/en2b/quizapi/game/GameControllerTest.java index d570e709..92a970c9 100644 --- a/api/src/test/java/lab/en2b/quizapi/game/GameControllerTest.java +++ b/api/src/test/java/lab/en2b/quizapi/game/GameControllerTest.java @@ -203,7 +203,7 @@ void getQuestionCategoriesShouldReturn403() throws Exception{ } @Test - void getGameModeshouldReturn200() throws Exception{ + void getGameModeShouldReturn200() throws Exception{ mockMvc.perform(get("/games/gamemodes") .with(user("test").roles("user")) .contentType("application/json") @@ -219,5 +219,39 @@ void getGameModesShouldReturn403() throws Exception{ .andExpect(status().isForbidden()); } + @Test + void getGameShouldReturn200() throws Exception{ + mockMvc.perform(get("/games/play") + .with(user("test").roles("user")) + .contentType("application/json") + .with(csrf())) + .andExpect(status().isOk()); + } + + @Test + void getGameShouldReturn403() throws Exception{ + mockMvc.perform(get("/games/play") + .contentType("application/json") + .with(csrf())) + .andExpect(status().isForbidden()); + } + + @Test + void getGameIsActiveShouldReturn200() throws Exception{ + mockMvc.perform(get("/games/is-active") + .with(user("test").roles("user")) + .contentType("application/json") + .with(csrf())) + .andExpect(status().isOk()); + } + + @Test + void getGameIsActiveShouldReturn403() throws Exception{ + mockMvc.perform(get("/games/is-active") + .contentType("application/json") + .with(csrf())) + .andExpect(status().isForbidden()); + } + } diff --git a/api/src/test/java/lab/en2b/quizapi/game/GameServiceTest.java b/api/src/test/java/lab/en2b/quizapi/game/GameServiceTest.java index ff9a948c..39f39365 100644 --- a/api/src/test/java/lab/en2b/quizapi/game/GameServiceTest.java +++ b/api/src/test/java/lab/en2b/quizapi/game/GameServiceTest.java @@ -234,6 +234,19 @@ public void getGameNotActive(){ assertThrows(NoSuchElementException.class, () -> gameService.getGame(authentication)); } + // IS GAME ACTIVE TESTS + @Test + public void isGameActive(){ + when(userService.getUserByAuthentication(authentication)).thenReturn(defaultUser); + when(gameRepository.findActiveGameForUser(1L)).thenReturn(Optional.of(defaultGame)); + assertTrue(gameService.isActive(authentication).isActive()); + } + @Test + public void isGameActiveNoActiveGame(){ + when(userService.getUserByAuthentication(authentication)).thenReturn(defaultUser); + assertFalse(gameService.isActive(authentication).isActive()); + } + // START ROUND TESTS @Test public void startRound(){ @@ -449,4 +462,9 @@ public void testGetQuestionCategories(){ assertEquals(Arrays.asList(QuestionCategory.values()), gameService.getQuestionCategories()); } + @Test + public void testGetGameModes(){ + assertFalse(gameService.getQuestionGameModes().isEmpty()); + } + } From 745c50a78903efa01f3169fd4c72280fafbd7551 Mon Sep 17 00:00:00 2001 From: "Dario G. Mori" Date: Tue, 16 Apr 2024 19:00:13 +0200 Subject: [PATCH 5/5] test: getCurrentQuestion no roundstarttime --- .../test/java/lab/en2b/quizapi/game/GameServiceTest.java | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/api/src/test/java/lab/en2b/quizapi/game/GameServiceTest.java b/api/src/test/java/lab/en2b/quizapi/game/GameServiceTest.java index 39f39365..e5d2d045 100644 --- a/api/src/test/java/lab/en2b/quizapi/game/GameServiceTest.java +++ b/api/src/test/java/lab/en2b/quizapi/game/GameServiceTest.java @@ -292,6 +292,14 @@ public void getCurrentQuestion() { assertEquals(defaultQuestionResponseDto, questionDto); } + @Test + public void getCurrentQuestionRoundTimeNull() { + defaultGame.setRoundStartTime(null); + when(gameRepository.findByIdForUser(any(), any())).thenReturn(Optional.of(defaultGame)); + when(userService.getUserByAuthentication(authentication)).thenReturn(defaultUser); + assertThrows(IllegalStateException.class, () -> gameService.getCurrentQuestion(1L,authentication)); + } + @Test public void getCurrentQuestionRoundNotStarted() { when(gameRepository.findByIdForUser(any(), any())).thenReturn(Optional.of(defaultGame));