Fixes bugs with top ten and duplications

Fixes https://github.com/BentoBoxWorld/Level/issues/161
This commit is contained in:
tastybento 2020-07-02 18:19:26 -07:00
parent baf0ba0974
commit f0e983db44
2 changed files with 74 additions and 27 deletions

View File

@ -21,6 +21,7 @@ import org.bukkit.World;
import org.eclipse.jdt.annotation.NonNull; import org.eclipse.jdt.annotation.NonNull;
import org.eclipse.jdt.annotation.Nullable; import org.eclipse.jdt.annotation.Nullable;
import world.bentobox.bentobox.BentoBox;
import world.bentobox.bentobox.api.events.addon.AddonEvent; import world.bentobox.bentobox.api.events.addon.AddonEvent;
import world.bentobox.bentobox.api.panels.PanelItem; import world.bentobox.bentobox.api.panels.PanelItem;
import world.bentobox.bentobox.api.panels.builders.PanelBuilder; import world.bentobox.bentobox.api.panels.builders.PanelBuilder;
@ -79,14 +80,24 @@ public class LevelsManager {
background = new PanelItemBuilder().icon(Material.BLACK_STAINED_GLASS_PANE).name(" ").build(); background = new PanelItemBuilder().icon(Material.BLACK_STAINED_GLASS_PANE).name(" ").build();
} }
/**
* Add a score to the top players list
* @param world - world
* @param targetPlayer - target player
* @param lv - island level
*/
private void addToTopTen(@NonNull World world, @NonNull UUID targetPlayer, long lv) { private void addToTopTen(@NonNull World world, @NonNull UUID targetPlayer, long lv) {
topTenLists.computeIfAbsent(world, k -> new TopTenData(world)); // Get top ten
if (hasTopTenPerm(world, targetPlayer)) { Map<UUID, Long> topTen = topTenLists.computeIfAbsent(world, k -> new TopTenData(world)).getTopTen();
topTenLists.get(world).getTopTen().put(targetPlayer, lv); // Remove this player from the top list no matter what (we'll put them back later if required)
} else { topTen.remove(targetPlayer);
topTenLists.get(world).getTopTen().remove(targetPlayer);
}
// Get the island
Island island = addon.getIslands().getIsland(world, targetPlayer);
if (island != null && island.getOwner() != null && hasTopTenPerm(world, island.getOwner())) {
// Insert the owner into the top ten
topTen.put(island.getOwner(), lv);
}
} }
/** /**
@ -110,8 +121,8 @@ public class LevelsManager {
result.complete(null); result.complete(null);
} }
// Save result // Save result
island.getMemberSet().forEach(uuid -> setIslandLevel(island.getWorld(), uuid, r.getLevel())); setIslandLevel(island.getWorld(), island.getOwner(), r.getLevel());
addToTopTen(island.getWorld(), island.getOwner(), r.getLevel()); addon.getManager().saveTopTen(island.getWorld());
result.complete(r); result.complete(r);
}); });
return result; return result;
@ -182,7 +193,7 @@ public class LevelsManager {
} }
// Show remaining slots // Show remaining slots
for (; i < SLOTS.length; i++) { for (; i < SLOTS.length; i++) {
panel.item(SLOTS[i], new PanelItemBuilder().icon(Material.GREEN_STAINED_GLASS_PANE).name("").build()); panel.item(SLOTS[i], new PanelItemBuilder().icon(Material.GREEN_STAINED_GLASS_PANE).name(String.valueOf(i + 1)).build());
} }
// Add yourself // Add yourself
@ -232,14 +243,17 @@ public class LevelsManager {
} }
/** /**
* Get level from cache for a player. * Get level of island from cache for a player.
* @param world - world where the island is * @param world - world where the island is
* @param targetPlayer - target player UUID * @param targetPlayer - target player UUID
* @return Level of player or zero if player is unknown or UUID is null * @return Level of the player's island or zero if player is unknown or UUID is null
*/ */
public long getIslandLevel(@NonNull World world, @Nullable UUID targetPlayer) { public long getIslandLevel(@NonNull World world, @Nullable UUID targetPlayer) {
if (targetPlayer == null) return 0L; if (targetPlayer == null) return 0L;
LevelsData ld = getLevelsData(targetPlayer); // Get the island owner
UUID owner = addon.getIslands().getOwner(world, targetPlayer);
if (owner == null) return 0L;
LevelsData ld = getLevelsData(owner);
return ld == null ? 0L : ld.getLevel(world); return ld == null ? 0L : ld.getLevel(world);
} }
@ -254,23 +268,23 @@ public class LevelsManager {
} }
/** /**
* Load a player from the cache or database * Load a level data for the island owner from the cache or database. Only island onwers are stored.
* @param targetPlayer - UUID of target player * @param islandOwner - UUID of island owner
* @return LevelsData object or null if not found * @return LevelsData object or null if not found
*/ */
@Nullable @Nullable
public LevelsData getLevelsData(@NonNull UUID targetPlayer) { public LevelsData getLevelsData(@NonNull UUID islandOwner) {
// Get from database if not in cache // Get from database if not in cache
if (!levelsCache.containsKey(targetPlayer) && handler.objectExists(targetPlayer.toString())) { if (!levelsCache.containsKey(islandOwner) && handler.objectExists(islandOwner.toString())) {
LevelsData ld = handler.loadObject(targetPlayer.toString()); LevelsData ld = handler.loadObject(islandOwner.toString());
if (ld != null) { if (ld != null) {
levelsCache.put(targetPlayer, ld); levelsCache.put(islandOwner, ld);
} else { } else {
handler.deleteID(targetPlayer.toString()); handler.deleteID(islandOwner.toString());
} }
} }
// Return cached value or null // Return cached value or null
return levelsCache.get(targetPlayer); return levelsCache.get(islandOwner);
} }
/** /**
@ -281,7 +295,9 @@ public class LevelsManager {
*/ */
public String getPointsToNextString(@NonNull World world, @Nullable UUID targetPlayer) { public String getPointsToNextString(@NonNull World world, @Nullable UUID targetPlayer) {
if (targetPlayer == null) return ""; if (targetPlayer == null) return "";
LevelsData ld = getLevelsData(targetPlayer); UUID owner = addon.getIslands().getOwner(world, targetPlayer);
if (owner == null) return "";
LevelsData ld = getLevelsData(owner);
return ld == null ? "" : String.valueOf(ld.getPointsToNextLevel(world)); return ld == null ? "" : String.valueOf(ld.getPointsToNextLevel(world));
} }
@ -298,6 +314,7 @@ public class LevelsManager {
topTenLists.get(world).getTopTen().keySet().removeIf(u -> !hasTopTenPerm(world, u)); topTenLists.get(world).getTopTen().keySet().removeIf(u -> !hasTopTenPerm(world, u));
// Return the sorted map // Return the sorted map
return Collections.unmodifiableMap(topTenLists.get(world).getTopTen().entrySet().stream() return Collections.unmodifiableMap(topTenLists.get(world).getTopTen().entrySet().stream()
.filter(e -> addon.getIslands().isOwner(world, e.getKey()))
.filter(l -> l.getValue() > 0) .filter(l -> l.getValue() > 0)
.sorted(Collections.reverseOrder(Map.Entry.comparingByValue())).limit(size) .sorted(Collections.reverseOrder(Map.Entry.comparingByValue())).limit(size)
.collect(Collectors.toMap( .collect(Collectors.toMap(
@ -308,11 +325,11 @@ public class LevelsManager {
* Checks if player has the correct top ten perm to have their level saved * Checks if player has the correct top ten perm to have their level saved
* @param world * @param world
* @param targetPlayer * @param targetPlayer
* @return true if player has the perm * @return true if player has the perm or the player is offline
*/ */
boolean hasTopTenPerm(@NonNull World world, @NonNull UUID targetPlayer) { boolean hasTopTenPerm(@NonNull World world, @NonNull UUID targetPlayer) {
String permPrefix = addon.getPlugin().getIWM().getPermissionPrefix(world); String permPrefix = addon.getPlugin().getIWM().getPermissionPrefix(world);
return Bukkit.getPlayer(targetPlayer) != null && Bukkit.getPlayer(targetPlayer).hasPermission(permPrefix + INTOPTEN); return Bukkit.getPlayer(targetPlayer) == null || Bukkit.getPlayer(targetPlayer).hasPermission(permPrefix + INTOPTEN);
} }
/** /**
@ -326,6 +343,8 @@ public class LevelsManager {
topTenLists.put(world, tt); topTenLists.put(world, tt);
addon.log("Loaded TopTen for " + world.getName()); addon.log("Loaded TopTen for " + world.getName());
// Update based on user data // Update based on user data
// Remove any non island owners
tt.getTopTen().keySet().removeIf(u -> !addon.getIslands().isOwner(world, u));
for (UUID uuid : tt.getTopTen().keySet()) { for (UUID uuid : tt.getTopTen().keySet()) {
tt.getTopTen().compute(uuid, (k,v) -> v = updateLevel(k, world)); tt.getTopTen().compute(uuid, (k,v) -> v = updateLevel(k, world));
} }
@ -361,6 +380,14 @@ public class LevelsManager {
topTenLists.values().forEach(topTenHandler::saveObjectAsync); topTenLists.values().forEach(topTenHandler::saveObjectAsync);
} }
/**
* Save the top ten for world
* @param world - world
*/
public void saveTopTen(World world) {
topTenHandler.saveObjectAsync(topTenLists.get(world));
}
/** /**
* Set an initial island level for player * Set an initial island level for player
* @param island - the island to set. Must have a non-null world and owner * @param island - the island to set. Must have a non-null world and owner
@ -372,11 +399,17 @@ public class LevelsManager {
handler.saveObjectAsync(levelsCache.get(island.getOwner())); handler.saveObjectAsync(levelsCache.get(island.getOwner()));
} }
/**
* Set the island level for the owner of the island that targetPlayer is a member
* @param world - world
* @param targetPlayer - player, may be a team member
* @param lv - level
*/
public void setIslandLevel(@NonNull World world, @NonNull UUID targetPlayer, long lv) { public void setIslandLevel(@NonNull World world, @NonNull UUID targetPlayer, long lv) {
levelsCache.computeIfAbsent(targetPlayer, LevelsData::new).setLevel(world, lv); levelsCache.computeIfAbsent(targetPlayer, LevelsData::new).setLevel(world, lv);
handler.saveObjectAsync(levelsCache.get(targetPlayer)); handler.saveObjectAsync(levelsCache.get(targetPlayer));
// Add to Top Ten // Update TopTen
addToTopTen(world, targetPlayer, lv); addToTopTen(world, targetPlayer, levelsCache.get(targetPlayer).getLevel(world));
} }
private Long updateLevel(UUID uuid, World world) { private Long updateLevel(UUID uuid, World world) {

View File

@ -154,7 +154,9 @@ public class LevelsManagerTest {
when(island.getMemberSet()).thenReturn(iset); when(island.getMemberSet()).thenReturn(iset);
when(island.getOwner()).thenReturn(uuid); when(island.getOwner()).thenReturn(uuid);
when(island.getWorld()).thenReturn(world); when(island.getWorld()).thenReturn(world);
// Default to uuid's being island owners
when(im.isOwner(eq(world), any())).thenReturn(true);
when(im.getOwner(any(), any(UUID.class))).thenAnswer(in -> in.getArgument(1, UUID.class));
// Player // Player
when(player.getUniqueId()).thenReturn(uuid); when(player.getUniqueId()).thenReturn(uuid);
@ -335,6 +337,17 @@ public class LevelsManagerTest {
assertEquals(1, lm.getTopTen(world, 1).size()); assertEquals(1, lm.getTopTen(world, 1).size());
} }
/**
* Test method for {@link world.bentobox.level.LevelsManager#getTopTen(org.bukkit.World, int)}.
*/
@Test
public void testGetTopTenNoOwners() {
when(im.isOwner(eq(world), any())).thenReturn(false);
testLoadTopTens();
Map<UUID, Long> tt = lm.getTopTen(world, 10);
assertTrue(tt.isEmpty());
}
/** /**
* Test method for {@link world.bentobox.level.LevelsManager#hasTopTenPerm(org.bukkit.World, java.util.UUID)}. * Test method for {@link world.bentobox.level.LevelsManager#hasTopTenPerm(org.bukkit.World, java.util.UUID)}.
*/ */
@ -403,6 +416,7 @@ public class LevelsManagerTest {
public void testSetIslandLevel() { public void testSetIslandLevel() {
lm.setIslandLevel(world, uuid, 1234); lm.setIslandLevel(world, uuid, 1234);
assertEquals(1234, lm.getIslandLevel(world, uuid)); assertEquals(1234, lm.getIslandLevel(world, uuid));
} }
/** /**