From 21cc436a834508858138528ab518c732913a0ba5 Mon Sep 17 00:00:00 2001 From: Rackochad <33836535+Rackover@users.noreply.github.com> Date: Mon, 23 Sep 2019 11:54:32 +0530 Subject: [PATCH 1/5] Updated clan model --- src/main/java/com/faforever/api/data/domain/Clan.java | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/src/main/java/com/faforever/api/data/domain/Clan.java b/src/main/java/com/faforever/api/data/domain/Clan.java index eb58b749d..b979316b8 100644 --- a/src/main/java/com/faforever/api/data/domain/Clan.java +++ b/src/main/java/com/faforever/api/data/domain/Clan.java @@ -46,6 +46,7 @@ public class Clan extends AbstractEntity implements OwnableEntity { private String description; private String tagColor; private String websiteUrl; + private boolean isOpen; private List memberships; @Column(name = "name") @@ -107,4 +108,11 @@ public String getWebsiteUrl() { public Login getEntityOwner() { return getLeader(); } + + @Column(name = "isOpen") + @NotNull + @UpdatePermission(expression = IsEntityOwner.EXPRESSION) + public bool getIsOpen() { + return isOpen; + } } From eb583cf99d3039a25370f7ce3e7805428f16682d Mon Sep 17 00:00:00 2001 From: Rackochad <33836535+Rackover@users.noreply.github.com> Date: Mon, 23 Sep 2019 12:19:45 +0530 Subject: [PATCH 2/5] Allow creation of generic clan links if the clan is open Disable player and expiration checks if the clan is marked as "open" --- .../java/com/faforever/api/clan/ClanService.java | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/src/main/java/com/faforever/api/clan/ClanService.java b/src/main/java/com/faforever/api/clan/ClanService.java index ba7d54a73..ec2faeeb8 100644 --- a/src/main/java/com/faforever/api/clan/ClanService.java +++ b/src/main/java/com/faforever/api/clan/ClanService.java @@ -96,9 +96,10 @@ String generatePlayerInvitationToken(Player requester, int newMemberId, int clan Player newMember = playerRepository.findById(newMemberId) .orElseThrow(() -> new ApiException(new Error(ErrorCode.CLAN_GENERATE_LINK_PLAYER_NOT_FOUND, newMemberId))); - long expire = Instant.now() - .plus(fafApiProperties.getClan().getInviteLinkExpireDurationMinutes(), ChronoUnit.MINUTES) - .toEpochMilli(); + long expire = clan.getIsOpen() ? 0 : + Instant.now() + .plus(fafApiProperties.getClan().getInviteLinkExpireDurationMinutes(), ChronoUnit.MINUTES) + .toEpochMilli(); InvitationResult result = new InvitationResult(expire, ClanResult.of(clan), @@ -120,11 +121,12 @@ void acceptPlayerInvitationToken(String stringToken, Authentication authenticati Clan clan = clanRepository.findById(clanId) .orElseThrow(() -> new ApiException(new Error(ErrorCode.CLAN_NOT_EXISTS, clanId))); - Player newMember = playerRepository.findById(invitation.getNewMember().getId()) - .orElseThrow(() -> new ProgrammingError("ClanMember does not exist: " + invitation.getNewMember().getId())); + Player newMember = clan.getIsOpen() ? null : + playerRepository.findById(invitation.getNewMember().getId()) + .orElseThrow(() -> new ProgrammingError("ClanMember does not exist: " + invitation.getNewMember().getId())); - if (player.getId() != newMember.getId()) { + if (!invitation.getClan().getIsOpen() && player.getId() != newMember.getId()) { throw new ApiException(new Error(ErrorCode.CLAN_ACCEPT_WRONG_PLAYER)); } if (newMember.getClan() != null) { @@ -133,7 +135,7 @@ void acceptPlayerInvitationToken(String stringToken, Authentication authenticati ClanMembership membership = new ClanMembership(); membership.setClan(clan); - membership.setPlayer(newMember); + membership.setPlayer(player); clanMembershipRepository.save(membership); } } From 0fbfc923802503fefe73f0e0bcea42c49c2eaa88 Mon Sep 17 00:00:00 2001 From: Rackochad <33836535+Rackover@users.noreply.github.com> Date: Mon, 23 Sep 2019 12:28:06 +0530 Subject: [PATCH 3/5] No new member as part of the clan invite is the clan is open --- src/main/java/com/faforever/api/clan/ClanService.java | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/src/main/java/com/faforever/api/clan/ClanService.java b/src/main/java/com/faforever/api/clan/ClanService.java index ec2faeeb8..8bcb8f8bc 100644 --- a/src/main/java/com/faforever/api/clan/ClanService.java +++ b/src/main/java/com/faforever/api/clan/ClanService.java @@ -93,8 +93,9 @@ String generatePlayerInvitationToken(Player requester, int newMemberId, int clan throw new ApiException(new Error(ErrorCode.CLAN_NOT_LEADER, clanId)); } - Player newMember = playerRepository.findById(newMemberId) - .orElseThrow(() -> new ApiException(new Error(ErrorCode.CLAN_GENERATE_LINK_PLAYER_NOT_FOUND, newMemberId))); + Player newMember = clan.getIsOpen() ? null : + playerRepository.findById(newMemberId) + .orElseThrow(() -> new ApiException(new Error(ErrorCode.CLAN_GENERATE_LINK_PLAYER_NOT_FOUND, newMemberId))); long expire = clan.getIsOpen() ? 0 : Instant.now() From e7ee8c2a631a723a8d2771f679850db6ae0a278e Mon Sep 17 00:00:00 2001 From: Rackochad <33836535+Rackover@users.noreply.github.com> Date: Mon, 23 Sep 2019 12:29:19 +0530 Subject: [PATCH 4/5] Added overloaded function for playerless invitation --- src/main/java/com/faforever/api/clan/ClanService.java | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/src/main/java/com/faforever/api/clan/ClanService.java b/src/main/java/com/faforever/api/clan/ClanService.java index 8bcb8f8bc..bfb135ec6 100644 --- a/src/main/java/com/faforever/api/clan/ClanService.java +++ b/src/main/java/com/faforever/api/clan/ClanService.java @@ -83,6 +83,12 @@ Clan create(String name, String tag, String description, Player creator) { clanRepository.save(clan); return clan; } + + @SneakyThrows + String generatePlayerInvitationToken(Player requester, int clanId) { + // Playerless invitation + return generatePlayerInvitationToken(requester, 0, int clanId); + } @SneakyThrows String generatePlayerInvitationToken(Player requester, int newMemberId, int clanId) { From 0860fc5f8f322e2cb53873e7a69c4a1966dc5e0b Mon Sep 17 00:00:00 2001 From: Rackochad <33836535+Rackover@users.noreply.github.com> Date: Mon, 23 Sep 2019 12:34:19 +0530 Subject: [PATCH 5/5] Added frantic comments all over the test script --- .../faforever/api/clan/ClanServiceTest.java | 50 +++++++++++++++++++ 1 file changed, 50 insertions(+) diff --git a/src/test/java/com/faforever/api/clan/ClanServiceTest.java b/src/test/java/com/faforever/api/clan/ClanServiceTest.java index 10fbee2b0..b1adb6674 100644 --- a/src/test/java/com/faforever/api/clan/ClanServiceTest.java +++ b/src/test/java/com/faforever/api/clan/ClanServiceTest.java @@ -144,6 +144,7 @@ public void createClanSuccessful() { assertEquals(creator, clanCaptor.getValue().getFounder()); assertEquals(1, clanCaptor.getValue().getMemberships().size()); assertEquals(creator, clanCaptor.getValue().getMemberships().get(0).getPlayer()); + assertEquals(false, clanCaptor.getValue().getIsOpen()); } @Test @@ -231,6 +232,35 @@ public void generatePlayerInvitationToken() throws IOException { assertEquals(clan.getName(), captor.getValue().getClan().getName()); } + @Test + public void generatePlayerOpenInvitationToken() throws IOException { + Player requester = new Player(); + requester.setId(1); + + Clan clan = ClanFactory.builder().leader(requester).build().setIsOpen(false); + + FafApiProperties props = new FafApiProperties(); + + when(clanRepository.findById(clan.getId())).thenReturn(Optional.of(clan)); + when(fafApiProperties.getClan()).thenReturn(props.getClan()); + + instance.generatePlayerInvitationToken(requester, clan.getId()); + + // What do i do with this?? + // Invitation result should be modified to accept null players but i have no IDE here so navigating is a pain + /* + ArgumentCaptor captor = ArgumentCaptor.forClass(InvitationResult.class); + verify(jwtService, Mockito.times(1)).sign(captor.capture()); + assertEquals(captor.getValue().getExpire(), 0); + assertEquals(null, captor.getValue().getNewMember()); + */ + assertEquals(clan.getId(), getValue().getClan().getId()); + /* + assertEquals(clan.getTag(), captor.getValue().getClan().getTag()); + assertEquals(clan.getName(), captor.getValue().getClan().getName()); + */ + } + @Test public void acceptPlayerInvitationTokenExpire() throws IOException { String stringToken = "1234"; @@ -249,6 +279,26 @@ public void acceptPlayerInvitationTokenExpire() throws IOException { } verify(clanMembershipRepository, Mockito.never()).save(any(ClanMembership.class)); } + + @Test + public void acceptPlayerOpenInvitationCannotExpire() throws IOException { + // Why is the clan passed nowhere here?? How do I make it not expire if the clan is open since no clan is passed as part of the test? + String stringToken = "1234"; + long expire = System.currentTimeMillis(); + Jwt jwtToken = Mockito.mock(Jwt.class); + + when(jwtToken.getClaims()).thenReturn( + String.format("{\"expire\":%s}", expire)); + when(jwtService.decodeAndVerify(any())).thenReturn(jwtToken); + + try { + instance.acceptPlayerInvitationToken(stringToken, null); + fail(); + } catch (ApiException e) { + assertThat(e, hasErrorCode(ErrorCode.CLAN_ACCEPT_TOKEN_EXPIRE)); + } + verify(clanMembershipRepository, Mockito.never()).save(any(ClanMembership.class)); + } @Test public void acceptPlayerInvitationTokenInvalidClan() throws IOException {