2018-03-26 03:50:46 +02:00
From dcdadafa3574152be245d8ebdad7d666a12006c1 Mon Sep 17 00:00:00 2001
2018-01-16 04:13:17 +01:00
From: Aikar <aikar@aikar.co>
Date: Mon, 15 Jan 2018 22:11:48 -0500
Subject: [PATCH] Basic PlayerProfile API
2018-01-19 06:03:09 +01:00
diff --git a/src/main/java/com/destroystokyo/paper/profile/CraftPlayerProfile.java b/src/main/java/com/destroystokyo/paper/profile/CraftPlayerProfile.java
2018-01-16 04:13:17 +01:00
new file mode 100644
2018-03-26 03:50:46 +02:00
index 000000000..8cc85be8f
2018-01-16 04:13:17 +01:00
--- /dev/null
2018-01-19 06:03:09 +01:00
+++ b/src/main/java/com/destroystokyo/paper/profile/CraftPlayerProfile.java
2018-03-26 03:50:46 +02:00
@@ -0,0 +1,276 @@
2018-01-16 04:13:17 +01:00
+package com.destroystokyo.paper.profile;
+
2018-03-22 06:28:22 +01:00
+import com.destroystokyo.paper.PaperConfig;
+import com.google.common.base.Charsets;
2018-01-16 04:13:17 +01:00
+import com.mojang.authlib.GameProfile;
+import com.mojang.authlib.properties.Property;
+import com.mojang.authlib.properties.PropertyMap;
2018-03-18 16:31:32 +01:00
+import net.minecraft.server.MinecraftServer;
2018-03-26 03:50:46 +02:00
+import net.minecraft.server.UserCache;
2018-03-22 06:28:22 +01:00
+import org.bukkit.craftbukkit.entity.CraftPlayer;
+import org.spigotmc.SpigotConfig;
2018-01-16 04:13:17 +01:00
+
+import javax.annotation.Nonnull;
+import javax.annotation.Nullable;
2018-01-21 20:09:09 +01:00
+import java.util.AbstractSet;
2018-01-16 04:13:17 +01:00
+import java.util.Collection;
2018-01-21 20:09:09 +01:00
+import java.util.Iterator;
2018-03-23 02:56:18 +01:00
+import java.util.Objects;
2018-01-16 04:13:17 +01:00
+import java.util.Set;
+import java.util.UUID;
+
2018-01-19 06:03:09 +01:00
+public class CraftPlayerProfile implements PlayerProfile {
2018-01-16 04:13:17 +01:00
+
2018-03-18 16:31:32 +01:00
+ private GameProfile profile;
2018-03-23 02:40:57 +01:00
+ private final PropertySet properties = new PropertySet();
2018-01-16 04:13:17 +01:00
+
2018-03-22 06:28:22 +01:00
+ public CraftPlayerProfile(CraftPlayer player) {
2018-03-23 02:40:57 +01:00
+ this.profile = player.getHandle().getProfile();
+ }
+
2018-01-19 06:03:09 +01:00
+ public CraftPlayerProfile(UUID id, String name) {
2018-03-26 02:05:30 +02:00
+ this.profile = new GameProfile(id, name);
2018-03-23 02:40:57 +01:00
+ }
+
+ public CraftPlayerProfile(GameProfile profile) {
+ this.profile = profile;
2018-03-22 06:28:22 +01:00
+ }
+
2018-03-26 02:05:30 +02:00
+ @Override
+ public boolean hasProperty(String property) {
+ return profile.getProperties().containsKey(property);
+ }
2018-03-22 06:28:22 +01:00
+
2018-03-26 02:05:30 +02:00
+ @Override
+ public void setProperty(ProfileProperty property) {
+ String name = property.getName();
+ PropertyMap properties = profile.getProperties();
+ properties.removeAll(name);
+ properties.put(name, new Property(name, property.getValue(), property.getSignature()));
2018-03-22 06:28:22 +01:00
+ }
+
2018-03-26 02:05:30 +02:00
+ public GameProfile getGameProfile() {
2018-03-23 02:40:57 +01:00
+ return profile;
+ }
+
2018-03-26 02:05:30 +02:00
+ @Nullable
+ @Override
+ public UUID getId() {
+ return profile.getId();
2018-01-16 04:13:17 +01:00
+ }
+
2018-03-26 03:50:46 +02:00
+ @Override
+ public UUID setId(@Nullable UUID uuid) {
+ GameProfile prev = this.profile;
+ this.profile = new GameProfile(uuid, prev.getName());
+ copyProfileProperties(prev, this.profile);
+ return prev.getId();
+ }
+
2018-03-26 02:05:30 +02:00
+ @Nullable
2018-01-16 04:13:17 +01:00
+ @Override
2018-03-26 02:05:30 +02:00
+ public String getName() {
+ return profile.getName();
2018-01-19 06:38:49 +01:00
+ }
+
2018-03-26 03:50:46 +02:00
+ @Override
+ public String setName(@Nullable String name) {
+ GameProfile prev = this.profile;
+ this.profile = new GameProfile(prev.getId(), name);
+ copyProfileProperties(prev, this.profile);
+ return prev.getName();
+ }
+
2018-03-26 02:05:30 +02:00
+ @Nonnull
2018-01-19 06:38:49 +01:00
+ @Override
2018-03-26 02:05:30 +02:00
+ public Set<ProfileProperty> getProperties() {
+ return properties;
2018-01-16 04:13:17 +01:00
+ }
+
+ @Override
+ public void setProperties(Collection<ProfileProperty> properties) {
+ properties.forEach(this::setProperty);
+ }
+
+ @Override
2018-03-26 02:05:30 +02:00
+ public void clearProperties() {
+ profile.getProperties().clear();
+ }
+
+ @Override
2018-01-16 04:13:17 +01:00
+ public boolean removeProperty(String property) {
+ return !profile.getProperties().removeAll(property).isEmpty();
+ }
+
+ @Override
2018-03-26 02:05:30 +02:00
+ public boolean equals(Object o) {
+ if (this == o) return true;
+ if (o == null || getClass() != o.getClass()) return false;
+ CraftPlayerProfile that = (CraftPlayerProfile) o;
+ return Objects.equals(profile, that.profile);
+ }
+
+ @Override
+ public int hashCode() {
+ return profile.hashCode();
+ }
+
+ @Override
+ public String toString() {
+ return profile.toString();
+ }
+
+ @Override
+ public CraftPlayerProfile clone() {
+ CraftPlayerProfile clone = new CraftPlayerProfile(this.getId(), this.getName());
+ clone.setProperties(getProperties());
+ return clone;
2018-01-16 04:13:17 +01:00
+ }
+
+ @Override
+ public boolean isComplete() {
+ return profile.isComplete();
+ }
+
2018-03-26 03:50:46 +02:00
+ @Override
+ public boolean completeFromCache() {
+ return completeFromCache(false);
+ }
+
+ public boolean completeFromCache(boolean lookupName) {
2018-03-23 02:40:57 +01:00
+ MinecraftServer server = MinecraftServer.getServer();
+ String name = profile.getName();
2018-03-26 03:50:46 +02:00
+ UserCache userCache = server.getUserCache();
2018-03-23 02:40:57 +01:00
+ if (profile.getId() == null) {
2018-03-26 03:50:46 +02:00
+ final GameProfile profile;
+ boolean isOnlineMode = server.getOnlineMode() || (SpigotConfig.bungee && PaperConfig.bungeeOnlineMode);
+ if (isOnlineMode) {
+ profile = lookupName ? userCache.getProfile(name) : userCache.getProfileIfCached(name);
+ } else {
+ // Make an OfflinePlayer using an offline mode UUID since the name has no profile
+ profile = new GameProfile(UUID.nameUUIDFromBytes(("OfflinePlayer:" + name).getBytes(Charsets.UTF_8)), name);
+ }
2018-03-23 02:40:57 +01:00
+ if (profile == null) {
+ throw new NullPointerException("Could not get UUID for Player " + name);
+ }
2018-03-23 02:56:18 +01:00
+ this.profile = profile;
2018-03-23 02:40:57 +01:00
+ }
2018-03-26 03:50:46 +02:00
+
+ if (profile.getName() == null) {
2018-03-26 02:05:30 +02:00
+ // If we need textures, skip this check, as we will get it below anyways.
2018-03-26 03:50:46 +02:00
+ GameProfile profile = userCache.getProfile(this.profile.getId());
2018-03-26 02:05:30 +02:00
+ if (profile != null) {
+ this.profile = profile;
+ }
+ }
2018-03-26 03:50:46 +02:00
+ return this.profile.isComplete();
+ }
+
+ public boolean complete(boolean textures) {
+ MinecraftServer server = MinecraftServer.getServer();
+
+ if (!this.completeFromCache(true) || textures && !hasTextures()) {
2018-03-23 02:40:57 +01:00
+ GameProfile result = server.getSessionService().fillProfileProperties(profile, true);
2018-03-22 06:28:22 +01:00
+ if (result != null) {
+ this.profile = result;
+ }
2018-03-22 00:12:02 +01:00
+ }
2018-03-23 02:40:57 +01:00
+ return profile.isComplete() && (!textures || hasTextures());
2018-03-18 16:31:32 +01:00
+ }
+
2018-03-26 02:05:30 +02:00
+ private static void copyProfileProperties(GameProfile source, GameProfile target) {
2018-03-26 03:50:46 +02:00
+ PropertyMap sourceProperties = source.getProperties();
+ if (sourceProperties.isEmpty()) {
+ return;
+ }
2018-03-26 02:05:30 +02:00
+ PropertyMap properties = target.getProperties();
+ properties.clear();
2018-03-26 03:50:46 +02:00
+
+ for (Property property : sourceProperties.values()) {
2018-03-26 02:05:30 +02:00
+ properties.put(property.getName(), property);
+ }
+ }
+
2018-01-19 06:03:09 +01:00
+ private static ProfileProperty toBukkit(Property property) {
2018-01-16 04:13:17 +01:00
+ return new ProfileProperty(property.getName(), property.getValue(), property.getSignature());
+ }
2018-01-19 06:03:09 +01:00
+
2018-01-19 06:38:49 +01:00
+ public static PlayerProfile asBukkitCopy(GameProfile gameProfile) {
2018-03-22 06:28:22 +01:00
+ CraftPlayerProfile profile = new CraftPlayerProfile(gameProfile.getId(), gameProfile.getName());
+ copyProfileProperties(gameProfile, profile.profile);
2018-01-19 06:03:09 +01:00
+ return profile;
+ }
+
2018-01-19 06:38:49 +01:00
+ public static PlayerProfile asBukkitMirror(GameProfile profile) {
+ return new CraftPlayerProfile(profile);
+ }
+
2018-01-19 06:03:09 +01:00
+ public static Property asAuthlib(ProfileProperty property) {
+ return new Property(property.getName(), property.getValue(), property.getSignature());
+ }
2018-03-26 02:05:30 +02:00
+
2018-01-19 06:38:49 +01:00
+ public static GameProfile asAuthlibCopy(PlayerProfile profile) {
+ CraftPlayerProfile craft = ((CraftPlayerProfile) profile);
+ return asAuthlib(craft.clone());
+ }
+
+ public static GameProfile asAuthlib(PlayerProfile profile) {
+ CraftPlayerProfile craft = ((CraftPlayerProfile) profile);
+ return craft.getGameProfile();
+ }
2018-01-21 20:09:09 +01:00
+
+ private class PropertySet extends AbstractSet<ProfileProperty> {
+
+ @Override
2018-03-08 03:03:01 +01:00
+ @Nonnull
2018-01-21 20:09:09 +01:00
+ public Iterator<ProfileProperty> iterator() {
2018-03-08 03:03:01 +01:00
+ return new ProfilePropertyIterator(profile.getProperties().values().iterator());
2018-01-21 20:09:09 +01:00
+ }
+
+ @Override
+ public int size() {
+ return profile.getProperties().size();
+ }
+
+ @Override
+ public boolean add(ProfileProperty property) {
+ setProperty(property);
+ return true;
+ }
+
+ @Override
+ public boolean addAll(Collection<? extends ProfileProperty> c) {
+ //noinspection unchecked
+ setProperties((Collection<ProfileProperty>) c);
+ return true;
+ }
+
+ @Override
+ public boolean contains(Object o) {
+ return o instanceof ProfileProperty && profile.getProperties().containsKey(((ProfileProperty) o).getName());
+ }
2018-03-08 03:03:01 +01:00
+
+ private class ProfilePropertyIterator implements Iterator<ProfileProperty> {
+ private final Iterator<Property> iterator;
+
+ ProfilePropertyIterator(Iterator<Property> iterator) {
+ this.iterator = iterator;
+ }
+
+ @Override
+ public boolean hasNext() {
+ return iterator.hasNext();
+ }
+
+ @Override
+ public ProfileProperty next() {
+ return toBukkit(iterator.next());
+ }
+
+ @Override
+ public void remove() {
+ iterator.remove();
+ }
+ }
2018-01-21 20:09:09 +01:00
+ }
2018-01-16 04:13:17 +01:00
+}
2018-01-19 06:12:03 +01:00
diff --git a/src/main/java/net/minecraft/server/MCUtil.java b/src/main/java/net/minecraft/server/MCUtil.java
2018-01-19 06:38:49 +01:00
index 02940d697..4539b5601 100644
2018-01-19 06:12:03 +01:00
--- a/src/main/java/net/minecraft/server/MCUtil.java
+++ b/src/main/java/net/minecraft/server/MCUtil.java
@@ -1,6 +1,9 @@
package net.minecraft.server;
+import com.destroystokyo.paper.profile.CraftPlayerProfile;
+import com.destroystokyo.paper.profile.PlayerProfile;
import com.google.common.util.concurrent.ThreadFactoryBuilder;
+import com.mojang.authlib.GameProfile;
import org.apache.commons.lang.exception.ExceptionUtils;
import org.bukkit.Location;
import org.bukkit.craftbukkit.CraftWorld;
@@ -66,6 +69,10 @@ public final class MCUtil {
return run.get();
}
+ public static PlayerProfile toBukkit(GameProfile profile) {
2018-01-19 06:38:49 +01:00
+ return CraftPlayerProfile.asBukkitMirror(profile);
2018-01-19 06:12:03 +01:00
+ }
+
/**
* Calculates distance between 2 entities
* @param e1
2018-03-18 16:31:32 +01:00
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
index e8bddc171..3b01ebd96 100644
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
@@ -1538,6 +1538,7 @@ public abstract class MinecraftServer implements ICommandListener, Runnable, IAs
this.H = i;
}
+ public MinecraftSessionService getSessionService() { return az(); } // Paper - OBFHELPER
public MinecraftSessionService az() {
return this.W;
}
2018-03-26 03:50:46 +02:00
diff --git a/src/main/java/net/minecraft/server/UserCache.java b/src/main/java/net/minecraft/server/UserCache.java
index 7ce08eb8b..6a750c25e 100644
--- a/src/main/java/net/minecraft/server/UserCache.java
+++ b/src/main/java/net/minecraft/server/UserCache.java
@@ -44,7 +44,7 @@ public class UserCache {
public static final SimpleDateFormat a = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss Z");
private static boolean c;
- private final Map<String, UserCache.UserCacheEntry> d = Maps.newHashMap();
+ private final Map<String, UserCache.UserCacheEntry> d = Maps.newHashMap();private final Map<String, UserCache.UserCacheEntry> nameCache = d; // Paper - OBFHELPER
private final Map<UUID, UserCache.UserCacheEntry> e = Maps.newHashMap();
private final Deque<GameProfile> f = new java.util.concurrent.LinkedBlockingDeque<GameProfile>(); // CraftBukkit
private final GameProfileRepository g;
@@ -173,6 +173,13 @@ public class UserCache {
return (String[]) arraylist.toArray(new String[arraylist.size()]);
}
+ // Paper start
+ @Nullable public GameProfile getProfileIfCached(String name) {
+ UserCache.UserCacheEntry entry = this.nameCache.get(name.toLowerCase(Locale.ROOT));
+ return entry == null ? null : entry.getProfile();
+ }
+ // Paper end
+
@Nullable public GameProfile getProfile(UUID uuid) { return a(uuid); } // Paper - OBFHELPER
@Nullable
public synchronized GameProfile a(UUID uuid) { // Paper - synchronize
@@ -282,7 +289,7 @@ public class UserCache {
class UserCacheEntry {
- private final GameProfile b;
+ private final GameProfile b;public GameProfile getProfile() { return b; } // Paper - OBFHELPER
private final Date c;
private UserCacheEntry(GameProfile gameprofile, Date date) {
2018-01-16 04:13:17 +01:00
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
2018-03-22 06:28:22 +01:00
index 77c16fe2c..2dd7ed96a 100644
2018-01-16 04:13:17 +01:00
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
@@ -135,6 +135,10 @@ import org.bukkit.craftbukkit.util.CraftNamespacedKey;
import org.bukkit.event.server.TabCompleteEvent;
import net.md_5.bungee.api.chat.BaseComponent;
+import javax.annotation.Nullable; // Paper
+import javax.annotation.Nonnull; // Paper
+
+
public final class CraftServer implements Server {
private final String serverName = "Paper";
private final String serverVersion;
2018-03-22 06:28:22 +01:00
@@ -1923,5 +1927,21 @@ public final class CraftServer implements Server {
2018-01-16 04:13:17 +01:00
public boolean suggestPlayerNamesWhenNullTabCompletions() {
return com.destroystokyo.paper.PaperConfig.suggestPlayersWhenNullTabCompletions;
}
+
+ public com.destroystokyo.paper.profile.PlayerProfile createProfile(@Nonnull UUID uuid) {
+ return createProfile(uuid, null);
+ }
+
+ public com.destroystokyo.paper.profile.PlayerProfile createProfile(@Nonnull String name) {
+ return createProfile(null, name);
+ }
+
+ public com.destroystokyo.paper.profile.PlayerProfile createProfile(@Nullable UUID uuid, @Nullable String name) {
2018-03-22 06:28:22 +01:00
+ Player player = uuid != null ? Bukkit.getPlayer(uuid) : (name != null ? Bukkit.getPlayerExact(name) : null);
+ if (player != null) {
+ return new com.destroystokyo.paper.profile.CraftPlayerProfile((CraftPlayer)player);
+ }
2018-01-19 06:03:09 +01:00
+ return new com.destroystokyo.paper.profile.CraftPlayerProfile(uuid, name);
2018-01-16 04:13:17 +01:00
+ }
// Paper end
}
--
2018-03-08 03:03:01 +01:00
2.16.2
2018-01-16 04:13:17 +01:00