Death Counter

This commit is contained in:
2025-09-01 21:19:38 +02:00
commit 954420ffd7
10 changed files with 873 additions and 0 deletions

113
.gitignore vendored Normal file
View File

@@ -0,0 +1,113 @@
# User-specific stuff
.idea/
*.iml
*.ipr
*.iws
# IntelliJ
out/
# Compiled class file
*.class
# Log file
*.log
# BlueJ files
*.ctxt
# Package Files #
*.jar
*.war
*.nar
*.ear
*.zip
*.tar.gz
*.rar
# virtual machine crash logs, see http://www.java.com/en/download/help/error_hotspot.xml
hs_err_pid*
*~
# temporary files which can be created if a process still has a handle open of a deleted file
.fuse_hidden*
# KDE directory preferences
.directory
# Linux trash folder which might appear on any partition or disk
.Trash-*
# .nfs files are created when an open file is removed but is still being accessed
.nfs*
# General
.DS_Store
.AppleDouble
.LSOverride
# Icon must end with two \r
Icon
# Thumbnails
._*
# Files that might appear in the root of a volume
.DocumentRevisions-V100
.fseventsd
.Spotlight-V100
.TemporaryItems
.Trashes
.VolumeIcon.icns
.com.apple.timemachine.donotpresent
# Directories potentially created on remote AFP share
.AppleDB
.AppleDesktop
Network Trash Folder
Temporary Items
.apdisk
# Windows thumbnail cache files
Thumbs.db
Thumbs.db:encryptable
ehthumbs.db
ehthumbs_vista.db
# Dump file
*.stackdump
# Folder config file
[Dd]esktop.ini
# Recycle Bin used on file shares
$RECYCLE.BIN/
# Windows Installer files
*.cab
*.msi
*.msix
*.msm
*.msp
# Windows shortcuts
*.lnk
target/
pom.xml.tag
pom.xml.releaseBackup
pom.xml.versionsBackup
pom.xml.next
release.properties
dependency-reduced-pom.xml
buildNumber.properties
.mvn/timing.properties
.mvn/wrapper/maven-wrapper.jar
.flattened-pom.xml
# Common working directory
run/

105
pom.xml Normal file
View File

@@ -0,0 +1,105 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>dev.tatsi.reloadmc.smp</groupId>
<artifactId>reloadmc_smp</artifactId>
<version>1.0.0</version>
<packaging>jar</packaging>
<name>reloadmc_smp</name>
<properties>
<java.version>21</java.version>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
<build>
<defaultGoal>clean package</defaultGoal>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.13.0</version>
<configuration>
<source>${java.version}</source>
<target>${java.version}</target>
</configuration>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-shade-plugin</artifactId>
<version>3.5.3</version>
<executions>
<execution>
<phase>package</phase>
<goals>
<goal>shade</goal>
</goals>
</execution>
</executions>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-surefire-plugin</artifactId>
<version>3.1.2</version>
<configuration>
<useSystemClassLoader>false</useSystemClassLoader>
</configuration>
</plugin>
</plugins>
<resources>
<resource>
<directory>src/main/resources</directory>
<filtering>true</filtering>
</resource>
</resources>
</build>
<repositories>
<repository>
<id>papermc-repo</id>
<url>https://repo.papermc.io/repository/maven-public/</url>
</repository>
<repository>
<id>jitpack.io</id>
<url>https://jitpack.io</url>
</repository>
</repositories>
<dependencies>
<dependency>
<groupId>io.papermc.paper</groupId>
<artifactId>paper-api</artifactId>
<version>1.21.8-R0.1-SNAPSHOT</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>com.google.code.gson</groupId>
<artifactId>gson</artifactId>
<version>2.10.1</version>
</dependency>
<!-- Testing dependencies -->
<dependency>
<groupId>org.junit.jupiter</groupId>
<artifactId>junit-jupiter</artifactId>
<version>5.10.0</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.mockito</groupId>
<artifactId>mockito-core</artifactId>
<version>5.5.0</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.mockito</groupId>
<artifactId>mockito-junit-jupiter</artifactId>
<version>5.5.0</version>
<scope>test</scope>
</dependency>
</dependencies>
</project>

View File

@@ -0,0 +1,39 @@
package dev.tatsi.reloadmc.smp;
import dev.tatsi.reloadmc.smp.command.DeathStatsCommand;
import dev.tatsi.reloadmc.smp.listener.PlayerDeathListener;
import dev.tatsi.reloadmc.smp.manager.DeathCounterManager;
import org.bukkit.plugin.java.JavaPlugin;
public final class ReloadMC extends JavaPlugin {
private DeathCounterManager deathCounterManager;
@Override
public void onEnable() {
// Initialize death counter manager
deathCounterManager = new DeathCounterManager(this);
// Register event listeners
getServer().getPluginManager().registerEvents(new PlayerDeathListener(deathCounterManager), this);
// Register commands
getCommand("deathstats").setExecutor(new DeathStatsCommand(deathCounterManager));
getLogger().info("ReloadMC SMP Plugin has been enabled!");
getLogger().info("Death counter system is now active.");
}
@Override
public void onDisable() {
// Save death counter data on shutdown
if (deathCounterManager != null) {
deathCounterManager.shutdown();
}
getLogger().info("ReloadMC SMP Plugin has been disabled!");
}
public DeathCounterManager getDeathCounterManager() {
return deathCounterManager;
}
}

View File

@@ -0,0 +1,95 @@
package dev.tatsi.reloadmc.smp.command;
import dev.tatsi.reloadmc.smp.manager.DeathCounterManager;
import dev.tatsi.reloadmc.smp.model.DeathRecord;
import dev.tatsi.reloadmc.smp.model.PlayerDeathData;
import org.bukkit.Bukkit;
import org.bukkit.command.Command;
import org.bukkit.command.CommandExecutor;
import org.bukkit.command.CommandSender;
import org.bukkit.entity.Player;
import java.util.List;
import java.util.UUID;
public class DeathStatsCommand implements CommandExecutor {
private final DeathCounterManager deathCounterManager;
public DeathStatsCommand(DeathCounterManager deathCounterManager) {
this.deathCounterManager = deathCounterManager;
}
@Override
public boolean onCommand(CommandSender sender, Command command, String label, String[] args) {
if (args.length == 0) {
// Show own stats if player, or usage if console
if (sender instanceof Player) {
showPlayerStats(sender, (Player) sender);
} else {
sender.sendMessage("§cUsage: /deathstats <player>");
}
return true;
}
// Show stats for specified player
String targetPlayerName = args[0];
Player targetPlayer = Bukkit.getPlayer(targetPlayerName);
if (targetPlayer == null) {
// Try to find offline player data
UUID targetUuid = null;
for (PlayerDeathData data : deathCounterManager.getAllPlayerData().values()) {
if (data.getPlayerName().equalsIgnoreCase(targetPlayerName)) {
targetUuid = data.getPlayerUuid();
break;
}
}
if (targetUuid == null) {
sender.sendMessage("§cPlayer '" + targetPlayerName + "' not found or has no death records.");
return true;
}
showPlayerStatsByUuid(sender, targetUuid, targetPlayerName);
} else {
showPlayerStats(sender, targetPlayer);
}
return true;
}
private void showPlayerStats(CommandSender sender, Player player) {
showPlayerStatsByUuid(sender, player.getUniqueId(), player.getName());
}
private void showPlayerStatsByUuid(CommandSender sender, UUID playerUuid, String playerName) {
PlayerDeathData deathData = deathCounterManager.getPlayerDeathData(playerUuid);
if (deathData == null || deathData.getDeathCount() == 0) {
sender.sendMessage("§a" + playerName + " has no recorded deaths. Lucky!");
return;
}
sender.sendMessage("§6=== Death Statistics for " + playerName + " ===");
sender.sendMessage("§eTotalDeaths: §c" + deathData.getDeathCount());
List<DeathRecord> deaths = deathData.getDeaths();
if (deaths.size() > 0) {
sender.sendMessage("§eRecent Deaths:");
int showCount = Math.min(5, deaths.size()); // Show last 5 deaths
for (int i = deaths.size() - showCount; i < deaths.size(); i++) {
DeathRecord death = deaths.get(i);
sender.sendMessage(String.format("§7- §f%s §7at §e%.1f, %.1f, %.1f §7in §b%s §7(%s)",
death.getDeathReason(),
death.getX(), death.getY(), death.getZ(),
death.getWorld(),
death.getTimestamp().replace("T", " ")
));
}
if (deaths.size() > 5) {
sender.sendMessage("§7... and " + (deaths.size() - 5) + " more deaths.");
}
}
}
}

View File

@@ -0,0 +1,51 @@
package dev.tatsi.reloadmc.smp.listener;
import dev.tatsi.reloadmc.smp.manager.DeathCounterManager;
import dev.tatsi.reloadmc.smp.model.DeathRecord;
import org.bukkit.Location;
import org.bukkit.entity.Player;
import org.bukkit.event.EventHandler;
import org.bukkit.event.Listener;
import org.bukkit.event.entity.PlayerDeathEvent;
public class PlayerDeathListener implements Listener {
private final DeathCounterManager deathCounterManager;
public PlayerDeathListener(DeathCounterManager deathCounterManager) {
this.deathCounterManager = deathCounterManager;
}
@EventHandler
public void onPlayerDeath(PlayerDeathEvent event) {
Player player = event.getEntity();
Location deathLocation = player.getLocation();
// Get death reason from the death message, or use a default if null
String deathReason = event.getDeathMessage();
if (deathReason == null || deathReason.trim().isEmpty()) {
deathReason = "Unknown cause";
}
// Create death record with location and reason
DeathRecord deathRecord = new DeathRecord(
deathLocation.getX(),
deathLocation.getY(),
deathLocation.getZ(),
deathLocation.getWorld().getName(),
deathReason
);
// Record the death
deathCounterManager.addDeath(
player.getUniqueId(),
player.getName(),
deathRecord
);
// Optional: Send a message to the player about their death count
int deathCount = deathCounterManager.getDeathCount(player.getUniqueId());
player.sendMessage(String.format("§cYou have died %d time(s). Death recorded at %.1f, %.1f, %.1f in %s",
deathCount, deathLocation.getX(), deathLocation.getY(), deathLocation.getZ(),
deathLocation.getWorld().getName()));
}
}

View File

@@ -0,0 +1,95 @@
package dev.tatsi.reloadmc.smp.manager;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.google.gson.reflect.TypeToken;
import dev.tatsi.reloadmc.smp.model.DeathRecord;
import dev.tatsi.reloadmc.smp.model.PlayerDeathData;
import org.bukkit.plugin.java.JavaPlugin;
import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.lang.reflect.Type;
import java.util.HashMap;
import java.util.Map;
import java.util.UUID;
import java.util.logging.Level;
public class DeathCounterManager {
private final JavaPlugin plugin;
private final File dataFile;
private final Gson gson;
private final Map<UUID, PlayerDeathData> playerDeathData;
public DeathCounterManager(JavaPlugin plugin) {
this.plugin = plugin;
this.dataFile = new File(plugin.getDataFolder(), "death_counter.json");
this.gson = new GsonBuilder().setPrettyPrinting().create();
this.playerDeathData = new HashMap<>();
// Create plugin data folder if it doesn't exist
if (!plugin.getDataFolder().exists()) {
plugin.getDataFolder().mkdirs();
}
loadData();
}
public void addDeath(UUID playerUuid, String playerName, DeathRecord deathRecord) {
PlayerDeathData data = playerDeathData.computeIfAbsent(playerUuid,
uuid -> new PlayerDeathData(uuid, playerName));
data.addDeath(deathRecord);
saveData();
plugin.getLogger().info(String.format("Recorded death for %s: %s", playerName, deathRecord));
}
public PlayerDeathData getPlayerDeathData(UUID playerUuid) {
return playerDeathData.get(playerUuid);
}
public int getDeathCount(UUID playerUuid) {
PlayerDeathData data = playerDeathData.get(playerUuid);
return data != null ? data.getDeathCount() : 0;
}
public Map<UUID, PlayerDeathData> getAllPlayerData() {
return new HashMap<>(playerDeathData);
}
private void loadData() {
if (!dataFile.exists()) {
plugin.getLogger().info("Death counter data file not found, starting with empty data.");
return;
}
try (FileReader reader = new FileReader(dataFile)) {
Type type = new TypeToken<Map<UUID, PlayerDeathData>>() {
}.getType();
Map<UUID, PlayerDeathData> loadedData = gson.fromJson(reader, type);
if (loadedData != null) {
playerDeathData.putAll(loadedData);
plugin.getLogger().info(String.format("Loaded death data for %d players.", loadedData.size()));
}
} catch (IOException e) {
plugin.getLogger().log(Level.SEVERE, "Failed to load death counter data", e);
}
}
private void saveData() {
try (FileWriter writer = new FileWriter(dataFile)) {
gson.toJson(playerDeathData, writer);
} catch (IOException e) {
plugin.getLogger().log(Level.SEVERE, "Failed to save death counter data", e);
}
}
public void shutdown() {
saveData();
plugin.getLogger().info("Death counter data saved on shutdown.");
}
}

View File

@@ -0,0 +1,53 @@
package dev.tatsi.reloadmc.smp.model;
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
public class DeathRecord {
private final double x;
private final double y;
private final double z;
private final String world;
private final String deathReason;
private final String timestamp;
public DeathRecord(double x, double y, double z, String world, String deathReason) {
this.x = x;
this.y = y;
this.z = z;
this.world = world;
this.deathReason = deathReason;
this.timestamp = LocalDateTime.now().format(DateTimeFormatter.ISO_LOCAL_DATE_TIME);
}
// Getters
public double getX() {
return x;
}
public double getY() {
return y;
}
public double getZ() {
return z;
}
public String getWorld() {
return world;
}
public String getDeathReason() {
return deathReason;
}
public String getTimestamp() {
return timestamp;
}
@Override
public String toString() {
return String.format("DeathRecord{x=%.2f, y=%.2f, z=%.2f, world='%s', reason='%s', time='%s'}",
x, y, z, world, deathReason, timestamp);
}
}

View File

@@ -0,0 +1,51 @@
package dev.tatsi.reloadmc.smp.model;
import java.util.ArrayList;
import java.util.List;
import java.util.UUID;
public class PlayerDeathData {
private final UUID playerUuid;
private final String playerName;
private final List<DeathRecord> deaths;
public PlayerDeathData(UUID playerUuid, String playerName) {
this.playerUuid = playerUuid;
this.playerName = playerName;
this.deaths = new ArrayList<>();
}
// Constructor for JSON deserialization
public PlayerDeathData(UUID playerUuid, String playerName, List<DeathRecord> deaths) {
this.playerUuid = playerUuid;
this.playerName = playerName;
this.deaths = deaths != null ? new ArrayList<>(deaths) : new ArrayList<>();
}
public void addDeath(DeathRecord deathRecord) {
deaths.add(deathRecord);
}
public int getDeathCount() {
return deaths.size();
}
// Getters
public UUID getPlayerUuid() {
return playerUuid;
}
public String getPlayerName() {
return playerName;
}
public List<DeathRecord> getDeaths() {
return new ArrayList<>(deaths); // Return copy to prevent external modification
}
@Override
public String toString() {
return String.format("PlayerDeathData{uuid=%s, name='%s', deathCount=%d}",
playerUuid, playerName, deaths.size());
}
}

View File

@@ -0,0 +1,14 @@
name: reloadmc_smp
version: '1.0.0'
main: dev.tatsi.reloadmc.smp.ReloadMC
api-version: '1.21'
prefix: CorePlugin
authors: [ Bummsa / Thatsaphorn Atchariyaphap ]
description: Custom Core Plugin for mini OGSquad
website: https://tatsi.dev
commands:
deathstats:
description: View death statistics for yourself or another player
usage: /deathstats [player]
aliases: [deaths, deathcount]

View File

@@ -0,0 +1,257 @@
package dev.tatsi.reloadmc.smp.manager;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import dev.tatsi.reloadmc.smp.model.DeathRecord;
import dev.tatsi.reloadmc.smp.model.PlayerDeathData;
import org.bukkit.plugin.java.JavaPlugin;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.extension.ExtendWith;
import org.junit.jupiter.api.io.TempDir;
import org.mockito.Mock;
import org.mockito.junit.jupiter.MockitoExtension;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.nio.file.Path;
import java.util.HashMap;
import java.util.Map;
import java.util.UUID;
import java.util.logging.Logger;
import static org.junit.jupiter.api.Assertions.*;
import static org.mockito.Mockito.*;
@ExtendWith(MockitoExtension.class)
class DeathCounterManagerTest {
@Mock
private JavaPlugin mockPlugin;
@Mock
private Logger mockLogger;
@TempDir
Path tempDir;
private DeathCounterManager deathCounterManager;
private File dataFolder;
private UUID testPlayerUuid;
private String testPlayerName;
@BeforeEach
void setUp() {
// Setup mock plugin
dataFolder = tempDir.toFile();
when(mockPlugin.getDataFolder()).thenReturn(dataFolder);
when(mockPlugin.getLogger()).thenReturn(mockLogger);
// Test data
testPlayerUuid = UUID.randomUUID();
testPlayerName = "TestPlayer";
// Create DeathCounterManager instance
deathCounterManager = new DeathCounterManager(mockPlugin);
}
@Test
void testConstructor_CreatesDataFolderIfNotExists() {
// Given: A new temp directory that doesn't exist
File newDataFolder = new File(tempDir.toFile(), "newFolder");
when(mockPlugin.getDataFolder()).thenReturn(newDataFolder);
// When: Creating a new DeathCounterManager
new DeathCounterManager(mockPlugin);
// Then: The data folder should be created
assertTrue(newDataFolder.exists());
}
@Test
void testAddDeath_NewPlayer() {
// Given: A new death record
DeathRecord deathRecord = new DeathRecord(100.0, 64.0, 200.0, "world", "Fell from a high place");
// When: Adding a death for a new player
deathCounterManager.addDeath(testPlayerUuid, testPlayerName, deathRecord);
// Then: Player data should be created and death should be recorded
PlayerDeathData playerData = deathCounterManager.getPlayerDeathData(testPlayerUuid);
assertNotNull(playerData);
assertEquals(testPlayerUuid, playerData.getPlayerUuid());
assertEquals(testPlayerName, playerData.getPlayerName());
assertEquals(1, playerData.getDeathCount());
assertEquals(1, playerData.getDeaths().size());
assertEquals(deathRecord.getDeathReason(), playerData.getDeaths().get(0).getDeathReason());
}
@Test
void testAddDeath_ExistingPlayer() {
// Given: A player with existing death data
DeathRecord firstDeath = new DeathRecord(100.0, 64.0, 200.0, "world", "Fell from a high place");
DeathRecord secondDeath = new DeathRecord(150.0, 70.0, 250.0, "nether", "Burned to death");
// When: Adding multiple deaths for the same player
deathCounterManager.addDeath(testPlayerUuid, testPlayerName, firstDeath);
deathCounterManager.addDeath(testPlayerUuid, testPlayerName, secondDeath);
// Then: Both deaths should be recorded
PlayerDeathData playerData = deathCounterManager.getPlayerDeathData(testPlayerUuid);
assertNotNull(playerData);
assertEquals(2, playerData.getDeathCount());
assertEquals(2, playerData.getDeaths().size());
}
@Test
void testGetPlayerDeathData_NonExistentPlayer() {
// Given: A UUID that doesn't exist in the data
UUID nonExistentUuid = UUID.randomUUID();
// When: Getting player data for non-existent player
PlayerDeathData result = deathCounterManager.getPlayerDeathData(nonExistentUuid);
// Then: Should return null
assertNull(result);
}
@Test
void testGetDeathCount_ExistingPlayer() {
// Given: A player with death data
DeathRecord deathRecord = new DeathRecord(100.0, 64.0, 200.0, "world", "Fell from a high place");
deathCounterManager.addDeath(testPlayerUuid, testPlayerName, deathRecord);
// When: Getting death count
int deathCount = deathCounterManager.getDeathCount(testPlayerUuid);
// Then: Should return correct count
assertEquals(1, deathCount);
}
@Test
void testGetDeathCount_NonExistentPlayer() {
// Given: A UUID that doesn't exist in the data
UUID nonExistentUuid = UUID.randomUUID();
// When: Getting death count for non-existent player
int deathCount = deathCounterManager.getDeathCount(nonExistentUuid);
// Then: Should return 0
assertEquals(0, deathCount);
}
@Test
void testGetAllPlayerData() {
// Given: Multiple players with death data
UUID player1Uuid = UUID.randomUUID();
UUID player2Uuid = UUID.randomUUID();
DeathRecord death1 = new DeathRecord(100.0, 64.0, 200.0, "world", "Fell from a high place");
DeathRecord death2 = new DeathRecord(150.0, 70.0, 250.0, "nether", "Burned to death");
deathCounterManager.addDeath(player1Uuid, "Player1", death1);
deathCounterManager.addDeath(player2Uuid, "Player2", death2);
// When: Getting all player data
Map<UUID, PlayerDeathData> allData = deathCounterManager.getAllPlayerData();
// Then: Should return copy of all data
assertEquals(2, allData.size());
assertTrue(allData.containsKey(player1Uuid));
assertTrue(allData.containsKey(player2Uuid));
// Verify it's a copy (modifying returned map shouldn't affect internal data)
allData.clear();
assertEquals(2, deathCounterManager.getAllPlayerData().size());
}
@Test
void testDataPersistence_SaveAndLoad() throws IOException {
// Given: Some death data
DeathRecord deathRecord = new DeathRecord(100.0, 64.0, 200.0, "world", "Fell from a high place");
deathCounterManager.addDeath(testPlayerUuid, testPlayerName, deathRecord);
// When: Creating a new DeathCounterManager (which should load existing data)
DeathCounterManager newManager = new DeathCounterManager(mockPlugin);
// Then: Data should be loaded correctly
PlayerDeathData loadedData = newManager.getPlayerDeathData(testPlayerUuid);
assertNotNull(loadedData);
assertEquals(testPlayerUuid, loadedData.getPlayerUuid());
assertEquals(testPlayerName, loadedData.getPlayerName());
assertEquals(1, loadedData.getDeathCount());
}
@Test
void testLoadData_EmptyFile() throws IOException {
// Given: An empty data file
File dataFile = new File(dataFolder, "death_counter.json");
// Create an empty file
dataFile.createNewFile();
assertTrue(dataFile.exists());
// When: Creating a new DeathCounterManager
DeathCounterManager newManager = new DeathCounterManager(mockPlugin);
// Then: Should handle empty data gracefully
assertEquals(0, newManager.getAllPlayerData().size());
}
@Test
void testShutdown_SavesData() {
// Given: Some death data
DeathRecord deathRecord = new DeathRecord(100.0, 64.0, 200.0, "world", "Fell from a high place");
deathCounterManager.addDeath(testPlayerUuid, testPlayerName, deathRecord);
// When: Calling shutdown
deathCounterManager.shutdown();
// Then: Should log shutdown message
verify(mockLogger).info("Death counter data saved on shutdown.");
}
@Test
void testDeathRecordDetails() {
// Given: A death record with specific details
double x = 123.45;
double y = 67.89;
double z = 234.56;
String world = "test_world";
String reason = "Killed by Zombie";
DeathRecord deathRecord = new DeathRecord(x, y, z, world, reason);
deathCounterManager.addDeath(testPlayerUuid, testPlayerName, deathRecord);
// When: Retrieving the death record
PlayerDeathData playerData = deathCounterManager.getPlayerDeathData(testPlayerUuid);
DeathRecord retrievedRecord = playerData.getDeaths().get(0);
// Then: All details should be preserved
assertEquals(x, retrievedRecord.getX(), 0.001);
assertEquals(y, retrievedRecord.getY(), 0.001);
assertEquals(z, retrievedRecord.getZ(), 0.001);
assertEquals(world, retrievedRecord.getWorld());
assertEquals(reason, retrievedRecord.getDeathReason());
assertNotNull(retrievedRecord.getTimestamp());
}
@Test
void testMultipleDeathsPreserveOrder() {
// Given: Multiple deaths added in sequence
DeathRecord death1 = new DeathRecord(100.0, 64.0, 200.0, "world", "First death");
DeathRecord death2 = new DeathRecord(150.0, 70.0, 250.0, "nether", "Second death");
DeathRecord death3 = new DeathRecord(200.0, 80.0, 300.0, "end", "Third death");
// When: Adding deaths in sequence
deathCounterManager.addDeath(testPlayerUuid, testPlayerName, death1);
deathCounterManager.addDeath(testPlayerUuid, testPlayerName, death2);
deathCounterManager.addDeath(testPlayerUuid, testPlayerName, death3);
// Then: Deaths should be preserved in order
PlayerDeathData playerData = deathCounterManager.getPlayerDeathData(testPlayerUuid);
assertEquals(3, playerData.getDeathCount());
assertEquals("First death", playerData.getDeaths().get(0).getDeathReason());
assertEquals("Second death", playerData.getDeaths().get(1).getDeathReason());
assertEquals("Third death", playerData.getDeaths().get(2).getDeathReason());
}
}