geforkt von Mirrors/Paper
#1475: Add tests for Minecraft registry <-> Bukkit fields
By: DerFrZocker <derrieple@gmail.com>
Dieser Commit ist enthalten in:
Ursprung
8b1426d818
Commit
eb72b9d8a3
@ -279,6 +279,16 @@ public class CraftRegistry<B extends Keyed, M> implements Registry<B> {
|
|||||||
return bukkit;
|
return bukkit;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
@Override
|
||||||
|
public B getOrThrow(@NotNull NamespacedKey namespacedKey) {
|
||||||
|
B object = get(namespacedKey);
|
||||||
|
|
||||||
|
Preconditions.checkArgument(object != null, "No registry entry found for key " + namespacedKey);
|
||||||
|
|
||||||
|
return object;
|
||||||
|
}
|
||||||
|
|
||||||
@NotNull
|
@NotNull
|
||||||
@Override
|
@Override
|
||||||
public Stream<B> stream() {
|
public Stream<B> stream() {
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
package org.bukkit.craftbukkit.generator.structure;
|
package org.bukkit.craftbukkit.generator.structure;
|
||||||
|
|
||||||
|
import com.google.common.base.Suppliers;
|
||||||
|
import java.util.function.Supplier;
|
||||||
import net.minecraft.core.registries.Registries;
|
import net.minecraft.core.registries.Registries;
|
||||||
import org.bukkit.NamespacedKey;
|
import org.bukkit.NamespacedKey;
|
||||||
import org.bukkit.Registry;
|
import org.bukkit.Registry;
|
||||||
@ -20,12 +22,12 @@ public class CraftStructure extends Structure implements Handleable<net.minecraf
|
|||||||
|
|
||||||
private final NamespacedKey key;
|
private final NamespacedKey key;
|
||||||
private final net.minecraft.world.level.levelgen.structure.Structure structure;
|
private final net.minecraft.world.level.levelgen.structure.Structure structure;
|
||||||
private final StructureType structureType;
|
private final Supplier<StructureType> structureType;
|
||||||
|
|
||||||
public CraftStructure(NamespacedKey key, net.minecraft.world.level.levelgen.structure.Structure structure) {
|
public CraftStructure(NamespacedKey key, net.minecraft.world.level.levelgen.structure.Structure structure) {
|
||||||
this.key = key;
|
this.key = key;
|
||||||
this.structure = structure;
|
this.structure = structure;
|
||||||
this.structureType = CraftStructureType.minecraftToBukkit(structure.type());
|
this.structureType = Suppliers.memoize(() -> CraftStructureType.minecraftToBukkit(structure.type()));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -35,7 +37,7 @@ public class CraftStructure extends Structure implements Handleable<net.minecraf
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public StructureType getStructureType() {
|
public StructureType getStructureType() {
|
||||||
return structureType;
|
return structureType.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
556
paper-server/src/test/java/org/bukkit/registry/RegistryClassTest.java
Normale Datei
556
paper-server/src/test/java/org/bukkit/registry/RegistryClassTest.java
Normale Datei
@ -0,0 +1,556 @@
|
|||||||
|
package org.bukkit.registry;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
|
import static org.mockito.Mockito.*;
|
||||||
|
import com.google.common.base.Joiner;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.lang.annotation.Annotation;
|
||||||
|
import java.lang.reflect.Field;
|
||||||
|
import java.lang.reflect.Modifier;
|
||||||
|
import java.net.URISyntaxException;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Optional;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.jar.JarEntry;
|
||||||
|
import java.util.jar.JarFile;
|
||||||
|
import java.util.logging.Logger;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
import net.minecraft.core.IRegistry;
|
||||||
|
import net.minecraft.core.IRegistryCustom;
|
||||||
|
import net.minecraft.resources.MinecraftKey;
|
||||||
|
import net.minecraft.resources.ResourceKey;
|
||||||
|
import net.minecraft.world.flag.FeatureElement;
|
||||||
|
import net.minecraft.world.flag.FeatureFlags;
|
||||||
|
import org.bukkit.Bukkit;
|
||||||
|
import org.bukkit.Keyed;
|
||||||
|
import org.bukkit.MinecraftExperimental;
|
||||||
|
import org.bukkit.NamespacedKey;
|
||||||
|
import org.bukkit.Registry;
|
||||||
|
import org.bukkit.craftbukkit.util.CraftNamespacedKey;
|
||||||
|
import org.bukkit.support.RegistryHelper;
|
||||||
|
import org.bukkit.support.environment.AllFeatures;
|
||||||
|
import org.bukkit.support.extension.AllFeaturesExtension;
|
||||||
|
import org.bukkit.support.provider.RegistriesArgumentProvider;
|
||||||
|
import org.bukkit.support.test.RegistriesTest;
|
||||||
|
import org.jetbrains.annotations.ApiStatus;
|
||||||
|
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.mockito.internal.util.MockUtil;
|
||||||
|
import org.objectweb.asm.AnnotationVisitor;
|
||||||
|
import org.objectweb.asm.ClassReader;
|
||||||
|
import org.objectweb.asm.ClassVisitor;
|
||||||
|
import org.objectweb.asm.FieldVisitor;
|
||||||
|
import org.objectweb.asm.Opcodes;
|
||||||
|
import org.objectweb.asm.Type;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Note: This test class assumes that feature flags only enable more features and do not disable vanilla ones.
|
||||||
|
*/
|
||||||
|
@AllFeatures
|
||||||
|
public class RegistryClassTest {
|
||||||
|
|
||||||
|
private static final Map<Class<? extends Keyed>, Data> INIT_DATA = new HashMap<>();
|
||||||
|
private static final List<Arguments> FIELD_DATA_CACHE = new ArrayList<>();
|
||||||
|
private static IRegistryCustom.Dimension vanilla_registry;
|
||||||
|
|
||||||
|
public static Stream<Arguments> fieldData() {
|
||||||
|
return FIELD_DATA_CACHE.stream();
|
||||||
|
}
|
||||||
|
|
||||||
|
@BeforeAll
|
||||||
|
public static void init() {
|
||||||
|
initValueClasses();
|
||||||
|
initFieldDataCache();
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterAll
|
||||||
|
public static void cleanUp() {
|
||||||
|
INIT_DATA.clear();
|
||||||
|
FIELD_DATA_CACHE.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void initValueClasses() {
|
||||||
|
vanilla_registry = RegistryHelper.createRegistry(FeatureFlags.VANILLA_SET);
|
||||||
|
|
||||||
|
Map<Class<? extends Keyed>, List<NamespacedKey>> outsideRequest = new LinkedHashMap<>();
|
||||||
|
|
||||||
|
// First init listening for outside requests
|
||||||
|
RegistriesArgumentProvider.getData()
|
||||||
|
.map(Arguments::get).map(args -> args[0])
|
||||||
|
.map(type -> (Class<? extends Keyed>) type)
|
||||||
|
.forEach(type -> {
|
||||||
|
Registry<?> spyRegistry = Bukkit.getRegistry(type);
|
||||||
|
spyOutsideRequests(outsideRequest, type, spyRegistry);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Init all registries and recorde the outcome
|
||||||
|
RegistriesArgumentProvider.getData()
|
||||||
|
.map(Arguments::get)
|
||||||
|
.map(args -> args[0])
|
||||||
|
.map(type -> (Class<? extends Keyed>) type)
|
||||||
|
.forEachOrdered(type -> {
|
||||||
|
try {
|
||||||
|
Registry<?> spyRegistry = Bukkit.getRegistry(type);
|
||||||
|
Registry<?> realRegistry = AllFeaturesExtension.getRealRegistry(type);
|
||||||
|
|
||||||
|
Set<NamespacedKey> nullAble = new HashSet<>();
|
||||||
|
Set<NamespacedKey> notNullAble = new HashSet<>();
|
||||||
|
Set<NamespacedKey> nullValue = new HashSet<>();
|
||||||
|
|
||||||
|
outsideRequest.clear();
|
||||||
|
MockUtil.resetMock(spyRegistry);
|
||||||
|
doAnswer(invocation -> {
|
||||||
|
Keyed item = realRegistry.get(invocation.getArgument(0));
|
||||||
|
|
||||||
|
if (item == null) {
|
||||||
|
nullValue.add(invocation.getArgument(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
nullAble.add(invocation.getArgument(0));
|
||||||
|
|
||||||
|
return item;
|
||||||
|
}).when(spyRegistry).get(any());
|
||||||
|
|
||||||
|
doAnswer(invocation -> {
|
||||||
|
Keyed item = realRegistry.get(invocation.getArgument(0));
|
||||||
|
|
||||||
|
if (item == null) {
|
||||||
|
nullValue.add(invocation.getArgument(0));
|
||||||
|
|
||||||
|
// We return a mock here so that it (hopefully) does not error out immediately.
|
||||||
|
// It may cause errors in other tests, but this test class should always pass before other tests can run safely.
|
||||||
|
// This is not necessary in the normal get method since the API contract states that the method can return null,
|
||||||
|
// whereas in this method, it is guaranteed to be non-null.
|
||||||
|
item = mock(type);
|
||||||
|
}
|
||||||
|
|
||||||
|
notNullAble.add(invocation.getArgument(0));
|
||||||
|
|
||||||
|
return item;
|
||||||
|
}).when(spyRegistry).getOrThrow(any());
|
||||||
|
|
||||||
|
// Load class
|
||||||
|
try {
|
||||||
|
Class.forName(type.getName());
|
||||||
|
} catch (Throwable e) {
|
||||||
|
e.printStackTrace(); // Print stacktrace, since JUnit eats the error in BeforeAll
|
||||||
|
fail(e);
|
||||||
|
} finally {
|
||||||
|
MockUtil.resetMock(spyRegistry);
|
||||||
|
spyOutsideRequests(outsideRequest, type, spyRegistry);
|
||||||
|
}
|
||||||
|
|
||||||
|
INIT_DATA.put(type, new Data(nullAble, notNullAble, nullValue, new LinkedHashMap<>(outsideRequest)));
|
||||||
|
} catch (Throwable e) {
|
||||||
|
e.printStackTrace(); // Print stacktrace, since JUnit eats the error in BeforeAll
|
||||||
|
fail(e);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Cleanup
|
||||||
|
RegistriesArgumentProvider.getData()
|
||||||
|
.map(Arguments::get)
|
||||||
|
.map(args -> args[0])
|
||||||
|
.map(type -> (Class<? extends Keyed>) type)
|
||||||
|
.forEach(type -> MockUtil.resetMock(Bukkit.getRegistry(type)));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void spyOutsideRequests(Map<Class<? extends Keyed>, List<NamespacedKey>> outsideRequest, Class<? extends Keyed> type, Registry<?> spyRegistry) {
|
||||||
|
doAnswer(invocation -> {
|
||||||
|
outsideRequest
|
||||||
|
.computeIfAbsent(type, ty -> new ArrayList<>()).add(invocation.getArgument(0));
|
||||||
|
return mock(type);
|
||||||
|
}).when(spyRegistry).get(any());
|
||||||
|
|
||||||
|
doAnswer(invocation -> {
|
||||||
|
outsideRequest
|
||||||
|
.computeIfAbsent(type, ty -> new ArrayList<>()).add(invocation.getArgument(0));
|
||||||
|
return mock(type);
|
||||||
|
}).when(spyRegistry).getOrThrow(any());
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void initFieldDataCache() {
|
||||||
|
RegistriesArgumentProvider.getData().map(arguments -> {
|
||||||
|
Class<? extends Keyed> type = (Class<? extends Keyed>) arguments.get()[0];
|
||||||
|
Map<String, List<Class<? extends Annotation>>> annotations = getFieldAnnotations(type);
|
||||||
|
|
||||||
|
List<FieldData> fields = new ArrayList<>();
|
||||||
|
for (Field field : type.getFields()) {
|
||||||
|
// We ignore each field that does not have the class itself as its type,
|
||||||
|
// is not static, public, or is deprecated.
|
||||||
|
if (!isValidField(type, field)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
Keyed keyed;
|
||||||
|
try {
|
||||||
|
keyed = (Keyed) field.get(null);
|
||||||
|
} catch (IllegalAccessException e) {
|
||||||
|
Logger.getGlobal().warning(String.format("Something went wrong while trying to read the field %s from class %s.\n"
|
||||||
|
+ "Please see the stack trace below for more information.", field, type));
|
||||||
|
e.printStackTrace();
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (keyed == null) {
|
||||||
|
// We have a test case for those scenarios,
|
||||||
|
// so we ignore them here to ensure that at least the other fields are being tested.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (MockUtil.isMock(keyed)) {
|
||||||
|
// If it is a mock, it means that there was no Minecraft registry item for that key.
|
||||||
|
// In this case, ignore it, as we already test and inform about this in another test.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
fields.add(new FieldData(field, annotations.computeIfAbsent(field.getName(), k -> new ArrayList<>())));
|
||||||
|
}
|
||||||
|
|
||||||
|
return Arguments.arguments(arguments.get()[0], arguments.get()[1], fields);
|
||||||
|
}).forEach(FIELD_DATA_CACHE::add);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Map<String, List<Class<? extends Annotation>>> getFieldAnnotations(Class<?> aClass) {
|
||||||
|
Map<String, List<Class<? extends Annotation>>> annotation = new HashMap<>();
|
||||||
|
|
||||||
|
try (JarFile jarFile = new JarFile(new File(aClass.getProtectionDomain().getCodeSource().getLocation().toURI()))) {
|
||||||
|
JarEntry jarEntry = jarFile.getJarEntry(aClass.getName().replace('.', '/') + ".class");
|
||||||
|
|
||||||
|
try (InputStream inputStream = jarFile.getInputStream(jarEntry)) {
|
||||||
|
ClassReader classReader = new ClassReader(inputStream);
|
||||||
|
classReader.accept(new ClassVisitor(Opcodes.ASM9) {
|
||||||
|
@Override
|
||||||
|
public FieldVisitor visitField(int access, String name, String descriptor, String signature, Object value) {
|
||||||
|
return new FieldVisitor(Opcodes.ASM9) {
|
||||||
|
@Override
|
||||||
|
public AnnotationVisitor visitAnnotation(String descriptor, boolean visible) {
|
||||||
|
try {
|
||||||
|
annotation.computeIfAbsent(name, k -> new ArrayList<>()).add((Class<? extends Annotation>) Class.forName(Type.getType(descriptor).getClassName()));
|
||||||
|
} catch (ClassNotFoundException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}, Opcodes.ASM9);
|
||||||
|
}
|
||||||
|
} catch (URISyntaxException | IOException e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
return annotation;
|
||||||
|
}
|
||||||
|
|
||||||
|
@RegistriesTest
|
||||||
|
public <T extends Keyed> void testOutsideRequests(Class<T> type) {
|
||||||
|
Data data = INIT_DATA.get(type);
|
||||||
|
|
||||||
|
assertNotNull(data, String.format("No data present for %s. This should never happen since the same data source is used.\n"
|
||||||
|
+ "Something has gone horribly wrong.", type));
|
||||||
|
|
||||||
|
assertTrue(data.outsideRequests.isEmpty(), String.format("""
|
||||||
|
There were outside registry accesses while loading registry items for class %s.
|
||||||
|
This can happen if you try to read any registry item in the constructor.
|
||||||
|
For ease of testing, please remove this from the constructor.
|
||||||
|
Subsequent tests may fail because of this.
|
||||||
|
|
||||||
|
You can use a Supplier instead. For easy caching, you can use com.google.common.base.Suppliers#memoize(Supplier)
|
||||||
|
Example:
|
||||||
|
|
||||||
|
private final Supplier<RegisterItem> otherRegisterItem;
|
||||||
|
|
||||||
|
public CraftRegisterItem([...]) {
|
||||||
|
[...]
|
||||||
|
this.otherRegisterItem = Suppliers.memoize(() -> CraftRegisterItem.getSome("other_key"));
|
||||||
|
}
|
||||||
|
|
||||||
|
public RegisterItem getOtherRegisterItem() {
|
||||||
|
return otherRegisterItem.get();
|
||||||
|
}
|
||||||
|
The following outside requests were registered:
|
||||||
|
%s""", type.getName(), Joiner.on('\n').withKeyValueSeparator(" <-> ").join(data.outsideRequests)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@RegistriesTest
|
||||||
|
public <T extends Keyed> void testNoNullValuePresent(Class<T> type) {
|
||||||
|
Data data = INIT_DATA.get(type);
|
||||||
|
|
||||||
|
assertNotNull(data, String.format("No data present for %s. This should never happen since the same data source is used.\n"
|
||||||
|
+ "Something has gone horribly wrong.", type));
|
||||||
|
|
||||||
|
assertTrue(data.nullValue.isEmpty(), String.format("""
|
||||||
|
%s tried to get registry items that are not present.
|
||||||
|
This can be caused if the affected registry item was removed, renamed, or if you mistyped the name.
|
||||||
|
Alternatively, there maybe was an attempt to read another registry item in the constructor of a registry item.
|
||||||
|
Subsequent tests may fail because of this.
|
||||||
|
The following registry items were requested, but there is no registry item for them:
|
||||||
|
%s""", type, Joiner.on('\n').join(data.nullValue)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@RegistriesTest
|
||||||
|
public <T extends Keyed> void testForNullValues(Class<T> type) throws IllegalAccessException {
|
||||||
|
List<Field> nullFields = new ArrayList<>();
|
||||||
|
|
||||||
|
for (Field field : type.getFields()) {
|
||||||
|
// We ignore each field that does not have the class itself as its type, is not static, public, or is deprecated.
|
||||||
|
if (!isValidField(type, field)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
Keyed keyed = (Keyed) field.get(null);
|
||||||
|
|
||||||
|
if (keyed == null) {
|
||||||
|
// This should not happen since, even if the item is not present in the Minecraft registry, we would use a mock.
|
||||||
|
nullFields.add(field);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue(nullFields.isEmpty(), String.format("""
|
||||||
|
The class %s has fields with no value assigned to them.
|
||||||
|
This should not normally happen, since all feature flags are set,
|
||||||
|
and a dummy is returned in cases where the registry item does not exist.
|
||||||
|
Please make sure the following fields are not null:
|
||||||
|
%s""", type, Joiner.on('\n').join(nullFields)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@MethodSource("fieldData")
|
||||||
|
public <T extends Keyed> void testExcessExperimentalAnnotation(Class<T> type, ResourceKey<IRegistry<?>> registryKey, List<FieldData> fieldDataList) throws IllegalAccessException {
|
||||||
|
List<Field> excess = new ArrayList<>();
|
||||||
|
|
||||||
|
for (FieldData fieldData : fieldDataList) {
|
||||||
|
if (!fieldData.annotations().contains(MinecraftExperimental.class) && !fieldData.annotations().contains(ApiStatus.Experimental.class)) {
|
||||||
|
// No annotation -> no problem
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
IRegistry<?> registry = vanilla_registry.registryOrThrow(registryKey);
|
||||||
|
|
||||||
|
Optional<?> optionalValue = registry.getOptional(CraftNamespacedKey.toMinecraft(((Keyed) fieldData.field().get(null)).getKey()));
|
||||||
|
|
||||||
|
if (optionalValue.isEmpty()) {
|
||||||
|
// The value is not present, which means it comes from a feature flag.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
Object value = optionalValue.get();
|
||||||
|
|
||||||
|
if (value instanceof FeatureElement featureElement && !featureElement.isEnabled(FeatureFlags.VANILLA_SET)) {
|
||||||
|
// It is a FeatureElement that is not enabled for vanilla FeatureSet so ignore it.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
excess.add(fieldData.field());
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue(excess.isEmpty(), String.format("""
|
||||||
|
The class %s has fields with the MinecraftExperimental and/or ApiStatus.Experimental annotation.
|
||||||
|
However, the value is present and active with the vanilla feature flags.
|
||||||
|
The annotation should be removed from the following fields:
|
||||||
|
%s""", type, Joiner.on('\n').join(excess)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@MethodSource("fieldData")
|
||||||
|
public <T extends Keyed> void testMissingExperimentalAnnotation(Class<T> type, ResourceKey<IRegistry<?>> registryKey, List<FieldData> fieldDataList) throws IllegalAccessException {
|
||||||
|
IRegistry<?> registry = vanilla_registry.registryOrThrow(registryKey);
|
||||||
|
List<Field> missing = new ArrayList<>();
|
||||||
|
|
||||||
|
for (FieldData fieldData : fieldDataList) {
|
||||||
|
if (fieldData.annotations().contains(MinecraftExperimental.class) && fieldData.annotations().contains(ApiStatus.Experimental.class)) {
|
||||||
|
// Annotation present -> no problem
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
Optional<?> optionalValue = registry.getOptional(CraftNamespacedKey.toMinecraft(((Keyed) fieldData.field().get(null)).getKey()));
|
||||||
|
|
||||||
|
if (optionalValue.isEmpty()) {
|
||||||
|
// The value is not present, which means it comes from a feature flag.
|
||||||
|
missing.add(fieldData.field());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
Object value = optionalValue.get();
|
||||||
|
|
||||||
|
if (value instanceof FeatureElement featureElement && !featureElement.isEnabled(FeatureFlags.VANILLA_SET)) {
|
||||||
|
// It is a FeatureElement that is not enabled for vanilla FeatureSet.
|
||||||
|
missing.add(fieldData.field());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue(missing.isEmpty(), String.format("""
|
||||||
|
The class %s has fields that don't have the MinecraftExperimental and/or ApiStatus.Experimental annotation.
|
||||||
|
However, the value is not present or active with the vanilla feature flags.
|
||||||
|
The annotation should be added to the following fields: %s
|
||||||
|
""", type, Joiner.on('\n').join(missing)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@MethodSource("fieldData")
|
||||||
|
public <T extends Keyed> void testExcessNullCheck(Class<T> type, ResourceKey<IRegistry<?>> registryKey, List<FieldData> fieldDataList) throws IllegalAccessException {
|
||||||
|
IRegistry<?> registry = vanilla_registry.registryOrThrow(registryKey);
|
||||||
|
List<Field> excess = new ArrayList<>();
|
||||||
|
Data data = INIT_DATA.get(type);
|
||||||
|
|
||||||
|
for (FieldData fieldData : fieldDataList) {
|
||||||
|
NamespacedKey key = ((Keyed) fieldData.field().get(null)).getKey();
|
||||||
|
|
||||||
|
Optional<?> optionalValue = registry.getOptional(CraftNamespacedKey.toMinecraft(key));
|
||||||
|
|
||||||
|
if (optionalValue.isPresent()) {
|
||||||
|
// The value is present, which means it cannot have an unnecessary null check, so ignore it.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (data.notNullAble.contains(key)) {
|
||||||
|
// The value is null, but there was a null check, which it should not have.
|
||||||
|
excess.add(fieldData.field());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue(excess.isEmpty(), String.format("""
|
||||||
|
The class %s has fields that do have a null check.
|
||||||
|
However, the value can be null, and there should not be a null check, so that it won't error out, and an IDE can handle it accordingly.
|
||||||
|
If there is no null check, make sure that org.bukkit.Registry#get(NamespaceKey) is used.
|
||||||
|
The following fields do have a null check:
|
||||||
|
%s""", type, Joiner.on('\n').join(excess)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@MethodSource("fieldData")
|
||||||
|
public <T extends Keyed> void testMissingNullCheck(Class<T> type, ResourceKey<IRegistry<?>> registryKey, List<FieldData> fieldDataList) throws IllegalAccessException {
|
||||||
|
IRegistry<?> registry = vanilla_registry.registryOrThrow(registryKey);
|
||||||
|
List<Field> missing = new ArrayList<>();
|
||||||
|
Data data = INIT_DATA.get(type);
|
||||||
|
|
||||||
|
for (FieldData fieldData : fieldDataList) {
|
||||||
|
NamespacedKey key = ((Keyed) fieldData.field().get(null)).getKey();
|
||||||
|
|
||||||
|
Optional<?> optionalValue = registry.getOptional(CraftNamespacedKey.toMinecraft(key));
|
||||||
|
|
||||||
|
if (optionalValue.isEmpty()) {
|
||||||
|
// The value is not present, which means there is no need for a null check, so ignore it.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (data.nullAble.contains(key)) {
|
||||||
|
// The value is not null, and there was no null check, which means it is missing.
|
||||||
|
missing.add(fieldData.field());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue(missing.isEmpty(), String.format("""
|
||||||
|
The class %s has fields that don't have a null check.
|
||||||
|
However, the value cannot be null, and there should be a null check so that an IDE can handle it accordingly.
|
||||||
|
If there is a null check, make sure that org.bukkit.Registry#getOrThrow(NamespaceKey) is used.
|
||||||
|
The following fields don't have a null check:
|
||||||
|
%s""", type, Joiner.on('\n').join(missing)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@MethodSource("fieldData")
|
||||||
|
public <T extends Keyed> void testMatchingFieldNames(Class<T> type, ResourceKey<IRegistry<?>> registryKey, List<FieldData> fieldDataList) throws IllegalAccessException {
|
||||||
|
Map<Field, NamespacedKey> notMatching = new LinkedHashMap<>();
|
||||||
|
|
||||||
|
for (FieldData fieldData : fieldDataList) {
|
||||||
|
NamespacedKey key = ((Keyed) fieldData.field().get(null)).getKey();
|
||||||
|
|
||||||
|
if (fieldData.field().getName().equals(convertToFieldName(key.getKey()))) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Field names to not match.
|
||||||
|
notMatching.put(fieldData.field(), key);
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue(notMatching.isEmpty(), String.format("""
|
||||||
|
The class %s has fields whose keys don't match up.
|
||||||
|
The names of each field should match up with their key.
|
||||||
|
If there is a '.' in the key, it should be replaced with '_' in the field name.
|
||||||
|
If the registry item no longer exists and is replaced by a new one,
|
||||||
|
add the @Deprecated annotation to the field and create a new field for the new registry item.
|
||||||
|
The following fields have mismatching keys:
|
||||||
|
%s""", type, Joiner.on('\n').withKeyValueSeparator(" <-> ").join(notMatching)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@MethodSource("fieldData")
|
||||||
|
public <T extends Keyed> void testMissingConstants(Class<T> type, ResourceKey<IRegistry<?>> registryKey) throws IllegalAccessException {
|
||||||
|
IRegistry<Object> registry = RegistryHelper.getRegistry().registryOrThrow(registryKey);
|
||||||
|
List<MinecraftKey> missingKeys = new ArrayList<>();
|
||||||
|
|
||||||
|
for (Object nmsObject : registry) {
|
||||||
|
MinecraftKey minecraftKey = registry.getKey(nmsObject);
|
||||||
|
|
||||||
|
try {
|
||||||
|
Field field = type.getField(convertToFieldName(minecraftKey.getPath()));
|
||||||
|
|
||||||
|
// Only fields which are not Deprecated
|
||||||
|
// and have the right registry item associated with the field count.
|
||||||
|
if (!isValidField(type, field)) {
|
||||||
|
missingKeys.add(minecraftKey);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
T keyed = (T) field.get(null);
|
||||||
|
|
||||||
|
if (keyed == null) {
|
||||||
|
missingKeys.add(minecraftKey);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!keyed.getKey().equals(CraftNamespacedKey.fromMinecraft(minecraftKey))) {
|
||||||
|
missingKeys.add(minecraftKey);
|
||||||
|
}
|
||||||
|
} catch (NoSuchFieldException e) {
|
||||||
|
missingKeys.add(minecraftKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assertTrue(missingKeys.isEmpty(), String.format("""
|
||||||
|
The class %s has missing fields for some registry items.
|
||||||
|
There should be a field for each registry item.
|
||||||
|
In case there is a field with the correct name (not misspelled), make sure that:
|
||||||
|
The field is not null, the field is not annotated with @Deprecated, the field has the right registry item key.
|
||||||
|
The following registry items don't have a field:
|
||||||
|
%s""", type, Joiner.on('\n').join(missingKeys)));
|
||||||
|
}
|
||||||
|
|
||||||
|
private String convertToFieldName(String value) {
|
||||||
|
return switch (value) {
|
||||||
|
// JukeboxSong does have keys which start with a number, which is not a valid field name
|
||||||
|
case "5" -> "FIVE";
|
||||||
|
case "11" -> "ELEVEN";
|
||||||
|
case "13" -> "THIRTEEN";
|
||||||
|
default -> value.toUpperCase(Locale.ROOT).replace('.', '_');
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean isValidField(Class<? extends Keyed> type, Field field) {
|
||||||
|
return type.isAssignableFrom(field.getType()) && Modifier.isStatic(field.getModifiers())
|
||||||
|
&& Modifier.isPublic(field.getModifiers()) && !field.isAnnotationPresent(Deprecated.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
private record Data(Set<NamespacedKey> nullAble, Set<NamespacedKey> notNullAble,
|
||||||
|
Set<NamespacedKey> nullValue,
|
||||||
|
Map<Class<? extends Keyed>, List<NamespacedKey>> outsideRequests) {
|
||||||
|
}
|
||||||
|
|
||||||
|
private record FieldData(Field field, List<Class<? extends Annotation>> annotations) {
|
||||||
|
}
|
||||||
|
}
|
@ -1,8 +1,12 @@
|
|||||||
package org.bukkit.support.suite;
|
package org.bukkit.support.suite;
|
||||||
|
|
||||||
|
import org.bukkit.registry.PerRegistryTest;
|
||||||
|
import org.bukkit.registry.RegistryClassTest;
|
||||||
|
import org.bukkit.registry.RegistryConversionTest;
|
||||||
import org.junit.platform.suite.api.ConfigurationParameter;
|
import org.junit.platform.suite.api.ConfigurationParameter;
|
||||||
import org.junit.platform.suite.api.ExcludeClassNamePatterns;
|
import org.junit.platform.suite.api.ExcludeClassNamePatterns;
|
||||||
import org.junit.platform.suite.api.IncludeTags;
|
import org.junit.platform.suite.api.IncludeTags;
|
||||||
|
import org.junit.platform.suite.api.SelectClasses;
|
||||||
import org.junit.platform.suite.api.SelectPackages;
|
import org.junit.platform.suite.api.SelectPackages;
|
||||||
import org.junit.platform.suite.api.Suite;
|
import org.junit.platform.suite.api.Suite;
|
||||||
import org.junit.platform.suite.api.SuiteDisplayName;
|
import org.junit.platform.suite.api.SuiteDisplayName;
|
||||||
@ -11,6 +15,7 @@ import org.junit.platform.suite.api.SuiteDisplayName;
|
|||||||
@SuiteDisplayName("Test suite for test which need registry values present, with all feature flags set")
|
@SuiteDisplayName("Test suite for test which need registry values present, with all feature flags set")
|
||||||
@IncludeTags("AllFeatures")
|
@IncludeTags("AllFeatures")
|
||||||
@SelectPackages("org.bukkit")
|
@SelectPackages("org.bukkit")
|
||||||
|
@SelectClasses({RegistryClassTest.class, PerRegistryTest.class, RegistryConversionTest.class}) // Make sure general registry tests are run first
|
||||||
@ExcludeClassNamePatterns("org.bukkit.craftbukkit.inventory.ItemStack.*Test")
|
@ExcludeClassNamePatterns("org.bukkit.craftbukkit.inventory.ItemStack.*Test")
|
||||||
@ConfigurationParameter(key = "TestSuite", value = "AllFeatures")
|
@ConfigurationParameter(key = "TestSuite", value = "AllFeatures")
|
||||||
public class AllFeaturesTestSuite {
|
public class AllFeaturesTestSuite {
|
||||||
|
Laden…
In neuem Issue referenzieren
Einen Benutzer sperren