Added support for cloing ServerPing packets.
Dieser Commit ist enthalten in:
Ursprung
ee53fc0d5f
Commit
922fb94804
@ -51,6 +51,7 @@ import com.comphenix.protocol.metrics.Updater.UpdateResult;
|
|||||||
import com.comphenix.protocol.metrics.Updater.UpdateType;
|
import com.comphenix.protocol.metrics.Updater.UpdateType;
|
||||||
import com.comphenix.protocol.reflect.compiler.BackgroundCompiler;
|
import com.comphenix.protocol.reflect.compiler.BackgroundCompiler;
|
||||||
import com.comphenix.protocol.utility.ChatExtensions;
|
import com.comphenix.protocol.utility.ChatExtensions;
|
||||||
|
import com.comphenix.protocol.utility.EnhancerFactory;
|
||||||
import com.comphenix.protocol.utility.MinecraftVersion;
|
import com.comphenix.protocol.utility.MinecraftVersion;
|
||||||
import com.google.common.base.Splitter;
|
import com.google.common.base.Splitter;
|
||||||
import com.google.common.collect.Iterables;
|
import com.google.common.collect.Iterables;
|
||||||
@ -119,7 +120,7 @@ public class ProtocolLibrary extends JavaPlugin {
|
|||||||
// Executors
|
// Executors
|
||||||
private static ListeningScheduledExecutorService executorAsync;
|
private static ListeningScheduledExecutorService executorAsync;
|
||||||
private static ListeningScheduledExecutorService executorSync;
|
private static ListeningScheduledExecutorService executorSync;
|
||||||
|
|
||||||
// Structure compiler
|
// Structure compiler
|
||||||
private BackgroundCompiler backgroundCompiler;
|
private BackgroundCompiler backgroundCompiler;
|
||||||
|
|
||||||
@ -156,6 +157,9 @@ public class ProtocolLibrary extends JavaPlugin {
|
|||||||
// Load configuration
|
// Load configuration
|
||||||
logger = getLoggerSafely();
|
logger = getLoggerSafely();
|
||||||
|
|
||||||
|
// Initialize enhancer factory
|
||||||
|
EnhancerFactory.getInstance().setClassLoader(getClassLoader());
|
||||||
|
|
||||||
// Initialize executors
|
// Initialize executors
|
||||||
executorAsync = BukkitExecutors.newAsynchronous(this);
|
executorAsync = BukkitExecutors.newAsynchronous(this);
|
||||||
executorSync = BukkitExecutors.newSynchronous(this);
|
executorSync = BukkitExecutors.newSynchronous(this);
|
||||||
|
@ -35,6 +35,8 @@ import org.bukkit.OfflinePlayer;
|
|||||||
import org.bukkit.World;
|
import org.bukkit.World;
|
||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
|
|
||||||
|
import com.comphenix.protocol.utility.EnhancerFactory;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents a player object that can be serialized by Java.
|
* Represents a player object that can be serialized by Java.
|
||||||
*
|
*
|
||||||
@ -208,7 +210,7 @@ class SerializedOfflinePlayer implements OfflinePlayer, Serializable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// MORE CGLIB magic!
|
// MORE CGLIB magic!
|
||||||
Enhancer ex = new Enhancer();
|
Enhancer ex = EnhancerFactory.getInstance().createEnhancer();
|
||||||
ex.setSuperclass(Player.class);
|
ex.setSuperclass(Player.class);
|
||||||
ex.setCallback(new MethodInterceptor() {
|
ex.setCallback(new MethodInterceptor() {
|
||||||
@Override
|
@Override
|
||||||
|
@ -68,6 +68,7 @@ import com.comphenix.protocol.injector.player.PlayerInjectionHandler.ConflictStr
|
|||||||
import com.comphenix.protocol.injector.spigot.SpigotPacketInjector;
|
import com.comphenix.protocol.injector.spigot.SpigotPacketInjector;
|
||||||
import com.comphenix.protocol.reflect.FieldAccessException;
|
import com.comphenix.protocol.reflect.FieldAccessException;
|
||||||
import com.comphenix.protocol.reflect.FuzzyReflection;
|
import com.comphenix.protocol.reflect.FuzzyReflection;
|
||||||
|
import com.comphenix.protocol.utility.EnhancerFactory;
|
||||||
import com.comphenix.protocol.utility.MinecraftReflection;
|
import com.comphenix.protocol.utility.MinecraftReflection;
|
||||||
import com.comphenix.protocol.utility.MinecraftVersion;
|
import com.comphenix.protocol.utility.MinecraftVersion;
|
||||||
import com.google.common.base.Objects;
|
import com.google.common.base.Objects;
|
||||||
@ -232,7 +233,7 @@ public final class PacketFilterManager implements ProtocolManager, ListenerInvok
|
|||||||
this.loginPackets = new LoginPackets(minecraftVersion);
|
this.loginPackets = new LoginPackets(minecraftVersion);
|
||||||
|
|
||||||
// The write packet interceptor
|
// The write packet interceptor
|
||||||
this.interceptWritePacket = new InterceptWritePacket(classLoader, reporter);
|
this.interceptWritePacket = new InterceptWritePacket(reporter);
|
||||||
|
|
||||||
// Use the correct injection type
|
// Use the correct injection type
|
||||||
if (MinecraftReflection.isUsingNetty()) {
|
if (MinecraftReflection.isUsingNetty()) {
|
||||||
@ -241,7 +242,7 @@ public final class PacketFilterManager implements ProtocolManager, ListenerInvok
|
|||||||
this.packetInjector = nettyInjector.getPacketInjector();
|
this.packetInjector = nettyInjector.getPacketInjector();
|
||||||
|
|
||||||
} else if (builder.isNettyEnabled()) {
|
} else if (builder.isNettyEnabled()) {
|
||||||
this.spigotInjector = new SpigotPacketInjector(classLoader, reporter, this, server);
|
this.spigotInjector = new SpigotPacketInjector(reporter, this, server);
|
||||||
this.playerInjection = spigotInjector.getPlayerHandler();
|
this.playerInjection = spigotInjector.getPlayerHandler();
|
||||||
this.packetInjector = spigotInjector.getPacketInjector();
|
this.packetInjector = spigotInjector.getPacketInjector();
|
||||||
|
|
||||||
@ -249,7 +250,6 @@ public final class PacketFilterManager implements ProtocolManager, ListenerInvok
|
|||||||
spigotInjector.setProxyPacketInjector(PacketInjectorBuilder.newBuilder().
|
spigotInjector.setProxyPacketInjector(PacketInjectorBuilder.newBuilder().
|
||||||
invoker(this).
|
invoker(this).
|
||||||
reporter(reporter).
|
reporter(reporter).
|
||||||
classLoader(classLoader).
|
|
||||||
playerInjection(playerInjection).
|
playerInjection(playerInjection).
|
||||||
buildInjector()
|
buildInjector()
|
||||||
);
|
);
|
||||||
@ -260,7 +260,6 @@ public final class PacketFilterManager implements ProtocolManager, ListenerInvok
|
|||||||
invoker(this).
|
invoker(this).
|
||||||
server(server).
|
server(server).
|
||||||
reporter(reporter).
|
reporter(reporter).
|
||||||
classLoader(classLoader).
|
|
||||||
packetListeners(packetListeners).
|
packetListeners(packetListeners).
|
||||||
injectionFilter(isInjectionNecessary).
|
injectionFilter(isInjectionNecessary).
|
||||||
version(builder.getMinecraftVersion()).
|
version(builder.getMinecraftVersion()).
|
||||||
@ -269,7 +268,6 @@ public final class PacketFilterManager implements ProtocolManager, ListenerInvok
|
|||||||
this.packetInjector = PacketInjectorBuilder.newBuilder().
|
this.packetInjector = PacketInjectorBuilder.newBuilder().
|
||||||
invoker(this).
|
invoker(this).
|
||||||
reporter(reporter).
|
reporter(reporter).
|
||||||
classLoader(classLoader).
|
|
||||||
playerInjection(playerInjection).
|
playerInjection(playerInjection).
|
||||||
buildInjector();
|
buildInjector();
|
||||||
}
|
}
|
||||||
@ -1077,9 +1075,9 @@ public final class PacketFilterManager implements ProtocolManager, ListenerInvok
|
|||||||
Method registerEvent = FuzzyReflection.fromObject(manager).getMethodByParameters("registerEvent",
|
Method registerEvent = FuzzyReflection.fromObject(manager).getMethodByParameters("registerEvent",
|
||||||
eventTypes, Listener.class, eventPriority, Plugin.class);
|
eventTypes, Listener.class, eventPriority, Plugin.class);
|
||||||
|
|
||||||
Enhancer playerLow = new Enhancer();
|
Enhancer playerLow = EnhancerFactory.getInstance().createEnhancer();
|
||||||
Enhancer playerEx = new Enhancer();
|
Enhancer playerEx = EnhancerFactory.getInstance().createEnhancer();
|
||||||
Enhancer serverEx = new Enhancer();
|
Enhancer serverEx = EnhancerFactory.getInstance().createEnhancer();
|
||||||
|
|
||||||
playerLow.setSuperclass(playerListener);
|
playerLow.setSuperclass(playerListener);
|
||||||
playerLow.setClassLoader(classLoader);
|
playerLow.setClassLoader(classLoader);
|
||||||
|
@ -15,6 +15,7 @@ import com.comphenix.protocol.events.NetworkMarker;
|
|||||||
import com.comphenix.protocol.events.PacketEvent;
|
import com.comphenix.protocol.events.PacketEvent;
|
||||||
import com.comphenix.protocol.reflect.MethodInfo;
|
import com.comphenix.protocol.reflect.MethodInfo;
|
||||||
import com.comphenix.protocol.reflect.fuzzy.FuzzyMethodContract;
|
import com.comphenix.protocol.reflect.fuzzy.FuzzyMethodContract;
|
||||||
|
import com.comphenix.protocol.utility.EnhancerFactory;
|
||||||
import com.comphenix.protocol.utility.MinecraftReflection;
|
import com.comphenix.protocol.utility.MinecraftReflection;
|
||||||
import com.google.common.collect.Maps;
|
import com.google.common.collect.Maps;
|
||||||
|
|
||||||
@ -39,15 +40,12 @@ public class InterceptWritePacket {
|
|||||||
private boolean writePacketIntercepted;
|
private boolean writePacketIntercepted;
|
||||||
|
|
||||||
private ConcurrentMap<Integer, Class<?>> proxyClasses = Maps.newConcurrentMap();
|
private ConcurrentMap<Integer, Class<?>> proxyClasses = Maps.newConcurrentMap();
|
||||||
|
|
||||||
private ClassLoader classLoader;
|
|
||||||
private ErrorReporter reporter;
|
private ErrorReporter reporter;
|
||||||
|
|
||||||
private WritePacketModifier modifierWrite;
|
private WritePacketModifier modifierWrite;
|
||||||
private WritePacketModifier modifierRest;
|
private WritePacketModifier modifierRest;
|
||||||
|
|
||||||
public InterceptWritePacket(ClassLoader classLoader, ErrorReporter reporter) {
|
public InterceptWritePacket(ErrorReporter reporter) {
|
||||||
this.classLoader = classLoader;
|
|
||||||
this.reporter = reporter;
|
this.reporter = reporter;
|
||||||
|
|
||||||
// Initialize modifiers
|
// Initialize modifiers
|
||||||
@ -57,7 +55,7 @@ public class InterceptWritePacket {
|
|||||||
|
|
||||||
private Class<?> createProxyClass(int packetId) {
|
private Class<?> createProxyClass(int packetId) {
|
||||||
// Construct the proxy object
|
// Construct the proxy object
|
||||||
Enhancer ex = new Enhancer();
|
Enhancer ex = EnhancerFactory.getInstance().createEnhancer();
|
||||||
|
|
||||||
// Attempt to share callback filter
|
// Attempt to share callback filter
|
||||||
if (filter == null) {
|
if (filter == null) {
|
||||||
@ -80,7 +78,6 @@ public class InterceptWritePacket {
|
|||||||
ex.setCallbackFilter(filter);
|
ex.setCallbackFilter(filter);
|
||||||
ex.setUseCache(false);
|
ex.setUseCache(false);
|
||||||
|
|
||||||
ex.setClassLoader(classLoader);
|
|
||||||
ex.setCallbackTypes( new Class[] { WritePacketModifier.class, WritePacketModifier.class });
|
ex.setCallbackTypes( new Class[] { WritePacketModifier.class, WritePacketModifier.class });
|
||||||
Class<?> proxyClass = ex.createClass();
|
Class<?> proxyClass = ex.createClass();
|
||||||
|
|
||||||
|
@ -28,23 +28,11 @@ public class PacketInjectorBuilder {
|
|||||||
public static PacketInjectorBuilder newBuilder() {
|
public static PacketInjectorBuilder newBuilder() {
|
||||||
return new PacketInjectorBuilder();
|
return new PacketInjectorBuilder();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected ClassLoader classLoader;
|
|
||||||
protected ListenerInvoker invoker;
|
protected ListenerInvoker invoker;
|
||||||
protected ErrorReporter reporter;
|
protected ErrorReporter reporter;
|
||||||
protected PlayerInjectionHandler playerInjection;
|
protected PlayerInjectionHandler playerInjection;
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the class loader to use during class generation.
|
|
||||||
* @param classLoader - new class loader.
|
|
||||||
* @return This builder, for chaining.
|
|
||||||
*/
|
|
||||||
public PacketInjectorBuilder classLoader(@Nonnull ClassLoader classLoader) {
|
|
||||||
Preconditions.checkNotNull(classLoader, "classLoader cannot be NULL");
|
|
||||||
this.classLoader = classLoader;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The error reporter used by the created injector.
|
* The error reporter used by the created injector.
|
||||||
* @param reporter - new error reporter.
|
* @param reporter - new error reporter.
|
||||||
@ -86,8 +74,6 @@ public class PacketInjectorBuilder {
|
|||||||
ProtocolManager manager = ProtocolLibrary.getProtocolManager();
|
ProtocolManager manager = ProtocolLibrary.getProtocolManager();
|
||||||
|
|
||||||
// Initialize with default values if we can
|
// Initialize with default values if we can
|
||||||
if (classLoader == null)
|
|
||||||
classLoader = this.getClass().getClassLoader();
|
|
||||||
if (reporter == null)
|
if (reporter == null)
|
||||||
reporter = ProtocolLibrary.getErrorReporter();
|
reporter = ProtocolLibrary.getErrorReporter();
|
||||||
if (invoker == null)
|
if (invoker == null)
|
||||||
@ -105,6 +91,6 @@ public class PacketInjectorBuilder {
|
|||||||
*/
|
*/
|
||||||
public PacketInjector buildInjector() throws FieldAccessException {
|
public PacketInjector buildInjector() throws FieldAccessException {
|
||||||
initializeDefaults();
|
initializeDefaults();
|
||||||
return new ProxyPacketInjector(classLoader, invoker, playerInjection, reporter);
|
return new ProxyPacketInjector(invoker, playerInjection, reporter);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -49,6 +49,7 @@ import com.comphenix.protocol.reflect.FieldUtils;
|
|||||||
import com.comphenix.protocol.reflect.FuzzyReflection;
|
import com.comphenix.protocol.reflect.FuzzyReflection;
|
||||||
import com.comphenix.protocol.reflect.MethodInfo;
|
import com.comphenix.protocol.reflect.MethodInfo;
|
||||||
import com.comphenix.protocol.reflect.fuzzy.FuzzyMethodContract;
|
import com.comphenix.protocol.reflect.fuzzy.FuzzyMethodContract;
|
||||||
|
import com.comphenix.protocol.utility.EnhancerFactory;
|
||||||
import com.comphenix.protocol.utility.MinecraftReflection;
|
import com.comphenix.protocol.utility.MinecraftReflection;
|
||||||
import com.comphenix.protocol.wrappers.WrappedIntHashMap;
|
import com.comphenix.protocol.wrappers.WrappedIntHashMap;
|
||||||
|
|
||||||
@ -147,19 +148,15 @@ class ProxyPacketInjector implements PacketInjector {
|
|||||||
// Allows us to determine the sender
|
// Allows us to determine the sender
|
||||||
private PlayerInjectionHandler playerInjection;
|
private PlayerInjectionHandler playerInjection;
|
||||||
|
|
||||||
// Class loader
|
|
||||||
private ClassLoader classLoader;
|
|
||||||
|
|
||||||
// Share callback filter
|
// Share callback filter
|
||||||
private CallbackFilter filter;
|
private CallbackFilter filter;
|
||||||
|
|
||||||
// Determine if the read packet method was found
|
// Determine if the read packet method was found
|
||||||
private boolean readPacketIntercepted = false;
|
private boolean readPacketIntercepted = false;
|
||||||
|
|
||||||
public ProxyPacketInjector(ClassLoader classLoader, ListenerInvoker manager,
|
public ProxyPacketInjector(ListenerInvoker manager, PlayerInjectionHandler playerInjection,
|
||||||
PlayerInjectionHandler playerInjection, ErrorReporter reporter) throws FieldAccessException {
|
ErrorReporter reporter) throws FieldAccessException {
|
||||||
|
|
||||||
this.classLoader = classLoader;
|
|
||||||
this.manager = manager;
|
this.manager = manager;
|
||||||
this.playerInjection = playerInjection;
|
this.playerInjection = playerInjection;
|
||||||
this.reporter = reporter;
|
this.reporter = reporter;
|
||||||
@ -211,7 +208,7 @@ class ProxyPacketInjector implements PacketInjector {
|
|||||||
if (hasPacketHandler(type))
|
if (hasPacketHandler(type))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
Enhancer ex = new Enhancer();
|
Enhancer ex = EnhancerFactory.getInstance().createEnhancer();
|
||||||
|
|
||||||
// Unfortunately, we can't easily distinguish between these two functions:
|
// Unfortunately, we can't easily distinguish between these two functions:
|
||||||
// * Object lookup(int par1)
|
// * Object lookup(int par1)
|
||||||
@ -255,7 +252,6 @@ class ProxyPacketInjector implements PacketInjector {
|
|||||||
ex.setSuperclass(old);
|
ex.setSuperclass(old);
|
||||||
ex.setCallbackFilter(filter);
|
ex.setCallbackFilter(filter);
|
||||||
ex.setCallbackTypes(new Class<?>[] { NoOp.class, ReadPacketModifier.class, ReadPacketModifier.class });
|
ex.setCallbackTypes(new Class<?>[] { NoOp.class, ReadPacketModifier.class, ReadPacketModifier.class });
|
||||||
ex.setClassLoader(classLoader);
|
|
||||||
Class proxy = ex.createClass();
|
Class proxy = ex.createClass();
|
||||||
|
|
||||||
// Create the proxy handlers
|
// Create the proxy handlers
|
||||||
|
@ -29,6 +29,7 @@ import com.comphenix.protocol.error.Report;
|
|||||||
import com.comphenix.protocol.error.ReportType;
|
import com.comphenix.protocol.error.ReportType;
|
||||||
import com.comphenix.protocol.injector.ListenerInvoker;
|
import com.comphenix.protocol.injector.ListenerInvoker;
|
||||||
import com.comphenix.protocol.injector.player.NetworkFieldInjector.FakePacket;
|
import com.comphenix.protocol.injector.player.NetworkFieldInjector.FakePacket;
|
||||||
|
import com.comphenix.protocol.utility.EnhancerFactory;
|
||||||
import com.comphenix.protocol.utility.MinecraftReflection;
|
import com.comphenix.protocol.utility.MinecraftReflection;
|
||||||
import com.google.common.collect.MapMaker;
|
import com.google.common.collect.MapMaker;
|
||||||
|
|
||||||
@ -55,12 +56,10 @@ class InjectedArrayList extends ArrayList<Object> {
|
|||||||
|
|
||||||
private transient PlayerInjector injector;
|
private transient PlayerInjector injector;
|
||||||
private transient Set<Object> ignoredPackets;
|
private transient Set<Object> ignoredPackets;
|
||||||
private transient ClassLoader classLoader;
|
|
||||||
|
|
||||||
private transient InvertedIntegerCallback callback;
|
private transient InvertedIntegerCallback callback;
|
||||||
|
|
||||||
public InjectedArrayList(ClassLoader classLoader, PlayerInjector injector, Set<Object> ignoredPackets) {
|
public InjectedArrayList(PlayerInjector injector, Set<Object> ignoredPackets) {
|
||||||
this.classLoader = classLoader;
|
|
||||||
this.injector = injector;
|
this.injector = injector;
|
||||||
this.ignoredPackets = ignoredPackets;
|
this.ignoredPackets = ignoredPackets;
|
||||||
this.callback = new InvertedIntegerCallback();
|
this.callback = new InvertedIntegerCallback();
|
||||||
@ -135,11 +134,10 @@ class InjectedArrayList extends ArrayList<Object> {
|
|||||||
// }
|
// }
|
||||||
// ect.
|
// ect.
|
||||||
// }
|
// }
|
||||||
Enhancer ex = new Enhancer();
|
Enhancer ex = EnhancerFactory.getInstance().createEnhancer();
|
||||||
ex.setSuperclass(MinecraftReflection.getPacketClass());
|
ex.setSuperclass(MinecraftReflection.getPacketClass());
|
||||||
ex.setInterfaces(new Class[] { FakePacket.class } );
|
ex.setInterfaces(new Class[] { FakePacket.class } );
|
||||||
ex.setUseCache(true);
|
ex.setUseCache(true);
|
||||||
ex.setClassLoader(classLoader);
|
|
||||||
ex.setCallbackType(InvertedIntegerCallback.class);
|
ex.setCallbackType(InvertedIntegerCallback.class);
|
||||||
|
|
||||||
Class<?> proxyClass = ex.createClass();
|
Class<?> proxyClass = ex.createClass();
|
||||||
|
@ -77,10 +77,10 @@ class NetworkFieldInjector extends PlayerInjector {
|
|||||||
// Determine if we're listening
|
// Determine if we're listening
|
||||||
private IntegerSet sendingFilters;
|
private IntegerSet sendingFilters;
|
||||||
|
|
||||||
public NetworkFieldInjector(ClassLoader classLoader, ErrorReporter reporter, Player player,
|
public NetworkFieldInjector(ErrorReporter reporter, Player player,
|
||||||
ListenerInvoker manager, IntegerSet sendingFilters) throws IllegalAccessException {
|
ListenerInvoker manager, IntegerSet sendingFilters) throws IllegalAccessException {
|
||||||
|
|
||||||
super(classLoader, reporter, player, manager);
|
super(reporter, player, manager);
|
||||||
this.sendingFilters = sendingFilters;
|
this.sendingFilters = sendingFilters;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -161,7 +161,7 @@ class NetworkFieldInjector extends PlayerInjector {
|
|||||||
|
|
||||||
synchronized(syncObject) {
|
synchronized(syncObject) {
|
||||||
// The list we'll be inserting
|
// The list we'll be inserting
|
||||||
List<Object> hackedList = new InjectedArrayList(classLoader, this, ignoredPackets);
|
List<Object> hackedList = new InjectedArrayList(this, ignoredPackets);
|
||||||
|
|
||||||
// Add every previously stored packet
|
// Add every previously stored packet
|
||||||
for (Object packet : minecraftList) {
|
for (Object packet : minecraftList) {
|
||||||
|
@ -41,6 +41,7 @@ import com.comphenix.protocol.injector.GamePhase;
|
|||||||
import com.comphenix.protocol.injector.ListenerInvoker;
|
import com.comphenix.protocol.injector.ListenerInvoker;
|
||||||
import com.comphenix.protocol.injector.PacketFilterManager.PlayerInjectHooks;
|
import com.comphenix.protocol.injector.PacketFilterManager.PlayerInjectHooks;
|
||||||
import com.comphenix.protocol.injector.server.TemporaryPlayerFactory;
|
import com.comphenix.protocol.injector.server.TemporaryPlayerFactory;
|
||||||
|
import com.comphenix.protocol.utility.EnhancerFactory;
|
||||||
import com.comphenix.protocol.utility.MinecraftVersion;
|
import com.comphenix.protocol.utility.MinecraftVersion;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -68,16 +69,16 @@ public class NetworkObjectInjector extends PlayerInjector {
|
|||||||
* Create a new network object injector.
|
* Create a new network object injector.
|
||||||
* <p>
|
* <p>
|
||||||
* Note: This class is intended to be internal. Do not use.
|
* Note: This class is intended to be internal. Do not use.
|
||||||
* @param classLoader - the class loader.
|
|
||||||
* @param reporter - the error reporter.
|
* @param reporter - the error reporter.
|
||||||
* @param player - the player Bukkit entity.
|
* @param player - the player Bukkit entity.
|
||||||
* @param invoker - the packet invoker.
|
* @param invoker - the packet invoker.
|
||||||
* @param sendingFilters - list of permitted packet IDs.
|
* @param sendingFilters - list of permitted packet IDs.
|
||||||
* @throws IllegalAccessException If reflection failed.
|
* @throws IllegalAccessException If reflection failed.
|
||||||
*/
|
*/
|
||||||
public NetworkObjectInjector(ClassLoader classLoader, ErrorReporter reporter, Player player,
|
public NetworkObjectInjector(ErrorReporter reporter, Player player,
|
||||||
ListenerInvoker invoker, IntegerSet sendingFilters) throws IllegalAccessException {
|
ListenerInvoker invoker, IntegerSet sendingFilters) throws IllegalAccessException {
|
||||||
super(classLoader, reporter, player, invoker);
|
|
||||||
|
super(reporter, player, invoker);
|
||||||
this.sendingFilters = sendingFilters;
|
this.sendingFilters = sendingFilters;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -194,8 +195,7 @@ public class NetworkObjectInjector extends PlayerInjector {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Create our proxy object
|
// Create our proxy object
|
||||||
Enhancer ex = new Enhancer();
|
Enhancer ex = EnhancerFactory.getInstance().createEnhancer();
|
||||||
ex.setClassLoader(classLoader);
|
|
||||||
ex.setSuperclass(networkInterface);
|
ex.setSuperclass(networkInterface);
|
||||||
ex.setCallbacks(new Callback[] { queueFilter, dispatch });
|
ex.setCallbacks(new Callback[] { queueFilter, dispatch });
|
||||||
ex.setCallbackFilter(callbackFilter);
|
ex.setCallbackFilter(callbackFilter);
|
||||||
|
@ -41,6 +41,7 @@ import com.comphenix.protocol.reflect.ObjectWriter;
|
|||||||
import com.comphenix.protocol.reflect.VolatileField;
|
import com.comphenix.protocol.reflect.VolatileField;
|
||||||
import com.comphenix.protocol.reflect.instances.DefaultInstances;
|
import com.comphenix.protocol.reflect.instances.DefaultInstances;
|
||||||
import com.comphenix.protocol.reflect.instances.ExistingGenerator;
|
import com.comphenix.protocol.reflect.instances.ExistingGenerator;
|
||||||
|
import com.comphenix.protocol.utility.EnhancerFactory;
|
||||||
import com.comphenix.protocol.utility.MinecraftMethods;
|
import com.comphenix.protocol.utility.MinecraftMethods;
|
||||||
import com.comphenix.protocol.utility.MinecraftReflection;
|
import com.comphenix.protocol.utility.MinecraftReflection;
|
||||||
import com.comphenix.protocol.utility.MinecraftVersion;
|
import com.comphenix.protocol.utility.MinecraftVersion;
|
||||||
@ -72,11 +73,11 @@ class NetworkServerInjector extends PlayerInjector {
|
|||||||
private final ObjectWriter writer = new ObjectWriter();
|
private final ObjectWriter writer = new ObjectWriter();
|
||||||
|
|
||||||
public NetworkServerInjector(
|
public NetworkServerInjector(
|
||||||
ClassLoader classLoader, ErrorReporter reporter, Player player,
|
ErrorReporter reporter, Player player,
|
||||||
ListenerInvoker invoker, IntegerSet sendingFilters,
|
ListenerInvoker invoker, IntegerSet sendingFilters,
|
||||||
InjectedServerConnection serverInjection) throws IllegalAccessException {
|
InjectedServerConnection serverInjection) throws IllegalAccessException {
|
||||||
|
|
||||||
super(classLoader, reporter, player, invoker);
|
super(reporter, player, invoker);
|
||||||
this.sendingFilters = sendingFilters;
|
this.sendingFilters = sendingFilters;
|
||||||
this.serverInjection = serverInjection;
|
this.serverInjection = serverInjection;
|
||||||
}
|
}
|
||||||
@ -148,7 +149,7 @@ class NetworkServerInjector extends PlayerInjector {
|
|||||||
private boolean tryInjectManager() {
|
private boolean tryInjectManager() {
|
||||||
Class<?> serverClass = serverHandler.getClass();
|
Class<?> serverClass = serverHandler.getClass();
|
||||||
|
|
||||||
Enhancer ex = new Enhancer();
|
Enhancer ex = EnhancerFactory.getInstance().createEnhancer();
|
||||||
Callback sendPacketCallback = new MethodInterceptor() {
|
Callback sendPacketCallback = new MethodInterceptor() {
|
||||||
@Override
|
@Override
|
||||||
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
|
public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
|
||||||
|
@ -140,8 +140,7 @@ public abstract class PlayerInjector implements SocketInjector {
|
|||||||
boolean updateOnLogin;
|
boolean updateOnLogin;
|
||||||
volatile Player updatedPlayer;
|
volatile Player updatedPlayer;
|
||||||
|
|
||||||
public PlayerInjector(ClassLoader classLoader, ErrorReporter reporter, Player player, ListenerInvoker invoker) throws IllegalAccessException {
|
public PlayerInjector(ErrorReporter reporter, Player player, ListenerInvoker invoker) throws IllegalAccessException {
|
||||||
this.classLoader = classLoader;
|
|
||||||
this.reporter = reporter;
|
this.reporter = reporter;
|
||||||
this.player = player;
|
this.player = player;
|
||||||
this.invoker = invoker;
|
this.invoker = invoker;
|
||||||
|
@ -32,7 +32,6 @@ public class PlayerInjectorBuilder {
|
|||||||
// Use the static method.
|
// Use the static method.
|
||||||
}
|
}
|
||||||
|
|
||||||
protected ClassLoader classLoader;
|
|
||||||
protected ErrorReporter reporter;
|
protected ErrorReporter reporter;
|
||||||
protected Predicate<GamePhase> injectionFilter;
|
protected Predicate<GamePhase> injectionFilter;
|
||||||
protected ListenerInvoker invoker;
|
protected ListenerInvoker invoker;
|
||||||
@ -40,17 +39,6 @@ public class PlayerInjectorBuilder {
|
|||||||
protected Server server;
|
protected Server server;
|
||||||
protected MinecraftVersion version;
|
protected MinecraftVersion version;
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the class loader to use during class generation.
|
|
||||||
* @param classLoader - new class loader.
|
|
||||||
* @return This builder, for chaining.
|
|
||||||
*/
|
|
||||||
public PlayerInjectorBuilder classLoader(@Nonnull ClassLoader classLoader) {
|
|
||||||
Preconditions.checkNotNull(classLoader, "classLoader cannot be NULL");
|
|
||||||
this.classLoader = classLoader;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The error reporter used by the created injector.
|
* The error reporter used by the created injector.
|
||||||
* @param reporter - new error reporter.
|
* @param reporter - new error reporter.
|
||||||
@ -126,8 +114,6 @@ public class PlayerInjectorBuilder {
|
|||||||
ProtocolManager manager = ProtocolLibrary.getProtocolManager();
|
ProtocolManager manager = ProtocolLibrary.getProtocolManager();
|
||||||
|
|
||||||
// Initialize with default values if we can
|
// Initialize with default values if we can
|
||||||
if (classLoader == null)
|
|
||||||
classLoader = this.getClass().getClassLoader();
|
|
||||||
if (reporter == null)
|
if (reporter == null)
|
||||||
reporter = ProtocolLibrary.getErrorReporter();
|
reporter = ProtocolLibrary.getErrorReporter();
|
||||||
if (invoker == null)
|
if (invoker == null)
|
||||||
@ -151,7 +137,7 @@ public class PlayerInjectorBuilder {
|
|||||||
initializeDefaults();
|
initializeDefaults();
|
||||||
|
|
||||||
return new ProxyPlayerInjectionHandler(
|
return new ProxyPlayerInjectionHandler(
|
||||||
classLoader, reporter, injectionFilter,
|
reporter, injectionFilter,
|
||||||
invoker, packetListeners, server, version);
|
invoker, packetListeners, server, version);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -124,17 +124,13 @@ class ProxyPlayerInjectionHandler implements PlayerInjectionHandler {
|
|||||||
// List of packet listeners
|
// List of packet listeners
|
||||||
private Set<PacketListener> packetListeners;
|
private Set<PacketListener> packetListeners;
|
||||||
|
|
||||||
// The class loader we're using
|
|
||||||
private ClassLoader classLoader;
|
|
||||||
|
|
||||||
// Used to filter injection attempts
|
// Used to filter injection attempts
|
||||||
private Predicate<GamePhase> injectionFilter;
|
private Predicate<GamePhase> injectionFilter;
|
||||||
|
|
||||||
public ProxyPlayerInjectionHandler(
|
public ProxyPlayerInjectionHandler(
|
||||||
ClassLoader classLoader, ErrorReporter reporter, Predicate<GamePhase> injectionFilter,
|
ErrorReporter reporter, Predicate<GamePhase> injectionFilter,
|
||||||
ListenerInvoker invoker, Set<PacketListener> packetListeners, Server server, MinecraftVersion version) {
|
ListenerInvoker invoker, Set<PacketListener> packetListeners, Server server, MinecraftVersion version) {
|
||||||
|
|
||||||
this.classLoader = classLoader;
|
|
||||||
this.reporter = reporter;
|
this.reporter = reporter;
|
||||||
this.invoker = invoker;
|
this.invoker = invoker;
|
||||||
this.injectionFilter = injectionFilter;
|
this.injectionFilter = injectionFilter;
|
||||||
@ -224,11 +220,11 @@ class ProxyPlayerInjectionHandler implements PlayerInjectionHandler {
|
|||||||
// Construct the correct player hook
|
// Construct the correct player hook
|
||||||
switch (hook) {
|
switch (hook) {
|
||||||
case NETWORK_HANDLER_FIELDS:
|
case NETWORK_HANDLER_FIELDS:
|
||||||
return new NetworkFieldInjector(classLoader, reporter, player, invoker, sendingFilters);
|
return new NetworkFieldInjector(reporter, player, invoker, sendingFilters);
|
||||||
case NETWORK_MANAGER_OBJECT:
|
case NETWORK_MANAGER_OBJECT:
|
||||||
return new NetworkObjectInjector(classLoader, reporter, player, invoker, sendingFilters);
|
return new NetworkObjectInjector(reporter, player, invoker, sendingFilters);
|
||||||
case NETWORK_SERVER_OBJECT:
|
case NETWORK_SERVER_OBJECT:
|
||||||
return new NetworkServerInjector(classLoader, reporter, player, invoker, sendingFilters, serverInjection);
|
return new NetworkServerInjector(reporter, player, invoker, sendingFilters, serverInjection);
|
||||||
default:
|
default:
|
||||||
throw new IllegalArgumentException("Cannot construct a player injector.");
|
throw new IllegalArgumentException("Cannot construct a player injector.");
|
||||||
}
|
}
|
||||||
|
@ -43,6 +43,7 @@ import com.comphenix.protocol.reflect.FieldUtils;
|
|||||||
import com.comphenix.protocol.reflect.FuzzyReflection;
|
import com.comphenix.protocol.reflect.FuzzyReflection;
|
||||||
import com.comphenix.protocol.reflect.MethodInfo;
|
import com.comphenix.protocol.reflect.MethodInfo;
|
||||||
import com.comphenix.protocol.reflect.fuzzy.FuzzyMethodContract;
|
import com.comphenix.protocol.reflect.fuzzy.FuzzyMethodContract;
|
||||||
|
import com.comphenix.protocol.utility.EnhancerFactory;
|
||||||
import com.comphenix.protocol.utility.MinecraftReflection;
|
import com.comphenix.protocol.utility.MinecraftReflection;
|
||||||
|
|
||||||
import com.google.common.collect.Iterables;
|
import com.google.common.collect.Iterables;
|
||||||
@ -98,7 +99,6 @@ public class SpigotPacketInjector implements SpigotPacketListener {
|
|||||||
private ListenerInvoker invoker;
|
private ListenerInvoker invoker;
|
||||||
private ErrorReporter reporter;
|
private ErrorReporter reporter;
|
||||||
private Server server;
|
private Server server;
|
||||||
private ClassLoader classLoader;
|
|
||||||
|
|
||||||
// The proxy packet injector
|
// The proxy packet injector
|
||||||
private PacketInjector proxyPacketInjector;
|
private PacketInjector proxyPacketInjector;
|
||||||
@ -106,8 +106,7 @@ public class SpigotPacketInjector implements SpigotPacketListener {
|
|||||||
/**
|
/**
|
||||||
* Create a new spigot injector.
|
* Create a new spigot injector.
|
||||||
*/
|
*/
|
||||||
public SpigotPacketInjector(ClassLoader classLoader, ErrorReporter reporter, ListenerInvoker invoker, Server server) {
|
public SpigotPacketInjector(ErrorReporter reporter, ListenerInvoker invoker, Server server) {
|
||||||
this.classLoader = classLoader;
|
|
||||||
this.reporter = reporter;
|
this.reporter = reporter;
|
||||||
this.invoker = invoker;
|
this.invoker = invoker;
|
||||||
this.server = server;
|
this.server = server;
|
||||||
@ -220,8 +219,7 @@ public class SpigotPacketInjector implements SpigotPacketListener {
|
|||||||
// Don't care for everything else
|
// Don't care for everything else
|
||||||
callbacks[2] = NoOp.INSTANCE;
|
callbacks[2] = NoOp.INSTANCE;
|
||||||
|
|
||||||
Enhancer enhancer = new Enhancer();
|
Enhancer enhancer = EnhancerFactory.getInstance().createEnhancer();
|
||||||
enhancer.setClassLoader(classLoader);
|
|
||||||
enhancer.setSuperclass(getSpigotListenerClass());
|
enhancer.setSuperclass(getSpigotListenerClass());
|
||||||
enhancer.setCallbacks(callbacks);
|
enhancer.setCallbacks(callbacks);
|
||||||
enhancer.setCallbackFilter(new CallbackFilter() {
|
enhancer.setCallbackFilter(new CallbackFilter() {
|
||||||
@ -316,7 +314,7 @@ public class SpigotPacketInjector implements SpigotPacketListener {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
NetworkObjectInjector created = new NetworkObjectInjector(
|
NetworkObjectInjector created = new NetworkObjectInjector(
|
||||||
classLoader, filterImpossibleWarnings(reporter), null, invoker, null);
|
filterImpossibleWarnings(reporter), null, invoker, null);
|
||||||
|
|
||||||
created.initializePlayer(player);
|
created.initializePlayer(player);
|
||||||
|
|
||||||
@ -344,7 +342,7 @@ public class SpigotPacketInjector implements SpigotPacketListener {
|
|||||||
// Inject the network manager
|
// Inject the network manager
|
||||||
try {
|
try {
|
||||||
NetworkObjectInjector created = new NetworkObjectInjector(
|
NetworkObjectInjector created = new NetworkObjectInjector(
|
||||||
classLoader, filterImpossibleWarnings(reporter), null, invoker, null);
|
filterImpossibleWarnings(reporter), null, invoker, null);
|
||||||
|
|
||||||
if (MinecraftReflection.isLoginHandler(connection)) {
|
if (MinecraftReflection.isLoginHandler(connection)) {
|
||||||
created.initialize(connection);
|
created.initialize(connection);
|
||||||
@ -501,7 +499,7 @@ public class SpigotPacketInjector implements SpigotPacketListener {
|
|||||||
void injectPlayer(Player player) {
|
void injectPlayer(Player player) {
|
||||||
try {
|
try {
|
||||||
NetworkObjectInjector dummy = new NetworkObjectInjector(
|
NetworkObjectInjector dummy = new NetworkObjectInjector(
|
||||||
classLoader, filterImpossibleWarnings(reporter), player, invoker, null);
|
filterImpossibleWarnings(reporter), player, invoker, null);
|
||||||
dummy.initializePlayer(player);
|
dummy.initializePlayer(player);
|
||||||
|
|
||||||
// Save this player for the network manager
|
// Save this player for the network manager
|
||||||
|
@ -22,6 +22,7 @@ import com.comphenix.protocol.utility.MinecraftReflection;
|
|||||||
import com.comphenix.protocol.wrappers.BukkitConverters;
|
import com.comphenix.protocol.wrappers.BukkitConverters;
|
||||||
import com.comphenix.protocol.wrappers.ChunkPosition;
|
import com.comphenix.protocol.wrappers.ChunkPosition;
|
||||||
import com.comphenix.protocol.wrappers.WrappedDataWatcher;
|
import com.comphenix.protocol.wrappers.WrappedDataWatcher;
|
||||||
|
import com.comphenix.protocol.wrappers.WrappedServerPing;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents an object that can clone a specific list of Bukkit- and Minecraft-related objects.
|
* Represents an object that can clone a specific list of Bukkit- and Minecraft-related objects.
|
||||||
@ -31,7 +32,7 @@ import com.comphenix.protocol.wrappers.WrappedDataWatcher;
|
|||||||
public class BukkitCloner implements Cloner {
|
public class BukkitCloner implements Cloner {
|
||||||
// List of classes we support
|
// List of classes we support
|
||||||
private Class<?>[] clonableClasses = { MinecraftReflection.getItemStackClass(), MinecraftReflection.getChunkPositionClass(),
|
private Class<?>[] clonableClasses = { MinecraftReflection.getItemStackClass(), MinecraftReflection.getChunkPositionClass(),
|
||||||
MinecraftReflection.getDataWatcherClass() };
|
MinecraftReflection.getDataWatcherClass(), MinecraftReflection.getServerPingClass() };
|
||||||
|
|
||||||
private int findMatchingClass(Class<?> type) {
|
private int findMatchingClass(Class<?> type) {
|
||||||
// See if is a subclass of any of our supported superclasses
|
// See if is a subclass of any of our supported superclasses
|
||||||
@ -66,6 +67,9 @@ public class BukkitCloner implements Cloner {
|
|||||||
case 2:
|
case 2:
|
||||||
EquivalentConverter<WrappedDataWatcher> dataConverter = BukkitConverters.getDataWatcherConverter();
|
EquivalentConverter<WrappedDataWatcher> dataConverter = BukkitConverters.getDataWatcherConverter();
|
||||||
return dataConverter.getGeneric(clonableClasses[2], dataConverter.getSpecific(source).deepClone());
|
return dataConverter.getGeneric(clonableClasses[2], dataConverter.getSpecific(source).deepClone());
|
||||||
|
case 3:
|
||||||
|
EquivalentConverter<WrappedServerPing> serverConverter = BukkitConverters.getWrappedServerPingConverter();
|
||||||
|
return serverConverter.getGeneric(clonableClasses[3], serverConverter.getSpecific(source).deepClone());
|
||||||
default:
|
default:
|
||||||
throw new IllegalArgumentException("Cannot clone objects of type " + source.getClass());
|
throw new IllegalArgumentException("Cannot clone objects of type " + source.getClass());
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,44 @@
|
|||||||
|
package com.comphenix.protocol.utility;
|
||||||
|
|
||||||
|
import net.sf.cglib.proxy.Enhancer;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents a shared enchancer factory.
|
||||||
|
* @author Kristian
|
||||||
|
*/
|
||||||
|
public class EnhancerFactory {
|
||||||
|
private static EnhancerFactory INSTANCE = new EnhancerFactory();
|
||||||
|
|
||||||
|
// The current class loader
|
||||||
|
private ClassLoader loader = EnhancerFactory.class.getClassLoader();
|
||||||
|
|
||||||
|
public static EnhancerFactory getInstance() {
|
||||||
|
return INSTANCE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new CGLib enhancer.
|
||||||
|
* @return The new enhancer.
|
||||||
|
*/
|
||||||
|
public Enhancer createEnhancer() {
|
||||||
|
Enhancer enhancer = new Enhancer();
|
||||||
|
enhancer.setClassLoader(loader);
|
||||||
|
return enhancer;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the current class loader to use when constructing enhancers.
|
||||||
|
* @param loader - the class loader
|
||||||
|
*/
|
||||||
|
public void setClassLoader(ClassLoader loader) {
|
||||||
|
this.loader = loader;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the current class loader we are using.
|
||||||
|
* @return The current class loader.
|
||||||
|
*/
|
||||||
|
public ClassLoader getClassLoader() {
|
||||||
|
return loader;
|
||||||
|
}
|
||||||
|
}
|
@ -5,6 +5,8 @@ import java.lang.reflect.Method;
|
|||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.bukkit.plugin.Plugin;
|
||||||
|
|
||||||
import net.minecraft.util.io.netty.buffer.ByteBuf;
|
import net.minecraft.util.io.netty.buffer.ByteBuf;
|
||||||
import net.minecraft.util.io.netty.buffer.UnpooledByteBufAllocator;
|
import net.minecraft.util.io.netty.buffer.UnpooledByteBufAllocator;
|
||||||
import net.minecraft.util.io.netty.channel.ChannelHandlerContext;
|
import net.minecraft.util.io.netty.channel.ChannelHandlerContext;
|
||||||
@ -161,7 +163,7 @@ public class MinecraftMethods {
|
|||||||
// Initialize the methods
|
// Initialize the methods
|
||||||
if (packetReadByteBuf == null || packetWriteByteBuf == null) {
|
if (packetReadByteBuf == null || packetWriteByteBuf == null) {
|
||||||
// This object will allow us to detect which methods were called
|
// This object will allow us to detect which methods were called
|
||||||
Enhancer enhancer = new Enhancer();
|
Enhancer enhancer = EnhancerFactory.getInstance().createEnhancer();
|
||||||
enhancer.setSuperclass(MinecraftReflection.getPacketDataSerializerClass());
|
enhancer.setSuperclass(MinecraftReflection.getPacketDataSerializerClass());
|
||||||
enhancer.setCallback(new MethodInterceptor() {
|
enhancer.setCallback(new MethodInterceptor() {
|
||||||
@Override
|
@Override
|
||||||
|
@ -97,6 +97,14 @@ public class WrappedChatComponent extends AbstractWrapper {
|
|||||||
this.cache = obj;
|
this.cache = obj;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieve a deep copy of the current chat component.
|
||||||
|
* @return A copy of the current component.
|
||||||
|
*/
|
||||||
|
public WrappedChatComponent deepClone() {
|
||||||
|
return fromJson(getJson());
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object obj) {
|
public boolean equals(Object obj) {
|
||||||
if (obj == this)
|
if (obj == this)
|
||||||
|
@ -234,6 +234,24 @@ public class WrappedServerPing extends AbstractWrapper {
|
|||||||
VERSION_PROTOCOL.set(version, protocol);
|
VERSION_PROTOCOL.set(version, protocol);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieve a deep copy of the current wrapper object.
|
||||||
|
* @return The current object.
|
||||||
|
*/
|
||||||
|
public WrappedServerPing deepClone() {
|
||||||
|
WrappedServerPing copy = new WrappedServerPing();
|
||||||
|
WrappedChatComponent motd = getMotD();
|
||||||
|
|
||||||
|
copy.setPlayers(getPlayers());
|
||||||
|
copy.setFavicon(getFavicon());
|
||||||
|
copy.setMotD(motd != null ? motd.deepClone() : null);
|
||||||
|
copy.setPlayersMaximum(getPlayersMaximum());
|
||||||
|
copy.setPlayersOnline(getPlayersOnline());
|
||||||
|
copy.setVersionName(getVersionName());
|
||||||
|
copy.setVersionProtocol(getVersionProtocol());
|
||||||
|
return copy;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents a compressed favicon.
|
* Represents a compressed favicon.
|
||||||
* @author Kristian
|
* @author Kristian
|
||||||
|
In neuem Issue referenzieren
Einen Benutzer sperren