package org.bukkit.craftbukkit; import com.mojang.authlib.GameProfile; import java.io.File; import java.time.Duration; import java.time.Instant; import java.util.Date; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; import java.util.UUID; import net.minecraft.core.GlobalPos; import net.minecraft.nbt.DynamicOpsNBT; import net.minecraft.nbt.NBTTagCompound; import net.minecraft.nbt.NBTTagList; import net.minecraft.server.players.WhiteListEntry; import net.minecraft.stats.ServerStatisticManager; import net.minecraft.world.level.storage.WorldNBTStorage; import org.bukkit.BanEntry; import org.bukkit.BanList; import org.bukkit.Bukkit; import org.bukkit.Location; import org.bukkit.Material; import org.bukkit.OfflinePlayer; import org.bukkit.Server; import org.bukkit.Statistic; import org.bukkit.ban.ProfileBanList; import org.bukkit.configuration.serialization.ConfigurationSerializable; import org.bukkit.configuration.serialization.SerializableAs; import org.bukkit.craftbukkit.entity.memory.CraftMemoryMapper; import org.bukkit.craftbukkit.profile.CraftPlayerProfile; import org.bukkit.entity.EntityType; import org.bukkit.entity.Player; import org.bukkit.metadata.MetadataValue; import org.bukkit.plugin.Plugin; import org.bukkit.profile.PlayerProfile; @SerializableAs("Player") public class CraftOfflinePlayer implements OfflinePlayer, ConfigurationSerializable { private final GameProfile profile; private final CraftServer server; private final WorldNBTStorage storage; protected CraftOfflinePlayer(CraftServer server, GameProfile profile) { this.server = server; this.profile = profile; this.storage = server.console.playerDataStorage; } @Override public boolean isOnline() { return getPlayer() != null; } @Override public String getName() { Player player = getPlayer(); if (player != null) { return player.getName(); } // This might not match lastKnownName but if not it should be more correct if (!profile.getName().isEmpty()) { return profile.getName(); } NBTTagCompound data = getBukkitData(); if (data != null) { if (data.contains("lastKnownName")) { return data.getString("lastKnownName"); } } return null; } @Override public UUID getUniqueId() { return profile.getId(); } @Override public PlayerProfile getPlayerProfile() { return new CraftPlayerProfile(profile); } public Server getServer() { return server; } @Override public boolean isOp() { return server.getHandle().isOp(profile); } @Override public void setOp(boolean value) { if (value == isOp()) { return; } if (value) { server.getHandle().op(profile); } else { server.getHandle().deop(profile); } } @Override public boolean isBanned() { return ((ProfileBanList) server.getBanList(BanList.Type.PROFILE)).isBanned(getPlayerProfile()); } @Override public BanEntry ban(String reason, Date expires, String source) { return ((ProfileBanList) server.getBanList(BanList.Type.PROFILE)).addBan(getPlayerProfile(), reason, expires, source); } @Override public BanEntry ban(String reason, Instant expires, String source) { return ((ProfileBanList) server.getBanList(BanList.Type.PROFILE)).addBan(getPlayerProfile(), reason, expires, source); } @Override public BanEntry ban(String reason, Duration duration, String source) { return ((ProfileBanList) server.getBanList(BanList.Type.PROFILE)).addBan(getPlayerProfile(), reason, duration, source); } public void setBanned(boolean value) { if (value) { ((ProfileBanList) server.getBanList(BanList.Type.PROFILE)).addBan(getPlayerProfile(), null, (Date) null, null); } else { ((ProfileBanList) server.getBanList(BanList.Type.PROFILE)).pardon(getPlayerProfile()); } } @Override public boolean isWhitelisted() { return server.getHandle().getWhiteList().isWhiteListed(profile); } @Override public void setWhitelisted(boolean value) { if (value) { server.getHandle().getWhiteList().add(new WhiteListEntry(profile)); } else { server.getHandle().getWhiteList().remove(profile); } } @Override public Map serialize() { Map result = new LinkedHashMap<>(); result.put("UUID", profile.getId().toString()); return result; } public static OfflinePlayer deserialize(Map args) { // Backwards comparability if (args.get("name") != null) { return Bukkit.getServer().getOfflinePlayer((String) args.get("name")); } return Bukkit.getServer().getOfflinePlayer(UUID.fromString((String) args.get("UUID"))); } @Override public String toString() { return getClass().getSimpleName() + "[UUID=" + profile.getId() + "]"; } @Override public Player getPlayer() { return server.getPlayer(getUniqueId()); } @Override public boolean equals(Object obj) { if (!(obj instanceof OfflinePlayer other)) { return false; } if ((this.getUniqueId() == null) || (other.getUniqueId() == null)) { return false; } return this.getUniqueId().equals(other.getUniqueId()); } @Override public int hashCode() { int hash = 5; hash = 97 * hash + (this.getUniqueId() != null ? this.getUniqueId().hashCode() : 0); return hash; } private NBTTagCompound getData() { return storage.load(profile.getName(), profile.getId().toString()).orElse(null); } private NBTTagCompound getBukkitData() { NBTTagCompound result = getData(); if (result != null) { if (!result.contains("bukkit")) { result.put("bukkit", new NBTTagCompound()); } result = result.getCompound("bukkit"); } return result; } private File getDataFile() { return new File(storage.getPlayerDir(), getUniqueId() + ".dat"); } @Override public long getFirstPlayed() { Player player = getPlayer(); if (player != null) return player.getFirstPlayed(); NBTTagCompound data = getBukkitData(); if (data != null) { if (data.contains("firstPlayed")) { return data.getLong("firstPlayed"); } else { File file = getDataFile(); return file.lastModified(); } } else { return 0; } } @Override public long getLastPlayed() { Player player = getPlayer(); if (player != null) return player.getLastPlayed(); NBTTagCompound data = getBukkitData(); if (data != null) { if (data.contains("lastPlayed")) { return data.getLong("lastPlayed"); } else { File file = getDataFile(); return file.lastModified(); } } else { return 0; } } @Override public boolean hasPlayedBefore() { return getData() != null; } @Override public Location getLastDeathLocation() { if (getData().contains("LastDeathLocation", 10)) { return GlobalPos.CODEC.parse(DynamicOpsNBT.INSTANCE, getData().get("LastDeathLocation")).result().map(CraftMemoryMapper::fromNms).orElse(null); } return null; } @Override public Location getLocation() { NBTTagCompound data = getData(); if (data == null) { return null; } if (data.contains("Pos") && data.contains("Rotation")) { NBTTagList position = (NBTTagList) data.get("Pos"); NBTTagList rotation = (NBTTagList) data.get("Rotation"); UUID uuid = new UUID(data.getLong("WorldUUIDMost"), data.getLong("WorldUUIDLeast")); return new Location(server.getWorld(uuid), position.getDouble(0), position.getDouble(1), position.getDouble(2), rotation.getFloat(0), rotation.getFloat(1) ); } return null; } @Override public Location getBedSpawnLocation() { return getRespawnLocation(); } @Override public Location getRespawnLocation() { NBTTagCompound data = getData(); if (data == null) return null; if (data.contains("SpawnX") && data.contains("SpawnY") && data.contains("SpawnZ")) { String spawnWorld = data.getString("SpawnWorld"); if (spawnWorld.equals("")) { spawnWorld = server.getWorlds().get(0).getName(); } return new Location(server.getWorld(spawnWorld), data.getInt("SpawnX"), data.getInt("SpawnY"), data.getInt("SpawnZ")); } return null; } public void setMetadata(String metadataKey, MetadataValue metadataValue) { server.getPlayerMetadata().setMetadata(this, metadataKey, metadataValue); } public List getMetadata(String metadataKey) { return server.getPlayerMetadata().getMetadata(this, metadataKey); } public boolean hasMetadata(String metadataKey) { return server.getPlayerMetadata().hasMetadata(this, metadataKey); } public void removeMetadata(String metadataKey, Plugin plugin) { server.getPlayerMetadata().removeMetadata(this, metadataKey, plugin); } private ServerStatisticManager getStatisticManager() { return server.getHandle().getPlayerStats(getUniqueId(), getName()); } @Override public void incrementStatistic(Statistic statistic) { if (isOnline()) { getPlayer().incrementStatistic(statistic); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.incrementStatistic(manager, statistic, null); manager.save(); } } @Override public void decrementStatistic(Statistic statistic) { if (isOnline()) { getPlayer().decrementStatistic(statistic); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.decrementStatistic(manager, statistic, null); manager.save(); } } @Override public int getStatistic(Statistic statistic) { if (isOnline()) { return getPlayer().getStatistic(statistic); } else { return CraftStatistic.getStatistic(getStatisticManager(), statistic); } } @Override public void incrementStatistic(Statistic statistic, int amount) { if (isOnline()) { getPlayer().incrementStatistic(statistic, amount); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.incrementStatistic(manager, statistic, amount, null); manager.save(); } } @Override public void decrementStatistic(Statistic statistic, int amount) { if (isOnline()) { getPlayer().decrementStatistic(statistic, amount); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.decrementStatistic(manager, statistic, amount, null); manager.save(); } } @Override public void setStatistic(Statistic statistic, int newValue) { if (isOnline()) { getPlayer().setStatistic(statistic, newValue); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.setStatistic(manager, statistic, newValue, null); manager.save(); } } @Override public void incrementStatistic(Statistic statistic, Material material) { if (isOnline()) { getPlayer().incrementStatistic(statistic, material); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.incrementStatistic(manager, statistic, material, null); manager.save(); } } @Override public void decrementStatistic(Statistic statistic, Material material) { if (isOnline()) { getPlayer().decrementStatistic(statistic, material); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.decrementStatistic(manager, statistic, material, null); manager.save(); } } @Override public int getStatistic(Statistic statistic, Material material) { if (isOnline()) { return getPlayer().getStatistic(statistic, material); } else { return CraftStatistic.getStatistic(getStatisticManager(), statistic, material); } } @Override public void incrementStatistic(Statistic statistic, Material material, int amount) { if (isOnline()) { getPlayer().incrementStatistic(statistic, material, amount); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.incrementStatistic(manager, statistic, material, amount, null); manager.save(); } } @Override public void decrementStatistic(Statistic statistic, Material material, int amount) { if (isOnline()) { getPlayer().decrementStatistic(statistic, material, amount); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.decrementStatistic(manager, statistic, material, amount, null); manager.save(); } } @Override public void setStatistic(Statistic statistic, Material material, int newValue) { if (isOnline()) { getPlayer().setStatistic(statistic, material, newValue); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.setStatistic(manager, statistic, material, newValue, null); manager.save(); } } @Override public void incrementStatistic(Statistic statistic, EntityType entityType) { if (isOnline()) { getPlayer().incrementStatistic(statistic, entityType); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.incrementStatistic(manager, statistic, entityType, null); manager.save(); } } @Override public void decrementStatistic(Statistic statistic, EntityType entityType) { if (isOnline()) { getPlayer().decrementStatistic(statistic, entityType); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.decrementStatistic(manager, statistic, entityType, null); manager.save(); } } @Override public int getStatistic(Statistic statistic, EntityType entityType) { if (isOnline()) { return getPlayer().getStatistic(statistic, entityType); } else { return CraftStatistic.getStatistic(getStatisticManager(), statistic, entityType); } } @Override public void incrementStatistic(Statistic statistic, EntityType entityType, int amount) { if (isOnline()) { getPlayer().incrementStatistic(statistic, entityType, amount); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.incrementStatistic(manager, statistic, entityType, amount, null); manager.save(); } } @Override public void decrementStatistic(Statistic statistic, EntityType entityType, int amount) { if (isOnline()) { getPlayer().decrementStatistic(statistic, entityType, amount); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.decrementStatistic(manager, statistic, entityType, amount, null); manager.save(); } } @Override public void setStatistic(Statistic statistic, EntityType entityType, int newValue) { if (isOnline()) { getPlayer().setStatistic(statistic, entityType, newValue); } else { ServerStatisticManager manager = getStatisticManager(); CraftStatistic.setStatistic(manager, statistic, entityType, newValue, null); manager.save(); } } }