Created
December 22, 2012 17:59
-
-
Save aadnk/4360240 to your computer and use it in GitHub Desktop.
Get the underlying net.minecraft.server-object of any Entity or ItemStack.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package com.comphenix.example; | |
/* | |
* 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 | |
*/ | |
import java.lang.reflect.Field; | |
import java.lang.reflect.InvocationTargetException; | |
import java.lang.reflect.Method; | |
import java.util.ArrayList; | |
import java.util.Collection; | |
import java.util.Map; | |
import java.util.concurrent.ConcurrentHashMap; | |
import com.google.common.primitives.Primitives; | |
/** | |
* Represents an object capable of converting wrapped Bukkit objects into NMS objects. | |
* <p> | |
* Typical conversions include: | |
* <ul> | |
* <li>org.bukkit.entity.Player -> net.minecraft.server.EntityPlayer</li> | |
* <li>org.bukkit.World -> net.minecraft.server.WorldServer</li> | |
* </ul> | |
* | |
* @author Kristian | |
*/ | |
public class BukkitUnwrapper { | |
private interface Unwrapper { | |
public Object unwrapItem(Object wrappedObject); | |
} | |
private static Map<Class<?>, Unwrapper> unwrapperCache = new ConcurrentHashMap<Class<?>, Unwrapper>(); | |
@SuppressWarnings("unchecked") | |
public static Object unwrapItem(Object wrappedObject) { | |
// Special case | |
if (wrappedObject == null) | |
return null; | |
Class<?> currentClass = wrappedObject.getClass(); | |
// Next, check for types that doesn't have a getHandle() | |
if (wrappedObject instanceof Collection) { | |
return handleCollection((Collection<Object>) wrappedObject); | |
} else if (Primitives.isWrapperType(currentClass) || wrappedObject instanceof String) { | |
return null; | |
} | |
Unwrapper specificUnwrapper = getSpecificUnwrapper(currentClass); | |
// Retrieve the handle | |
if (specificUnwrapper != null) | |
return specificUnwrapper.unwrapItem(wrappedObject); | |
else | |
return null; | |
} | |
// Handle a collection of items | |
private static Object handleCollection(Collection<Object> wrappedObject) { | |
Collection<Object> copy = new ArrayList<Object>(); | |
// Unwrap every element | |
for (Object element : wrappedObject) { | |
copy.add(unwrapItem(element)); | |
} | |
return copy; | |
} | |
/** | |
* Retrieve a cached class unwrapper for the given class. | |
* @param type - the type of the class. | |
* @return An unwrapper for the given class. | |
*/ | |
private static Unwrapper getSpecificUnwrapper(Class<?> type) { | |
// See if we're already determined this | |
if (unwrapperCache.containsKey(type)) { | |
// We will never remove from the cache, so this ought to be thread safe | |
return unwrapperCache.get(type); | |
} | |
try { | |
final Method find = type.getMethod("getHandle"); | |
// It's thread safe, as getMethod should return the same handle | |
Unwrapper methodUnwrapper = new Unwrapper() { | |
@Override | |
public Object unwrapItem(Object wrappedObject) { | |
try { | |
return find.invoke(wrappedObject); | |
} catch (IllegalArgumentException e) { | |
throw new RuntimeException("Illegal argument.", e); | |
} catch (IllegalAccessException e) { | |
// Should not occur either | |
return null; | |
} catch (InvocationTargetException e) { | |
// This is really bad | |
throw new RuntimeException("Minecraft error.", e); | |
} | |
} | |
}; | |
unwrapperCache.put(type, methodUnwrapper); | |
return methodUnwrapper; | |
} catch (SecurityException e) { | |
throw new RuntimeException("Security limitation for " + type.getName(), e); | |
} catch (NoSuchMethodException e) { | |
// Try getting the field unwrapper too | |
Unwrapper fieldUnwrapper = getFieldUnwrapper(type); | |
if (fieldUnwrapper != null) | |
return fieldUnwrapper; | |
else | |
throw new RuntimeException("Cannot find method getHande() in " + type.getName(), e); | |
} | |
} | |
/** | |
* Retrieve a cached unwrapper using the handle field. | |
* @param type - a cached field unwrapper. | |
* @return The cached field unwrapper. | |
*/ | |
private static Unwrapper getFieldUnwrapper(Class<?> type) { | |
try { | |
final Field find = type.getDeclaredField("handle"); | |
Unwrapper fieldUnwrapper = new Unwrapper() { | |
@Override | |
public Object unwrapItem(Object wrappedObject) { | |
try { | |
find.setAccessible(true); | |
return find.get(wrappedObject); | |
} catch (IllegalAccessException e) { | |
throw new RuntimeException("Cannot read field 'handle' in " + find.getName(), e); | |
} | |
} | |
}; | |
unwrapperCache.put(type, fieldUnwrapper); | |
return fieldUnwrapper; | |
} catch (NoSuchFieldException e1) { | |
// Inform about this too | |
throw new RuntimeException("Could not find field 'handle' in " + type.getName()); | |
} | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment