From 781a945ef463cdf731c9a55a5a18434bbe34ef3e Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Wed, 12 Dec 2012 07:41:19 +0100 Subject: [PATCH 01/37] Bumping to 1.8.1-SNAPSHOT for development towards the next version. --- ProtocolLib/pom.xml | 2 +- ProtocolLib/src/main/resources/plugin.yml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/ProtocolLib/pom.xml b/ProtocolLib/pom.xml index 421b5e48..b2c422c0 100644 --- a/ProtocolLib/pom.xml +++ b/ProtocolLib/pom.xml @@ -2,7 +2,7 @@ 4.0.0 com.comphenix.protocol ProtocolLib - 1.8.0 + 1.8.1-SNAPSHOT jar Provides read/write access to the Minecraft protocol. diff --git a/ProtocolLib/src/main/resources/plugin.yml b/ProtocolLib/src/main/resources/plugin.yml index cd537899..5f087b09 100644 --- a/ProtocolLib/src/main/resources/plugin.yml +++ b/ProtocolLib/src/main/resources/plugin.yml @@ -1,5 +1,5 @@ name: ProtocolLib -version: 1.8.0 +version: 1.8.1-SNAPSHOT description: Provides read/write access to the Minecraft protocol. author: Comphenix website: http://www.comphenix.net/ProtocolLib From e855be64ad1f156c3a8c0291266c81482e90bead Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Wed, 12 Dec 2012 07:41:58 +0100 Subject: [PATCH 02/37] Default settings matter. Make auto-update opt-in. --- ProtocolLib/src/main/resources/config.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ProtocolLib/src/main/resources/config.yml b/ProtocolLib/src/main/resources/config.yml index 0b5a4f64..32f4e4c3 100644 --- a/ProtocolLib/src/main/resources/config.yml +++ b/ProtocolLib/src/main/resources/config.yml @@ -2,7 +2,7 @@ global: # Settings for the automatic version updater auto updater: notify: true - download: true + download: false # Number of seconds to wait until a new update is downloaded delay: 43200 # 12 hours From 711754550911ba7cda88d38ed8db202d05b0d625 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Wed, 12 Dec 2012 08:32:37 +0100 Subject: [PATCH 03/37] Prevent plugins from spamming the console. --- ProtocolLib/dependency-reduced-pom.xml | 2 +- .../protocol/error/DetailedErrorReporter.java | 70 ++++++++++++++++--- 2 files changed, 60 insertions(+), 12 deletions(-) diff --git a/ProtocolLib/dependency-reduced-pom.xml b/ProtocolLib/dependency-reduced-pom.xml index 00d4c784..5a8f8b86 100644 --- a/ProtocolLib/dependency-reduced-pom.xml +++ b/ProtocolLib/dependency-reduced-pom.xml @@ -4,7 +4,7 @@ com.comphenix.protocol ProtocolLib ProtocolLib - 1.8.0 + 1.8.1-SNAPSHOT Provides read/write access to the Minecraft protocol. http://dev.bukkit.org/server-mods/protocollib/ diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java b/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java index 9b6ef329..c14e6f33 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java @@ -6,6 +6,9 @@ import java.lang.ref.WeakReference; import java.util.HashMap; import java.util.Map; import java.util.Set; +import java.util.concurrent.ConcurrentHashMap; +import java.util.concurrent.ConcurrentMap; +import java.util.concurrent.atomic.AtomicInteger; import java.util.logging.Level; import java.util.logging.Logger; @@ -35,6 +38,9 @@ public class DetailedErrorReporter implements ErrorReporter { // We don't want to spam the server public static final int DEFAULT_MAX_ERROR_COUNT = 20; + // Prevent spam per plugin too + private ConcurrentMap warningCount = new ConcurrentHashMap(); + protected String prefix; protected String supportURL; @@ -97,23 +103,65 @@ public class DetailedErrorReporter implements ErrorReporter { @Override public void reportMinimal(Plugin sender, String methodName, Throwable error, Object... parameters) { - reportMinimal(sender, methodName, error); - - // Print parameters, if they are given - if (parameters != null && parameters.length > 0) { - logger.log(Level.SEVERE, " Parameters:"); - - // Print each parameter - for (Object parameter : parameters) { - logger.log(Level.SEVERE, " " + getStringDescription(parameter)); + if (reportMinimalNoSpam(sender, methodName, error)) { + // Print parameters, if they are given + if (parameters != null && parameters.length > 0) { + logger.log(Level.SEVERE, " Parameters:"); + + // Print each parameter + for (Object parameter : parameters) { + logger.log(Level.SEVERE, " " + getStringDescription(parameter)); + } } } } @Override public void reportMinimal(Plugin sender, String methodName, Throwable error) { - logger.log(Level.SEVERE, "[" + PLUGIN_NAME + "] Unhandled exception occured in " + methodName + " for " + - PacketAdapter.getPluginName(sender), error); + reportMinimalNoSpam(sender, methodName, error); + } + + public boolean reportMinimalNoSpam(Plugin sender, String methodName, Throwable error) { + String pluginName = PacketAdapter.getPluginName(sender); + AtomicInteger counter = warningCount.get(pluginName); + + // Thread safe pattern + if (counter == null) { + AtomicInteger created = new AtomicInteger(); + counter = warningCount.putIfAbsent(pluginName, created); + + if (counter == null) { + counter = created; + } + } + + final int errorCount = counter.incrementAndGet(); + + // See if we should print the full error + if (errorCount < getMaxErrorCount()) { + logger.log(Level.SEVERE, "[" + PLUGIN_NAME + "] Unhandled exception occured in " + + methodName + " for " + pluginName, error); + return true; + + } else { + // Nope - only print the error count occationally + if (isPowerOfTwo(errorCount)) { + logger.log(Level.SEVERE, "[" + PLUGIN_NAME + "] Unhandled exception number " + errorCount + " occured in " + + methodName + " for " + pluginName); + } + return false; + } + } + + /** + * Determine if a given number is a power of two. + *

+ * That is, if there exists an N such that 2^N = number. + * @param number - the number to check. + * @return TRUE if the given number is a power of two, FALSE otherwise. + */ + private boolean isPowerOfTwo(int number) { + return (number & (number - 1)) == 0; } @Override From 15961f6ccef6150c19e4949384ed9d7d7aa0e278 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Wed, 12 Dec 2012 08:41:55 +0100 Subject: [PATCH 04/37] We can include the stack trace - it will only very rarely be printed. --- .../com/comphenix/protocol/error/DetailedErrorReporter.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java b/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java index c14e6f33..baec9121 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java @@ -147,7 +147,7 @@ public class DetailedErrorReporter implements ErrorReporter { // Nope - only print the error count occationally if (isPowerOfTwo(errorCount)) { logger.log(Level.SEVERE, "[" + PLUGIN_NAME + "] Unhandled exception number " + errorCount + " occured in " + - methodName + " for " + pluginName); + methodName + " for " + pluginName, error); } return false; } From 4440e6fcaa5f8a22e9e31137526aac871f3d3a90 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Wed, 12 Dec 2012 18:34:02 +0100 Subject: [PATCH 05/37] Prevent ProtocolLib from spamming the console too. --- .../protocol/error/DetailedErrorReporter.java | 21 ++++++++++++------- 1 file changed, 14 insertions(+), 7 deletions(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java b/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java index baec9121..9964f2c7 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java @@ -44,7 +44,8 @@ public class DetailedErrorReporter implements ErrorReporter { protected String prefix; protected String supportURL; - protected int errorCount; + protected AtomicInteger internalErrorCount = new AtomicInteger(); + protected int maxErrorCount; protected Logger logger; @@ -185,12 +186,18 @@ public class DetailedErrorReporter implements ErrorReporter { public void reportDetailed(Object sender, String message, Throwable error, Object... parameters) { final Plugin plugin = pluginReference.get(); + final int errorCount = internalErrorCount.incrementAndGet(); // Do not overtly spam the server! - if (++errorCount > maxErrorCount) { - String maxReached = String.format("Reached maxmimum error count. Cannot pass error %s from %s.", error, sender); - logger.severe(maxReached); - return; + if (errorCount > getMaxErrorCount()) { + // Only allow the error count at rare occations + if (isPowerOfTwo(errorCount)) { + // Permit it - but print the number of exceptions first + reportWarning(this, "Internal exception count: " + errorCount + "!"); + } else { + // NEVER SPAM THE CONSOLE + return; + } } StringWriter text = new StringWriter(); @@ -297,11 +304,11 @@ public class DetailedErrorReporter implements ErrorReporter { } public int getErrorCount() { - return errorCount; + return internalErrorCount.get(); } public void setErrorCount(int errorCount) { - this.errorCount = errorCount; + internalErrorCount.set(errorCount); } public int getMaxErrorCount() { From 86f15c867450e1c04bcdd1555a1b831bb4c9206e Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Wed, 12 Dec 2012 18:35:58 +0100 Subject: [PATCH 06/37] Fallback on our own custom object printer. --- .../protocol/error/DetailedErrorReporter.java | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java b/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java index 9964f2c7..ec878a62 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java @@ -18,6 +18,7 @@ import org.bukkit.Bukkit; import org.bukkit.plugin.Plugin; import com.comphenix.protocol.events.PacketAdapter; +import com.comphenix.protocol.reflect.PrettyPrinter; import com.google.common.primitives.Primitives; /** @@ -285,12 +286,15 @@ public class DetailedErrorReporter implements ErrorReporter { return (ToStringBuilder.reflectionToString(value, ToStringStyle.MULTI_LINE_STYLE, false, null)); } catch (Throwable ex) { // Apache is probably missing - logger.warning("Cannot find Apache Commons. Object introspection disabled."); apacheCommonsMissing = true; } - // Just use toString() - return String.format("%s", value); + // Use our custom object printer instead + try { + return PrettyPrinter.printObject(value, value.getClass(), Object.class); + } catch (IllegalAccessException e) { + return "[Error: " + e.getMessage() + "]"; + } } } From 79afb0bcdcb0300a308b9430545ec509048090b4 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 13 Dec 2012 00:39:50 +0100 Subject: [PATCH 07/37] Forgot to add the target object in the ChunkPosition wrapper. --- .../java/com/comphenix/protocol/wrappers/ChunkPosition.java | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/ChunkPosition.java b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/ChunkPosition.java index fd7bc679..f9e60fa1 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/ChunkPosition.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/ChunkPosition.java @@ -166,7 +166,8 @@ public class ChunkPosition { if (intModifier.size() >= 3) { try { - return new ChunkPosition(intModifier.read(0), intModifier.read(1), intModifier.read(2)); + StructureModifier instance = intModifier.withTarget(generic); + return new ChunkPosition(instance.read(0), instance.read(1), instance.read(2)); } catch (FieldAccessException e) { // This is an exeptional work-around, so we don't want to burden the caller with the messy details throw new RuntimeException("Field access error.", e); From b1bd7e9fcde178857f540f269a41b88154532c4f Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Sat, 15 Dec 2012 07:17:04 +0100 Subject: [PATCH 08/37] Remove unused classes from CGLib. Saves us 136 kB. --- ProtocolLib/dependency-reduced-pom.xml | 1 + ProtocolLib/pom.xml | 1 + 2 files changed, 2 insertions(+) diff --git a/ProtocolLib/dependency-reduced-pom.xml b/ProtocolLib/dependency-reduced-pom.xml index 5a8f8b86..5872d5cc 100644 --- a/ProtocolLib/dependency-reduced-pom.xml +++ b/ProtocolLib/dependency-reduced-pom.xml @@ -55,6 +55,7 @@ shade + true false true diff --git a/ProtocolLib/pom.xml b/ProtocolLib/pom.xml index b2c422c0..f5fe7b03 100644 --- a/ProtocolLib/pom.xml +++ b/ProtocolLib/pom.xml @@ -55,6 +55,7 @@ shade + true false true From d7d6e24506dc8779f8cd7f3f23cee03df93c4dc2 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Wed, 19 Dec 2012 17:51:21 +0100 Subject: [PATCH 09/37] Added support for the new way of converting CraftItemStacks in 1.4.5 --- ProtocolLib/dependency-reduced-pom.xml | 1 - .../protocol/utility/MinecraftReflection.java | 66 +++++++++++++++++-- 2 files changed, 61 insertions(+), 6 deletions(-) diff --git a/ProtocolLib/dependency-reduced-pom.xml b/ProtocolLib/dependency-reduced-pom.xml index 5872d5cc..5a8f8b86 100644 --- a/ProtocolLib/dependency-reduced-pom.xml +++ b/ProtocolLib/dependency-reduced-pom.xml @@ -55,7 +55,6 @@ shade - true false true diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java index ded29674..e21be948 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java @@ -35,10 +35,14 @@ public class MinecraftReflection { private static CachedPackage craftbukkitPackage; // org.bukkit.craftbukkit - private static Class craftItemStackClass; private static Constructor craftNMSConstructor; private static Constructor craftBukkitConstructor; + // New in 1.4.5 + private static Method craftNMSMethod; + private static Method craftBukkitMethod; + private static boolean craftItemStackFailed; + // net.minecraft.server private static Class itemStackArrayClass; @@ -370,9 +374,7 @@ public class MinecraftReflection { */ @SuppressWarnings("rawtypes") public static Class getCraftItemStackClass() { - if (craftItemStackClass == null) - craftItemStackClass = getCraftBukkitClass("inventory.CraftItemStack"); - return craftItemStackClass; + return getCraftBukkitClass("inventory.CraftItemStack"); } /** @@ -382,10 +384,18 @@ public class MinecraftReflection { */ @SuppressWarnings("unchecked") public static ItemStack getBukkitItemStack(ItemStack bukkitItemStack) { + // Delegate this task to the method that can execute it + if (craftBukkitMethod != null) + return getBukkitItemByMethod(bukkitItemStack); + if (craftBukkitConstructor == null) { try { craftBukkitConstructor = getCraftItemStackClass().getConstructor(ItemStack.class); } catch (Exception e) { + // See if this method works + if (!craftItemStackFailed) + return getBukkitItemByMethod(bukkitItemStack); + throw new RuntimeException("Cannot find CraftItemStack(org.bukkit.inventory.ItemStack).", e); } } @@ -397,7 +407,26 @@ public class MinecraftReflection { throw new RuntimeException("Cannot construct CraftItemStack.", e); } } - + + @SuppressWarnings("unchecked") + private static ItemStack getBukkitItemByMethod(ItemStack bukkitItemStack) { + if (craftBukkitMethod == null) { + try { + craftBukkitMethod = getCraftItemStackClass().getMethod("asCraftCopy", ItemStack.class); + } catch (Exception e) { + craftItemStackFailed = true; + throw new RuntimeException("Cannot find CraftItemStack.asCraftCopy(org.bukkit.inventory.ItemStack).", e); + } + } + + // Next, construct it + try { + return (ItemStack) craftBukkitMethod.invoke(null, bukkitItemStack); + } catch (Exception e) { + throw new RuntimeException("Cannot construct CraftItemStack.", e); + } + } + /** * Retrieve the Bukkit ItemStack from a given net.minecraft.server ItemStack. * @param minecraftItemStack - the NMS ItemStack to wrap. @@ -405,10 +434,18 @@ public class MinecraftReflection { */ @SuppressWarnings("unchecked") public static ItemStack getBukkitItemStack(Object minecraftItemStack) { + // Delegate this task to the method that can execute it + if (craftNMSMethod != null) + return getBukkitItemByMethod(minecraftItemStack); + if (craftNMSConstructor == null) { try { craftNMSConstructor = getCraftItemStackClass().getConstructor(minecraftItemStack.getClass()); } catch (Exception e) { + // Give it a try + if (!craftItemStackFailed) + return getBukkitItemByMethod(minecraftItemStack); + throw new RuntimeException("Cannot find CraftItemStack(net.mineraft.server.ItemStack).", e); } } @@ -421,6 +458,25 @@ public class MinecraftReflection { } } + @SuppressWarnings("unchecked") + private static ItemStack getBukkitItemByMethod(Object minecraftItemStack) { + if (craftNMSMethod == null) { + try { + craftNMSMethod = getCraftItemStackClass().getMethod("asCraftMirror", minecraftItemStack.getClass()); + } catch (Exception e) { + craftItemStackFailed = true; + throw new RuntimeException("Cannot find CraftItemStack.asCraftMirror(net.mineraft.server.ItemStack).", e); + } + } + + // Next, construct it + try { + return (ItemStack) craftNMSMethod.invoke(null, minecraftItemStack); + } catch (Exception e) { + throw new RuntimeException("Cannot construct CraftItemStack.", e); + } + } + /** * Retrieve the net.minecraft.server ItemStack from a Bukkit ItemStack. * @param stack - the Bukkit ItemStack to convert. From b0cc1440ddfb03810001cfcdd851e1b071260fc2 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Wed, 19 Dec 2012 17:52:56 +0100 Subject: [PATCH 10/37] Revert "Remove unused classes from CGLib. Saves us 136 kB. " This reverts commit b1bd7e9fcde178857f540f269a41b88154532c4f. --- ProtocolLib/pom.xml | 1 - 1 file changed, 1 deletion(-) diff --git a/ProtocolLib/pom.xml b/ProtocolLib/pom.xml index f5fe7b03..b2c422c0 100644 --- a/ProtocolLib/pom.xml +++ b/ProtocolLib/pom.xml @@ -55,7 +55,6 @@ shade - true false true From 3b0790f136dab1b4debffd0c20265e9eaeb46f43 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Wed, 19 Dec 2012 18:04:44 +0100 Subject: [PATCH 11/37] Bumping to release of 1.8.1 --- ProtocolLib/pom.xml | 2 +- ProtocolLib/src/main/resources/plugin.yml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/ProtocolLib/pom.xml b/ProtocolLib/pom.xml index b2c422c0..7b0502a5 100644 --- a/ProtocolLib/pom.xml +++ b/ProtocolLib/pom.xml @@ -2,7 +2,7 @@ 4.0.0 com.comphenix.protocol ProtocolLib - 1.8.1-SNAPSHOT + 1.8.1 jar Provides read/write access to the Minecraft protocol. diff --git a/ProtocolLib/src/main/resources/plugin.yml b/ProtocolLib/src/main/resources/plugin.yml index 5f087b09..f0f1e961 100644 --- a/ProtocolLib/src/main/resources/plugin.yml +++ b/ProtocolLib/src/main/resources/plugin.yml @@ -1,5 +1,5 @@ name: ProtocolLib -version: 1.8.1-SNAPSHOT +version: 1.8.1 description: Provides read/write access to the Minecraft protocol. author: Comphenix website: http://www.comphenix.net/ProtocolLib From 1509f5cd8bef819561bd5cce50bb912b91f01e54 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Wed, 19 Dec 2012 23:10:28 +0100 Subject: [PATCH 12/37] Increasing to 1.8.2-SNAPSHOT for development toward the next version. --- ProtocolLib/dependency-reduced-pom.xml | 2 +- ProtocolLib/pom.xml | 2 +- ProtocolLib/src/main/resources/plugin.yml | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/ProtocolLib/dependency-reduced-pom.xml b/ProtocolLib/dependency-reduced-pom.xml index 5a8f8b86..1be13b08 100644 --- a/ProtocolLib/dependency-reduced-pom.xml +++ b/ProtocolLib/dependency-reduced-pom.xml @@ -4,7 +4,7 @@ com.comphenix.protocol ProtocolLib ProtocolLib - 1.8.1-SNAPSHOT + 1.8.2-SNAPSHOT Provides read/write access to the Minecraft protocol. http://dev.bukkit.org/server-mods/protocollib/ diff --git a/ProtocolLib/pom.xml b/ProtocolLib/pom.xml index 7b0502a5..252339c9 100644 --- a/ProtocolLib/pom.xml +++ b/ProtocolLib/pom.xml @@ -2,7 +2,7 @@ 4.0.0 com.comphenix.protocol ProtocolLib - 1.8.1 + 1.8.2-SNAPSHOT jar Provides read/write access to the Minecraft protocol. diff --git a/ProtocolLib/src/main/resources/plugin.yml b/ProtocolLib/src/main/resources/plugin.yml index f0f1e961..f990e825 100644 --- a/ProtocolLib/src/main/resources/plugin.yml +++ b/ProtocolLib/src/main/resources/plugin.yml @@ -1,5 +1,5 @@ name: ProtocolLib -version: 1.8.1 +version: 1.8.2-SNAPSHOT description: Provides read/write access to the Minecraft protocol. author: Comphenix website: http://www.comphenix.net/ProtocolLib From 3b142db569539e06acd0afe848c2f802fb213a15 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Wed, 19 Dec 2012 23:10:50 +0100 Subject: [PATCH 13/37] Don't ignore critical exceptions - print them. --- .../java/com/comphenix/protocol/ProtocolLibrary.java | 11 ++++++++++- .../comphenix/protocol/injector/BukkitUnwrapper.java | 3 +++ 2 files changed, 13 insertions(+), 1 deletion(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolLibrary.java b/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolLibrary.java index 2f5b4774..3cf0e1ab 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolLibrary.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolLibrary.java @@ -65,7 +65,7 @@ public class ProtocolLibrary extends JavaPlugin { private static PacketFilterManager protocolManager; // Error reporter - private ErrorReporter reporter; + private static ErrorReporter reporter; // Metrics and statistisc private Statistics statistisc; @@ -353,6 +353,7 @@ public class ProtocolLibrary extends JavaPlugin { protocolManager.close(); protocolManager = null; statistisc = null; + reporter = null; // Leaky ClassLoader begone! CleanupStaticMembers cleanup = new CleanupStaticMembers(getClassLoader(), reporter); @@ -373,6 +374,14 @@ public class ProtocolLibrary extends JavaPlugin { return log; } + /** + * Retrieve the current error reporter. + * @return Current error reporter. + */ + public static ErrorReporter getErrorReporter() { + return reporter; + } + /** * Retrieves the packet protocol manager. * @return Packet protocol manager, or NULL if it has been disabled. diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/BukkitUnwrapper.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/BukkitUnwrapper.java index 147e2087..40420214 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/BukkitUnwrapper.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/BukkitUnwrapper.java @@ -23,6 +23,7 @@ import java.util.Collection; import java.util.Map; import java.util.concurrent.ConcurrentHashMap; +import com.comphenix.protocol.ProtocolLibrary; import com.comphenix.protocol.injector.PacketConstructor.Unwrapper; import com.comphenix.protocol.reflect.instances.DefaultInstances; @@ -107,8 +108,10 @@ public class BukkitUnwrapper implements Unwrapper { return find; } catch (SecurityException e) { + ProtocolLibrary.getErrorReporter().reportDetailed(this, "Security limitation.", e, type); return null; } catch (NoSuchMethodException e) { + ProtocolLibrary.getErrorReporter().reportDetailed(this, "Cannot unwrap object.", e, type); return null; } } From 220c0e4bc507f75d9fbddde7fed89b30d63c9427 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Wed, 19 Dec 2012 23:48:31 +0100 Subject: [PATCH 14/37] Ensure that CraftItemStacks can be converted to NMS item stacks. --- .../protocol/injector/BukkitUnwrapper.java | 131 +++++++++++++----- 1 file changed, 97 insertions(+), 34 deletions(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/BukkitUnwrapper.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/BukkitUnwrapper.java index 40420214..ae444147 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/BukkitUnwrapper.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/BukkitUnwrapper.java @@ -17,6 +17,7 @@ package com.comphenix.protocol.injector; +import java.lang.reflect.Field; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.util.Collection; @@ -25,7 +26,9 @@ import java.util.concurrent.ConcurrentHashMap; import com.comphenix.protocol.ProtocolLibrary; import com.comphenix.protocol.injector.PacketConstructor.Unwrapper; +import com.comphenix.protocol.reflect.FieldUtils; import com.comphenix.protocol.reflect.instances.DefaultInstances; +import com.google.common.primitives.Primitives; /** * Represents an object capable of converting wrapped Bukkit objects into NMS objects. @@ -39,41 +42,33 @@ import com.comphenix.protocol.reflect.instances.DefaultInstances; * @author Kristian */ public class BukkitUnwrapper implements Unwrapper { + private static Map, Unwrapper> unwrapperCache = new ConcurrentHashMap, Unwrapper>(); - private static Map, Method> cache = new ConcurrentHashMap, Method>(); - @SuppressWarnings("unchecked") @Override public Object unwrapItem(Object wrappedObject) { - - // Special cases - if (wrappedObject == null) { + // Special case + if (wrappedObject == null) return null; - } else if (wrappedObject instanceof Collection) { - return handleCollection((Collection) wrappedObject); - } - Class currentClass = wrappedObject.getClass(); - Method cachedMethod = initializeCache(currentClass); - try { - // Retrieve the handle - if (cachedMethod != null) - return cachedMethod.invoke(wrappedObject); - else - return null; - - } catch (IllegalArgumentException e) { - // Impossible + // Next, check for types that doesn't have a getHandle() + if (wrappedObject instanceof Collection) { + return handleCollection((Collection) wrappedObject); + } else if (Primitives.isWrapperType(currentClass) || wrappedObject instanceof String) { return null; - } catch (IllegalAccessException e) { - return null; - } catch (InvocationTargetException e) { - // This is REALLY bad - throw new RuntimeException("Minecraft error.", e); } + + Unwrapper specificUnwrapper = getSpecificUnwrapper(currentClass); + + // Retrieve the handle + if (specificUnwrapper != null) + return specificUnwrapper.unwrapItem(wrappedObject); + else + return null; } + // Handle a collection of items private Object handleCollection(Collection wrappedObject) { @SuppressWarnings("unchecked") @@ -92,26 +87,94 @@ public class BukkitUnwrapper implements Unwrapper { } } - private Method initializeCache(Class type) { - + /** + * Retrieve a cached class unwrapper for the given class. + * @param type - the type of the class. + * @return An unwrapper for the given class. + */ + private Unwrapper getSpecificUnwrapper(Class type) { // See if we're already determined this - if (cache.containsKey(type)) { + if (unwrapperCache.containsKey(type)) { // We will never remove from the cache, so this ought to be thread safe - return cache.get(type); + return unwrapperCache.get(type); } try { - Method find = type.getMethod("getHandle"); + final Method find = type.getMethod("getHandle"); // It's thread safe, as getMethod should return the same handle - cache.put(type, find); - return find; + Unwrapper methodUnwrapper = new Unwrapper() { + @Override + public Object unwrapItem(Object wrappedObject) { + + try { + return find.invoke(wrappedObject); + + } catch (IllegalArgumentException e) { + ProtocolLibrary.getErrorReporter().reportDetailed( + this, "Illegal argument.", e, wrappedObject, find); + } catch (IllegalAccessException e) { + // Should not occur either + return null; + } catch (InvocationTargetException e) { + // This is really bad + throw new RuntimeException("Minecraft error.", e); + } + + return null; + } + }; + + unwrapperCache.put(type, methodUnwrapper); + return methodUnwrapper; } catch (SecurityException e) { - ProtocolLibrary.getErrorReporter().reportDetailed(this, "Security limitation.", e, type); - return null; + ProtocolLibrary.getErrorReporter().reportDetailed(this, "Security limitation.", e, type.getName()); } catch (NoSuchMethodException e) { - ProtocolLibrary.getErrorReporter().reportDetailed(this, "Cannot unwrap object.", e, type); + // Try getting the field unwrapper too + Unwrapper fieldUnwrapper = getFieldUnwrapper(type); + + if (fieldUnwrapper != null) + return fieldUnwrapper; + else + ProtocolLibrary.getErrorReporter().reportDetailed(this, "Cannot find method.", e, type.getName()); + } + + // Default method + return null; + } + + /** + * Retrieve a cached unwrapper using the handle field. + * @param type - a cached field unwrapper. + * @return The cached field unwrapper. + */ + private Unwrapper getFieldUnwrapper(Class type) { + final Field find = FieldUtils.getField(type, "handle", true); + + // See if we succeeded + if (find != null) { + Unwrapper fieldUnwrapper = new Unwrapper() { + @Override + public Object unwrapItem(Object wrappedObject) { + try { + return FieldUtils.readField(find, wrappedObject, true); + } catch (IllegalAccessException e) { + ProtocolLibrary.getErrorReporter().reportDetailed( + this, "Cannot read field 'handle'.", e, wrappedObject, find.getName()); + return null; + } + } + }; + + unwrapperCache.put(type, fieldUnwrapper); + return fieldUnwrapper; + + } else { + // Inform about this too + ProtocolLibrary.getErrorReporter().reportDetailed( + this, "Could not find field 'handle'.", + new Exception("Unable to find 'handle'"), type.getName()); return null; } } From 5a96a63ff85aaba91c061285e801041e9dcaa5ec Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Fri, 21 Dec 2012 00:27:13 +0100 Subject: [PATCH 15/37] Added support for 1.4.6. Yes, it's essentially a hack, but I don't have enough time today to find a better way. It will have to do until 1.5.0 is out. --- .../player/InjectedServerConnection.java | 3 +- .../injector/player/NetLoginInjector.java | 8 +- .../player/NetworkServerInjector.java | 3 +- .../injector/player/PlayerInjector.java | 16 ++-- .../protocol/utility/CachedPackage.java | 10 +++ .../protocol/utility/MinecraftReflection.java | 90 ++++++++++++++++++- 6 files changed, 118 insertions(+), 12 deletions(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedServerConnection.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedServerConnection.java index 1d94a9c9..1886cac9 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedServerConnection.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedServerConnection.java @@ -77,7 +77,8 @@ class InjectedServerConnection { return; if (minecraftServerField == null) - minecraftServerField = FuzzyReflection.fromObject(server, true).getFieldByType(".*MinecraftServer"); + minecraftServerField = FuzzyReflection.fromObject(server, true). + getFieldByType("MinecraftServer", MinecraftReflection.getMinecraftServerClass()); try { minecraftServer = FieldUtils.readField(minecraftServerField, server, true); diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetLoginInjector.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetLoginInjector.java index 64e2f005..9ac2f5f8 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetLoginInjector.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetLoginInjector.java @@ -8,6 +8,7 @@ import org.bukkit.entity.Player; import com.comphenix.protocol.error.ErrorReporter; import com.comphenix.protocol.injector.GamePhase; import com.comphenix.protocol.injector.player.TemporaryPlayerFactory.InjectContainer; +import com.comphenix.protocol.utility.MinecraftReflection; import com.google.common.collect.Maps; /** @@ -62,9 +63,9 @@ class NetLoginInjector { } catch (Throwable e) { // Minecraft can't handle this, so we'll deal with it here - reporter.reportDetailed(this, "Unable to hook NetLoginHandler.", e, inserting); + reporter.reportDetailed(this, "Unable to hook " + + MinecraftReflection.getNetLoginHandlerName() + ".", e, inserting); return inserting; - } } @@ -103,7 +104,8 @@ class NetLoginInjector { } catch (Throwable e) { // Don't leak this to Minecraft - reporter.reportDetailed(this, "Cannot cleanup NetLoginHandler.", e, removing); + reporter.reportDetailed(this, "Cannot cleanup " + + MinecraftReflection.getNetLoginHandlerName() + ".", e, removing); } } } diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetworkServerInjector.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetworkServerInjector.java index a7b42de2..eacc83de 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetworkServerInjector.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetworkServerInjector.java @@ -141,7 +141,8 @@ public class NetworkServerInjector extends PlayerInjector { } throw new RuntimeException( - "Cannot hook player: Unable to find a valid constructor for the NetServerHandler object."); + "Cannot hook player: Unable to find a valid constructor for the " + + MinecraftReflection.getNetServerHandlerClass().getName() + " object."); } } diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/PlayerInjector.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/PlayerInjector.java index 239f6b7a..25d683c6 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/PlayerInjector.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/PlayerInjector.java @@ -156,7 +156,8 @@ abstract class PlayerInjector { // Retrieve the server handler if (serverHandlerField == null) { - serverHandlerField = FuzzyReflection.fromObject(notchEntity).getFieldByType(".*NetServerHandler"); + serverHandlerField = FuzzyReflection.fromObject(notchEntity).getFieldByType( + "NetServerHandler", MinecraftReflection.getNetServerHandlerClass()); proxyServerField = getProxyField(notchEntity, serverHandlerField); } @@ -166,7 +167,8 @@ abstract class PlayerInjector { // Next, get the network manager if (networkManagerField == null) - networkManagerField = FuzzyReflection.fromObject(serverHandler).getFieldByType(".*NetworkManager"); + networkManagerField = FuzzyReflection.fromObject(serverHandler). + getFieldByType(".*" + MinecraftReflection.getNetworkManagerName()); initializeNetworkManager(networkManagerField, serverHandler); } } @@ -181,7 +183,8 @@ abstract class PlayerInjector { loginHandler = netLoginHandler; if (netLoginNetworkField == null) - netLoginNetworkField = FuzzyReflection.fromObject(netLoginHandler).getFieldByType(".*NetworkManager"); + netLoginNetworkField = FuzzyReflection.fromObject(netLoginHandler). + getFieldByType(".*" + MinecraftReflection.getNetworkManagerName()); initializeNetworkManager(netLoginNetworkField, netLoginHandler); } } @@ -341,7 +344,7 @@ abstract class PlayerInjector { FuzzyReflection reflection = FuzzyReflection.fromObject(handler, true); // It might be - return reflection.getFieldByType(".*NetServerHandler"); + return reflection.getFieldByType("NetServerHandler", MinecraftReflection.getNetServerHandlerClass()); } catch (RuntimeException e) { // Damn @@ -367,7 +370,7 @@ abstract class PlayerInjector { try { if (netHandlerField == null) netHandlerField = FuzzyReflection.fromClass(networkManager.getClass(), true). - getFieldByType("net\\.minecraft\\.NetHandler"); + getFieldByType("NetHandler", MinecraftReflection.getNetHandlerClass()); } catch (RuntimeException e1) { // Swallow it } @@ -398,7 +401,8 @@ abstract class PlayerInjector { */ private Object getEntityPlayer(Object netHandler) throws IllegalAccessException { if (entityPlayerField == null) - entityPlayerField = FuzzyReflection.fromObject(netHandler).getFieldByType(".*EntityPlayer"); + entityPlayerField = FuzzyReflection.fromObject(netHandler).getFieldByType( + "EntityPlayer", MinecraftReflection.getEntityPlayerClass()); return FieldUtils.readField(entityPlayerField, netHandler); } diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/CachedPackage.java b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/CachedPackage.java index e69f956c..7d552fb7 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/CachedPackage.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/CachedPackage.java @@ -18,6 +18,16 @@ class CachedPackage { this.cache = Maps.newConcurrentMap(); } + /** + * Associate a given class with a class name. + * @param className - class name. + * @param clazz - type of class. + */ + @SuppressWarnings("rawtypes") + public void setPackageClass(String className, Class clazz) { + cache.put(className, clazz); + } + /** * Retrieve the class object of a specific class in the current package. * @param className - the specific class. diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java index e21be948..15ba850f 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java @@ -11,6 +11,7 @@ import org.bukkit.Server; import org.bukkit.inventory.ItemStack; import com.comphenix.protocol.injector.BukkitUnwrapper; +import com.google.common.base.Joiner; /** * Methods and constants specifically used in conjuction with reflecting Minecraft object. @@ -289,7 +290,31 @@ public class MinecraftReflection { */ @SuppressWarnings("rawtypes") public static Class getNetLoginHandlerClass() { - return getMinecraftClass("NetLoginHandler"); + return getMinecraftClass("NetLoginHandler", "PendingConnection"); + } + + /** + * Retrieve the NetServerHandler class. + * @return The NetServerHandler class. + */ + public static Class getNetServerHandlerClass() { + return getMinecraftClass("NetServerHandler", "PlayerConnection"); + } + + /** + * Retrieve the NetworkManager class. + * @return The NetworkManager class. + */ + public static Class getNetworkManagerClass() { + return getMinecraftClass("NetworkManager"); + } + + /** + * Retrieve the NetHandler class. + * @return The NetHandler class. + */ + public static Class getNetHandlerClass() { + return getMinecraftClass("NetHandler"); } /** @@ -310,6 +335,14 @@ public class MinecraftReflection { return getMinecraftClass("WorldType"); } + /** + * Retrieve the MinecraftServer class. + * @return MinecraftServer class. + */ + public static Class getMinecraftServerClass() { + return getMinecraftClass("MinecraftServer"); + } + /** * Retrieve the DataWatcher class. * @return The DataWatcher class. @@ -516,4 +549,59 @@ public class MinecraftReflection { minecraftPackage = new CachedPackage(getMinecraftPackage()); return minecraftPackage.getPackageClass(className); } + + /** + * Retrieve the first class that matches a specified Minecraft name. + * @param classes - the specific Minecraft class. + * @return Class object. + * @throws RuntimeException If we are unable to find any of the given classes. + */ + @SuppressWarnings("rawtypes") + public static Class getMinecraftClass(String className, String... aliases) { + try { + // Try the main class first + return getMinecraftClass(className); + } catch (RuntimeException e1) { + Class success = null; + + // Try every alias too + for (String alias : aliases) { + try { + success = getMinecraftClass(alias); + break; + } catch (RuntimeException e2) { + // Swallov + } + } + + if (success != null) { + // Save it for later + minecraftPackage.setPackageClass(className, success); + return success; + } else { + // Hack failed + throw new RuntimeException( + String.format("Unable to find %s (%s)", + className, + Joiner.on(", ").join(aliases)) + ); + } + } + } + + /** + * Dynamically retrieve the NetworkManager name. + * @return Name of the NetworkManager class. + */ + public static String getNetworkManagerName() { + return getNetworkManagerClass().getSimpleName(); + } + + /** + * Dynamically retrieve the name of the current NetLoginHandler. + * @return Name of the NetLoginHandler class. + */ + public static String getNetLoginHandlerName() { + return getNetLoginHandlerClass().getSimpleName(); + } } From c71bf615119017ef040a4c3baad065b0dfa238d4 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Fri, 21 Dec 2012 00:36:04 +0100 Subject: [PATCH 16/37] 1.4.6 appears to be working properly. --- .../src/main/java/com/comphenix/protocol/ProtocolLibrary.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolLibrary.java b/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolLibrary.java index 3cf0e1ab..e82f47ff 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolLibrary.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolLibrary.java @@ -52,7 +52,7 @@ public class ProtocolLibrary extends JavaPlugin { /** * The maximum version ProtocolLib has been tested with, */ - private static final String MAXIMUM_MINECRAFT_VERSION = "1.4.5"; + private static final String MAXIMUM_MINECRAFT_VERSION = "1.4.6"; /** * The number of milliseconds per second. From 35083f87fe38434877c24c88fb3c0d45be7e1928 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Fri, 21 Dec 2012 00:37:46 +0100 Subject: [PATCH 17/37] Increment to version 1.8.3-SNAPSHOT. --- ProtocolLib/pom.xml | 2 +- ProtocolLib/src/main/resources/plugin.yml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/ProtocolLib/pom.xml b/ProtocolLib/pom.xml index 252339c9..23377c48 100644 --- a/ProtocolLib/pom.xml +++ b/ProtocolLib/pom.xml @@ -2,7 +2,7 @@ 4.0.0 com.comphenix.protocol ProtocolLib - 1.8.2-SNAPSHOT + 1.8.3-SNAPSHOT jar Provides read/write access to the Minecraft protocol. diff --git a/ProtocolLib/src/main/resources/plugin.yml b/ProtocolLib/src/main/resources/plugin.yml index f990e825..34772ea8 100644 --- a/ProtocolLib/src/main/resources/plugin.yml +++ b/ProtocolLib/src/main/resources/plugin.yml @@ -1,5 +1,5 @@ name: ProtocolLib -version: 1.8.2-SNAPSHOT +version: 1.8.3-SNAPSHOT description: Provides read/write access to the Minecraft protocol. author: Comphenix website: http://www.comphenix.net/ProtocolLib From 555b5d0795b8b541d50614dc31275b8c44aa654b Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Fri, 21 Dec 2012 18:33:09 +0100 Subject: [PATCH 18/37] Create a typed ItemStack array when writing it to a field. FIXES 24. After the package versioning commit, ProtocolLib switched to performing everything exclusively with reflection. Unfortunately, this may have introduced a couple of new bugs. This problem is caused by the fact that ProtocolLib converts a Bukkit ItemStack array to an Object array instead of an array of NMS ItemStacks. This object array can't be assigned to a ItemStack array, so the write operation fails. The fix is to create a typed array with reflection. --- .../com/comphenix/protocol/events/PacketContainer.java | 7 ++++--- .../comphenix/protocol/utility/MinecraftReflection.java | 6 +++--- 2 files changed, 7 insertions(+), 6 deletions(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java b/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java index a7b4a265..373c7d98 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java @@ -25,6 +25,7 @@ import java.io.IOException; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.io.Serializable; +import java.lang.reflect.Array; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.util.Collection; @@ -241,12 +242,12 @@ public class PacketContainer implements Serializable { BukkitConverters.getIgnoreNull(new EquivalentConverter() { public Object getGeneric(ClassgenericType, ItemStack[] specific) { - Object[] result = new Object[specific.length]; + Class nmsStack = MinecraftReflection.getItemStackClass(); + Object[] result = (Object[]) Array.newInstance(nmsStack, specific.length); // Unwrap every item for (int i = 0; i < result.length; i++) { - result[i] = stackConverter.getGeneric( - MinecraftReflection.getItemStackClass(), specific[i]); + result[i] = stackConverter.getGeneric(nmsStack, specific[i]); } return result; } diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java index 15ba850f..bf6ffc20 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java @@ -314,12 +314,12 @@ public class MinecraftReflection { * @return The NetHandler class. */ public static Class getNetHandlerClass() { - return getMinecraftClass("NetHandler"); + return getMinecraftClass("NetHandler", "Connection"); } /** - * Retrieve the NetLoginHandler class. - * @return The NetLoginHandler class. + * Retrieve the NMS ItemStack class. + * @return The ItemStack class. */ @SuppressWarnings("rawtypes") public static Class getItemStackClass() { From 5d06b9e84c2a8b85cf91c607f1f8ee4a6582e54d Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Fri, 21 Dec 2012 18:33:40 +0100 Subject: [PATCH 19/37] Bumping to version 1.8.4-SNAPSHOT. --- ProtocolLib/dependency-reduced-pom.xml | 2 +- ProtocolLib/pom.xml | 2 +- ProtocolLib/src/main/resources/plugin.yml | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/ProtocolLib/dependency-reduced-pom.xml b/ProtocolLib/dependency-reduced-pom.xml index 1be13b08..ed3a2888 100644 --- a/ProtocolLib/dependency-reduced-pom.xml +++ b/ProtocolLib/dependency-reduced-pom.xml @@ -4,7 +4,7 @@ com.comphenix.protocol ProtocolLib ProtocolLib - 1.8.2-SNAPSHOT + 1.8.3-SNAPSHOT Provides read/write access to the Minecraft protocol. http://dev.bukkit.org/server-mods/protocollib/ diff --git a/ProtocolLib/pom.xml b/ProtocolLib/pom.xml index 23377c48..2a29936f 100644 --- a/ProtocolLib/pom.xml +++ b/ProtocolLib/pom.xml @@ -2,7 +2,7 @@ 4.0.0 com.comphenix.protocol ProtocolLib - 1.8.3-SNAPSHOT + 1.8.4-SNAPSHOT jar Provides read/write access to the Minecraft protocol. diff --git a/ProtocolLib/src/main/resources/plugin.yml b/ProtocolLib/src/main/resources/plugin.yml index 34772ea8..04d9f25e 100644 --- a/ProtocolLib/src/main/resources/plugin.yml +++ b/ProtocolLib/src/main/resources/plugin.yml @@ -1,5 +1,5 @@ name: ProtocolLib -version: 1.8.3-SNAPSHOT +version: 1.8.4-SNAPSHOT description: Provides read/write access to the Minecraft protocol. author: Comphenix website: http://www.comphenix.net/ProtocolLib From d328111b38d16850e02df3db2678fea2f8b0818d Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Sat, 22 Dec 2012 23:16:21 +0100 Subject: [PATCH 20/37] No need to suppress any warnings - we can use wildcards. --- ProtocolLib/dependency-reduced-pom.xml | 2 +- .../protocol/utility/CachedPackage.java | 8 +-- .../protocol/utility/MinecraftReflection.java | 61 +++++-------------- 3 files changed, 20 insertions(+), 51 deletions(-) diff --git a/ProtocolLib/dependency-reduced-pom.xml b/ProtocolLib/dependency-reduced-pom.xml index ed3a2888..4c88f2ae 100644 --- a/ProtocolLib/dependency-reduced-pom.xml +++ b/ProtocolLib/dependency-reduced-pom.xml @@ -4,7 +4,7 @@ com.comphenix.protocol ProtocolLib ProtocolLib - 1.8.3-SNAPSHOT + 1.8.4-SNAPSHOT Provides read/write access to the Minecraft protocol. http://dev.bukkit.org/server-mods/protocollib/ diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/CachedPackage.java b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/CachedPackage.java index 7d552fb7..ba4a19f8 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/CachedPackage.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/CachedPackage.java @@ -23,8 +23,7 @@ class CachedPackage { * @param className - class name. * @param clazz - type of class. */ - @SuppressWarnings("rawtypes") - public void setPackageClass(String className, Class clazz) { + public void setPackageClass(String className, Class clazz) { cache.put(className, clazz); } @@ -34,10 +33,9 @@ class CachedPackage { * @return Class object. * @throws RuntimeException If we are unable to find the given class. */ - @SuppressWarnings("rawtypes") - public Class getPackageClass(String className) { + public Class getPackageClass(String className) { try { - Class result = cache.get(className); + Class result = cache.get(className); // Concurrency is not a problem - we don't care if we look up a class twice if (result == null) { diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java index bf6ffc20..027d3f6a 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java @@ -162,7 +162,6 @@ public class MinecraftReflection { * @param obj - the object to test. * @return TRUE if it can, FALSE otherwise. */ - @SuppressWarnings("unchecked") public static boolean isChunkPosition(Object obj) { return getChunkPositionClass().isAssignableFrom(obj.getClass()); } @@ -172,7 +171,6 @@ public class MinecraftReflection { * @param obj - the object to test. * @return TRUE if it can, FALSE otherwise. */ - @SuppressWarnings("unchecked") public static boolean isChunkCoordinates(Object obj) { return getChunkCoordinatesClass().isAssignableFrom(obj.getClass()); } @@ -182,7 +180,6 @@ public class MinecraftReflection { * @param obj - the given object. * @return TRUE if it is, FALSE otherwise. */ - @SuppressWarnings("unchecked") public static boolean isPacketClass(Object obj) { return getPacketClass().isAssignableFrom(obj.getClass()); } @@ -192,7 +189,6 @@ public class MinecraftReflection { * @param obj - the given object. * @return TRUE if it is, FALSE otherwise. */ - @SuppressWarnings("unchecked") public static boolean isLoginHandler(Object obj) { return getNetLoginHandlerClass().isAssignableFrom(obj.getClass()); } @@ -202,7 +198,6 @@ public class MinecraftReflection { * @param obj - the given object. * @return TRUE if it is, FALSE otherwise. */ - @SuppressWarnings("unchecked") public static boolean isMinecraftEntity(Object obj) { return getEntityClass().isAssignableFrom(obj.getClass()); } @@ -212,7 +207,6 @@ public class MinecraftReflection { * @param obj - the given object. * @return TRUE if it is, FALSE otherwise. */ - @SuppressWarnings("unchecked") public static boolean isItemStack(Object value) { return getItemStackClass().isAssignableFrom(value.getClass()); } @@ -222,7 +216,6 @@ public class MinecraftReflection { * @param obj - the given object. * @return TRUE if it is, FALSE otherwise. */ - @SuppressWarnings("unchecked") public static boolean isMinecraftPlayer(Object obj) { return getEntityPlayerClass().isAssignableFrom(obj.getClass()); } @@ -232,7 +225,6 @@ public class MinecraftReflection { * @param obj - the given object. * @return TRUE if it is, FALSE otherwise. */ - @SuppressWarnings("unchecked") public static boolean isWatchableObject(Object obj) { return getWatchableObjectClass().isAssignableFrom(obj.getClass()); } @@ -242,7 +234,6 @@ public class MinecraftReflection { * @param obj - the given object. * @return TRUE if it is, FALSE otherwise. */ - @SuppressWarnings("unchecked") public static boolean isDataWatcher(Object obj) { return getDataWatcherClass().isAssignableFrom(obj.getClass()); } @@ -252,7 +243,6 @@ public class MinecraftReflection { * @param obj - the given object. * @return TRUE if it is, FALSE otherwise. */ - @SuppressWarnings("unchecked") public static boolean isCraftItemStack(Object obj) { return getCraftItemStackClass().isAssignableFrom(obj.getClass()); } @@ -261,8 +251,7 @@ public class MinecraftReflection { * Retrieve the EntityPlayer (NMS) class. * @return The entity class. */ - @SuppressWarnings("rawtypes") - public static Class getEntityPlayerClass() { + public static Class getEntityPlayerClass() { return getMinecraftClass("EntityPlayer"); } @@ -270,8 +259,7 @@ public class MinecraftReflection { * Retrieve the entity (NMS) class. * @return The entity class. */ - @SuppressWarnings("rawtypes") - public static Class getEntityClass() { + public static Class getEntityClass() { return getMinecraftClass("Entity"); } @@ -279,8 +267,7 @@ public class MinecraftReflection { * Retrieve the packet class. * @return The packet class. */ - @SuppressWarnings("rawtypes") - public static Class getPacketClass() { + public static Class getPacketClass() { return getMinecraftClass("Packet"); } @@ -288,8 +275,7 @@ public class MinecraftReflection { * Retrieve the NetLoginHandler class. * @return The NetLoginHandler class. */ - @SuppressWarnings("rawtypes") - public static Class getNetLoginHandlerClass() { + public static Class getNetLoginHandlerClass() { return getMinecraftClass("NetLoginHandler", "PendingConnection"); } @@ -321,8 +307,7 @@ public class MinecraftReflection { * Retrieve the NMS ItemStack class. * @return The ItemStack class. */ - @SuppressWarnings("rawtypes") - public static Class getItemStackClass() { + public static Class getItemStackClass() { return getMinecraftClass("ItemStack"); } @@ -330,8 +315,7 @@ public class MinecraftReflection { * Retrieve the WorldType class. * @return The WorldType class. */ - @SuppressWarnings("rawtypes") - public static Class getWorldTypeClass() { + public static Class getWorldTypeClass() { return getMinecraftClass("WorldType"); } @@ -347,8 +331,7 @@ public class MinecraftReflection { * Retrieve the DataWatcher class. * @return The DataWatcher class. */ - @SuppressWarnings("rawtypes") - public static Class getDataWatcherClass() { + public static Class getDataWatcherClass() { return getMinecraftClass("DataWatcher"); } @@ -356,8 +339,7 @@ public class MinecraftReflection { * Retrieve the ChunkPosition class. * @return The ChunkPosition class. */ - @SuppressWarnings("rawtypes") - public static Class getChunkPositionClass() { + public static Class getChunkPositionClass() { return getMinecraftClass("ChunkPosition"); } @@ -365,8 +347,7 @@ public class MinecraftReflection { * Retrieve the ChunkPosition class. * @return The ChunkPosition class. */ - @SuppressWarnings("rawtypes") - public static Class getChunkCoordinatesClass() { + public static Class getChunkCoordinatesClass() { return getMinecraftClass("ChunkCoordinates"); } @@ -374,8 +355,7 @@ public class MinecraftReflection { * Retrieve the WatchableObject class. * @return The WatchableObject class. */ - @SuppressWarnings("rawtypes") - public static Class getWatchableObjectClass() { + public static Class getWatchableObjectClass() { return getMinecraftClass("WatchableObject"); } @@ -383,8 +363,7 @@ public class MinecraftReflection { * Retrieve the ItemStack[] class. * @return The ItemStack[] class. */ - @SuppressWarnings("rawtypes") - public static Class getItemStackArrayClass() { + public static Class getItemStackArrayClass() { if (itemStackArrayClass == null) itemStackArrayClass = getArrayClass(getItemStackClass()); return itemStackArrayClass; @@ -395,8 +374,7 @@ public class MinecraftReflection { * @param componentType - type of each element in the array. * @return The class of the array. */ - @SuppressWarnings("rawtypes") - public static Class getArrayClass(Class componentType) { + public static Class getArrayClass(Class componentType) { // Bit of a hack, but it works return Array.newInstance(componentType, 0).getClass(); } @@ -405,8 +383,7 @@ public class MinecraftReflection { * Retrieve the CraftItemStack class. * @return The CraftItemStack class. */ - @SuppressWarnings("rawtypes") - public static Class getCraftItemStackClass() { + public static Class getCraftItemStackClass() { return getCraftBukkitClass("inventory.CraftItemStack"); } @@ -415,7 +392,6 @@ public class MinecraftReflection { * @param bukkitItemStack - the Bukkit ItemStack to convert. * @return A CraftItemStack as an ItemStack. */ - @SuppressWarnings("unchecked") public static ItemStack getBukkitItemStack(ItemStack bukkitItemStack) { // Delegate this task to the method that can execute it if (craftBukkitMethod != null) @@ -441,7 +417,6 @@ public class MinecraftReflection { } } - @SuppressWarnings("unchecked") private static ItemStack getBukkitItemByMethod(ItemStack bukkitItemStack) { if (craftBukkitMethod == null) { try { @@ -465,7 +440,6 @@ public class MinecraftReflection { * @param minecraftItemStack - the NMS ItemStack to wrap. * @return The wrapped ItemStack. */ - @SuppressWarnings("unchecked") public static ItemStack getBukkitItemStack(Object minecraftItemStack) { // Delegate this task to the method that can execute it if (craftNMSMethod != null) @@ -491,7 +465,6 @@ public class MinecraftReflection { } } - @SuppressWarnings("unchecked") private static ItemStack getBukkitItemByMethod(Object minecraftItemStack) { if (craftNMSMethod == null) { try { @@ -543,8 +516,7 @@ public class MinecraftReflection { * @return Class object. * @throws RuntimeException If we are unable to find the given class. */ - @SuppressWarnings("rawtypes") - public static Class getMinecraftClass(String className) { + public static Class getMinecraftClass(String className) { if (minecraftPackage == null) minecraftPackage = new CachedPackage(getMinecraftPackage()); return minecraftPackage.getPackageClass(className); @@ -556,13 +528,12 @@ public class MinecraftReflection { * @return Class object. * @throws RuntimeException If we are unable to find any of the given classes. */ - @SuppressWarnings("rawtypes") - public static Class getMinecraftClass(String className, String... aliases) { + public static Class getMinecraftClass(String className, String... aliases) { try { // Try the main class first return getMinecraftClass(className); } catch (RuntimeException e1) { - Class success = null; + Class success = null; // Try every alias too for (String alias : aliases) { From 450fbd8e8d1ef55026cbaf927e2b9a51269caec8 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Sun, 23 Dec 2012 04:22:02 +0100 Subject: [PATCH 21/37] Bumping to beta 1.4.6 of CraftBukkit --- ProtocolLib/pom.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ProtocolLib/pom.xml b/ProtocolLib/pom.xml index 2a29936f..1c8bec49 100644 --- a/ProtocolLib/pom.xml +++ b/ProtocolLib/pom.xml @@ -202,7 +202,7 @@ org.bukkit craftbukkit - 1.4.5-R0.3-SNAPSHOT + 1.4.6-R0.1 provided From cc3441dfeb1b7eb819497b28931ee697a631f198 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Sun, 23 Dec 2012 18:52:00 +0100 Subject: [PATCH 22/37] Continue to remove unnecessary raw types. --- .../com/comphenix/protocol/wrappers/BukkitConverters.java | 4 +--- .../java/com/comphenix/protocol/wrappers/ChunkPosition.java | 1 - .../comphenix/protocol/wrappers/WrappedWatchableObject.java | 5 ++--- 3 files changed, 3 insertions(+), 7 deletions(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/BukkitConverters.java b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/BukkitConverters.java index 7b9bfe49..d8da3e31 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/BukkitConverters.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/BukkitConverters.java @@ -155,7 +155,6 @@ public class BukkitConverters { return null; return getIgnoreNull(new EquivalentConverter() { - @SuppressWarnings("unchecked") @Override public Object getGeneric(Class genericType, WorldType specific) { try { @@ -169,8 +168,7 @@ public class BukkitConverters { throw new FieldAccessException("Cannot find the WorldType.getType() method.", e); } } - - @SuppressWarnings("unchecked") + @Override public WorldType getSpecific(Object generic) { try { diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/ChunkPosition.java b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/ChunkPosition.java index f9e60fa1..718f227f 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/ChunkPosition.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/ChunkPosition.java @@ -133,7 +133,6 @@ public class ChunkPosition { */ public static EquivalentConverter getConverter() { return new EquivalentConverter() { - @SuppressWarnings("unchecked") @Override public Object getGeneric(Class genericType, ChunkPosition specific) { if (chunkPositionConstructor == null) { diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedWatchableObject.java b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedWatchableObject.java index c3af7039..003385d1 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedWatchableObject.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedWatchableObject.java @@ -49,7 +49,6 @@ public class WrappedWatchableObject { * @param index - the index. * @param value - non-null value of specific types. */ - @SuppressWarnings("unchecked") public WrappedWatchableObject(int index, Object value) { if (value == null) throw new IllegalArgumentException("Value cannot be NULL."); @@ -296,8 +295,8 @@ public class WrappedWatchableObject { * @throws FieldAccessException If we're unable to use reflection. */ public WrappedWatchableObject deepClone() throws FieldAccessException { - @SuppressWarnings("unchecked") - WrappedWatchableObject clone = new WrappedWatchableObject(DefaultInstances.DEFAULT.getDefault(MinecraftReflection.getWatchableObjectClass())); + WrappedWatchableObject clone = new WrappedWatchableObject( + DefaultInstances.DEFAULT.getDefault(MinecraftReflection.getWatchableObjectClass())); clone.setDirtyState(getDirtyState()); clone.setIndex(getIndex()); From 425c5fd682653edd1d281658c37677f3026cfb4e Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Mon, 24 Dec 2012 21:42:48 +0100 Subject: [PATCH 23/37] Add equals() and hashcode() to the wrapped DataWatcher. --- .../protocol/wrappers/WrappedDataWatcher.java | 31 +++++++++++++++++++ 1 file changed, 31 insertions(+) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedDataWatcher.java b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedDataWatcher.java index 90eeb283..d1b45bae 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedDataWatcher.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedDataWatcher.java @@ -275,6 +275,37 @@ public class WrappedDataWatcher implements Iterable { } } + @Override + public boolean equals(Object obj) { + // Quick checks + if (obj == this) + return true; + if (obj == null) + return false; + + if (obj instanceof WrappedDataWatcher) { + WrappedDataWatcher other = (WrappedDataWatcher) obj; + Iterator first = iterator(), second = other.iterator(); + + // Make sure they're the same size + if (size() != other.size()) + return false; + + for (; first.hasNext() && second.hasNext(); ) { + // See if the two elements are equal + if (!first.next().equals(second.next())) + return false; + } + return true; + } + return false; + } + + @Override + public int hashCode() { + return getWatchableObjects().hashCode(); + } + /** * Retrieve a copy of every index associated with a watched object. * @return Every watched object index. From 655fbf479915d61da0a8b6a5505f9e145b5b31a8 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Tue, 25 Dec 2012 15:44:14 +0100 Subject: [PATCH 24/37] Added StructureModifier tests for different packet types. --- ProtocolLib/pom.xml | 18 + .../protocol/events/PacketContainerTest.java | 340 ++++++++++++++++++ 2 files changed, 358 insertions(+) create mode 100644 ProtocolLib/src/test/java/com/comphenix/protocol/events/PacketContainerTest.java diff --git a/ProtocolLib/pom.xml b/ProtocolLib/pom.xml index 1c8bec49..4fa87e38 100644 --- a/ProtocolLib/pom.xml +++ b/ProtocolLib/pom.xml @@ -8,6 +8,7 @@ cp1252 + 1.5 @@ -211,5 +212,22 @@ 4.10 test + + org.mockito + mockito-all + 1.8.4 + + + org.powermock + powermock-module-junit4 + ${powermock.version} + test + + + org.powermock + powermock-api-mockito + ${powermock.version} + test + \ No newline at end of file diff --git a/ProtocolLib/src/test/java/com/comphenix/protocol/events/PacketContainerTest.java b/ProtocolLib/src/test/java/com/comphenix/protocol/events/PacketContainerTest.java new file mode 100644 index 00000000..ce907afe --- /dev/null +++ b/ProtocolLib/src/test/java/com/comphenix/protocol/events/PacketContainerTest.java @@ -0,0 +1,340 @@ +package com.comphenix.protocol.events; + +import static org.junit.Assert.*; +import static org.mockito.Matchers.any; +import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.when; + +import java.util.List; + +// Will have to be updated for every version though +import org.bukkit.craftbukkit.v1_4_6.inventory.CraftItemFactory; + +import org.bukkit.Bukkit; +import org.bukkit.Material; +import org.bukkit.Server; +import org.bukkit.WorldType; +import org.bukkit.inventory.ItemFactory; +import org.bukkit.inventory.ItemStack; +import org.bukkit.inventory.meta.ItemMeta; +import org.junit.BeforeClass; +import org.junit.Test; +import org.junit.runner.RunWith; +import org.powermock.core.classloader.annotations.PrepareForTest; + +import com.comphenix.protocol.Packets; +import com.comphenix.protocol.reflect.FieldUtils; +import com.comphenix.protocol.reflect.StructureModifier; +import com.comphenix.protocol.utility.MinecraftReflection; +import com.comphenix.protocol.wrappers.ChunkPosition; +import com.comphenix.protocol.wrappers.WrappedDataWatcher; +import com.comphenix.protocol.wrappers.WrappedWatchableObject; +import com.google.common.collect.Iterables; +import com.google.common.collect.Lists; + +// Ensure that the CraftItemFactory is mockable +@RunWith(org.powermock.modules.junit4.PowerMockRunner.class) +@PrepareForTest(CraftItemFactory.class) +public class PacketContainerTest { + + @BeforeClass + public static void initializeBukkit() throws IllegalAccessException { + // Initialize reflection + MinecraftReflection.setMinecraftPackage("net.minecraft.server.v1_4_6", "org.bukkit.craftbukkit.v1_4_6"); + + // Mock the server object + Server mockedServer = mock(Server.class); + ItemFactory mockedFactory = mock(CraftItemFactory.class); + ItemMeta mockedMeta = mock(ItemMeta.class); + + when(mockedServer.getItemFactory()).thenReturn(mockedFactory); + when(mockedFactory.getItemMeta(any(Material.class))).thenReturn(mockedMeta); + + // Inject this fake server + FieldUtils.writeStaticField(Bukkit.class, "server", mockedServer, true); + + // And the fake item factory + FieldUtils.writeStaticField(CraftItemFactory.class, "instance", mockedFactory, true); + } + + private void testPrimitive(StructureModifier modifier, int index, T initialValue, T testValue) { + // Check initial value + assertEquals(initialValue, modifier.read(index)); + + // Test assignment + modifier.write(index, testValue); + assertEquals(testValue, modifier.read(0)); + } + + private void testObjectArray(StructureModifier modifier, int index, T[] initialValue, T[] testValue) { + // Check initial value + assertNull(modifier.read(index)); + modifier.writeDefaults(); + + // Test initial + assertArrayEquals(initialValue, modifier.read(index)); + + // Test assignment + modifier.write(index, testValue); + assertArrayEquals(testValue, modifier.read(0)); + } + + @Test + public void testGetByteArrays() { + // Contains a byte array we will test + PacketContainer customPayload = new PacketContainer(Packets.Server.CUSTOM_PAYLOAD); + StructureModifier bytes = customPayload.getByteArrays(); + byte[] testArray = new byte[] { 1, 2, 3 }; + + // It's NULL at first + assertArrayEquals(null, (byte[]) bytes.read(0)); + customPayload.getModifier().writeDefaults(); + + // Then it should create an empty array + assertArrayEquals(new byte[0], (byte[]) bytes.read(0)); + + // Check and see if we can write to it + bytes.write(0, testArray); + assertArrayEquals(testArray, (byte[]) bytes.read(0)); + } + + @Test + public void testGetBytes() { + PacketContainer spawnMob = new PacketContainer(Packets.Server.MOB_SPAWN); + testPrimitive(spawnMob.getBytes(), 0, (byte)0, (byte)1); + } + + @Test + public void testGetShorts() { + PacketContainer itemData = new PacketContainer(Packets.Server.ITEM_DATA); + testPrimitive(itemData.getShorts(), 0, (short)0, (short)1); + } + + @Test + public void testGetIntegers() { + PacketContainer updateSign = new PacketContainer(Packets.Server.UPDATE_SIGN); + testPrimitive(updateSign.getIntegers(), 0, (int)0, (int)1); + } + + @Test + public void testGetLongs() { + PacketContainer updateTime = new PacketContainer(Packets.Server.UPDATE_TIME); + testPrimitive(updateTime.getLongs(), 0, (long)0, (long)1); + } + + @Test + public void testGetFloat() { + PacketContainer explosion = new PacketContainer(Packets.Server.EXPLOSION); + testPrimitive(explosion.getFloat(), 0, (float)0, (float)0.8); + } + + @Test + public void testGetDoubles() { + PacketContainer explosion = new PacketContainer(Packets.Server.EXPLOSION); + testPrimitive(explosion.getDoubles(), 0, (double)0, (double)0.8); + } + + @Test + public void testGetStrings() { + PacketContainer explosion = new PacketContainer(Packets.Server.CHAT); + testPrimitive(explosion.getStrings(), 0, null, "hello"); + } + + @Test + public void testGetStringArrays() { + PacketContainer explosion = new PacketContainer(Packets.Server.UPDATE_SIGN); + testObjectArray(explosion.getStringArrays(), 0, new String[0], new String[] { "hello", "world" }); + } + + @Test + public void testGetIntegerArrays() { + // Contains a byte array we will test + PacketContainer mapChunkBulk = new PacketContainer(Packets.Server.MAP_CHUNK_BULK); + StructureModifier integers = mapChunkBulk.getIntegerArrays(); + int[] testArray = new int[] { 1, 2, 3 }; + + // Pre and post conditions + assertArrayEquals(null, (int[]) integers.read(0)); + mapChunkBulk.getModifier().writeDefaults(); + assertArrayEquals(new int[0], (int[]) integers.read(0)); + + integers.write(0, testArray); + assertArrayEquals(testArray, (int[]) integers.read(0)); + } + + @Test + public void testGetItemModifier() { + PacketContainer windowClick = new PacketContainer(Packets.Client.WINDOW_CLICK); + + StructureModifier items = windowClick.getItemModifier(); + ItemStack goldAxe = new ItemStack(Material.GOLD_AXE); + + assertNull(items.read(0)); + + // Insert the goldaxe and check if it's there + items.write(0, goldAxe); + assertTrue(equivalentItem(goldAxe, items.read(0))); + } + + @Test + public void testGetItemArrayModifier() { + PacketContainer windowItems = new PacketContainer(Packets.Server.WINDOW_ITEMS); + StructureModifier itemAccess = windowItems.getItemArrayModifier(); + + ItemStack[] itemArray = new ItemStack[] { + new ItemStack(Material.GOLD_AXE), + new ItemStack(Material.DIAMOND_AXE) + }; + + assertNull(itemAccess.read(0)); + + // Insert and check that it was succesful + itemAccess.write(0, itemArray); + + // Read back array + ItemStack[] comparision = itemAccess.read(0); + assertEquals(itemArray.length, comparision.length); + + // Check that it is equivalent + for (int i = 0; i < itemArray.length; i++) { + assertTrue(String.format("Array element %s is not the same: %s != %s", + i, itemArray[i], comparision[i]), equivalentItem(itemArray[i], comparision[i])); + } + } + + private boolean equivalentItem(ItemStack first, ItemStack second) { + if (first == null) + return second == null; + else if (second == null) + return false; + else + return first.getType().equals(second.getType()); + } + + private boolean equivalentPacket(PacketContainer first, PacketContainer second) { + if (first == null) + return second == null; + else if (second == null) + return false; + else + return first.getModifier().getValues().equals(second.getModifier().getValues()); + } + + @Test + public void testGetWorldTypeModifier() { + PacketContainer loginPacket = new PacketContainer(Packets.Server.LOGIN); + StructureModifier worldAccess = loginPacket.getWorldTypeModifier(); + + WorldType testValue = WorldType.LARGE_BIOMES; + + assertNull(worldAccess.read(0)); + + // Insert and read back + worldAccess.write(0, testValue); + assertEquals(testValue, worldAccess.read(0)); + } + + @Test + public void testGetDataWatcherModifier() { + PacketContainer mobSpawnPacket = new PacketContainer(Packets.Server.MOB_SPAWN); + StructureModifier watcherAccessor = mobSpawnPacket.getDataWatcherModifier(); + + WrappedDataWatcher dataWatcher = new WrappedDataWatcher(); + dataWatcher.setObject(1, 100); + dataWatcher.setObject(2, 125); + + assertNull(watcherAccessor.read(0)); + + // Insert and read back + watcherAccessor.write(0, dataWatcher); + assertEquals(dataWatcher, watcherAccessor.read(0)); + } + + // Unfortunately, it might be too difficult to mock this one + // + // @Test + // public void testGetEntityModifier() { } + + // No packet expose this type directly. + // + // @Test + // public void testGetPositionModifier() { } + + @Test + public void testGetPositionCollectionModifier() { + PacketContainer explosionPacket = new PacketContainer(Packets.Server.EXPLOSION); + StructureModifier> positionAccessor = explosionPacket.getPositionCollectionModifier(); + + assertNull(positionAccessor.read(0)); + + List positions = Lists.newArrayList(); + positions.add(new ChunkPosition(1, 2, 3)); + positions.add(new ChunkPosition(3, 4, 5)); + + // Insert and read back + positionAccessor.write(0, positions); + assertEquals(positions, positionAccessor.read(0)); + } + + @Test + public void testGetWatchableCollectionModifier() { + PacketContainer entityMetadata = new PacketContainer(Packets.Server.ENTITY_METADATA); + StructureModifier> watchableAccessor = + entityMetadata.getWatchableCollectionModifier(); + + assertNull(watchableAccessor.read(0)); + + WrappedDataWatcher watcher = new WrappedDataWatcher(); + watcher.setObject(1, 10); + watcher.setObject(8, 10); + + List list = watcher.getWatchableObjects(); + + // Insert and read back + watchableAccessor.write(0, list); + assertEquals(list, watchableAccessor.read(0)); + } + + @Test + public void testDeepClone() { + // Try constructing all the packets + for (Integer id : Iterables.concat( + Packets.getClientRegistry().values(), + Packets.getServerRegistry().values() )) { + + // Whether or not this packet has been registered + boolean registered = Packets.Server.isSupported(id) || + Packets.Client.isSupported(id); + + try { + PacketContainer constructed = new PacketContainer(id); + + if (!registered) { + fail("Expected IllegalArgumentException(Packet " + id + " not registered"); + } + + // Make sure these packets contains fields as well + assertTrue("Constructed packet with no known fields (" + id + ")", + constructed.getModifier().size() > 0); + + // Initialize default values + constructed.getModifier().writeDefaults(); + + // Clone the packet + PacketContainer cloned = constructed.deepClone(); + + // Make sure they're equivalent + assertTrue("Packet " + id + " could not be cloned.", equivalentPacket(constructed, cloned)); + + } catch (IllegalArgumentException e) { + if (!registered) { + // Check the same + assertEquals(e.getMessage(), "The packet ID " + id + " is not registered."); + } else { + // Something is very wrong + throw e; + } + } + } + } +} From fdbd3d64955ba364f87bfc51d6dd3602ccd73743 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Tue, 25 Dec 2012 15:46:33 +0100 Subject: [PATCH 25/37] Rename ObjectCloner to ObjectWriter. It's more accurately thought off as a way of writing the content of object A to object B, than a generic cloning method. --- .../java/com/comphenix/protocol/CleanupStaticMembers.java | 4 ++-- .../protocol/injector/player/InjectedServerConnection.java | 4 ++-- .../protocol/injector/player/NetworkServerInjector.java | 4 ++-- .../reflect/{ObjectCloner.java => ObjectWriter.java} | 5 ++--- 4 files changed, 8 insertions(+), 9 deletions(-) rename ProtocolLib/src/main/java/com/comphenix/protocol/reflect/{ObjectCloner.java => ObjectWriter.java} (95%) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/CleanupStaticMembers.java b/ProtocolLib/src/main/java/com/comphenix/protocol/CleanupStaticMembers.java index f04aebdc..69d82099 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/CleanupStaticMembers.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/CleanupStaticMembers.java @@ -13,7 +13,7 @@ import com.comphenix.protocol.injector.BukkitUnwrapper; import com.comphenix.protocol.reflect.FieldUtils; import com.comphenix.protocol.reflect.FuzzyReflection; import com.comphenix.protocol.reflect.MethodUtils; -import com.comphenix.protocol.reflect.ObjectCloner; +import com.comphenix.protocol.reflect.ObjectWriter; import com.comphenix.protocol.reflect.compiler.BackgroundCompiler; import com.comphenix.protocol.reflect.compiler.StructureCompiler; import com.comphenix.protocol.reflect.instances.CollectionGenerator; @@ -48,7 +48,7 @@ class CleanupStaticMembers { BukkitUnwrapper.class, DefaultInstances.class, CollectionGenerator.class, PrimitiveGenerator.class, FuzzyReflection.class, MethodUtils.class, BackgroundCompiler.class, StructureCompiler.class, - ObjectCloner.class, Packets.Server.class, Packets.Client.class, + ObjectWriter.class, Packets.Server.class, Packets.Client.class, ChunkPosition.class, WrappedDataWatcher.class, WrappedWatchableObject.class }; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedServerConnection.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedServerConnection.java index 1886cac9..307e8b24 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedServerConnection.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedServerConnection.java @@ -29,7 +29,7 @@ import org.bukkit.Server; import com.comphenix.protocol.error.ErrorReporter; import com.comphenix.protocol.reflect.FieldUtils; import com.comphenix.protocol.reflect.FuzzyReflection; -import com.comphenix.protocol.reflect.ObjectCloner; +import com.comphenix.protocol.reflect.ObjectWriter; import com.comphenix.protocol.reflect.VolatileField; import com.comphenix.protocol.utility.MinecraftReflection; @@ -219,7 +219,7 @@ class InjectedServerConnection { if (!(inserting instanceof Factory)) { // If so, copy the content of the old element to the new try { - ObjectCloner.copyTo(inserting, replacement, inserting.getClass()); + ObjectWriter.copyTo(inserting, replacement, inserting.getClass()); } catch (Throwable e) { reporter.reportDetailed(InjectedServerConnection.this, "Cannot copy old " + inserting + " to new.", e, inserting, replacement); diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetworkServerInjector.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetworkServerInjector.java index eacc83de..f13b261c 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetworkServerInjector.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetworkServerInjector.java @@ -38,7 +38,7 @@ import com.comphenix.protocol.injector.ListenerInvoker; import com.comphenix.protocol.injector.PacketFilterManager.PlayerInjectHooks; import com.comphenix.protocol.reflect.FieldUtils; import com.comphenix.protocol.reflect.FuzzyReflection; -import com.comphenix.protocol.reflect.ObjectCloner; +import com.comphenix.protocol.reflect.ObjectWriter; import com.comphenix.protocol.reflect.VolatileField; import com.comphenix.protocol.reflect.instances.DefaultInstances; import com.comphenix.protocol.reflect.instances.ExistingGenerator; @@ -248,7 +248,7 @@ public class NetworkServerInjector extends PlayerInjector { @Override protected void cleanHook() { if (serverHandlerRef != null && serverHandlerRef.isCurrentSet()) { - ObjectCloner.copyTo(serverHandlerRef.getValue(), serverHandlerRef.getOldValue(), serverHandler.getClass()); + ObjectWriter.copyTo(serverHandlerRef.getValue(), serverHandlerRef.getOldValue(), serverHandler.getClass()); serverHandlerRef.revertValue(); try { diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectWriter.java similarity index 95% rename from ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectCloner.java rename to ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectWriter.java index 5475e60d..8e7e1d1f 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectCloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectWriter.java @@ -20,13 +20,12 @@ package com.comphenix.protocol.reflect; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentMap; - /** * Can copy an object field by field. * * @author Kristian */ -public class ObjectCloner { +public class ObjectWriter { // Cache structure modifiers @SuppressWarnings("rawtypes") @@ -34,7 +33,7 @@ public class ObjectCloner { new ConcurrentHashMap>(); /** - * Copy every field in object A to object B. + * Copy every field in object A to object B. Each value is copied directly, and is not cloned. *

* The two objects must have the same number of fields of the same type. * @param source - fields to copy. From 8b91e3034ddc9b75874711db77e23122ee44a0aa Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 27 Dec 2012 09:23:07 +0100 Subject: [PATCH 26/37] Added a generic cloning library to properly support deepClone(). --- .../protocol/events/PacketContainer.java | 34 +-- .../protocol/injector/MinecraftRegistry.java | 16 ++ .../protocol/injector/StructureCache.java | 21 ++ .../protocol/reflect/ObjectWriter.java | 90 +++++-- .../protocol/reflect/StructureModifier.java | 29 ++- .../reflect/cloning/AggregateCloner.java | 236 ++++++++++++++++++ .../reflect/cloning/BukkitCloner.java | 56 +++++ .../protocol/reflect/cloning/Cloner.java | 25 ++ .../reflect/cloning/CollectionCloner.java | 146 +++++++++++ .../protocol/reflect/cloning/FieldCloner.java | 61 +++++ .../reflect/cloning/IdentityCloner.java | 18 ++ .../reflect/cloning/ImmutableDetector.java | 70 ++++++ .../reflect/cloning/NullableCloner.java | 32 +++ .../reflect/instances/DefaultInstances.java | 9 +- .../reflect/instances/ExistingGenerator.java | 141 ++++++++++- .../protocol/utility/MinecraftReflection.java | 38 +-- .../protocol/events/PacketContainerTest.java | 66 ++++- 17 files changed, 992 insertions(+), 96 deletions(-) create mode 100644 ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java create mode 100644 ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/BukkitCloner.java create mode 100644 ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/Cloner.java create mode 100644 ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/CollectionCloner.java create mode 100644 ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java create mode 100644 ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/IdentityCloner.java create mode 100644 ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/ImmutableDetector.java create mode 100644 ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/NullableCloner.java diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java b/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java index 373c7d98..eff03790 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java @@ -17,8 +17,6 @@ package com.comphenix.protocol.events; -import java.io.ByteArrayInputStream; -import java.io.ByteArrayOutputStream; import java.io.DataInputStream; import java.io.DataOutputStream; import java.io.IOException; @@ -41,6 +39,7 @@ import com.comphenix.protocol.injector.StructureCache; import com.comphenix.protocol.reflect.EquivalentConverter; import com.comphenix.protocol.reflect.FuzzyReflection; import com.comphenix.protocol.reflect.StructureModifier; +import com.comphenix.protocol.reflect.cloning.AggregateCloner; import com.comphenix.protocol.utility.MinecraftReflection; import com.comphenix.protocol.wrappers.BukkitConverters; import com.comphenix.protocol.wrappers.ChunkPosition; @@ -370,35 +369,8 @@ public class PacketContainer implements Serializable { * @return A deep copy of the current packet. */ public PacketContainer deepClone() { - ObjectOutputStream output = null; - ObjectInputStream input = null; - - try { - // Use a small buffer of 32 bytes initially. - ByteArrayOutputStream bufferOut = new ByteArrayOutputStream(); - output = new ObjectOutputStream(bufferOut); - output.writeObject(this); - - ByteArrayInputStream bufferIn = new ByteArrayInputStream(bufferOut.toByteArray()); - input = new ObjectInputStream(bufferIn); - return (PacketContainer) input.readObject(); - - } catch (IOException e) { - throw new IllegalStateException("Unexpected error occured during object cloning.", e); - } catch (ClassNotFoundException e) { - // Cannot happen - throw new IllegalStateException("Unexpected failure with serialization.", e); - } finally { - try { - if (output != null) - output.close(); - if (input != null) - input.close(); - - } catch (IOException e) { - // STOP IT - } - } + Object clonedPacket = AggregateCloner.DEFAULT.clone(getHandle()); + return new PacketContainer(getID(), clonedPacket); } private void writeObject(ObjectOutputStream output) throws IOException { diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/MinecraftRegistry.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/MinecraftRegistry.java index 5cdddd7c..7ad014cb 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/MinecraftRegistry.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/MinecraftRegistry.java @@ -182,6 +182,22 @@ class MinecraftRegistry { throw new IllegalArgumentException("The packet ID " + packetID + " is not registered."); } + /** + * Retrieve the packet ID of a given packet. + * @param packet - the type of packet to check. + * @return The ID of the given packet. + * @throws IllegalArgumentException If this is not a valid packet. + */ + public static int getPacketID(Class packet) { + if (packet == null) + throw new IllegalArgumentException("Packet type class cannot be NULL."); + if (!MinecraftReflection.getPacketClass().isAssignableFrom(packet)) + throw new IllegalArgumentException("Type must be a packet."); + + // The registry contains both the overridden and original packets + return getPacketToID().get(packet); + } + /** * Find the first superclass that is not a CBLib proxy object. * @param clazz - the class whose hierachy we're going to search through. diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/StructureCache.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/StructureCache.java index bc4f7aa2..b76f2179 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/StructureCache.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/StructureCache.java @@ -64,6 +64,27 @@ public class StructureCache { return getStructure(id, true); } + /** + * Retrieve a cached structure modifier given a packet type. + * @param packetType - packet type. + * @return A structure modifier. + */ + public static StructureModifier getStructure(Class packetType) { + // Compile structures by default + return getStructure(packetType, true); + } + + /** + * Retrieve a cached structure modifier given a packet type. + * @param packetType - packet type. + * @param compile - whether or not to asynchronously compile the structure modifier. + * @return A structure modifier. + */ + public static StructureModifier getStructure(Class packetType, boolean compile) { + // Get the ID from the class + return getStructure(MinecraftRegistry.getPacketID(packetType), compile); + } + /** * Retrieve a cached structure modifier for the given packet id. * @param id - packet ID. diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectWriter.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectWriter.java index 8e7e1d1f..82224912 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectWriter.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectWriter.java @@ -17,21 +17,63 @@ package com.comphenix.protocol.reflect; +import java.lang.reflect.Field; +import java.lang.reflect.Modifier; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentMap; +import com.comphenix.protocol.injector.StructureCache; +import com.comphenix.protocol.reflect.cloning.Cloner; +import com.comphenix.protocol.reflect.cloning.IdentityCloner; +import com.comphenix.protocol.utility.MinecraftReflection; + /** * Can copy an object field by field. * * @author Kristian */ public class ObjectWriter { - // Cache structure modifiers @SuppressWarnings("rawtypes") private static ConcurrentMap> cache = new ConcurrentHashMap>(); + /** + * The default value cloner to use. + */ + private static final Cloner DEFAULT_CLONER = new IdentityCloner(); + + /** + * Retrieve a usable structure modifier for the given object type. + *

+ * Will attempt to reuse any other structure modifiers we have cached. + * @param type - the type of the object we are modifying. + * @return A structure modifier for the given type. + */ + private static StructureModifier getModifier(Class type) { + Class packetClass = MinecraftReflection.getPacketClass(); + + // Handle subclasses of the packet class with our custom structure cache + if (!type.equals(packetClass) && packetClass.isAssignableFrom(type)) { + // Delegate to our already existing registry of structure modifiers + return StructureCache.getStructure(type); + } + + StructureModifier modifier = cache.get(type); + + // Create the structure modifier if we haven't already + if (modifier == null) { + StructureModifier value = new StructureModifier(type, null, false); + modifier = cache.putIfAbsent(type, value); + + if (modifier == null) + modifier = value; + } + + // And we're done + return modifier; + } + /** * Copy every field in object A to object B. Each value is copied directly, and is not cloned. *

@@ -41,22 +83,31 @@ public class ObjectWriter { * @param commonType - type containing each field to copy. */ public static void copyTo(Object source, Object destination, Class commonType) { - + // Note that we indicate that public fields will be copied the first time around + copyToInternal(source, destination, commonType, DEFAULT_CLONER, true); + } + + /** + * Copy every field in object A to object B. Each value is copied using the supplied cloner. + *

+ * The two objects must have the same number of fields of the same type. + * @param source - fields to copy. + * @param destination - fields to copy to. + * @param commonType - type containing each field to copy. + * @param valueCloner - a object responsible for copying the content of each field. + */ + public static void copyTo(Object source, Object destination, Class commonType, Cloner valueCloner) { + copyToInternal(source, destination, commonType, valueCloner, true); + } + + // Internal method that will actually implement the recursion + private static void copyToInternal(Object source, Object destination, Class commonType, Cloner valueCloner, boolean copyPublic) { if (source == null) throw new IllegalArgumentException("Source cannot be NULL"); if (destination == null) throw new IllegalArgumentException("Destination cannot be NULL"); - StructureModifier modifier = cache.get(commonType); - - // Create the structure modifier if we haven't already - if (modifier == null) { - StructureModifier value = new StructureModifier(commonType, null, false); - modifier = cache.putIfAbsent(commonType, value); - - if (modifier == null) - modifier = value; - } + StructureModifier modifier = getModifier(commonType); // Add target StructureModifier modifierSource = modifier.withTarget(source); @@ -65,20 +116,21 @@ public class ObjectWriter { // Copy every field try { for (int i = 0; i < modifierSource.size(); i++) { - if (!modifierDest.isReadOnly(i)) { - Object value = modifierSource.read(i); - modifierDest.write(i, value); - } + Field field = modifierSource.getField(i); + int mod = field.getModifiers(); - // System.out.println(String.format("Writing value %s to %s", - // value, modifier.getFields().get(i).getName())); + // Skip static fields. We also get the "public" field fairly often, so we'll skip that. + if (!Modifier.isStatic(mod) && (!Modifier.isPublic(mod) || copyPublic)) { + Object value = modifierSource.read(i); + modifierDest.write(i, valueCloner.clone(value)); + } } // Copy private fields underneath Class superclass = commonType.getSuperclass(); if (!superclass.equals(Object.class)) { - copyTo(source, destination, superclass); + copyToInternal(source, destination, superclass, valueCloner, false); } } catch (FieldAccessException e) { diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/StructureModifier.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/StructureModifier.java index 87a309fa..9d90d0b8 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/StructureModifier.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/StructureModifier.java @@ -201,10 +201,16 @@ public class StructureModifier { * @return TRUE if the field by the given index is read-only, FALSE otherwise. */ public boolean isReadOnly(int fieldIndex) { - if (fieldIndex < 0 || fieldIndex >= data.size()) - throw new IllegalArgumentException("Index parameter is not within [0 - " + data.size() + ")"); - - return Modifier.isFinal(data.get(fieldIndex).getModifiers()); + return Modifier.isFinal(getField(fieldIndex).getModifiers()); + } + + /** + * Determine if a given field is public or not. + * @param fieldIndex - field index. + * @return TRUE if the field is public, FALSE otherwise. + */ + public boolean isPublic(int fieldIndex) { + return Modifier.isPublic(getField(fieldIndex).getModifiers()); } /** @@ -499,6 +505,19 @@ public class StructureModifier { return ImmutableList.copyOf(data); } + /** + * Retrieve a field by index. + * @param fieldIndex - index of the field to retrieve. + * @return The field represented with the given index. + * @throws IllegalArgumentException If no field with the given index can be found. + */ + public Field getField(int fieldIndex) { + if (fieldIndex < 0 || fieldIndex >= data.size()) + throw new IllegalArgumentException("Index parameter is not within [0 - " + data.size() + ")"); + + return data.get(fieldIndex); + } + /** * Retrieve every value stored in the fields of the current type. * @return Every field value. @@ -560,4 +579,6 @@ public class StructureModifier { return result; } + + } diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java new file mode 100644 index 00000000..233afbbb --- /dev/null +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java @@ -0,0 +1,236 @@ +package com.comphenix.protocol.reflect.cloning; + +import java.lang.ref.WeakReference; +import java.util.Collections; +import java.util.List; + +import javax.annotation.Nullable; + +import com.comphenix.protocol.reflect.instances.DefaultInstances; +import com.comphenix.protocol.reflect.instances.ExistingGenerator; +import com.comphenix.protocol.reflect.instances.InstanceProvider; +import com.google.common.base.Function; +import com.google.common.collect.Lists; + +/** + * Implements a cloning procedure by trying multiple methods in turn until one is successful. + * + * @author Kristian + */ +public class AggregateCloner implements Cloner { + public static class BuilderParameters { + // Can only be modified by the builder + private InstanceProvider instanceProvider; + private Cloner aggregateCloner; + + // Used to construct the different types + private InstanceProvider typeConstructor; + + private BuilderParameters() { + // Only allow inner classes to construct it. + } + + /** + * Retrieve the instance provider last set in the builder. + * @return Current instance provider. + */ + public InstanceProvider getInstanceProvider() { + return instanceProvider; + } + + /** + * Retrieve the aggregate cloner that is being built. + * @return The parent cloner. + */ + public Cloner getAggregateCloner() { + return aggregateCloner; + } + } + + public static class Builder { + private List> factories = Lists.newArrayList(); + private BuilderParameters parameters; + + /** + * Create a new aggregate builder. + */ + public Builder() { + this.parameters = new BuilderParameters(); + } + + /** + * Set the instance provider supplied to all cloners in this builder. + * @param provider - new instance provider. + * @return The current builder. + */ + public Builder instanceProvider(InstanceProvider provider) { + this.parameters.instanceProvider = provider; + return this; + } + + /** + * Add the next cloner that will be considered in turn. + * @param type - the type of the next cloner. + * @return This builder. + */ + public Builder andThen(final Class type) { + // Use reflection to generate a factory on the fly + return orCloner(new Function() { + @Override + public Cloner apply(@Nullable BuilderParameters param) { + Object result = param.typeConstructor.create(type); + + if (result == null) { + throw new IllegalStateException("Constructed NULL instead of " + type); + } + + if (type.isAssignableFrom(result.getClass())) + return (Cloner) result; + else + throw new IllegalStateException("Constructed " + result.getClass() + " instead of " + type); + } + }); + } + + /** + * Add the next cloner that will be considered in turn. + * @param factory - factory constructing the next cloner. + * @return This builder. + */ + public Builder orCloner(Function factory) { + factories.add(factory); + return this; + } + + /** + * Build a new aggregate cloner using the supplied values. + * @return A new aggregate cloner. + */ + public AggregateCloner build() { + AggregateCloner newCloner = new AggregateCloner(); + + // The parameters we will pass to our cloners + Cloner paramCloner = new NullableCloner(newCloner); + InstanceProvider paramProvider = parameters.instanceProvider; + + // Initialize parameters + parameters.aggregateCloner = paramCloner; + parameters.typeConstructor = DefaultInstances.fromArray( + ExistingGenerator.fromObjectArray(new Object[] { paramCloner, paramProvider }) + ); + + // Build every cloner in the correct order + List cloners = Lists.newArrayList(); + + for (int i = 0; i < factories.size(); i++) { + Cloner cloner = factories.get(i).apply(parameters); + + // See if we were successful + if (cloner != null) + cloners.add(cloner); + else + throw new IllegalArgumentException( + String.format("Cannot create cloner from %s (%s)", factories.get(i), i) + ); + } + + // We're done + newCloner.setCloners(cloners); + return newCloner; + } + } + + /** + * Represents a default aggregate cloner. + */ + public static final AggregateCloner DEFAULT = newBuilder(). + instanceProvider(DefaultInstances.DEFAULT). + andThen(BukkitCloner.class). + andThen(ImmutableDetector.class). + andThen(CollectionCloner.class). + andThen(FieldCloner.class). + build(); + + // List of clone methods + private List cloners; + + private WeakReference lastObject; + private int lastResult; + + /** + * Begins constructing a new aggregate cloner. + * @return A builder for a new aggregate cloner. + */ + public static Builder newBuilder() { + return new Builder(); + } + + /** + * Construct a new, empty aggregate cloner. + */ + private AggregateCloner() { + // Only used by our builder above. + } + + /** + * Retrieves a view of the current list of cloners. + * @return Current cloners. + */ + public List getCloners() { + return Collections.unmodifiableList(cloners); + } + + /** + * Set the cloners that will be used. + * @param cloners - the cloners that will be used. + */ + private void setCloners(Iterable cloners) { + this.cloners = Lists.newArrayList(cloners); + } + + @Override + public boolean canClone(Object source) { + // Optimize a bit + lastResult = getFirstCloner(source); + lastObject = new WeakReference(source); + return lastResult >= 0 && lastResult < cloners.size(); + } + + /** + * Retrieve the index of the first cloner capable of cloning the given object. + *

+ * Returns an invalid index if no cloner is able to clone the object. + * @param source - the object to clone. + * @return The index of the cloner object. + */ + private int getFirstCloner(Object source) { + for (int i = 0; i < cloners.size(); i++) { + if (cloners.get(i).canClone(source)) + return i; + } + + return cloners.size(); + } + + @Override + public Object clone(Object source) { + if (source == null) + throw new IllegalAccessError("source cannot be NULL."); + int index = 0; + + // Are we dealing with the same object? + if (lastObject != null && lastObject.get() == source) { + index = lastResult; + } else { + index = getFirstCloner(source); + } + + // Make sure the object is valid + if (index < cloners.size()) { + return cloners.get(index).clone(source); + } + + // Damn - failure + throw new IllegalArgumentException("Cannot clone " + source + ": No cloner is sutable."); + } +} diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/BukkitCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/BukkitCloner.java new file mode 100644 index 00000000..290fb48a --- /dev/null +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/BukkitCloner.java @@ -0,0 +1,56 @@ +package com.comphenix.protocol.reflect.cloning; + +import com.comphenix.protocol.reflect.EquivalentConverter; +import com.comphenix.protocol.utility.MinecraftReflection; +import com.comphenix.protocol.wrappers.BukkitConverters; +import com.comphenix.protocol.wrappers.ChunkPosition; +import com.comphenix.protocol.wrappers.WrappedDataWatcher; + +/** + * Represents an object that can clone a specific list of Bukkit- and Minecraft-related objects. + * + * @author Kristian + */ +public class BukkitCloner implements Cloner { + // List of classes we support + private Class[] clonableClasses = { MinecraftReflection.getItemStackClass(), MinecraftReflection.getChunkPositionClass(), + MinecraftReflection.getDataWatcherClass() }; + + private int findMatchingClass(Class type) { + // See if is a subclass of any of our supported superclasses + for (int i = 0; i < clonableClasses.length; i++) { + if (clonableClasses[i].isAssignableFrom(type)) + return i; + } + + return -1; + } + + @Override + public boolean canClone(Object source) { + if (source == null) + return false; + + return findMatchingClass(source.getClass()) >= 0; + } + + @Override + public Object clone(Object source) { + if (source == null) + throw new IllegalArgumentException("source cannot be NULL."); + + // Convert to a wrapper + switch (findMatchingClass(source.getClass())) { + case 0: + return MinecraftReflection.getMinecraftItemStack(MinecraftReflection.getBukkitItemStack(source).clone()); + case 1: + EquivalentConverter chunkConverter = ChunkPosition.getConverter(); + return chunkConverter.getGeneric(clonableClasses[1], chunkConverter.getSpecific(source)); + case 2: + EquivalentConverter dataConverter = BukkitConverters.getDataWatcherConverter(); + return dataConverter.getGeneric(clonableClasses[2], dataConverter.getSpecific(source).deepClone()); + default: + throw new IllegalArgumentException("Cannot clone objects of type " + source.getClass()); + } + } +} diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/Cloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/Cloner.java new file mode 100644 index 00000000..c7fd0cf6 --- /dev/null +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/Cloner.java @@ -0,0 +1,25 @@ +package com.comphenix.protocol.reflect.cloning; + +/** + * Represents an object that is capable of cloning other objects. + * + * @author Kristian + */ +public interface Cloner { + /** + * Determine whether or not the current cloner can clone the given object. + * @param source - the object that is being considered. + * @return TRUE if this cloner can actually clone the given object, FALSE otherwise. + */ + public boolean canClone(Object source); + + /** + * Perform the clone. + *

+ * This method should never be called unless a corresponding {@link #canClone(Object)} returns TRUE. + * @param source - the value to clone. + * @return A cloned value. + * @throws IllegalArgumentException If this cloner cannot perform the clone. + */ + public Object clone(Object source); +} diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/CollectionCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/CollectionCloner.java new file mode 100644 index 00000000..5fd9f3bc --- /dev/null +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/CollectionCloner.java @@ -0,0 +1,146 @@ +package com.comphenix.protocol.reflect.cloning; + +import java.lang.reflect.Array; +import java.lang.reflect.Constructor; +import java.util.Collection; + +/** + * Attempts to clone collection and array classes. + * + * @author Kristian + */ +public class CollectionCloner implements Cloner { + private final Cloner defaultCloner; + + /** + * Constructs a new collection and array cloner with the given inner element cloner. + * @param defaultCloner - default inner element cloner. + */ + public CollectionCloner(Cloner defaultCloner) { + this.defaultCloner = defaultCloner; + } + + @Override + public boolean canClone(Object source) { + if (source == null) + return false; + + Class clazz = source.getClass(); + return Collection.class.isAssignableFrom(clazz) || clazz.isArray(); + } + + @SuppressWarnings("unchecked") + @Override + public Object clone(Object source) { + if (source == null) + throw new IllegalArgumentException("source cannot be NULL."); + + Class clazz = source.getClass(); + + if (source instanceof Collection) { + Collection copy = null; + + // Not all collections implement "clone", but most *do* implement the "copy constructor" pattern + try { + Constructor constructCopy = clazz.getConstructor(Collection.class); + copy = (Collection) constructCopy.newInstance(source); + } catch (NoSuchMethodException e) { + copy = (Collection) cloneObject(clazz, source); + } catch (Exception e) { + throw new RuntimeException("Cannot construct collection.", e); + } + + // Next, clone each element in the collection + copy.clear(); + + for (Object element : (Collection) source) { + if (defaultCloner.canClone(element)) + copy.add(defaultCloner.clone(element)); + else + throw new IllegalArgumentException("Cannot clone " + element + " in collection " + source); + } + + return copy; + + // Second possibility + } else if (clazz.isArray()) { + // Get the length + int lenght = Array.getLength(source); + Class component = clazz.getComponentType(); + + // Can we speed things up by making a shallow copy instead? + if (ImmutableDetector.isImmutable(component)) { + return clonePrimitive(component, source); + } + + // Create a new copy + Object copy = Array.newInstance(clazz.getComponentType(), lenght); + + // Set each element + for (int i = 0; i < lenght; i++) { + Object element = Array.get(source, i); + + if (defaultCloner.canClone(element)) + Array.set(copy, i, defaultCloner.clone(element)); + else + throw new IllegalArgumentException("Cannot clone " + element + " in array " + source); + } + + // And we're done + return copy; + } + + throw new IllegalArgumentException(source + " is not an array nor a Collection."); + } + + /** + * Clone a primitive or immutable array by calling its clone method. + * @param component - the component type of the array. + * @param source - the array itself. + * @return The cloned array. + */ + private Object clonePrimitive(Class component, Object source) { + // Cast and call the correct version + if (byte.class.equals(component)) + return ((byte[]) source).clone(); + else if (short.class.equals(component)) + return ((short[]) source).clone(); + else if (int.class.equals(component)) + return ((int[]) source).clone(); + else if (long.class.equals(component)) + return ((long[]) source).clone(); + else if (float.class.equals(component)) + return ((float[]) source).clone(); + else if (double.class.equals(component)) + return ((double[]) source).clone(); + else if (char.class.equals(component)) + return ((char[]) source).clone(); + else if (boolean.class.equals(component)) + return ((boolean[]) source).clone(); + else + return ((Object[]) source).clone(); + } + + /** + * Clone an object by calling "clone" using reflection. + * @param clazz - the class type. + * @param obj - the object to clone. + * @return The cloned object. + */ + private Object cloneObject(Class clazz, Object source) { + // Try to clone it instead + try { + return clazz.getMethod("clone").invoke(source); + } catch (Exception e1) { + throw new RuntimeException("Cannot copy " + source + " (" + clazz + ")", e1); + } + } + + /** + * Retrieve the default cloner used to clone the content of each element in the collection. + * @return Cloner used to clone elements. + */ + public Cloner getDefaultCloner() { + return defaultCloner; + } +} diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java new file mode 100644 index 00000000..18d6554b --- /dev/null +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java @@ -0,0 +1,61 @@ +package com.comphenix.protocol.reflect.cloning; + +import com.comphenix.protocol.reflect.ObjectWriter; +import com.comphenix.protocol.reflect.instances.InstanceProvider; + +/** + * Represents a class capable of cloning objects by deeply copying its fields. + * + * @author Kristian + */ +public class FieldCloner implements Cloner { + private final Cloner defaultCloner; + private final InstanceProvider instanceProvider; + + /** + * Constructs a field cloner that copies objects by reading and writing the internal fields directly. + * @param defaultCloner - the default cloner used while copying fields. + * @param instanceProvider - used to construct new, empty copies of a given type. + */ + public FieldCloner(Cloner defaultCloner, InstanceProvider instanceProvider) { + this.defaultCloner = defaultCloner; + this.instanceProvider = instanceProvider; + } + + @Override + public boolean canClone(Object source) { + if (source == null) + return false; + + // Attempt to create the type + return instanceProvider.create(source.getClass()) != null; + } + + @Override + public Object clone(Object source) { + if (source == null) + throw new IllegalArgumentException("source cannot be NULL."); + + Object copy = instanceProvider.create(source.getClass()); + + // Copy public and private fields alike. Skip static and transient fields. + ObjectWriter.copyTo(source, copy, source.getClass(), defaultCloner); + return copy; + } + + /** + * Retrieve the default cloner used to clone the content of each field. + * @return Cloner used to clone fields. + */ + public Cloner getDefaultCloner() { + return defaultCloner; + } + + /** + * Retrieve the instance provider this cloner is using to create new, empty classes. + * @return The instance provider in use. + */ + public InstanceProvider getInstanceProvider() { + return instanceProvider; + } +} diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/IdentityCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/IdentityCloner.java new file mode 100644 index 00000000..576baee1 --- /dev/null +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/IdentityCloner.java @@ -0,0 +1,18 @@ +package com.comphenix.protocol.reflect.cloning; + +/** + * Represents a cloner that simply returns the given object. + * + * @author Kristian + */ +public class IdentityCloner implements Cloner { + @Override + public boolean canClone(Object source) { + return true; + } + + @Override + public Object clone(Object source) { + return source; + } +} diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/ImmutableDetector.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/ImmutableDetector.java new file mode 100644 index 00000000..7ac3fee0 --- /dev/null +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/ImmutableDetector.java @@ -0,0 +1,70 @@ +package com.comphenix.protocol.reflect.cloning; + +import java.math.BigDecimal; +import java.math.BigInteger; +import java.net.Inet4Address; +import java.net.Inet6Address; +import java.net.InetSocketAddress; +import java.net.URI; +import java.net.URL; +import java.util.Locale; +import java.util.UUID; + +import com.google.common.primitives.Primitives; + +/** + * Detects classes that are immutable, and thus doesn't require cloning. + *

+ * This ought to have no false positives, but plenty of false negatives. + * + * @author Kristian + */ +public class ImmutableDetector implements Cloner { + // Notable immutable classes we might encounter + private static final Class[] immutableClasses = { + StackTraceElement.class, BigDecimal.class, + BigInteger.class, Locale.class, UUID.class, + URL.class, URI.class, Inet4Address.class, + Inet6Address.class, InetSocketAddress.class + }; + + @Override + public boolean canClone(Object source) { + // Don't accept NULL + if (source == null) + return false; + + return isImmutable(source.getClass()); + } + + /** + * Determine if the given type is probably immutable. + * @param type - the type to check. + * @return TRUE if the type is immutable, FALSE otherwise. + */ + public static boolean isImmutable(Class type) { + // Cases that are definitely not true + if (type.isArray()) + return false; + + // All primitive types + if (Primitives.isWrapperType(type) || String.class.equals(type)) + return true; + // May not be true, but if so, that kind of code is broken anyways + if (type.isEnum()) + return true; + + for (Class clazz : immutableClasses) + if (clazz.equals(type)) + return true; + + // Probably not + return false; + } + + @Override + public Object clone(Object source) { + // Safe if the class is immutable + return source; + } +} diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/NullableCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/NullableCloner.java new file mode 100644 index 00000000..cfeaea55 --- /dev/null +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/NullableCloner.java @@ -0,0 +1,32 @@ +package com.comphenix.protocol.reflect.cloning; + +/** + * Creates a cloner wrapper that accepts and clones NULL values. + * + * @author Kristian + */ +public class NullableCloner implements Cloner { + protected Cloner wrapped; + + public NullableCloner(Cloner wrapped) { + this.wrapped = wrapped; + } + + @Override + public boolean canClone(Object source) { + return true; + } + + @Override + public Object clone(Object source) { + // Don't pass the NULL value to the cloner + if (source == null) + return null; + else + return wrapped.clone(source); + } + + public Cloner getWrapped() { + return wrapped; + } +} diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/instances/DefaultInstances.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/instances/DefaultInstances.java index 38066613..fe7233da 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/instances/DefaultInstances.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/instances/DefaultInstances.java @@ -20,6 +20,8 @@ package com.comphenix.protocol.reflect.instances; import java.lang.reflect.Constructor; import java.util.*; +import javax.annotation.Nullable; + import net.sf.cglib.proxy.Enhancer; import com.google.common.base.Objects; @@ -30,7 +32,7 @@ import com.google.common.collect.ImmutableList; * @author Kristian * */ -public class DefaultInstances { +public class DefaultInstances implements InstanceProvider { /** * Standard default instance provider. @@ -326,4 +328,9 @@ public class DefaultInstances { } return false; } + + @Override + public Object create(@Nullable Class type) { + return getDefault(type); + } } diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/instances/ExistingGenerator.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/instances/ExistingGenerator.java index 43527d17..ca7d85d2 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/instances/ExistingGenerator.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/instances/ExistingGenerator.java @@ -18,13 +18,16 @@ package com.comphenix.protocol.reflect.instances; import java.lang.reflect.Field; +import java.util.Collection; import java.util.HashMap; +import java.util.LinkedList; import java.util.Map; import javax.annotation.Nullable; import com.comphenix.protocol.reflect.FieldUtils; import com.comphenix.protocol.reflect.FuzzyReflection; +import com.google.common.collect.Lists; /** * Provides instance constructors using a list of existing values. @@ -33,8 +36,52 @@ import com.comphenix.protocol.reflect.FuzzyReflection; * @author Kristian */ public class ExistingGenerator implements InstanceProvider { + /** + * Represents a single node in the tree of possible values. + * + * @author Kristian + */ + private static final class Node { + private Map, Node> children; + private Class key; + private Object value; + private int level; + + public Node(Class key, Object value, int level) { + this.children = new HashMap, Node>(); + this.key = key; + this.value = value; + this.level = level; + } - private Map existingValues = new HashMap(); + public Node addChild(Node node) { + children.put(node.key, node); + return node; + } + + public int getLevel() { + return level; + } + + public Collection getChildren() { + return children.values(); + } + + public Object getValue() { + return value; + } + + public void setValue(Object value) { + this.value = value; + } + + public Node getChild(Class clazz) { + return children.get(clazz); + } + } + + // Represents the root node + private Node root = new Node(null, null, 0); private ExistingGenerator() { // Only accessible to the constructors @@ -110,18 +157,94 @@ public class ExistingGenerator implements InstanceProvider { if (value == null) throw new IllegalArgumentException("Value cannot be NULL."); - existingValues.put(value.getClass().getName(), value); - } - - private void addObject(Class type, Object value) { - existingValues.put(type.getName(), value); + addObject(value.getClass(), value); } + private void addObject(Class type, Object value) { + Node node = getLeafNode(root, type, false); + + // Set the value + node.setValue(value); + } + + private Node getLeafNode(final Node start, Class type, boolean readOnly) { + Class[] path = getHierachy(type); + Node current = start; + + for (int i = 0; i < path.length; i++) { + Node next = getNext(current, path[i], readOnly); + + // Try every interface too + if (next == null && readOnly) { + current = null; + break; + } + + current = next; + } + + // And we're done + return current; + } + + private Node getNext(Node current, Class clazz, boolean readOnly) { + Node next = current.getChild(clazz); + + // Add a new node if needed + if (next == null && !readOnly) { + next = current.addChild(new Node(clazz, null, current.getLevel() + 1)); + } + + // Add interfaces + if (next != null && !readOnly && !clazz.isInterface()) { + for (Class clazzInterface : clazz.getInterfaces()) { + getLeafNode(root, clazzInterface, readOnly).addChild(next); + } + } + return next; + } + + private Node getLowestLeaf(Node current) { + Node candidate = current; + + // Depth-first search + for (Node child : current.getChildren()) { + Node subtree = getLowestLeaf(child); + + // Get the lowest node + if (subtree.getValue() != null && candidate.getLevel() < subtree.getLevel()) { + candidate = subtree; + } + } + + return candidate; + } + + private Class[] getHierachy(Class type) { + LinkedList> levels = Lists.newLinkedList(); + + // Add each class from the hierachy + for (; type != null; type = type.getSuperclass()) { + levels.addFirst(type); + } + + return levels.toArray(new Class[0]); + } + @Override public Object create(@Nullable Class type) { - Object value = existingValues.get(type.getName()); - + // Locate the type in the hierachy + Node node = getLeafNode(root, type, true); + + // Next, get the lowest leaf node + if (node != null) { + node = getLowestLeaf(node); + } + // NULL values indicate that the generator failed - return value; + if (node != null) + return node.getValue(); + else + return null; } } diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java index 027d3f6a..b02e39a1 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java @@ -112,6 +112,24 @@ public class MinecraftReflection { return fullName.substring(0, fullName.lastIndexOf(".")); } + /** + * Dynamically retrieve the Bukkit entity from a given entity. + * @param nmsObject - the NMS entity. + * @return A bukkit entity. + * @throws RuntimeException If we were unable to retrieve the Bukkit entity. + */ + public static Object getBukkitEntity(Object nmsObject) { + if (nmsObject == null) + return null; + + // We will have to do this dynamically, unfortunately + try { + return nmsObject.getClass().getMethod("getBukkitEntity").invoke(nmsObject); + } catch (Exception e) { + throw new RuntimeException("Cannot get Bukkit entity from " + nmsObject, e); + } + } + /** * Determine if a given object can be found within the package net.minecraft.server. * @param obj - the object to test. @@ -138,25 +156,7 @@ public class MinecraftReflection { String javaName = obj.getClass().getName(); return javaName.startsWith(MINECRAFT_PREFIX_PACKAGE) && javaName.endsWith(className); } - - /** - * Dynamically retrieve the Bukkit entity from a given entity. - * @param nmsObject - the NMS entity. - * @return A bukkit entity. - * @throws RuntimeException If we were unable to retrieve the Bukkit entity. - */ - public static Object getBukkitEntity(Object nmsObject) { - if (nmsObject == null) - return null; - - // We will have to do this dynamically, unfortunately - try { - return nmsObject.getClass().getMethod("getBukkitEntity").invoke(nmsObject); - } catch (Exception e) { - throw new RuntimeException("Cannot get Bukkit entity from " + nmsObject, e); - } - } - + /** * Determine if a given object is a ChunkPosition. * @param obj - the object to test. diff --git a/ProtocolLib/src/test/java/com/comphenix/protocol/events/PacketContainerTest.java b/ProtocolLib/src/test/java/com/comphenix/protocol/events/PacketContainerTest.java index ce907afe..1782518b 100644 --- a/ProtocolLib/src/test/java/com/comphenix/protocol/events/PacketContainerTest.java +++ b/ProtocolLib/src/test/java/com/comphenix/protocol/events/PacketContainerTest.java @@ -5,6 +5,7 @@ import static org.mockito.Matchers.any; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.when; +import java.lang.reflect.Array; import java.util.List; // Will have to be updated for every version though @@ -23,12 +24,15 @@ import org.junit.runner.RunWith; import org.powermock.core.classloader.annotations.PrepareForTest; import com.comphenix.protocol.Packets; +import com.comphenix.protocol.reflect.EquivalentConverter; import com.comphenix.protocol.reflect.FieldUtils; import com.comphenix.protocol.reflect.StructureModifier; import com.comphenix.protocol.utility.MinecraftReflection; +import com.comphenix.protocol.wrappers.BukkitConverters; import com.comphenix.protocol.wrappers.ChunkPosition; import com.comphenix.protocol.wrappers.WrappedDataWatcher; import com.comphenix.protocol.wrappers.WrappedWatchableObject; + import com.google.common.collect.Iterables; import com.google.common.collect.Lists; @@ -36,6 +40,9 @@ import com.google.common.collect.Lists; @RunWith(org.powermock.modules.junit4.PowerMockRunner.class) @PrepareForTest(CraftItemFactory.class) public class PacketContainerTest { + // Helper converters + private EquivalentConverter watchConvert = BukkitConverters.getDataWatcherConverter(); + private EquivalentConverter itemConvert = BukkitConverters.getItemStackConverter(); @BeforeClass public static void initializeBukkit() throws IllegalAccessException { @@ -203,23 +210,15 @@ public class PacketContainerTest { } private boolean equivalentItem(ItemStack first, ItemStack second) { - if (first == null) + if (first == null) { return second == null; - else if (second == null) + } else if (second == null) { return false; - else + } else { return first.getType().equals(second.getType()); + } } - private boolean equivalentPacket(PacketContainer first, PacketContainer second) { - if (first == null) - return second == null; - else if (second == null) - return false; - else - return first.getModifier().getValues().equals(second.getModifier().getValues()); - } - @Test public void testGetWorldTypeModifier() { PacketContainer loginPacket = new PacketContainer(Packets.Server.LOGIN); @@ -324,7 +323,16 @@ public class PacketContainerTest { PacketContainer cloned = constructed.deepClone(); // Make sure they're equivalent - assertTrue("Packet " + id + " could not be cloned.", equivalentPacket(constructed, cloned)); + StructureModifier firstMod = constructed.getModifier(), secondMod = cloned.getModifier(); + assertEquals(firstMod.size(), secondMod.size()); + + // Make sure all the fields are equivalent + for (int i = 0; i < firstMod.size(); i++) { + if (firstMod.getField(i).getType().isArray()) + assertArrayEquals(getArray(firstMod.read(i)), getArray(secondMod.read(i))); + else + testEquality(firstMod.read(i), secondMod.read(i)); + } } catch (IllegalArgumentException e) { if (!registered) { @@ -337,4 +345,36 @@ public class PacketContainerTest { } } } + + // Convert to objects that support equals() + private void testEquality(Object a, Object b) { + if (a != null && b != null) { + if (MinecraftReflection.isDataWatcher(a)) { + a = watchConvert.getSpecific(a); + b = watchConvert.getSpecific(b); + } else if (MinecraftReflection.isItemStack(a)) { + a = itemConvert.getSpecific(a); + b = itemConvert.getSpecific(b); + } + } + + assertEquals(a, b); + } + + /** + * Get the underlying array as an object array. + * @param val - array wrapped as an Object. + * @return An object array. + */ + private Object[] getArray(Object val) { + if (val instanceof Object[]) + return (Object[]) val; + + int arrlength = Array.getLength(val); + Object[] outputArray = new Object[arrlength]; + + for (int i = 0; i < arrlength; ++i) + outputArray[i] = Array.get(val, i); + return outputArray; + } } From dea725e1e30ba7d4f281a90e71f6e0b34a7c7b97 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 27 Dec 2012 10:32:26 +0100 Subject: [PATCH 27/37] Move the field cloning reponsibility to FieldCloner. --- .../player/InjectedServerConnection.java | 5 ++- .../player/NetworkServerInjector.java | 5 ++- .../protocol/reflect/ObjectWriter.java | 38 +++++++------------ .../protocol/reflect/cloning/FieldCloner.java | 18 ++++++++- .../protocol/events/PacketContainerTest.java | 20 +++++----- 5 files changed, 49 insertions(+), 37 deletions(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedServerConnection.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedServerConnection.java index 307e8b24..6123ec0f 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedServerConnection.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedServerConnection.java @@ -212,6 +212,9 @@ class InjectedServerConnection { * Shut up Eclipse! */ private static final long serialVersionUID = 2070481080950500367L; + + // Object writer we'll use + private final ObjectWriter writer = new ObjectWriter(); @Override protected void onReplacing(Object inserting, Object replacement) { @@ -219,7 +222,7 @@ class InjectedServerConnection { if (!(inserting instanceof Factory)) { // If so, copy the content of the old element to the new try { - ObjectWriter.copyTo(inserting, replacement, inserting.getClass()); + writer.copyTo(inserting, replacement, inserting.getClass()); } catch (Throwable e) { reporter.reportDetailed(InjectedServerConnection.this, "Cannot copy old " + inserting + " to new.", e, inserting, replacement); diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetworkServerInjector.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetworkServerInjector.java index f13b261c..74a0a3e6 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetworkServerInjector.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetworkServerInjector.java @@ -66,6 +66,9 @@ public class NetworkServerInjector extends PlayerInjector { // Whether or not the player has disconnected private boolean hasDisconnected; + // Used to copy fields + private final ObjectWriter writer = new ObjectWriter(); + public NetworkServerInjector( ClassLoader classLoader, ErrorReporter reporter, Player player, ListenerInvoker invoker, IntegerSet sendingFilters, @@ -248,7 +251,7 @@ public class NetworkServerInjector extends PlayerInjector { @Override protected void cleanHook() { if (serverHandlerRef != null && serverHandlerRef.isCurrentSet()) { - ObjectWriter.copyTo(serverHandlerRef.getValue(), serverHandlerRef.getOldValue(), serverHandler.getClass()); + writer.copyTo(serverHandlerRef.getValue(), serverHandlerRef.getOldValue(), serverHandler.getClass()); serverHandlerRef.revertValue(); try { diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectWriter.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectWriter.java index 82224912..5ab3eb2d 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectWriter.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/ObjectWriter.java @@ -23,8 +23,6 @@ import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentMap; import com.comphenix.protocol.injector.StructureCache; -import com.comphenix.protocol.reflect.cloning.Cloner; -import com.comphenix.protocol.reflect.cloning.IdentityCloner; import com.comphenix.protocol.utility.MinecraftReflection; /** @@ -38,11 +36,6 @@ public class ObjectWriter { private static ConcurrentMap> cache = new ConcurrentHashMap>(); - /** - * The default value cloner to use. - */ - private static final Cloner DEFAULT_CLONER = new IdentityCloner(); - /** * Retrieve a usable structure modifier for the given object type. *

@@ -50,7 +43,7 @@ public class ObjectWriter { * @param type - the type of the object we are modifying. * @return A structure modifier for the given type. */ - private static StructureModifier getModifier(Class type) { + private StructureModifier getModifier(Class type) { Class packetClass = MinecraftReflection.getPacketClass(); // Handle subclasses of the packet class with our custom structure cache @@ -82,26 +75,24 @@ public class ObjectWriter { * @param destination - fields to copy to. * @param commonType - type containing each field to copy. */ - public static void copyTo(Object source, Object destination, Class commonType) { + public void copyTo(Object source, Object destination, Class commonType) { // Note that we indicate that public fields will be copied the first time around - copyToInternal(source, destination, commonType, DEFAULT_CLONER, true); + copyToInternal(source, destination, commonType, true); } /** - * Copy every field in object A to object B. Each value is copied using the supplied cloner. - *

- * The two objects must have the same number of fields of the same type. - * @param source - fields to copy. - * @param destination - fields to copy to. - * @param commonType - type containing each field to copy. - * @param valueCloner - a object responsible for copying the content of each field. + * Called for every non-static field that will be copied. + * @param modifierSource - modifier for the original object. + * @param modifierDest - modifier for the new cloned object. + * @param fieldIndex - the current field index. */ - public static void copyTo(Object source, Object destination, Class commonType, Cloner valueCloner) { - copyToInternal(source, destination, commonType, valueCloner, true); + protected void transformField(StructureModifier modifierSource, StructureModifier modifierDest, int fieldIndex) { + Object value = modifierSource.read(fieldIndex); + modifierDest.write(fieldIndex, value); } // Internal method that will actually implement the recursion - private static void copyToInternal(Object source, Object destination, Class commonType, Cloner valueCloner, boolean copyPublic) { + private void copyToInternal(Object source, Object destination, Class commonType, boolean copyPublic) { if (source == null) throw new IllegalArgumentException("Source cannot be NULL"); if (destination == null) @@ -119,10 +110,9 @@ public class ObjectWriter { Field field = modifierSource.getField(i); int mod = field.getModifiers(); - // Skip static fields. We also get the "public" field fairly often, so we'll skip that. + // Skip static fields. We also get the "public" fields fairly often, so we'll skip that. if (!Modifier.isStatic(mod) && (!Modifier.isPublic(mod) || copyPublic)) { - Object value = modifierSource.read(i); - modifierDest.write(i, valueCloner.clone(value)); + transformField(modifierSource, modifierDest, i); } } @@ -130,7 +120,7 @@ public class ObjectWriter { Class superclass = commonType.getSuperclass(); if (!superclass.equals(Object.class)) { - copyToInternal(source, destination, superclass, valueCloner, false); + copyToInternal(source, destination, superclass, false); } } catch (FieldAccessException e) { diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java index 18d6554b..5eb7f514 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java @@ -1,6 +1,7 @@ package com.comphenix.protocol.reflect.cloning; import com.comphenix.protocol.reflect.ObjectWriter; +import com.comphenix.protocol.reflect.StructureModifier; import com.comphenix.protocol.reflect.instances.InstanceProvider; /** @@ -12,6 +13,9 @@ public class FieldCloner implements Cloner { private final Cloner defaultCloner; private final InstanceProvider instanceProvider; + // Used to clone objects + private final ObjectWriter writer; + /** * Constructs a field cloner that copies objects by reading and writing the internal fields directly. * @param defaultCloner - the default cloner used while copying fields. @@ -20,6 +24,16 @@ public class FieldCloner implements Cloner { public FieldCloner(Cloner defaultCloner, InstanceProvider instanceProvider) { this.defaultCloner = defaultCloner; this.instanceProvider = instanceProvider; + + // Remember to clone the value too + this.writer = new ObjectWriter() { + @Override + protected void transformField(StructureModifier modifierSource, + StructureModifier modifierDest, int fieldIndex) { + Object value = modifierSource.read(fieldIndex); + modifierDest.write(fieldIndex, getDefaultCloner().clone(value)); + } + }; } @Override @@ -38,8 +52,8 @@ public class FieldCloner implements Cloner { Object copy = instanceProvider.create(source.getClass()); - // Copy public and private fields alike. Skip static and transient fields. - ObjectWriter.copyTo(source, copy, source.getClass(), defaultCloner); + // Copy public and private fields alike. Skip static fields. + writer.copyTo(source, copy, source.getClass()); return copy; } diff --git a/ProtocolLib/src/test/java/com/comphenix/protocol/events/PacketContainerTest.java b/ProtocolLib/src/test/java/com/comphenix/protocol/events/PacketContainerTest.java index 1782518b..bbee0a76 100644 --- a/ProtocolLib/src/test/java/com/comphenix/protocol/events/PacketContainerTest.java +++ b/ProtocolLib/src/test/java/com/comphenix/protocol/events/PacketContainerTest.java @@ -367,14 +367,16 @@ public class PacketContainerTest { * @return An object array. */ private Object[] getArray(Object val) { - if (val instanceof Object[]) - return (Object[]) val; - - int arrlength = Array.getLength(val); - Object[] outputArray = new Object[arrlength]; - - for (int i = 0; i < arrlength; ++i) - outputArray[i] = Array.get(val, i); - return outputArray; + if (val instanceof Object[]) + return (Object[]) val; + if (val == null) + return null; + + int arrlength = Array.getLength(val); + Object[] outputArray = new Object[arrlength]; + + for (int i = 0; i < arrlength; ++i) + outputArray[i] = Array.get(val, i); + return outputArray; } } From 695a4f3b0e49c9b23c0eb41acb4d5cca6ed2d3fc Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 27 Dec 2012 11:13:42 +0100 Subject: [PATCH 28/37] Added a shallow clone method. Removed the identity cloner, as it's no longer needed. --- .../protocol/events/PacketContainer.java | 42 ++++++++++++++++++- .../reflect/cloning/AggregateCloner.java | 9 +++- .../protocol/reflect/cloning/FieldCloner.java | 6 +-- .../reflect/cloning/IdentityCloner.java | 18 -------- 4 files changed, 51 insertions(+), 24 deletions(-) delete mode 100644 ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/IdentityCloner.java diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java b/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java index eff03790..266d45cd 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java @@ -30,6 +30,8 @@ import java.util.Collection; import java.util.List; import java.util.concurrent.ConcurrentMap; +import javax.annotation.Nullable; + import org.bukkit.World; import org.bukkit.WorldType; import org.bukkit.entity.Entity; @@ -38,13 +40,19 @@ import org.bukkit.inventory.ItemStack; import com.comphenix.protocol.injector.StructureCache; import com.comphenix.protocol.reflect.EquivalentConverter; import com.comphenix.protocol.reflect.FuzzyReflection; +import com.comphenix.protocol.reflect.ObjectWriter; import com.comphenix.protocol.reflect.StructureModifier; import com.comphenix.protocol.reflect.cloning.AggregateCloner; +import com.comphenix.protocol.reflect.cloning.Cloner; +import com.comphenix.protocol.reflect.cloning.FieldCloner; +import com.comphenix.protocol.reflect.cloning.AggregateCloner.BuilderParameters; +import com.comphenix.protocol.reflect.instances.DefaultInstances; import com.comphenix.protocol.utility.MinecraftReflection; import com.comphenix.protocol.wrappers.BukkitConverters; import com.comphenix.protocol.wrappers.ChunkPosition; import com.comphenix.protocol.wrappers.WrappedDataWatcher; import com.comphenix.protocol.wrappers.WrappedWatchableObject; +import com.google.common.base.Function; import com.google.common.collect.Maps; /** @@ -69,6 +77,21 @@ public class PacketContainer implements Serializable { private static ConcurrentMap, Method> writeMethods = Maps.newConcurrentMap(); private static ConcurrentMap, Method> readMethods = Maps.newConcurrentMap(); + // Used to clone packets + private static final AggregateCloner DEEP_CLONER = AggregateCloner.DEFAULT; + private static final AggregateCloner SHALLOW_CLONER = AggregateCloner.newBuilder(). + instanceProvider(DefaultInstances.DEFAULT). + andThen(new Function() { + @Override + public Cloner apply(@Nullable BuilderParameters param) { + return new FieldCloner(param.getAggregateCloner(), param.getInstanceProvider()) {{ + // Use a default writer with no concept of cloning + writer = new ObjectWriter(); + }}; + } + }). + build(); + /** * Creates a packet container for a new packet. * @param id - ID of the packet to create. @@ -364,12 +387,29 @@ public class PacketContainer implements Serializable { return id; } + /** + * Create a shallow copy of the current packet. + *

+ * This merely writes the content of each field to the new class directly, + * without performing any expensive copies. + * + * @return A shallow copy of the current packet. + */ + public PacketContainer shallowClone() { + Object clonedPacket = SHALLOW_CLONER.clone(getHandle()); + return new PacketContainer(getID(), clonedPacket); + } + /** * Create a deep copy of the current packet. + *

+ * This will perform a full copy of the entire object tree, only skipping + * known immutable objects and primitive types. + * * @return A deep copy of the current packet. */ public PacketContainer deepClone() { - Object clonedPacket = AggregateCloner.DEFAULT.clone(getHandle()); + Object clonedPacket = DEEP_CLONER.clone(getHandle()); return new PacketContainer(getID(), clonedPacket); } diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java index 233afbbb..07e6fc49 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java @@ -47,6 +47,11 @@ public class AggregateCloner implements Cloner { } } + /** + * Represents a builder for aggregate (combined) cloners. + * + * @author Kristian + */ public static class Builder { private List> factories = Lists.newArrayList(); private BuilderParameters parameters; @@ -75,7 +80,7 @@ public class AggregateCloner implements Cloner { */ public Builder andThen(final Class type) { // Use reflection to generate a factory on the fly - return orCloner(new Function() { + return andThen(new Function() { @Override public Cloner apply(@Nullable BuilderParameters param) { Object result = param.typeConstructor.create(type); @@ -97,7 +102,7 @@ public class AggregateCloner implements Cloner { * @param factory - factory constructing the next cloner. * @return This builder. */ - public Builder orCloner(Function factory) { + public Builder andThen(Function factory) { factories.add(factory); return this; } diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java index 5eb7f514..20149156 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java @@ -10,11 +10,11 @@ import com.comphenix.protocol.reflect.instances.InstanceProvider; * @author Kristian */ public class FieldCloner implements Cloner { - private final Cloner defaultCloner; - private final InstanceProvider instanceProvider; + protected Cloner defaultCloner; + protected InstanceProvider instanceProvider; // Used to clone objects - private final ObjectWriter writer; + protected ObjectWriter writer; /** * Constructs a field cloner that copies objects by reading and writing the internal fields directly. diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/IdentityCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/IdentityCloner.java deleted file mode 100644 index 576baee1..00000000 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/IdentityCloner.java +++ /dev/null @@ -1,18 +0,0 @@ -package com.comphenix.protocol.reflect.cloning; - -/** - * Represents a cloner that simply returns the given object. - * - * @author Kristian - */ -public class IdentityCloner implements Cloner { - @Override - public boolean canClone(Object source) { - return true; - } - - @Override - public Object clone(Object source) { - return source; - } -} From 447f01a3dda28b5c27502978aff19b60158abe24 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 27 Dec 2012 12:12:11 +0100 Subject: [PATCH 29/37] Crash if multiple copies of ProtocolLib are running simultaneously. --- ProtocolLib/dependency-reduced-pom.xml | 33 ++++++++++- ProtocolLib/pom.xml | 1 + .../comphenix/protocol/MinecraftVersion.java | 42 ++++++++++++-- .../comphenix/protocol/ProtocolLibrary.java | 56 ++++++++++++++++++- 4 files changed, 125 insertions(+), 7 deletions(-) diff --git a/ProtocolLib/dependency-reduced-pom.xml b/ProtocolLib/dependency-reduced-pom.xml index 4c88f2ae..c69963b3 100644 --- a/ProtocolLib/dependency-reduced-pom.xml +++ b/ProtocolLib/dependency-reduced-pom.xml @@ -139,7 +139,7 @@ org.bukkit craftbukkit - 1.4.5-R0.3-SNAPSHOT + 1.4.6-R0.1 provided @@ -154,6 +154,36 @@ + + org.mockito + mockito-all + 1.8.4 + test + + + org.powermock + powermock-module-junit4 + 1.5 + test + + + powermock-module-junit4-common + org.powermock + + + + + org.powermock + powermock-api-mockito + 1.5 + test + + + powermock-api-support + org.powermock + + + @@ -181,6 +211,7 @@ + 1.5 cp1252 diff --git a/ProtocolLib/pom.xml b/ProtocolLib/pom.xml index 4fa87e38..d1803e54 100644 --- a/ProtocolLib/pom.xml +++ b/ProtocolLib/pom.xml @@ -216,6 +216,7 @@ org.mockito mockito-all 1.8.4 + test org.powermock diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/MinecraftVersion.java b/ProtocolLib/src/main/java/com/comphenix/protocol/MinecraftVersion.java index 6544ac11..963c8430 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/MinecraftVersion.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/MinecraftVersion.java @@ -7,13 +7,14 @@ import org.bukkit.Server; import com.google.common.base.Objects; import com.google.common.collect.ComparisonChain; +import com.google.common.collect.Ordering; /** * Determine the current Minecraft version. * * @author Kristian */ -class MinecraftVersion implements Comparable { +public class MinecraftVersion implements Comparable { /** * Regular expression used to parse version strings. */ @@ -22,6 +23,9 @@ class MinecraftVersion implements Comparable { private final int major; private final int minor; private final int build; + + // The development stage + private final String development; /** * Determine the current Minecraft version. @@ -36,11 +40,13 @@ class MinecraftVersion implements Comparable { * @param versionOnly - the version in text form. */ public MinecraftVersion(String versionOnly) { - int[] numbers = parseVersion(versionOnly); + String[] section = versionOnly.split("-"); + int[] numbers = parseVersion(section[0]); this.major = numbers[0]; this.minor = numbers[1]; this.build = numbers[2]; + this.development = section.length > 1 ? section[1] : null; } /** @@ -50,9 +56,21 @@ class MinecraftVersion implements Comparable { * @param build - build version number. */ public MinecraftVersion(int major, int minor, int build) { + this(major, minor, build, null); + } + + /** + * Construct a version object directly. + * @param major - major version number. + * @param minor - minor version number. + * @param build - build version number. + * @param development - development stage. + */ + public MinecraftVersion(int major, int minor, int build, String development) { this.major = major; this.minor = minor; this.build = build; + this.development = development; } private int[] parseVersion(String version) { @@ -65,7 +83,7 @@ class MinecraftVersion implements Comparable { // The String 1 or 1.2 is interpreted as 1.0.0 and 1.2.0 respectively. for (int i = 0; i < Math.min(numbers.length, elements.length); i++) - numbers[i] = Integer.parseInt(elements[i].trim()); + numbers[i] = Integer.parseInt(elements[i].trim()); return numbers; } @@ -93,12 +111,23 @@ class MinecraftVersion implements Comparable { return build; } + /** + * Retrieve the development stage. + * @return Development stage, or NULL if this is a release. + */ + public String getDevelopmentStage() { + return development; + } + /** * Retrieve the version String (major.minor.build) only. * @return A normal version string. */ public String getVersion() { - return String.format("%s.%s.%s", major, minor, build); + if (development == null) + return String.format("%s.%s.%s", major, minor, build); + else + return String.format("%s.%s.%s-%s", major, minor, build, development); } @Override @@ -110,6 +139,8 @@ class MinecraftVersion implements Comparable { compare(major, o.major). compare(minor, o.minor). compare(build, o.build). + // No development String means it's a release + compare(development, o.development, Ordering.natural().nullsLast()). result(); } @@ -125,7 +156,8 @@ class MinecraftVersion implements Comparable { return major == other.major && minor == other.minor && - build == other.build; + build == other.build && + Objects.equal(development, other.development); } return false; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolLibrary.java b/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolLibrary.java index e82f47ff..c8c3e0a8 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolLibrary.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolLibrary.java @@ -17,11 +17,14 @@ package com.comphenix.protocol; +import java.io.File; import java.io.IOException; import java.util.logging.Handler; import java.util.logging.Level; import java.util.logging.LogRecord; import java.util.logging.Logger; +import java.util.regex.Matcher; +import java.util.regex.Pattern; import org.bukkit.Server; import org.bukkit.command.CommandExecutor; @@ -97,6 +100,9 @@ public class ProtocolLibrary extends JavaPlugin { private CommandProtocol commandProtocol; private CommandPacket commandPacket; + // Whether or not disable is not needed + private boolean skipDisable; + @Override public void onLoad() { // Load configuration @@ -104,7 +110,6 @@ public class ProtocolLibrary extends JavaPlugin { // Add global parameters DetailedErrorReporter detailedReporter = new DetailedErrorReporter(this); - updater = new Updater(this, logger, "protocollib", getFile(), "protocol.info"); reporter = detailedReporter; try { @@ -121,6 +126,12 @@ public class ProtocolLibrary extends JavaPlugin { } try { + // Check for other versions + checkConflictingVersions(); + + // Set updater + updater = new Updater(this, logger, "protocollib", getFile(), "protocol.info"); + unhookTask = new DelayedSingleTask(this); protocolManager = new PacketFilterManager(getClassLoader(), getServer(), unhookTask, detailedReporter); detailedReporter.addGlobalParameter("manager", protocolManager); @@ -252,6 +263,45 @@ public class ProtocolLibrary extends JavaPlugin { reporter.reportWarning(this, "Unable to retrieve current Minecraft version.", e); } } + + private void checkConflictingVersions() { + Pattern ourPlugin = Pattern.compile("ProtocolLib-(.*)\\.jar"); + MinecraftVersion currentVersion = new MinecraftVersion(this.getDescription().getVersion()); + MinecraftVersion newestVersion = null; + + try { + // Scan the plugin folder for newer versions of ProtocolLib + File pluginFolder = new File("plugins/"); + + for (File candidate : pluginFolder.listFiles()) { + if (candidate.isFile()) { + Matcher match = ourPlugin.matcher(candidate.getName()); + + if (match.matches()) { + MinecraftVersion version = new MinecraftVersion(match.group(1)); + + if (newestVersion == null || newestVersion.compareTo(version) < 0) { + newestVersion = version; + } + } + } + } + + } catch (Exception e) { + reporter.reportWarning(this, "Unable to detect conflicting plugin versions.", e); + } + + // See if the newest version is actually higher + if (newestVersion != null && currentVersion.compareTo(newestVersion) < 0) { + // We don't need to set internal classes or instances to NULL - that would break the other loaded plugin + skipDisable = true; + + throw new IllegalStateException( + String.format("Detected a newer version of ProtocolLib (%s) in plugin folder than the current (%s). Disabling.", + newestVersion.getVersion(), currentVersion.getVersion()) + ); + } + } private void registerCommand(String name, CommandExecutor executor) { try { @@ -331,6 +381,10 @@ public class ProtocolLibrary extends JavaPlugin { @Override public void onDisable() { + if (skipDisable) { + return; + } + // Disable compiler if (backgroundCompiler != null) { backgroundCompiler.shutdownAll(); From 0f30a1151062e1c499af33b898586ddca170bfb6 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 27 Dec 2012 12:12:44 +0100 Subject: [PATCH 30/37] Bumping version to 1.8.5-SNAPSHOT. --- ProtocolLib/pom.xml | 2 +- ProtocolLib/src/main/resources/plugin.yml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/ProtocolLib/pom.xml b/ProtocolLib/pom.xml index d1803e54..5b25746e 100644 --- a/ProtocolLib/pom.xml +++ b/ProtocolLib/pom.xml @@ -2,7 +2,7 @@ 4.0.0 com.comphenix.protocol ProtocolLib - 1.8.4-SNAPSHOT + 1.8.5-SNAPSHOT jar Provides read/write access to the Minecraft protocol. diff --git a/ProtocolLib/src/main/resources/plugin.yml b/ProtocolLib/src/main/resources/plugin.yml index 04d9f25e..eec500c3 100644 --- a/ProtocolLib/src/main/resources/plugin.yml +++ b/ProtocolLib/src/main/resources/plugin.yml @@ -1,5 +1,5 @@ name: ProtocolLib -version: 1.8.4-SNAPSHOT +version: 1.8.5-SNAPSHOT description: Provides read/write access to the Minecraft protocol. author: Comphenix website: http://www.comphenix.net/ProtocolLib From 01b9e15930e88bd7c5bf305f52bce02fff0e9dac Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 27 Dec 2012 14:23:22 +0100 Subject: [PATCH 31/37] No need to expose the MinecraftVersion class. --- ProtocolLib/dependency-reduced-pom.xml | 2 +- .../src/main/java/com/comphenix/protocol/MinecraftVersion.java | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/ProtocolLib/dependency-reduced-pom.xml b/ProtocolLib/dependency-reduced-pom.xml index c69963b3..893aad2e 100644 --- a/ProtocolLib/dependency-reduced-pom.xml +++ b/ProtocolLib/dependency-reduced-pom.xml @@ -4,7 +4,7 @@ com.comphenix.protocol ProtocolLib ProtocolLib - 1.8.4-SNAPSHOT + 1.8.5-SNAPSHOT Provides read/write access to the Minecraft protocol. http://dev.bukkit.org/server-mods/protocollib/ diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/MinecraftVersion.java b/ProtocolLib/src/main/java/com/comphenix/protocol/MinecraftVersion.java index 963c8430..87ddbac2 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/MinecraftVersion.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/MinecraftVersion.java @@ -14,7 +14,7 @@ import com.google.common.collect.Ordering; * * @author Kristian */ -public class MinecraftVersion implements Comparable { +class MinecraftVersion implements Comparable { /** * Regular expression used to parse version strings. */ From f0e7b9aec7a433b5d664970d44eecfa86e3180cd Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 27 Dec 2012 14:28:33 +0100 Subject: [PATCH 32/37] Added missing copyright notices (GPL). --- .../protocol/CleanupStaticMembers.java | 17 +++++++++++++++++ .../com/comphenix/protocol/CommandBase.java | 17 +++++++++++++++++ .../com/comphenix/protocol/CommandPacket.java | 17 +++++++++++++++++ .../com/comphenix/protocol/CommandProtocol.java | 17 +++++++++++++++++ .../comphenix/protocol/MinecraftVersion.java | 17 +++++++++++++++++ .../com/comphenix/protocol/ProtocolConfig.java | 17 +++++++++++++++++ .../com/comphenix/protocol/RangeParser.java | 17 +++++++++++++++++ .../protocol/async/PlayerSendingHandler.java | 17 +++++++++++++++++ .../concurrency/AbstractIntervalTree.java | 17 +++++++++++++++++ .../protocol/concurrency/BlockingHashMap.java | 17 +++++++++++++++++ .../protocol/error/DetailedErrorReporter.java | 17 +++++++++++++++++ .../comphenix/protocol/error/ErrorReporter.java | 17 +++++++++++++++++ .../protocol/events/MonitorAdapter.java | 17 +++++++++++++++++ .../protocol/injector/DelayedSingleTask.java | 17 +++++++++++++++++ .../comphenix/protocol/injector/GamePhase.java | 17 +++++++++++++++++ .../protocol/injector/player/IntegerSet.java | 17 +++++++++++++++++ .../injector/player/NetLoginInjector.java | 17 +++++++++++++++++ .../injector/player/TemporaryPlayerFactory.java | 17 +++++++++++++++++ .../injector/player/UnsupportedListener.java | 17 +++++++++++++++++ .../protocol/reflect/PrettyPrinter.java | 17 +++++++++++++++++ .../reflect/cloning/AggregateCloner.java | 17 +++++++++++++++++ .../protocol/reflect/cloning/BukkitCloner.java | 17 +++++++++++++++++ .../protocol/reflect/cloning/Cloner.java | 17 +++++++++++++++++ .../reflect/cloning/CollectionCloner.java | 17 +++++++++++++++++ .../protocol/reflect/cloning/FieldCloner.java | 17 +++++++++++++++++ .../reflect/cloning/ImmutableDetector.java | 17 +++++++++++++++++ .../reflect/cloning/NullableCloner.java | 17 +++++++++++++++++ .../protocol/utility/CachedPackage.java | 17 +++++++++++++++++ .../protocol/utility/ChatExtensions.java | 17 +++++++++++++++++ .../protocol/utility/MinecraftReflection.java | 17 +++++++++++++++++ .../protocol/wrappers/BukkitConverters.java | 17 +++++++++++++++++ .../protocol/wrappers/ChunkPosition.java | 17 +++++++++++++++++ .../wrappers/WrappedChunkCoordinate.java | 17 +++++++++++++++++ .../protocol/wrappers/WrappedDataWatcher.java | 17 +++++++++++++++++ .../wrappers/WrappedWatchableObject.java | 17 +++++++++++++++++ 35 files changed, 595 insertions(+) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/CleanupStaticMembers.java b/ProtocolLib/src/main/java/com/comphenix/protocol/CleanupStaticMembers.java index 69d82099..f4940bee 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/CleanupStaticMembers.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/CleanupStaticMembers.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol; import java.lang.reflect.Field; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/CommandBase.java b/ProtocolLib/src/main/java/com/comphenix/protocol/CommandBase.java index 7cda5346..18ddb25e 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/CommandBase.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/CommandBase.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol; import org.bukkit.ChatColor; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/CommandPacket.java b/ProtocolLib/src/main/java/com/comphenix/protocol/CommandPacket.java index 5719a097..4da7a3c4 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/CommandPacket.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/CommandPacket.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol; import java.lang.reflect.InvocationTargetException; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/CommandProtocol.java b/ProtocolLib/src/main/java/com/comphenix/protocol/CommandProtocol.java index d4c6aa05..c35f7632 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/CommandProtocol.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/CommandProtocol.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol; import org.bukkit.ChatColor; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/MinecraftVersion.java b/ProtocolLib/src/main/java/com/comphenix/protocol/MinecraftVersion.java index 87ddbac2..53fb9898 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/MinecraftVersion.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/MinecraftVersion.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol; import java.util.regex.Matcher; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolConfig.java b/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolConfig.java index f135fa97..040a8ef1 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolConfig.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/ProtocolConfig.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol; import java.io.File; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/RangeParser.java b/ProtocolLib/src/main/java/com/comphenix/protocol/RangeParser.java index d396be43..f893f731 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/RangeParser.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/RangeParser.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol; import java.util.ArrayList; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/async/PlayerSendingHandler.java b/ProtocolLib/src/main/java/com/comphenix/protocol/async/PlayerSendingHandler.java index 8dc4d0be..e2aa0f09 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/async/PlayerSendingHandler.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/async/PlayerSendingHandler.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.async; import java.util.ArrayList; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/concurrency/AbstractIntervalTree.java b/ProtocolLib/src/main/java/com/comphenix/protocol/concurrency/AbstractIntervalTree.java index 9b241e90..7d29ddf2 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/concurrency/AbstractIntervalTree.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/concurrency/AbstractIntervalTree.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.concurrency; import java.util.HashSet; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/concurrency/BlockingHashMap.java b/ProtocolLib/src/main/java/com/comphenix/protocol/concurrency/BlockingHashMap.java index 793f3c11..df867951 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/concurrency/BlockingHashMap.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/concurrency/BlockingHashMap.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.concurrency; import java.util.Collection; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java b/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java index ec878a62..09786d1a 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/error/DetailedErrorReporter.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.error; import java.io.PrintWriter; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/error/ErrorReporter.java b/ProtocolLib/src/main/java/com/comphenix/protocol/error/ErrorReporter.java index 54b45f48..e1b7f927 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/error/ErrorReporter.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/error/ErrorReporter.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.error; import org.bukkit.plugin.Plugin; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/events/MonitorAdapter.java b/ProtocolLib/src/main/java/com/comphenix/protocol/events/MonitorAdapter.java index 832c2d8f..358c6bd0 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/events/MonitorAdapter.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/events/MonitorAdapter.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.events; import java.util.logging.Level; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/DelayedSingleTask.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/DelayedSingleTask.java index b44de2ee..41242067 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/DelayedSingleTask.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/DelayedSingleTask.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.injector; import org.bukkit.plugin.Plugin; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/GamePhase.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/GamePhase.java index 07c7e7a5..399b3350 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/GamePhase.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/GamePhase.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.injector; /** diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/IntegerSet.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/IntegerSet.java index b877bc2a..3deea142 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/IntegerSet.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/IntegerSet.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.injector.player; import java.util.Arrays; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetLoginInjector.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetLoginInjector.java index 9ac2f5f8..dd03ea5d 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetLoginInjector.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/NetLoginInjector.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.injector.player; import java.util.concurrent.ConcurrentMap; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/TemporaryPlayerFactory.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/TemporaryPlayerFactory.java index 891edc7c..45942257 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/TemporaryPlayerFactory.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/TemporaryPlayerFactory.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.injector.player; import java.lang.reflect.InvocationTargetException; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/UnsupportedListener.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/UnsupportedListener.java index 2e4cbe5a..121428d8 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/UnsupportedListener.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/UnsupportedListener.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.injector.player; import java.util.Arrays; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/PrettyPrinter.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/PrettyPrinter.java index 33ceb604..ad285df8 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/PrettyPrinter.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/PrettyPrinter.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.reflect; import java.lang.reflect.Array; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java index 07e6fc49..eefc3235 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.reflect.cloning; import java.lang.ref.WeakReference; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/BukkitCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/BukkitCloner.java index 290fb48a..63456619 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/BukkitCloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/BukkitCloner.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.reflect.cloning; import com.comphenix.protocol.reflect.EquivalentConverter; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/Cloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/Cloner.java index c7fd0cf6..a15455dd 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/Cloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/Cloner.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.reflect.cloning; /** diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/CollectionCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/CollectionCloner.java index 5fd9f3bc..3f5c203e 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/CollectionCloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/CollectionCloner.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.reflect.cloning; import java.lang.reflect.Array; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java index 20149156..d5e3daa7 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.reflect.cloning; import com.comphenix.protocol.reflect.ObjectWriter; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/ImmutableDetector.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/ImmutableDetector.java index 7ac3fee0..b1ca310c 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/ImmutableDetector.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/ImmutableDetector.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.reflect.cloning; import java.math.BigDecimal; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/NullableCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/NullableCloner.java index cfeaea55..2531edf2 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/NullableCloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/NullableCloner.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.reflect.cloning; /** diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/CachedPackage.java b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/CachedPackage.java index ba4a19f8..9081705e 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/CachedPackage.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/CachedPackage.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.utility; import java.util.Map; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/ChatExtensions.java b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/ChatExtensions.java index cdacdbdd..92af6cca 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/ChatExtensions.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/ChatExtensions.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.utility; import java.lang.reflect.InvocationTargetException; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java index b02e39a1..31094d57 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/utility/MinecraftReflection.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.utility; import java.lang.reflect.Array; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/BukkitConverters.java b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/BukkitConverters.java index d8da3e31..7d6f421e 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/BukkitConverters.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/BukkitConverters.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.wrappers; import java.lang.ref.WeakReference; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/ChunkPosition.java b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/ChunkPosition.java index 718f227f..d5326a85 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/ChunkPosition.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/ChunkPosition.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.wrappers; import java.lang.reflect.Constructor; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedChunkCoordinate.java b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedChunkCoordinate.java index 2b6bb244..e673eec4 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedChunkCoordinate.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedChunkCoordinate.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.wrappers; import com.comphenix.protocol.reflect.StructureModifier; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedDataWatcher.java b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedDataWatcher.java index d1b45bae..012d9458 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedDataWatcher.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedDataWatcher.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.wrappers; import java.lang.reflect.Field; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedWatchableObject.java b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedWatchableObject.java index 003385d1..c1b2edb7 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedWatchableObject.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/wrappers/WrappedWatchableObject.java @@ -1,3 +1,20 @@ +/* + * ProtocolLib - Bukkit server library that allows access to the Minecraft protocol. + * Copyright (C) 2012 Kristian S. Stangeland + * + * This program is free software; you can redistribute it and/or modify it under the terms of the + * GNU General Public License as published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; + * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. + * See the GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with this program; + * if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA + * 02111-1307 USA + */ + package com.comphenix.protocol.wrappers; import java.lang.reflect.Constructor; From d16e0bf1c60a166586c08aebabd069044a7970d4 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 27 Dec 2012 15:06:25 +0100 Subject: [PATCH 33/37] Add support for cloning maps. --- ProtocolLib/.project | 6 ++ .../reflect/cloning/CollectionCloner.java | 73 ++++++++++++++----- 2 files changed, 61 insertions(+), 18 deletions(-) diff --git a/ProtocolLib/.project b/ProtocolLib/.project index 3b5df9da..a1960c9e 100644 --- a/ProtocolLib/.project +++ b/ProtocolLib/.project @@ -15,9 +15,15 @@ + + net.sourceforge.metrics.builder + + + org.eclipse.m2e.core.maven2Nature org.eclipse.jdt.core.javanature + net.sourceforge.metrics.nature diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/CollectionCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/CollectionCloner.java index 3f5c203e..dbb56115 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/CollectionCloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/CollectionCloner.java @@ -20,6 +20,8 @@ package com.comphenix.protocol.reflect.cloning; import java.lang.reflect.Array; import java.lang.reflect.Constructor; import java.util.Collection; +import java.util.Map; +import java.util.Map.Entry; /** * Attempts to clone collection and array classes. @@ -43,9 +45,9 @@ public class CollectionCloner implements Cloner { return false; Class clazz = source.getClass(); - return Collection.class.isAssignableFrom(clazz) || clazz.isArray(); + return Collection.class.isAssignableFrom(clazz) || Map.class.isAssignableFrom(clazz) || clazz.isArray(); } - + @SuppressWarnings("unchecked") @Override public Object clone(Object source) { @@ -55,31 +57,32 @@ public class CollectionCloner implements Cloner { Class clazz = source.getClass(); if (source instanceof Collection) { - Collection copy = null; - - // Not all collections implement "clone", but most *do* implement the "copy constructor" pattern - try { - Constructor constructCopy = clazz.getConstructor(Collection.class); - copy = (Collection) constructCopy.newInstance(source); - } catch (NoSuchMethodException e) { - copy = (Collection) cloneObject(clazz, source); - } catch (Exception e) { - throw new RuntimeException("Cannot construct collection.", e); - } + Collection copy = cloneConstructor(Collection.class, clazz, source); // Next, clone each element in the collection copy.clear(); for (Object element : (Collection) source) { - if (defaultCloner.canClone(element)) - copy.add(defaultCloner.clone(element)); - else - throw new IllegalArgumentException("Cannot clone " + element + " in collection " + source); + copy.add(getClone(element, source)); + } + + return copy; + + } else if (source instanceof Map) { + + Map copy = cloneConstructor(Map.class, clazz, source); + + // Next, clone each element in the collection + copy.clear(); + + for (Entry element : ((Map) source).entrySet()) { + Object key = getClone(element.getKey(), source); + Object value = getClone(element.getValue(), source); + copy.put(key, value); } return copy; - // Second possibility } else if (clazz.isArray()) { // Get the length int lenght = Array.getLength(source); @@ -110,6 +113,19 @@ public class CollectionCloner implements Cloner { throw new IllegalArgumentException(source + " is not an array nor a Collection."); } + /** + * Clone an element using the default cloner. + * @param element - the element to clone. + * @param container - where the element is stored. + * @return The cloned element. + */ + private Object getClone(Object element, Object container) { + if (defaultCloner.canClone(element)) + return defaultCloner.clone(element); + else + throw new IllegalArgumentException("Cannot clone " + element + " in container " + container); + } + /** * Clone a primitive or immutable array by calling its clone method. * @param component - the component type of the array. @@ -138,6 +154,27 @@ public class CollectionCloner implements Cloner { return ((Object[]) source).clone(); } + /** + * Clone an object by calling its clone constructor, or alternatively, a "clone" method. + * @param superclass - the superclass we expect in the clone constructor. + * @param clazz - the class of the object. + * @param source - the object itself. + * @return A cloned object. + */ + @SuppressWarnings("unchecked") + private T cloneConstructor(Class superclass, Class clazz, Object source) { + + // Not all collections or maps implement "clone", but most *do* implement the "copy constructor" pattern + try { + Constructor constructCopy = clazz.getConstructor(Collection.class); + return (T) constructCopy.newInstance(source); + } catch (NoSuchMethodException e) { + return (T) cloneObject(clazz, source); + } catch (Exception e) { + throw new RuntimeException("Cannot construct collection.", e); + } + } + /** * Clone an object by calling "clone" using reflection. * @param clazz - the class type. From 20ab2a71ac27ac1c0f4e0b5cdd3f169dd165ebf4 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 27 Dec 2012 15:11:08 +0100 Subject: [PATCH 34/37] Assume SecretKey and PublicKey is immutable too. --- .../protocol/reflect/cloning/ImmutableDetector.java | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/ImmutableDetector.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/ImmutableDetector.java index b1ca310c..143e9d04 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/ImmutableDetector.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/ImmutableDetector.java @@ -24,9 +24,12 @@ import java.net.Inet6Address; import java.net.InetSocketAddress; import java.net.URI; import java.net.URL; +import java.security.PublicKey; import java.util.Locale; import java.util.UUID; +import javax.crypto.SecretKey; + import com.google.common.primitives.Primitives; /** @@ -42,7 +45,8 @@ public class ImmutableDetector implements Cloner { StackTraceElement.class, BigDecimal.class, BigInteger.class, Locale.class, UUID.class, URL.class, URI.class, Inet4Address.class, - Inet6Address.class, InetSocketAddress.class + Inet6Address.class, InetSocketAddress.class, + SecretKey.class, PublicKey.class }; @Override From 0d34b82d7208d9ea9399efa2365a59fb8bd2cdb7 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 27 Dec 2012 16:28:30 +0100 Subject: [PATCH 35/37] Don't copy inflated buffers - they won't be modified in any case. Resulted in some pretty ugly factory code. --- .../protocol/events/PacketContainer.java | 36 ++++++++++++++++++- .../reflect/cloning/AggregateCloner.java | 5 +++ .../protocol/reflect/cloning/FieldCloner.java | 17 +++++++-- 3 files changed, 55 insertions(+), 3 deletions(-) diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java b/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java index 266d45cd..c724a266 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java @@ -43,8 +43,11 @@ import com.comphenix.protocol.reflect.FuzzyReflection; import com.comphenix.protocol.reflect.ObjectWriter; import com.comphenix.protocol.reflect.StructureModifier; import com.comphenix.protocol.reflect.cloning.AggregateCloner; +import com.comphenix.protocol.reflect.cloning.BukkitCloner; import com.comphenix.protocol.reflect.cloning.Cloner; +import com.comphenix.protocol.reflect.cloning.CollectionCloner; import com.comphenix.protocol.reflect.cloning.FieldCloner; +import com.comphenix.protocol.reflect.cloning.ImmutableDetector; import com.comphenix.protocol.reflect.cloning.AggregateCloner.BuilderParameters; import com.comphenix.protocol.reflect.instances.DefaultInstances; import com.comphenix.protocol.utility.MinecraftReflection; @@ -78,7 +81,14 @@ public class PacketContainer implements Serializable { private static ConcurrentMap, Method> readMethods = Maps.newConcurrentMap(); // Used to clone packets - private static final AggregateCloner DEEP_CLONER = AggregateCloner.DEFAULT; + private static final AggregateCloner DEEP_CLONER = AggregateCloner.newBuilder(). + instanceProvider(DefaultInstances.DEFAULT). + andThen(BukkitCloner.class). + andThen(ImmutableDetector.class). + andThen(CollectionCloner.class). + andThen(getSpecializedDeepClonerFactory()). + build(); + private static final AggregateCloner SHALLOW_CLONER = AggregateCloner.newBuilder(). instanceProvider(DefaultInstances.DEFAULT). andThen(new Function() { @@ -405,6 +415,8 @@ public class PacketContainer implements Serializable { *

* This will perform a full copy of the entire object tree, only skipping * known immutable objects and primitive types. + *

+ * Note that the inflated buffers in packet 51 and 56 will be copied directly to save memory. * * @return A deep copy of the current packet. */ @@ -413,6 +425,28 @@ public class PacketContainer implements Serializable { return new PacketContainer(getID(), clonedPacket); } + // To save space, we'll skip copying the inflated buffers in packet 51 and 56 + private static Function getSpecializedDeepClonerFactory() { + // Look at what you've made me do Java, look at it!! + return new Function() { + @Override + public Cloner apply(@Nullable BuilderParameters param) { + return new FieldCloner(param.getAggregateCloner(), param.getInstanceProvider()) {{ + this.writer = new ObjectWriter() { + protected void transformField(StructureModifier modifierSource, + StructureModifier modifierDest, int fieldIndex) { + // No need to clone inflated buffers + if (modifierSource.getField(fieldIndex).getName().startsWith("inflatedBuffer")) + modifierDest.write(fieldIndex, modifierSource.read(fieldIndex)); + else + defaultTransform(modifierSource, modifierDest, getDefaultCloner(), fieldIndex); + }; + }; + }}; + } + }; + } + private void writeObject(ObjectOutputStream output) throws IOException { // Default serialization output.defaultWriteObject(); diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java index eefc3235..19874b5a 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/AggregateCloner.java @@ -35,6 +35,11 @@ import com.google.common.collect.Lists; * @author Kristian */ public class AggregateCloner implements Cloner { + /** + * Supplies the cloner factories with necessary parameters. + * + * @author Kristian + */ public static class BuilderParameters { // Can only be modified by the builder private InstanceProvider instanceProvider; diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java index d5e3daa7..62a07bc7 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/reflect/cloning/FieldCloner.java @@ -47,12 +47,25 @@ public class FieldCloner implements Cloner { @Override protected void transformField(StructureModifier modifierSource, StructureModifier modifierDest, int fieldIndex) { - Object value = modifierSource.read(fieldIndex); - modifierDest.write(fieldIndex, getDefaultCloner().clone(value)); + defaultTransform(modifierDest, modifierDest, getDefaultCloner(), fieldIndex); } }; } + /** + * Default implementation of the field transform. Applies a clone operation before a field value is written. + * @param modifierSource - modifier for the original object. + * @param modifierDest - modifier for the new cloned object. + * @param defaultCloner - cloner to use. + * @param fieldIndex - the current field index. + */ + protected void defaultTransform(StructureModifier modifierSource, + StructureModifier modifierDest, Cloner defaultCloner, int fieldIndex) { + + Object value = modifierSource.read(fieldIndex); + modifierDest.write(fieldIndex, defaultCloner.clone(value)); + } + @Override public boolean canClone(Object source) { if (source == null) From 82ec3dc5f1cf8b06113e40f6f850208f1b66f021 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 27 Dec 2012 17:03:30 +0100 Subject: [PATCH 36/37] Bumping to 1.9.0 to indicate that the cloning API is improved. --- ProtocolLib/pom.xml | 2 +- ProtocolLib/src/main/resources/plugin.yml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/ProtocolLib/pom.xml b/ProtocolLib/pom.xml index 5b25746e..c95ebec1 100644 --- a/ProtocolLib/pom.xml +++ b/ProtocolLib/pom.xml @@ -2,7 +2,7 @@ 4.0.0 com.comphenix.protocol ProtocolLib - 1.8.5-SNAPSHOT + 1.9.0 jar Provides read/write access to the Minecraft protocol. diff --git a/ProtocolLib/src/main/resources/plugin.yml b/ProtocolLib/src/main/resources/plugin.yml index eec500c3..68783116 100644 --- a/ProtocolLib/src/main/resources/plugin.yml +++ b/ProtocolLib/src/main/resources/plugin.yml @@ -1,5 +1,5 @@ name: ProtocolLib -version: 1.8.5-SNAPSHOT +version: 1.9.0 description: Provides read/write access to the Minecraft protocol. author: Comphenix website: http://www.comphenix.net/ProtocolLib From fd69f08a1d4be7b0cc8db8e03e35cca2265411ab Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Thu, 27 Dec 2012 17:37:45 +0100 Subject: [PATCH 37/37] Maven crap. --- ProtocolLib/dependency-reduced-pom.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ProtocolLib/dependency-reduced-pom.xml b/ProtocolLib/dependency-reduced-pom.xml index 893aad2e..ea37303c 100644 --- a/ProtocolLib/dependency-reduced-pom.xml +++ b/ProtocolLib/dependency-reduced-pom.xml @@ -4,7 +4,7 @@ com.comphenix.protocol ProtocolLib ProtocolLib - 1.8.5-SNAPSHOT + 1.9.0 Provides read/write access to the Minecraft protocol. http://dev.bukkit.org/server-mods/protocollib/