2013-11-22 21:10:20 +01:00
|
|
|
package me.libraryaddict.disguise.utilities;
|
2013-11-18 04:24:25 +01:00
|
|
|
|
2013-11-18 12:49:04 +01:00
|
|
|
import java.lang.reflect.Field;
|
2013-11-18 04:24:25 +01:00
|
|
|
import java.lang.reflect.Method;
|
2013-11-18 20:25:48 +01:00
|
|
|
import java.lang.reflect.Modifier;
|
2014-01-15 11:14:21 +01:00
|
|
|
import java.util.UUID;
|
2013-11-18 04:24:25 +01:00
|
|
|
|
2013-11-18 12:49:04 +01:00
|
|
|
import org.bukkit.Art;
|
2013-11-18 04:24:25 +01:00
|
|
|
import org.bukkit.Bukkit;
|
2013-12-31 06:33:42 +01:00
|
|
|
import org.bukkit.Location;
|
2013-11-18 12:49:04 +01:00
|
|
|
import org.bukkit.Sound;
|
2013-11-18 20:25:48 +01:00
|
|
|
import org.bukkit.World;
|
|
|
|
import org.bukkit.entity.Entity;
|
2014-01-05 01:13:49 +01:00
|
|
|
import org.bukkit.entity.Player;
|
2013-11-18 04:24:25 +01:00
|
|
|
import org.bukkit.inventory.ItemStack;
|
|
|
|
|
|
|
|
public class ReflectionManager {
|
2014-01-18 20:21:55 +01:00
|
|
|
public enum LibVersion {
|
2014-04-12 07:19:46 +02:00
|
|
|
V1_6, V1_7;
|
2014-01-18 20:21:55 +01:00
|
|
|
private static LibVersion currentVersion;
|
|
|
|
static {
|
|
|
|
if (getBukkitVersion().startsWith("v1_")) {
|
|
|
|
try {
|
|
|
|
int version = Integer.parseInt(getBukkitVersion().split("_")[1]);
|
|
|
|
if (version == 7) {
|
2014-04-12 07:19:46 +02:00
|
|
|
currentVersion = LibVersion.V1_7;
|
2014-01-18 20:21:55 +01:00
|
|
|
} else {
|
|
|
|
if (version < 7) {
|
|
|
|
currentVersion = LibVersion.V1_6;
|
|
|
|
} else {
|
|
|
|
currentVersion = LibVersion.V1_7;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (Exception ex) {
|
2014-04-12 07:07:27 +02:00
|
|
|
ex.printStackTrace();
|
2014-01-18 20:21:55 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static LibVersion getGameVersion() {
|
|
|
|
return currentVersion;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static boolean is1_6() {
|
|
|
|
return getGameVersion() == V1_6;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static boolean is1_7() {
|
2014-04-12 07:19:46 +02:00
|
|
|
return getGameVersion() == V1_7;
|
2014-01-18 20:21:55 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-18 04:24:25 +01:00
|
|
|
private static String bukkitVersion = Bukkit.getServer().getClass().getName().split("\\.")[3];
|
2014-01-18 20:39:23 +01:00
|
|
|
private static Method damageAndIdleSoundMethod;
|
2014-01-18 21:18:58 +01:00
|
|
|
private static Class itemClass;
|
2014-01-20 17:29:14 +01:00
|
|
|
private static Field pingField;
|
|
|
|
|
2013-11-18 04:24:25 +01:00
|
|
|
static {
|
2013-11-18 20:25:48 +01:00
|
|
|
for (Method method : getNmsClass("EntityLiving").getDeclaredMethods()) {
|
|
|
|
try {
|
|
|
|
if (method.getReturnType() == float.class && Modifier.isProtected(method.getModifiers())
|
|
|
|
&& method.getParameterTypes().length == 0) {
|
2014-01-18 20:39:23 +01:00
|
|
|
Object entity = createEntityInstance("Cow");
|
2013-11-18 20:25:48 +01:00
|
|
|
method.setAccessible(true);
|
2014-01-18 20:39:23 +01:00
|
|
|
float value = (Float) method.invoke(entity);
|
|
|
|
if (value == 0.4F) {
|
|
|
|
damageAndIdleSoundMethod = method;
|
2013-11-18 20:25:48 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
}
|
2013-11-18 04:24:25 +01:00
|
|
|
try {
|
2013-11-18 20:25:48 +01:00
|
|
|
itemClass = getCraftClass("inventory.CraftItemStack");
|
2014-01-20 17:29:14 +01:00
|
|
|
pingField = getNmsClass("EntityPlayer").getField("ping");
|
2013-11-18 04:24:25 +01:00
|
|
|
} catch (Exception e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
}
|
2014-01-05 01:13:49 +01:00
|
|
|
}
|
|
|
|
|
2013-12-22 01:03:47 +01:00
|
|
|
public static Object createEntityInstance(String entityName) {
|
|
|
|
try {
|
|
|
|
Class entityClass = getNmsClass("Entity" + entityName);
|
|
|
|
Object entityObject;
|
|
|
|
Object world = getWorld(Bukkit.getWorlds().get(0));
|
|
|
|
if (entityName.equals("Player")) {
|
|
|
|
Object minecraftServer = getNmsClass("MinecraftServer").getMethod("getServer").invoke(null);
|
|
|
|
Object playerinteractmanager = getNmsClass("PlayerInteractManager").getConstructor(getNmsClass("World"))
|
|
|
|
.newInstance(world);
|
2014-01-18 20:21:55 +01:00
|
|
|
if (LibVersion.is1_7()) {
|
2014-04-08 18:13:54 +02:00
|
|
|
Object gameProfile = getGameProfile(null, "LibsDisguises");
|
2013-12-22 01:03:47 +01:00
|
|
|
entityObject = entityClass.getConstructor(getNmsClass("MinecraftServer"), getNmsClass("WorldServer"),
|
|
|
|
gameProfile.getClass(), playerinteractmanager.getClass()).newInstance(minecraftServer, world,
|
|
|
|
gameProfile, playerinteractmanager);
|
|
|
|
} else {
|
|
|
|
entityObject = entityClass.getConstructor(getNmsClass("MinecraftServer"), getNmsClass("World"), String.class,
|
|
|
|
playerinteractmanager.getClass()).newInstance(minecraftServer, world, "LibsDisguises",
|
|
|
|
playerinteractmanager);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
entityObject = entityClass.getConstructor(getNmsClass("World")).newInstance(world);
|
|
|
|
}
|
|
|
|
return entityObject;
|
|
|
|
} catch (Exception e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2013-12-22 00:36:06 +01:00
|
|
|
public static FakeBoundingBox getBoundingBox(Entity entity) {
|
|
|
|
try {
|
2013-12-22 04:58:49 +01:00
|
|
|
Object boundingBox = getNmsClass("Entity").getField("boundingBox").get(getNmsEntity(entity));
|
|
|
|
double x = 0, y = 0, z = 0;
|
|
|
|
int stage = 0;
|
|
|
|
for (Field field : boundingBox.getClass().getFields()) {
|
|
|
|
if (field.getType().getSimpleName().equals("double")) {
|
|
|
|
stage++;
|
|
|
|
switch (stage) {
|
|
|
|
case 1:
|
|
|
|
x -= field.getDouble(boundingBox);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
y -= field.getDouble(boundingBox);
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
z -= field.getDouble(boundingBox);
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
x += field.getDouble(boundingBox);
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
y += field.getDouble(boundingBox);
|
|
|
|
break;
|
|
|
|
case 6:
|
|
|
|
z += field.getDouble(boundingBox);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Exception("Error while setting the bounding box, more doubles than I thought??");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return new FakeBoundingBox(x, y, z);
|
|
|
|
|
2013-12-22 00:36:06 +01:00
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2013-11-27 04:55:03 +01:00
|
|
|
public static Entity getBukkitEntity(Object nmsEntity) {
|
|
|
|
try {
|
|
|
|
Entity bukkitEntity = (Entity) ReflectionManager.getNmsClass("Entity").getMethod("getBukkitEntity").invoke(nmsEntity);
|
|
|
|
return bukkitEntity;
|
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2013-11-22 20:52:15 +01:00
|
|
|
public static ItemStack getBukkitItem(Object nmsItem) {
|
2013-11-18 04:24:25 +01:00
|
|
|
try {
|
2013-11-22 20:52:15 +01:00
|
|
|
return (ItemStack) itemClass.getMethod("asBukkitCopy", getNmsClass("ItemStack")).invoke(null, nmsItem);
|
2013-11-18 04:24:25 +01:00
|
|
|
} catch (Exception e) {
|
2013-11-22 20:52:15 +01:00
|
|
|
e.printStackTrace();
|
2013-11-18 04:24:25 +01:00
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2014-01-18 20:21:55 +01:00
|
|
|
public static String getBukkitVersion() {
|
|
|
|
return bukkitVersion;
|
|
|
|
}
|
|
|
|
|
2013-11-18 20:25:48 +01:00
|
|
|
public static Class getCraftClass(String className) {
|
|
|
|
try {
|
2014-01-18 20:21:55 +01:00
|
|
|
return Class.forName("org.bukkit.craftbukkit." + getBukkitVersion() + "." + className);
|
2013-11-18 20:25:48 +01:00
|
|
|
} catch (Exception e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2013-11-22 20:52:15 +01:00
|
|
|
public static String getCraftSound(Sound sound) {
|
2013-11-18 04:24:25 +01:00
|
|
|
try {
|
2013-11-22 20:52:15 +01:00
|
|
|
Class c = getCraftClass("CraftSound");
|
|
|
|
return (String) c.getMethod("getSound", Sound.class).invoke(null, sound);
|
2013-11-18 04:24:25 +01:00
|
|
|
} catch (Exception ex) {
|
2013-11-22 20:52:15 +01:00
|
|
|
ex.printStackTrace();
|
2013-11-18 04:24:25 +01:00
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2013-11-22 20:52:15 +01:00
|
|
|
public static String getEnumArt(Art art) {
|
2013-11-18 20:25:48 +01:00
|
|
|
try {
|
2014-01-18 20:21:55 +01:00
|
|
|
Class craftArt = Class.forName("org.bukkit.craftbukkit." + getBukkitVersion() + ".CraftArt");
|
2013-11-22 20:52:15 +01:00
|
|
|
Object enumArt = craftArt.getMethod("BukkitToNotch", Art.class).invoke(null, art);
|
|
|
|
for (Field field : enumArt.getClass().getFields()) {
|
|
|
|
if (field.getType() == String.class) {
|
|
|
|
return (String) field.get(enumArt);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2014-04-12 07:07:27 +02:00
|
|
|
public static Object getGameProfile(Player player) {
|
2014-04-12 07:19:46 +02:00
|
|
|
if (LibVersion.is1_7()) {
|
2014-04-12 07:07:27 +02:00
|
|
|
try {
|
|
|
|
return getNmsClass("EntityHuman").getMethod("getProfile").invoke(getNmsEntity(player));
|
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2014-04-08 18:13:54 +02:00
|
|
|
public static Object getGameProfile(UUID uuid, String playerName) {
|
2013-12-05 09:05:58 +01:00
|
|
|
try {
|
2014-04-12 06:22:47 +02:00
|
|
|
try {
|
|
|
|
return Class.forName("net.minecraft.util.com.mojang.authlib.GameProfile")
|
|
|
|
.getConstructor(UUID.class, String.class)
|
2014-04-18 22:14:02 +02:00
|
|
|
.newInstance(uuid != null ? uuid : UUID.randomUUID(), playerName);
|
2014-04-12 06:22:47 +02:00
|
|
|
} catch (NoSuchMethodException ex) {
|
|
|
|
return Class.forName("net.minecraft.util.com.mojang.authlib.GameProfile")
|
|
|
|
.getConstructor(String.class, String.class).newInstance(uuid != null ? uuid.toString() : "", playerName);
|
|
|
|
}
|
2013-12-05 09:05:58 +01:00
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2013-11-22 20:52:15 +01:00
|
|
|
public static Class getNmsClass(String className) {
|
|
|
|
try {
|
2014-01-18 20:21:55 +01:00
|
|
|
return Class.forName("net.minecraft.server." + getBukkitVersion() + "." + className);
|
2013-11-18 04:24:25 +01:00
|
|
|
} catch (Exception e) {
|
2013-11-27 04:38:51 +01:00
|
|
|
// e.printStackTrace();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2013-11-22 20:52:15 +01:00
|
|
|
public static Object getNmsEntity(Entity entity) {
|
|
|
|
try {
|
|
|
|
return getCraftClass("entity.CraftEntity").getMethod("getHandle").invoke(entity);
|
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
2013-11-18 04:24:25 +01:00
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Object getNmsItem(ItemStack itemstack) {
|
|
|
|
try {
|
2013-11-18 17:50:31 +01:00
|
|
|
return itemClass.getMethod("asNMSCopy", ItemStack.class).invoke(null, itemstack);
|
2013-11-18 04:24:25 +01:00
|
|
|
} catch (Exception e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2014-01-20 18:01:49 +01:00
|
|
|
public static double getPing(Player player) {
|
|
|
|
try {
|
|
|
|
return (double) pingField.getInt(ReflectionManager.getNmsEntity(player));
|
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
return 0D;
|
|
|
|
}
|
|
|
|
|
2013-12-22 05:35:57 +01:00
|
|
|
public static float[] getSize(Entity entity) {
|
|
|
|
try {
|
|
|
|
float length = getNmsClass("Entity").getField("length").getFloat(getNmsEntity(entity));
|
|
|
|
float width = getNmsClass("Entity").getField("width").getFloat(getNmsEntity(entity));
|
|
|
|
float height = getNmsClass("Entity").getField("height").getFloat(getNmsEntity(entity));
|
|
|
|
return new float[] { length, width, height };
|
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2013-11-22 20:52:15 +01:00
|
|
|
public static Float getSoundModifier(Object entity) {
|
2013-11-18 04:24:25 +01:00
|
|
|
try {
|
2014-01-18 20:39:23 +01:00
|
|
|
damageAndIdleSoundMethod.setAccessible(true);
|
|
|
|
return (Float) damageAndIdleSoundMethod.invoke(entity);
|
2013-11-22 20:52:15 +01:00
|
|
|
} catch (Exception ex) {
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Object getWorld(World world) {
|
|
|
|
try {
|
|
|
|
return getCraftClass("CraftWorld").getMethod("getHandle").invoke(world);
|
2013-11-18 04:24:25 +01:00
|
|
|
} catch (Exception e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2014-04-18 22:14:02 +02:00
|
|
|
public static Object grabProfileAddUUID(String playername) {
|
2014-04-14 16:26:31 +02:00
|
|
|
try {
|
|
|
|
Object minecraftServer = getNmsClass("MinecraftServer").getMethod("getServer").invoke(null);
|
|
|
|
for (Method method : getNmsClass("MinecraftServer").getMethods()) {
|
2014-04-18 22:14:02 +02:00
|
|
|
if (method.getReturnType().getSimpleName().equals("GameProfileRepository")) {
|
|
|
|
Object profileRepo = method.invoke(minecraftServer);
|
|
|
|
Object agent = Class.forName("net.minecraft.util.com.mojang.authlib.Agent").getField("MINECRAFT").get(null);
|
|
|
|
LibsProfileLookupCaller callback = new LibsProfileLookupCaller();
|
|
|
|
profileRepo
|
|
|
|
.getClass()
|
|
|
|
.getMethod("findProfilesByNames", String[].class, agent.getClass(),
|
|
|
|
Class.forName("net.minecraft.util.com.mojang.authlib.ProfileLookupCallback"))
|
|
|
|
.invoke(profileRepo, new String[] { playername }, agent, callback);
|
|
|
|
return callback.getGameProfile();
|
2014-04-14 16:26:31 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2014-04-19 16:13:56 +02:00
|
|
|
public static Object getSkullBlob(Object gameProfile) {
|
2014-04-14 16:54:04 +02:00
|
|
|
try {
|
|
|
|
Object minecraftServer = getNmsClass("MinecraftServer").getMethod("getServer").invoke(null);
|
|
|
|
for (Method method : getNmsClass("MinecraftServer").getMethods()) {
|
|
|
|
if (method.getReturnType().getSimpleName().equals("MinecraftSessionService")) {
|
|
|
|
Object session = method.invoke(minecraftServer);
|
2014-04-18 22:14:02 +02:00
|
|
|
return session.getClass().getMethod("fillProfileProperties", gameProfile.getClass(), boolean.class)
|
|
|
|
.invoke(session, gameProfile, true);
|
2014-04-14 16:54:04 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2014-04-18 22:14:02 +02:00
|
|
|
public static boolean hasSkinBlob(Object gameprofile) {
|
|
|
|
try {
|
|
|
|
Field propField = gameprofile.getClass().getDeclaredField("properties");
|
|
|
|
propField.setAccessible(true);
|
|
|
|
Object propMap = propField.get(gameprofile);
|
|
|
|
propField = propMap.getClass().getDeclaredField("properties");
|
|
|
|
propField.setAccessible(true);
|
|
|
|
propMap = propField.get(propMap);
|
|
|
|
return !(Boolean) propMap.getClass().getMethod("isEmpty").invoke(propMap);
|
|
|
|
} catch (NoSuchFieldException ex) {
|
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-01-18 20:21:55 +01:00
|
|
|
public static void setAllowSleep(Player player) {
|
|
|
|
try {
|
|
|
|
Object nmsEntity = getNmsEntity(player);
|
|
|
|
Object connection = nmsEntity.getClass().getField("playerConnection").get(nmsEntity);
|
|
|
|
Field check = connection.getClass().getField("checkMovement");
|
|
|
|
check.setBoolean(connection, true);
|
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
2013-12-07 19:51:37 +01:00
|
|
|
}
|
|
|
|
|
2013-12-31 08:08:14 +01:00
|
|
|
public static void setBoundingBox(Entity entity, FakeBoundingBox newBox, float[] entitySize) {
|
2013-12-22 01:03:47 +01:00
|
|
|
try {
|
|
|
|
Object boundingBox = getNmsClass("Entity").getField("boundingBox").get(getNmsEntity(entity));
|
|
|
|
int stage = 0;
|
2013-12-31 06:33:42 +01:00
|
|
|
Location loc = entity.getLocation();
|
2013-12-22 01:03:47 +01:00
|
|
|
for (Field field : boundingBox.getClass().getFields()) {
|
2013-12-22 02:30:15 +01:00
|
|
|
if (field.getType().getSimpleName().equals("double")) {
|
2013-12-22 01:03:47 +01:00
|
|
|
stage++;
|
|
|
|
switch (stage) {
|
|
|
|
case 1:
|
2013-12-31 06:33:42 +01:00
|
|
|
field.setDouble(boundingBox, loc.getX() - newBox.getX());
|
2013-12-22 01:03:47 +01:00
|
|
|
break;
|
|
|
|
case 2:
|
2013-12-31 06:37:28 +01:00
|
|
|
// field.setDouble(boundingBox, loc.getY() - newBox.getY());
|
2013-12-22 01:03:47 +01:00
|
|
|
break;
|
|
|
|
case 3:
|
2013-12-31 06:33:42 +01:00
|
|
|
field.setDouble(boundingBox, loc.getZ() - newBox.getZ());
|
2013-12-22 01:03:47 +01:00
|
|
|
break;
|
|
|
|
case 4:
|
2013-12-31 06:33:42 +01:00
|
|
|
field.setDouble(boundingBox, loc.getX() + newBox.getX());
|
2013-12-22 04:23:55 +01:00
|
|
|
break;
|
2013-12-22 01:03:47 +01:00
|
|
|
case 5:
|
2013-12-31 08:08:27 +01:00
|
|
|
field.setDouble(boundingBox, loc.getY() + newBox.getY());
|
2013-12-22 04:23:55 +01:00
|
|
|
break;
|
2013-12-22 01:03:47 +01:00
|
|
|
case 6:
|
2013-12-31 06:33:42 +01:00
|
|
|
field.setDouble(boundingBox, loc.getZ() + newBox.getZ());
|
2013-12-22 01:03:47 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Exception("Error while setting the bounding box, more doubles than I thought??");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (Exception ex) {
|
|
|
|
ex.printStackTrace();
|
|
|
|
}
|
|
|
|
}
|
2013-11-18 04:24:25 +01:00
|
|
|
}
|