mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-21 07:41:30 +01:00
#1515: Add a Class reader and Class node argument provider
By: DerFrZocker <derrieple@gmail.com>
This commit is contained in:
parent
eba2b1e948
commit
a11ca51ee1
@ -2,31 +2,12 @@ package org.bukkit.event;
|
||||
|
||||
import static org.junit.jupiter.api.Assertions.*;
|
||||
import com.google.common.base.Joiner;
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.FileNotFoundException;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.net.URI;
|
||||
import java.net.URISyntaxException;
|
||||
import java.nio.file.Files;
|
||||
import java.nio.file.Path;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
import java.util.jar.JarFile;
|
||||
import java.util.stream.Stream;
|
||||
import net.minecraft.WorldVersion;
|
||||
import net.minecraft.server.Main;
|
||||
import net.minecraft.world.level.entity.EntityAccess;
|
||||
import org.bukkit.support.environment.Normal;
|
||||
import org.junit.jupiter.api.AfterAll;
|
||||
import org.junit.jupiter.api.BeforeAll;
|
||||
import org.junit.jupiter.params.ParameterizedTest;
|
||||
import org.junit.jupiter.params.provider.Arguments;
|
||||
import org.junit.jupiter.params.provider.MethodSource;
|
||||
import org.objectweb.asm.ClassReader;
|
||||
import org.bukkit.support.test.ClassNodeTest;
|
||||
import org.objectweb.asm.Handle;
|
||||
import org.objectweb.asm.Opcodes;
|
||||
import org.objectweb.asm.tree.AbstractInsnNode;
|
||||
import org.objectweb.asm.tree.ClassNode;
|
||||
import org.objectweb.asm.tree.InvokeDynamicInsnNode;
|
||||
@ -37,84 +18,12 @@ import org.objectweb.asm.tree.MethodNode;
|
||||
@Normal
|
||||
public class EntityRemoveEventTest {
|
||||
|
||||
// Needs to be a class, which is present in the source, and not a test class
|
||||
private static final URI CRAFT_BUKKIT_CLASSES;
|
||||
// Needs to be a class, which is from the minecraft package and not patch by CraftBukkit
|
||||
private static final URI MINECRAFT_CLASSES;
|
||||
|
||||
static {
|
||||
try {
|
||||
CRAFT_BUKKIT_CLASSES = Main.class.getProtectionDomain().getCodeSource().getLocation().toURI();
|
||||
MINECRAFT_CLASSES = WorldVersion.class.getProtectionDomain().getCodeSource().getLocation().toURI();
|
||||
} catch (URISyntaxException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}
|
||||
|
||||
private static JarFile jarFile = null;
|
||||
private static Stream<Path> files = null;
|
||||
|
||||
public static Stream<Arguments> craftBukkitData() {
|
||||
return files
|
||||
.map(Path::toFile)
|
||||
.filter(File::isFile)
|
||||
.filter(file -> file.getName().endsWith(".class"))
|
||||
.filter(file -> !file.getName().equals("EntityAccess.class"))
|
||||
.map(file -> {
|
||||
try {
|
||||
return new FileInputStream(file);
|
||||
} catch (FileNotFoundException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}).map(Arguments::of);
|
||||
}
|
||||
|
||||
public static Stream<Arguments> minecraftData() {
|
||||
return jarFile
|
||||
.stream()
|
||||
.filter(entry -> entry.getName().endsWith(".class"))
|
||||
.filter(entry -> !new File(CRAFT_BUKKIT_CLASSES.resolve(entry.getName())).exists())
|
||||
.filter(entry -> !entry.getName().startsWith("net/minecraft/gametest/framework"))
|
||||
.map(entry -> {
|
||||
try {
|
||||
return jarFile.getInputStream(entry);
|
||||
} catch (IOException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}).map(Arguments::arguments);
|
||||
}
|
||||
|
||||
@BeforeAll
|
||||
public static void beforeAll() throws IOException {
|
||||
assertNotEquals(CRAFT_BUKKIT_CLASSES, MINECRAFT_CLASSES, """
|
||||
The minecraft and craft bukkit uri point to the same directory / file.
|
||||
Please make sure the CRAFT_BUKKIT_CLASSES points to the test class directory and MINECRAFT_CLASSES to the minecraft server jar.
|
||||
""");
|
||||
|
||||
jarFile = new JarFile(new File(MINECRAFT_CLASSES));
|
||||
files = Files.walk(Path.of(CRAFT_BUKKIT_CLASSES));
|
||||
}
|
||||
|
||||
@ParameterizedTest
|
||||
@MethodSource("minecraftData")
|
||||
public void testMinecraftClasses(InputStream inputStream) throws IOException, ClassNotFoundException {
|
||||
test(inputStream);
|
||||
}
|
||||
|
||||
@ParameterizedTest
|
||||
@MethodSource("craftBukkitData")
|
||||
public void testCraftBukkitModifiedClasses(InputStream inputStream) throws IOException, ClassNotFoundException {
|
||||
test(inputStream);
|
||||
}
|
||||
|
||||
private void test(InputStream inputStream) throws IOException, ClassNotFoundException {
|
||||
@ClassNodeTest(value = {ClassNodeTest.ClassType.CRAFT_BUKKIT, ClassNodeTest.ClassType.MINECRAFT_MODIFIED, ClassNodeTest.ClassType.MINECRAFT_UNMODIFIED},
|
||||
excludedClasses = EntityAccess.class,
|
||||
excludedPackages = "net/minecraft/gametest/framework")
|
||||
public void testForMissing(ClassNode classNode) throws ClassNotFoundException {
|
||||
List<String> missingReason = new ArrayList<>();
|
||||
|
||||
try (inputStream) {
|
||||
ClassReader classReader = new ClassReader(inputStream);
|
||||
ClassNode classNode = new ClassNode(Opcodes.ASM9);
|
||||
|
||||
classReader.accept(classNode, Opcodes.ASM9);
|
||||
boolean minecraftCause = false;
|
||||
boolean bukkitCause = false;
|
||||
|
||||
@ -181,7 +90,6 @@ public class EntityRemoveEventTest {
|
||||
Please remove the bukkit method override, since it is no longer needed.
|
||||
""", classNode.name));
|
||||
}
|
||||
}
|
||||
|
||||
private boolean check(String owner, String name, String desc) throws ClassNotFoundException {
|
||||
if (!name.equals("discard") && !name.equals("remove") && !name.equals("setRemoved")) {
|
||||
@ -207,15 +115,4 @@ public class EntityRemoveEventTest {
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
@AfterAll
|
||||
public static void clear() throws IOException {
|
||||
if (jarFile != null) {
|
||||
jarFile.close();
|
||||
}
|
||||
|
||||
if (files != null) {
|
||||
files.close();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -0,0 +1,70 @@
|
||||
package org.bukkit.support.provider;
|
||||
|
||||
import java.lang.annotation.Annotation;
|
||||
import java.util.stream.Stream;
|
||||
import org.bukkit.support.test.ClassNodeTest;
|
||||
import org.bukkit.support.test.ClassReaderTest;
|
||||
import org.junit.jupiter.api.extension.ExtensionContext;
|
||||
import org.junit.jupiter.params.provider.Arguments;
|
||||
import org.junit.jupiter.params.provider.ArgumentsProvider;
|
||||
import org.junit.jupiter.params.support.AnnotationConsumer;
|
||||
import org.objectweb.asm.ClassReader;
|
||||
import org.objectweb.asm.Opcodes;
|
||||
import org.objectweb.asm.tree.ClassNode;
|
||||
|
||||
public class ClassNodeArgumentProvider implements ArgumentsProvider, AnnotationConsumer<ClassNodeTest> {
|
||||
|
||||
private ClassNodeTest.ClassType[] classTypes;
|
||||
private Class<?>[] excludedClasses;
|
||||
private String[] excludedPackages;
|
||||
|
||||
@Override
|
||||
public void accept(ClassNodeTest classNodeTest) {
|
||||
this.classTypes = classNodeTest.value();
|
||||
this.excludedClasses = classNodeTest.excludedClasses();
|
||||
this.excludedPackages = classNodeTest.excludedPackages();
|
||||
|
||||
for (int i = 0; i < excludedPackages.length; i++) {
|
||||
this.excludedPackages[i] = this.excludedPackages[i].replace('.', '/');
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public Stream<? extends Arguments> provideArguments(ExtensionContext extensionContext) throws Exception {
|
||||
ClassReaderArgumentProvider classReaderArgumentProvider = new ClassReaderArgumentProvider();
|
||||
classReaderArgumentProvider.accept(new ClassReaderArguments(classReaderClassType(), excludedClasses, excludedPackages));
|
||||
|
||||
return classReaderArgumentProvider.getClassReaders().map(this::toClassNode).map(Arguments::of);
|
||||
}
|
||||
|
||||
private ClassReaderTest.ClassType[] classReaderClassType() {
|
||||
ClassReaderTest.ClassType[] newValues = new ClassReaderTest.ClassType[classTypes.length];
|
||||
|
||||
for (int i = 0; i < classTypes.length; i++) {
|
||||
newValues[i] = switch (classTypes[i]) {
|
||||
case BUKKIT -> ClassReaderTest.ClassType.BUKKIT;
|
||||
case CRAFT_BUKKIT -> ClassReaderTest.ClassType.CRAFT_BUKKIT;
|
||||
case MINECRAFT_UNMODIFIED -> ClassReaderTest.ClassType.MINECRAFT_UNMODIFIED;
|
||||
case MINECRAFT_MODIFIED -> ClassReaderTest.ClassType.MINECRAFT_MODIFIED;
|
||||
};
|
||||
}
|
||||
|
||||
return newValues;
|
||||
}
|
||||
|
||||
private ClassNode toClassNode(ClassReader classReader) {
|
||||
ClassNode classNode = new ClassNode(Opcodes.ASM9);
|
||||
|
||||
classReader.accept(classNode, Opcodes.ASM9);
|
||||
|
||||
return classNode;
|
||||
}
|
||||
|
||||
private record ClassReaderArguments(ClassType[] value, Class<?>[] excludedClasses, String[] excludedPackages) implements ClassReaderTest {
|
||||
|
||||
@Override
|
||||
public Class<? extends Annotation> annotationType() {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,210 @@
|
||||
package org.bukkit.support.provider;
|
||||
|
||||
import static org.junit.jupiter.api.Assertions.*;
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.FileNotFoundException;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.net.URI;
|
||||
import java.net.URISyntaxException;
|
||||
import java.nio.file.Files;
|
||||
import java.nio.file.Path;
|
||||
import java.util.jar.JarEntry;
|
||||
import java.util.jar.JarFile;
|
||||
import java.util.stream.Stream;
|
||||
import net.minecraft.WorldVersion;
|
||||
import net.minecraft.server.Main;
|
||||
import org.bukkit.Bukkit;
|
||||
import org.bukkit.support.test.ClassReaderTest;
|
||||
import org.junit.jupiter.api.extension.ExtensionContext;
|
||||
import org.junit.jupiter.params.provider.Arguments;
|
||||
import org.junit.jupiter.params.provider.ArgumentsProvider;
|
||||
import org.junit.jupiter.params.support.AnnotationConsumer;
|
||||
import org.objectweb.asm.ClassReader;
|
||||
|
||||
public class ClassReaderArgumentProvider implements ArgumentsProvider, AnnotationConsumer<ClassReaderTest> {
|
||||
|
||||
// Needs to be a class, which is present in the source, and not a test class
|
||||
private static final URI CRAFT_BUKKIT_CLASSES;
|
||||
// Needs to be a class, which is from the minecraft package and not patch by CraftBukkit
|
||||
private static final URI MINECRAFT_CLASSES;
|
||||
// Needs to be a class, which is from the bukkit package and not a CraftBukkit class
|
||||
private static final URI BUKKIT_CLASSES;
|
||||
|
||||
static {
|
||||
try {
|
||||
CRAFT_BUKKIT_CLASSES = Main.class.getProtectionDomain().getCodeSource().getLocation().toURI();
|
||||
MINECRAFT_CLASSES = WorldVersion.class.getProtectionDomain().getCodeSource().getLocation().toURI();
|
||||
BUKKIT_CLASSES = Bukkit.class.getProtectionDomain().getCodeSource().getLocation().toURI();
|
||||
} catch (URISyntaxException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}
|
||||
|
||||
private ClassReaderTest.ClassType[] classTypes;
|
||||
private Class<?>[] excludedClasses;
|
||||
private String[] excludedPackages;
|
||||
|
||||
@Override
|
||||
public void accept(ClassReaderTest classReaderTest) {
|
||||
this.classTypes = classReaderTest.value();
|
||||
this.excludedClasses = classReaderTest.excludedClasses();
|
||||
this.excludedPackages = classReaderTest.excludedPackages();
|
||||
|
||||
for (int i = 0; i < excludedPackages.length; i++) {
|
||||
this.excludedPackages[i] = this.excludedPackages[i].replace('.', '/');
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public Stream<? extends Arguments> provideArguments(ExtensionContext extensionContext) throws Exception {
|
||||
return getClassReaders().map(Arguments::of);
|
||||
}
|
||||
|
||||
public Stream<ClassReader> getClassReaders() {
|
||||
assertNotEquals(CRAFT_BUKKIT_CLASSES, MINECRAFT_CLASSES, """
|
||||
The Minecraft and CraftBukkit uri point to the same directory / file.
|
||||
Please make sure the CRAFT_BUKKIT_CLASSES points to the test class directory and MINECRAFT_CLASSES to the minecraft server jar.
|
||||
""");
|
||||
|
||||
Stream<InputStream> result = Stream.empty();
|
||||
|
||||
if (contains(ClassReaderTest.ClassType.MINECRAFT_UNMODIFIED)) {
|
||||
result = Stream.concat(result, readMinecraftClasses());
|
||||
}
|
||||
|
||||
if (contains(ClassReaderTest.ClassType.CRAFT_BUKKIT) || contains(ClassReaderTest.ClassType.MINECRAFT_MODIFIED)) {
|
||||
result = Stream.concat(result, readCraftBukkitAndOrMinecraftModifiedClasses(contains(ClassReaderTest.ClassType.CRAFT_BUKKIT), contains(ClassReaderTest.ClassType.MINECRAFT_MODIFIED)));
|
||||
}
|
||||
|
||||
if (contains(ClassReaderTest.ClassType.BUKKIT)) {
|
||||
result = Stream.concat(result, readBukkitClasses());
|
||||
}
|
||||
|
||||
return result.map(this::toClassReader);
|
||||
}
|
||||
|
||||
private ClassReader toClassReader(InputStream stream) {
|
||||
try (stream) {
|
||||
return new ClassReader(stream);
|
||||
} catch (IOException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}
|
||||
|
||||
private boolean contains(ClassReaderTest.ClassType classType) {
|
||||
for (ClassReaderTest.ClassType c : classTypes) {
|
||||
if (c == classType) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
private Stream<InputStream> readMinecraftClasses() {
|
||||
return readJarFile(MINECRAFT_CLASSES, true);
|
||||
}
|
||||
|
||||
private Stream<InputStream> readBukkitClasses() {
|
||||
return readJarFile(BUKKIT_CLASSES, false);
|
||||
}
|
||||
|
||||
private Stream<InputStream> readJarFile(URI uri, boolean filterModified) {
|
||||
try {
|
||||
JarFile jarFile = new JarFile(new File(uri));
|
||||
return jarFile.stream().onClose(() -> closeJarFile(jarFile))
|
||||
.filter(entry -> entry.getName().endsWith(".class"))
|
||||
.filter(entry -> filterModifiedIfNeeded(entry, filterModified))
|
||||
.filter(entry -> filterPackageNames(entry.getName()))
|
||||
.filter(entry -> filterClass(entry.getName()))
|
||||
.map(entry -> {
|
||||
try {
|
||||
return jarFile.getInputStream(entry);
|
||||
} catch (IOException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
});
|
||||
} catch (IOException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}
|
||||
|
||||
private boolean filterModifiedIfNeeded(JarEntry entry, boolean needed) {
|
||||
if (!needed) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return !new File(CRAFT_BUKKIT_CLASSES.resolve(entry.getName())).exists();
|
||||
}
|
||||
|
||||
private boolean filterPackageNames(String name) {
|
||||
for (String packageName : excludedPackages) {
|
||||
if (name.startsWith(packageName)) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
private boolean filterClass(String name) {
|
||||
for (Class<?> clazz : excludedClasses) {
|
||||
if (name.equals(clazz.getName().replace('.', '/') + ".class")) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
private Stream<InputStream> readCraftBukkitAndOrMinecraftModifiedClasses(boolean craftBukkit, boolean minecraftModified) {
|
||||
try {
|
||||
return Files.walk(Path.of(CRAFT_BUKKIT_CLASSES))
|
||||
.map(Path::toFile)
|
||||
.filter(File::isFile)
|
||||
.filter(file -> file.getName().endsWith(".class"))
|
||||
.filter(file -> shouldInclude(removeHomeDirectory(file), craftBukkit, minecraftModified))
|
||||
.filter(file -> filterPackageNames(removeHomeDirectory(file)))
|
||||
.filter(file -> filterClass(removeHomeDirectory(file)))
|
||||
.map(file -> {
|
||||
try {
|
||||
return new FileInputStream(file);
|
||||
} catch (FileNotFoundException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
});
|
||||
} catch (IOException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}
|
||||
|
||||
private String removeHomeDirectory(File file) {
|
||||
return file.getAbsolutePath().substring(CRAFT_BUKKIT_CLASSES.getPath().length());
|
||||
}
|
||||
|
||||
private boolean shouldInclude(String name, boolean craftBukkit, boolean minecraftModified) {
|
||||
if (craftBukkit && minecraftModified) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (craftBukkit) {
|
||||
return name.startsWith("org/bukkit/craftbukkit/");
|
||||
}
|
||||
|
||||
if (minecraftModified) {
|
||||
return name.startsWith("net/minecraft/");
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
private void closeJarFile(JarFile jarFile) {
|
||||
try {
|
||||
jarFile.close();
|
||||
} catch (IOException e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,29 @@
|
||||
package org.bukkit.support.test;
|
||||
|
||||
import java.lang.annotation.ElementType;
|
||||
import java.lang.annotation.Retention;
|
||||
import java.lang.annotation.RetentionPolicy;
|
||||
import java.lang.annotation.Target;
|
||||
import org.bukkit.support.provider.ClassNodeArgumentProvider;
|
||||
import org.junit.jupiter.params.ParameterizedTest;
|
||||
import org.junit.jupiter.params.provider.ArgumentsSource;
|
||||
|
||||
@Target({ElementType.METHOD, ElementType.ANNOTATION_TYPE})
|
||||
@Retention(RetentionPolicy.RUNTIME)
|
||||
@ArgumentsSource(ClassNodeArgumentProvider.class)
|
||||
@ParameterizedTest
|
||||
public @interface ClassNodeTest {
|
||||
|
||||
ClassType[] value() default {ClassType.BUKKIT, ClassType.CRAFT_BUKKIT, ClassType.MINECRAFT_UNMODIFIED, ClassType.MINECRAFT_MODIFIED};
|
||||
|
||||
Class<?>[] excludedClasses() default {};
|
||||
|
||||
String[] excludedPackages() default {};
|
||||
|
||||
enum ClassType {
|
||||
BUKKIT,
|
||||
CRAFT_BUKKIT,
|
||||
MINECRAFT_UNMODIFIED,
|
||||
MINECRAFT_MODIFIED,
|
||||
}
|
||||
}
|
@ -0,0 +1,29 @@
|
||||
package org.bukkit.support.test;
|
||||
|
||||
import java.lang.annotation.ElementType;
|
||||
import java.lang.annotation.Retention;
|
||||
import java.lang.annotation.RetentionPolicy;
|
||||
import java.lang.annotation.Target;
|
||||
import org.bukkit.support.provider.ClassReaderArgumentProvider;
|
||||
import org.junit.jupiter.params.ParameterizedTest;
|
||||
import org.junit.jupiter.params.provider.ArgumentsSource;
|
||||
|
||||
@Target({ElementType.METHOD, ElementType.ANNOTATION_TYPE})
|
||||
@Retention(RetentionPolicy.RUNTIME)
|
||||
@ArgumentsSource(ClassReaderArgumentProvider.class)
|
||||
@ParameterizedTest
|
||||
public @interface ClassReaderTest {
|
||||
|
||||
ClassType[] value() default {ClassType.BUKKIT, ClassType.CRAFT_BUKKIT, ClassType.MINECRAFT_UNMODIFIED, ClassType.MINECRAFT_MODIFIED};
|
||||
|
||||
Class<?>[] excludedClasses() default {};
|
||||
|
||||
String[] excludedPackages() default {};
|
||||
|
||||
enum ClassType {
|
||||
BUKKIT,
|
||||
CRAFT_BUKKIT,
|
||||
MINECRAFT_UNMODIFIED,
|
||||
MINECRAFT_MODIFIED,
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user