mirror of
https://github.com/plan-player-analytics/Plan.git
synced 2024-12-28 12:07:35 +01:00
Refactored ExtensionExtractor, Added Group parameter
This commit is contained in:
parent
aa8fe70e1f
commit
245c083237
@ -20,8 +20,27 @@ package com.djrapitops.plan.extension;
|
|||||||
* Interface to implement data extensions with.
|
* Interface to implement data extensions with.
|
||||||
* <p>
|
* <p>
|
||||||
* The class implementing this interface should be annotated with {@link com.djrapitops.plan.extension.annotation.PluginInfo}.
|
* The class implementing this interface should be annotated with {@link com.djrapitops.plan.extension.annotation.PluginInfo}.
|
||||||
* <p>
|
|
||||||
* If the extension is given to Plan API without the annotation it will be rejected.
|
* If the extension is given to Plan API without the annotation it will be rejected.
|
||||||
|
* <p>
|
||||||
|
* Public methods in the class should be annotated with appropriate Provider annotations.
|
||||||
|
* Provider annotations:
|
||||||
|
* {@link com.djrapitops.plan.extension.annotation.BooleanProvider} for {@code boolean} values and conditions for {@link com.djrapitops.plan.extension.annotation.Conditional}.
|
||||||
|
* {@link com.djrapitops.plan.extension.annotation.NumberProvider} for {@code long} values. (Use this for integers by casting to long) Has option for formatting.
|
||||||
|
* {@link com.djrapitops.plan.extension.annotation.DoubleProvider} for {@code double} values.
|
||||||
|
* {@link com.djrapitops.plan.extension.annotation.PercentageProvider} for {@code double} values that represent a percentage.
|
||||||
|
* {@link com.djrapitops.plan.extension.annotation.StringProvider} for {@link String} values.
|
||||||
|
* <p>
|
||||||
|
* Methods can have one of the following as method parameters:
|
||||||
|
* {@code UUID playerUUID} - UUID of the player the data is about
|
||||||
|
* {@code String playerName} - Name of the player the data is about
|
||||||
|
* {@link Group group} - Provided group the data is about (In case a group needs additional information)
|
||||||
|
* nothing - The data is interpreted to be about the server.
|
||||||
|
* <p>
|
||||||
|
* Some additional annotations are available for controlling appearance of the results:
|
||||||
|
* {@link com.djrapitops.plan.extension.annotation.Conditional} A {@code boolean} returned by {@link com.djrapitops.plan.extension.annotation.BooleanProvider} has to be {@code true} for this method to be called.
|
||||||
|
* {@link com.djrapitops.plan.extension.annotation.Tab} The value of this provider should be placed on a tab with a specific name
|
||||||
|
* {@link com.djrapitops.plan.extension.annotation.TabInfo} Optional Structure information about a tab
|
||||||
|
* {@link com.djrapitops.plan.extension.annotation.TabOrder} Optional information about preferred tab order
|
||||||
*
|
*
|
||||||
* @author Rsl1122
|
* @author Rsl1122
|
||||||
*/
|
*/
|
||||||
|
@ -0,0 +1,34 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of Player Analytics (Plan).
|
||||||
|
*
|
||||||
|
* Plan is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Lesser General Public License v3 as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* Plan 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 Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public License
|
||||||
|
* along with Plan. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
package com.djrapitops.plan.extension;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Method parameter for providing values about a group with provider annotations.
|
||||||
|
* <p>
|
||||||
|
* Usage Example: {@code @StringProvider String provideStringAboutGroup(Group group)}
|
||||||
|
* <p>
|
||||||
|
* Group names of users are provided with {@code @GroupProvider String[] provideGroups(UUID playerUUID)}
|
||||||
|
* <p>
|
||||||
|
* This method parameter is used since it is not possible to differentiate String playerName and String groupName.
|
||||||
|
*
|
||||||
|
* @author Rsl1122
|
||||||
|
*/
|
||||||
|
public interface Group {
|
||||||
|
|
||||||
|
String getGroupName();
|
||||||
|
|
||||||
|
}
|
@ -17,14 +17,17 @@
|
|||||||
package com.djrapitops.plan.extension.extractor;
|
package com.djrapitops.plan.extension.extractor;
|
||||||
|
|
||||||
import com.djrapitops.plan.extension.DataExtension;
|
import com.djrapitops.plan.extension.DataExtension;
|
||||||
|
import com.djrapitops.plan.extension.Group;
|
||||||
import com.djrapitops.plan.extension.annotation.*;
|
import com.djrapitops.plan.extension.annotation.*;
|
||||||
|
|
||||||
|
import java.lang.annotation.Annotation;
|
||||||
import java.lang.reflect.Method;
|
import java.lang.reflect.Method;
|
||||||
|
import java.lang.reflect.Modifier;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Implementation details for extracting methods from {@link com.djrapitops.plan.extension.DataExtension}.
|
* Implementation detail, for extracting methods from {@link com.djrapitops.plan.extension.DataExtension}.
|
||||||
* <p>
|
* <p>
|
||||||
* This class can be used for testing validity of annotation implementations
|
* This class can be used for testing validity of annotation implementations
|
||||||
* in your unit tests to avoid runtime errors. {@link ExtensionExtractor#validateAnnotations()}
|
* in your unit tests to avoid runtime errors. {@link ExtensionExtractor#validateAnnotations()}
|
||||||
@ -34,11 +37,18 @@ import java.util.stream.Collectors;
|
|||||||
public final class ExtensionExtractor {
|
public final class ExtensionExtractor {
|
||||||
|
|
||||||
private final DataExtension extension;
|
private final DataExtension extension;
|
||||||
|
private final String extensionName;
|
||||||
|
|
||||||
private final List<String> errors = new ArrayList<>();
|
private final List<String> warnings = new ArrayList<>();
|
||||||
|
|
||||||
|
private PluginInfo pluginInfo;
|
||||||
|
private TabOrder tabOrder;
|
||||||
|
private List<TabInfo> tabInformation;
|
||||||
|
private MethodAnnotations methodAnnotations;
|
||||||
|
|
||||||
public ExtensionExtractor(DataExtension extension) {
|
public ExtensionExtractor(DataExtension extension) {
|
||||||
this.extension = extension;
|
this.extension = extension;
|
||||||
|
extensionName = extension.getClass().getSimpleName();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -47,295 +57,276 @@ public final class ExtensionExtractor {
|
|||||||
* @throws IllegalArgumentException If an implementation error is found.
|
* @throws IllegalArgumentException If an implementation error is found.
|
||||||
*/
|
*/
|
||||||
public void validateAnnotations() {
|
public void validateAnnotations() {
|
||||||
|
extractAnnotationInformation();
|
||||||
|
|
||||||
|
if (!warnings.isEmpty()) {
|
||||||
|
throw new IllegalArgumentException("Warnings: " + warnings.toString());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private <T extends Annotation> Optional<T> getClassAnnotation(Class<T> ofClass) {
|
||||||
|
return Optional.ofNullable(extension.getClass().getAnnotation(ofClass));
|
||||||
|
}
|
||||||
|
|
||||||
|
private Method[] getMethods() {
|
||||||
|
return extension.getClass().getDeclaredMethods();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void extractAnnotationInformation() {
|
||||||
extractPluginInfo();
|
extractPluginInfo();
|
||||||
|
|
||||||
|
extractMethodAnnotations();
|
||||||
|
validateMethodAnnotations();
|
||||||
|
|
||||||
|
validateConditionals();
|
||||||
|
|
||||||
extractTabInfo();
|
extractTabInfo();
|
||||||
List<Method> booleanMethods = extractBooleanMethods();
|
|
||||||
extractConditionals(booleanMethods);
|
|
||||||
extractNumberMethods();
|
|
||||||
extractDoubleMethods();
|
|
||||||
extractPercentageMethods();
|
|
||||||
extractStringMethods();
|
|
||||||
if (errors.isEmpty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
throw new IllegalArgumentException("Found errors: " + errors.toString());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public PluginInfo extractPluginInfo() {
|
private void extractMethodAnnotations() {
|
||||||
Class<? extends DataExtension> extClass = extension.getClass();
|
methodAnnotations = new MethodAnnotations();
|
||||||
PluginInfo info = extClass.getAnnotation(PluginInfo.class);
|
|
||||||
|
|
||||||
if (info.name().length() > 50) {
|
for (Method method : getMethods()) {
|
||||||
errors.add(extClass.getName() + "Plugin name was over 50 characters.");
|
if (!Modifier.isPublic(method.getModifiers())) {
|
||||||
}
|
|
||||||
|
|
||||||
return info;
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<TabInfo> extractTabInfo() {
|
|
||||||
Set<String> tabNames = new HashSet<>();
|
|
||||||
List<TabInfo> tabInformation = new ArrayList<>();
|
|
||||||
|
|
||||||
Class<? extends DataExtension> extClass = extension.getClass();
|
|
||||||
|
|
||||||
for (Method method : extClass.getDeclaredMethods()) {
|
|
||||||
Tab tab = method.getAnnotation(Tab.class);
|
|
||||||
if (tab == null) {
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
String tabName = tab.value();
|
|
||||||
// Length restriction check
|
MethodAnnotations.get(method, BooleanProvider.class).ifPresent(annotation -> methodAnnotations.put(method, BooleanProvider.class, annotation));
|
||||||
if (tabName.length() > 50) {
|
MethodAnnotations.get(method, NumberProvider.class).ifPresent(annotation -> methodAnnotations.put(method, NumberProvider.class, annotation));
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " tab name was over 50 characters.");
|
MethodAnnotations.get(method, DoubleProvider.class).ifPresent(annotation -> methodAnnotations.put(method, DoubleProvider.class, annotation));
|
||||||
|
MethodAnnotations.get(method, PercentageProvider.class).ifPresent(annotation -> methodAnnotations.put(method, PercentageProvider.class, annotation));
|
||||||
|
MethodAnnotations.get(method, StringProvider.class).ifPresent(annotation -> methodAnnotations.put(method, StringProvider.class, annotation));
|
||||||
|
|
||||||
|
MethodAnnotations.get(method, Conditional.class).ifPresent(annotation -> methodAnnotations.put(method, Conditional.class, annotation));
|
||||||
|
MethodAnnotations.get(method, Tab.class).ifPresent(annotation -> methodAnnotations.put(method, Tab.class, annotation));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (methodAnnotations.isEmpty()) {
|
||||||
|
throw new IllegalArgumentException(extensionName + " class had no methods annotated with a Provider annotation");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private <T> void validateReturnType(Method method, Class<T> expectedType) {
|
||||||
|
Class<?> returnType = method.getReturnType();
|
||||||
|
if (!expectedType.isAssignableFrom(returnType)) {
|
||||||
|
throw new IllegalArgumentException(extensionName + "." + method.getName() + " has invalid return type. was: " + returnType.getName() + ", expected: " + expectedType.getName());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void validateMethodAnnotationPropertyLength(String property, String name, int maxLength, Method method) {
|
||||||
|
if (property.length() > maxLength) {
|
||||||
|
warnings.add(extensionName + "." + method.getName() + " '" + name + "' was over 50 characters.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void validateMethodArguments(Method method, boolean parameterIsRequired, Class... parameterOptions) {
|
||||||
|
Class<?>[] parameterTypes = method.getParameterTypes();
|
||||||
|
|
||||||
|
// Possible parameters for the methods:
|
||||||
|
// UUID playerUUID, String playerName, Group group, none
|
||||||
|
|
||||||
|
int parameters = parameterTypes.length;
|
||||||
|
|
||||||
|
if (parameterIsRequired && parameters == 0) {
|
||||||
|
// Does not have parameters, but one is required
|
||||||
|
throw new IllegalArgumentException(extensionName + "." + method.getName() + " requires one of " + Arrays.toString(parameterOptions) + " as a parameter.");
|
||||||
|
} else if (parameters == 0) {
|
||||||
|
// Has no parameters & it is acceptable.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parameters > 1) {
|
||||||
|
// Has too many parameters
|
||||||
|
throw new IllegalArgumentException(extensionName + "." + method.getName() + " has too many parameters, only one of " + Arrays.toString(parameterOptions) + " is required as a parameter.");
|
||||||
|
}
|
||||||
|
|
||||||
|
Class<?> methodParameter = parameterTypes[0];
|
||||||
|
|
||||||
|
boolean validParameter = false;
|
||||||
|
for (Class option : parameterOptions) {
|
||||||
|
if (option.equals(methodParameter)) {
|
||||||
|
validParameter = true;
|
||||||
}
|
}
|
||||||
tabNames.add(tabName);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TabInfo.Multiple tabInfoAnnotations = extClass.getAnnotation(TabInfo.Multiple.class);
|
if (!validParameter) {
|
||||||
if (tabInfoAnnotations == null) {
|
// Has invalid parameter
|
||||||
// No tab info, go with default order
|
throw new IllegalArgumentException(extensionName + "." + method.getName() + " has invalid parameter: '" + methodParameter.getName() + "' one of " + Arrays.toString(parameterOptions) + " is required as a parameter.");
|
||||||
return tabInformation;
|
}
|
||||||
|
// Has valid parameter & it is acceptable.
|
||||||
|
}
|
||||||
|
|
||||||
|
private void validateMethodAnnotations() {
|
||||||
|
validateBooleanProviderAnnotations();
|
||||||
|
validateNumberProviderAnnotations();
|
||||||
|
validateDoubleProviderAnnotations();
|
||||||
|
validatePercentageProviderAnnotations();
|
||||||
|
validateStringProviderAnnotations();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void validateBooleanProviderAnnotations() {
|
||||||
|
for (Map.Entry<Method, BooleanProvider> booleanProvider : methodAnnotations.getMethodAnnotations(BooleanProvider.class).entrySet()) {
|
||||||
|
Method method = booleanProvider.getKey();
|
||||||
|
BooleanProvider annotation = booleanProvider.getValue();
|
||||||
|
|
||||||
|
validateReturnType(method, boolean.class);
|
||||||
|
validateMethodAnnotationPropertyLength(annotation.text(), "text", 50, method);
|
||||||
|
validateMethodAnnotationPropertyLength(annotation.description(), "description", 150, method);
|
||||||
|
validateMethodAnnotationPropertyLength(annotation.conditionName(), "conditionName", 50, method);
|
||||||
|
validateMethodArguments(method, false, UUID.class, String.class, Group.class);
|
||||||
|
|
||||||
|
String condition = MethodAnnotations.get(method, Conditional.class).map(Conditional::value).orElse(null);
|
||||||
|
if (annotation.conditionName().equals(condition)) {
|
||||||
|
warnings.add(extensionName + "." + method.getName() + " can not be conditional of itself. required condition: " + condition + ", provided condition: " + annotation.conditionName());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void validateNumberProviderAnnotations() {
|
||||||
|
for (Map.Entry<Method, NumberProvider> numberProvider : methodAnnotations.getMethodAnnotations(NumberProvider.class).entrySet()) {
|
||||||
|
Method method = numberProvider.getKey();
|
||||||
|
NumberProvider annotation = numberProvider.getValue();
|
||||||
|
|
||||||
|
validateReturnType(method, long.class);
|
||||||
|
validateMethodAnnotationPropertyLength(annotation.text(), "text", 50, method);
|
||||||
|
validateMethodAnnotationPropertyLength(annotation.description(), "description", 150, method);
|
||||||
|
validateMethodArguments(method, false, UUID.class, String.class, Group.class);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void validateDoubleProviderAnnotations() {
|
||||||
|
for (Map.Entry<Method, DoubleProvider> numberProvider : methodAnnotations.getMethodAnnotations(DoubleProvider.class).entrySet()) {
|
||||||
|
Method method = numberProvider.getKey();
|
||||||
|
DoubleProvider annotation = numberProvider.getValue();
|
||||||
|
|
||||||
|
validateReturnType(method, double.class);
|
||||||
|
validateMethodAnnotationPropertyLength(annotation.text(), "text", 50, method);
|
||||||
|
validateMethodAnnotationPropertyLength(annotation.description(), "description", 150, method);
|
||||||
|
validateMethodArguments(method, false, UUID.class, String.class, Group.class);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void validatePercentageProviderAnnotations() {
|
||||||
|
for (Map.Entry<Method, PercentageProvider> numberProvider : methodAnnotations.getMethodAnnotations(PercentageProvider.class).entrySet()) {
|
||||||
|
Method method = numberProvider.getKey();
|
||||||
|
PercentageProvider annotation = numberProvider.getValue();
|
||||||
|
|
||||||
|
validateReturnType(method, double.class);
|
||||||
|
validateMethodAnnotationPropertyLength(annotation.text(), "text", 50, method);
|
||||||
|
validateMethodAnnotationPropertyLength(annotation.description(), "description", 150, method);
|
||||||
|
validateMethodArguments(method, false, UUID.class, String.class, Group.class);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void validateStringProviderAnnotations() {
|
||||||
|
for (Map.Entry<Method, StringProvider> numberProvider : methodAnnotations.getMethodAnnotations(StringProvider.class).entrySet()) {
|
||||||
|
Method method = numberProvider.getKey();
|
||||||
|
StringProvider annotation = numberProvider.getValue();
|
||||||
|
|
||||||
|
validateReturnType(method, String.class);
|
||||||
|
validateMethodAnnotationPropertyLength(annotation.text(), "text", 50, method);
|
||||||
|
validateMethodAnnotationPropertyLength(annotation.description(), "description", 150, method);
|
||||||
|
validateMethodArguments(method, false, UUID.class, String.class, Group.class);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void validateConditionals() {
|
||||||
|
Collection<Conditional> conditionals = methodAnnotations.getAnnotations(Conditional.class);
|
||||||
|
Collection<BooleanProvider> conditionProviders = methodAnnotations.getAnnotations(BooleanProvider.class);
|
||||||
|
|
||||||
|
Set<String> providedConditions = conditionProviders.stream().map(BooleanProvider::conditionName).collect(Collectors.toSet());
|
||||||
|
|
||||||
|
for (Conditional condition : conditionals) {
|
||||||
|
String conditionName = condition.value();
|
||||||
|
|
||||||
|
if (conditionName.length() > 50) {
|
||||||
|
warnings.add(extensionName + ": '" + conditionName + "' conditionName was over 50 characters.");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!providedConditions.contains(conditionName)) {
|
||||||
|
warnings.add(extensionName + ": '" + conditionName + "' Condition was not provided by any BooleanProvider.");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (TabInfo tabInfo : tabInfoAnnotations.value()) {
|
// Make sure that all methods annotated with Conditional have a Provider annotation
|
||||||
|
Collection<Method> conditionalMethods = methodAnnotations.getMethodAnnotations(Conditional.class).keySet();
|
||||||
|
for (Method conditionalMethod : conditionalMethods) {
|
||||||
|
if (!MethodAnnotations.hasAnyOf(conditionalMethod,
|
||||||
|
BooleanProvider.class, DoubleProvider.class, NumberProvider.class,
|
||||||
|
PercentageProvider.class, StringProvider.class
|
||||||
|
)) {
|
||||||
|
throw new IllegalArgumentException(extensionName + "." + conditionalMethod.getName() + " did not have any associated Provider for Conditional.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void extractPluginInfo() {
|
||||||
|
pluginInfo = getClassAnnotation(PluginInfo.class).orElseThrow(() -> new IllegalArgumentException("Given class had no PluginInfo annotation"));
|
||||||
|
|
||||||
|
if (pluginInfo.name().length() > 50) {
|
||||||
|
warnings.add(extensionName + " PluginInfo 'name' was over 50 characters.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void extractTabInfo() {
|
||||||
|
tabInformation = new ArrayList<>();
|
||||||
|
getClassAnnotation(TabInfo.Multiple.class).ifPresent(tabs -> {
|
||||||
|
for (TabInfo tabInfo : tabs.value()) {
|
||||||
|
String tabName = tabInfo.tab();
|
||||||
|
|
||||||
|
// Length restriction check
|
||||||
|
if (tabName.length() > 50) {
|
||||||
|
warnings.add(extensionName + " tabName '" + tabName + "' was over 50 characters.");
|
||||||
|
}
|
||||||
|
|
||||||
|
tabInformation.add(tabInfo);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
tabOrder = getClassAnnotation(TabOrder.class).orElse(null);
|
||||||
|
|
||||||
|
Map<Method, Tab> tabs = this.methodAnnotations.getMethodAnnotations(Tab.class);
|
||||||
|
Set<String> tabNames = tabs.values().stream().map(Tab::value).collect(Collectors.toSet());
|
||||||
|
|
||||||
|
// Check for unused TabInfo annotations
|
||||||
|
for (TabInfo tabInfo : tabInformation) {
|
||||||
String tabName = tabInfo.tab();
|
String tabName = tabInfo.tab();
|
||||||
|
|
||||||
// Length restriction check
|
|
||||||
if (tabName.length() > 50) {
|
if (tabName.length() > 50) {
|
||||||
errors.add(extClass.getName() + " tabName '" + tabName + "' was over 50 characters.");
|
warnings.add(extensionName + " TabInfo " + tabName + " name was over 50 characters.");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!tabNames.contains(tabName)) {
|
if (!tabNames.contains(tabName)) {
|
||||||
errors.add(extClass.getName() + " tab for '" + tabName + "' was not used.");
|
warnings.add(extensionName + " has TabInfo for " + tabName + ", but it is not used.");
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
tabInformation.add(tabInfo);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TabOrder tabOrder = extClass.getAnnotation(TabOrder.class);
|
// Check Tab name lengths
|
||||||
if (tabOrder != null) {
|
for (Map.Entry<Method, Tab> tab : tabs.entrySet()) {
|
||||||
for (String tabName : tabOrder.value()) {
|
String tabName = tab.getValue().value();
|
||||||
// Length restriction check
|
if (tabName.length() > 50) {
|
||||||
if (tabName.length() > 50) {
|
warnings.add(extensionName + "." + tab.getKey().getName() + " Tab '" + tabName + "' name was over 50 characters.");
|
||||||
errors.add(extClass.getName() + " tabName '" + tabName + "' found in TabOrder was over 50 characters.");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!tabNames.contains(tabName)) {
|
|
||||||
errors.add(extClass.getName() + " tab '" + tabName + "' found in TabOrder was not used.");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Set<String> tabOrderNames = Arrays.stream(tabOrder.value()).collect(Collectors.toSet());
|
|
||||||
for (String tabName : tabNames) {
|
|
||||||
if (!tabOrderNames.contains(tabName)) {
|
|
||||||
errors.add(extClass.getName() + " tab '" + tabName + "' was not in TabOrder.");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<String> getWarnings() {
|
||||||
|
return warnings;
|
||||||
|
}
|
||||||
|
|
||||||
|
public PluginInfo getPluginInfo() {
|
||||||
|
return pluginInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Optional<TabOrder> getTabOrder() {
|
||||||
|
return Optional.ofNullable(tabOrder);
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<TabInfo> getTabInformation() {
|
||||||
return tabInformation;
|
return tabInformation;
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<Conditional> extractConditionals(List<Method> booleanMethods) {
|
public MethodAnnotations getMethodAnnotations() {
|
||||||
Set<String> conditionNames = booleanMethods.stream()
|
return methodAnnotations;
|
||||||
.map(method -> method.getAnnotation(BooleanProvider.class).conditionName())
|
|
||||||
.collect(Collectors.toSet());
|
|
||||||
|
|
||||||
List<Conditional> conditionals = new ArrayList<>();
|
|
||||||
|
|
||||||
Class<? extends DataExtension> extClass = extension.getClass();
|
|
||||||
for (Method method : extClass.getMethods()) {
|
|
||||||
Conditional conditional = method.getAnnotation(Conditional.class);
|
|
||||||
if (conditional == null) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
conditionals.add(conditional);
|
|
||||||
}
|
|
||||||
|
|
||||||
for (Conditional conditional : conditionals) {
|
|
||||||
String conditionName = conditional.value();
|
|
||||||
if (conditionName.length() > 50) {
|
|
||||||
errors.add(extClass.getName() + " '" + conditionName + "' conditionName was over 50 characters.");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!conditionNames.contains(conditionName)) {
|
|
||||||
errors.add(extClass.getName() + " '" + conditionName + "' Condition was not provided by any BooleanProvider.");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return conditionals;
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<Method> extractBooleanMethods() {
|
|
||||||
List<Method> booleanProviderMethods = new ArrayList<>();
|
|
||||||
|
|
||||||
Class<? extends DataExtension> extClass = extension.getClass();
|
|
||||||
for (Method method : extClass.getMethods()) {
|
|
||||||
BooleanProvider provider = method.getAnnotation(BooleanProvider.class);
|
|
||||||
if (provider == null) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return type check
|
|
||||||
Class<?> returnType = method.getReturnType();
|
|
||||||
if (!boolean.class.isAssignableFrom(returnType)) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " has invalid return type. was: " + returnType.getName() + ", expected: " + boolean.class.getName());
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Cyclic conditional check
|
|
||||||
Conditional conditional = method.getAnnotation(Conditional.class);
|
|
||||||
if (conditional != null) {
|
|
||||||
String conditionName = provider.conditionName();
|
|
||||||
String requiredConditionName = conditional.value();
|
|
||||||
|
|
||||||
if (!conditionName.isEmpty() && conditionName.equals(requiredConditionName)) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " can not be conditional of itself. required condition: " + requiredConditionName + ", provided condition: " + conditionName);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Length restriction checks
|
|
||||||
if (provider.text().length() > 50) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " text was over 50 characters.");
|
|
||||||
}
|
|
||||||
if (provider.description().length() > 150) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " description was over 150 characters.");
|
|
||||||
}
|
|
||||||
if (provider.conditionName().length() > 50) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " conditionName was over 50 characters.");
|
|
||||||
}
|
|
||||||
|
|
||||||
booleanProviderMethods.add(method);
|
|
||||||
}
|
|
||||||
|
|
||||||
return booleanProviderMethods;
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<Method> extractNumberMethods() {
|
|
||||||
List<Method> numberProviderMethods = new ArrayList<>();
|
|
||||||
|
|
||||||
Class<? extends DataExtension> extClass = extension.getClass();
|
|
||||||
for (Method method : extClass.getMethods()) {
|
|
||||||
NumberProvider provider = method.getAnnotation(NumberProvider.class);
|
|
||||||
if (provider == null) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return type check
|
|
||||||
Class<?> returnType = method.getReturnType();
|
|
||||||
if (!long.class.isAssignableFrom(returnType)) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " has invalid return type. was: " + returnType.getName() + ", expected: " + long.class.getName());
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Length restriction checks
|
|
||||||
if (provider.text().length() > 50) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " text was over 50 characters.");
|
|
||||||
}
|
|
||||||
if (provider.description().length() > 150) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " description was over 150 characters.");
|
|
||||||
}
|
|
||||||
|
|
||||||
numberProviderMethods.add(method);
|
|
||||||
}
|
|
||||||
|
|
||||||
return numberProviderMethods;
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<Method> extractDoubleMethods() {
|
|
||||||
List<Method> doubleProviderMethods = new ArrayList<>();
|
|
||||||
|
|
||||||
Class<? extends DataExtension> extClass = extension.getClass();
|
|
||||||
for (Method method : extClass.getMethods()) {
|
|
||||||
DoubleProvider provider = method.getAnnotation(DoubleProvider.class);
|
|
||||||
if (provider == null) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return type check
|
|
||||||
Class<?> returnType = method.getReturnType();
|
|
||||||
if (!double.class.isAssignableFrom(returnType)) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " has invalid return type. was: " + returnType.getName() + ", expected: " + double.class.getName());
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Length restriction checks
|
|
||||||
if (provider.text().length() > 50) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " text was over 50 characters.");
|
|
||||||
}
|
|
||||||
if (provider.description().length() > 150) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " description was over 150 characters.");
|
|
||||||
}
|
|
||||||
|
|
||||||
doubleProviderMethods.add(method);
|
|
||||||
}
|
|
||||||
|
|
||||||
return doubleProviderMethods;
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<Method> extractPercentageMethods() {
|
|
||||||
List<Method> percentageProviderMethods = new ArrayList<>();
|
|
||||||
|
|
||||||
Class<? extends DataExtension> extClass = extension.getClass();
|
|
||||||
for (Method method : extClass.getMethods()) {
|
|
||||||
PercentageProvider provider = method.getAnnotation(PercentageProvider.class);
|
|
||||||
if (provider == null) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return type check
|
|
||||||
Class<?> returnType = method.getReturnType();
|
|
||||||
if (!double.class.isAssignableFrom(returnType)) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " has invalid return type. was: " + returnType.getName() + ", expected: " + double.class.getName());
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Length restriction checks
|
|
||||||
if (provider.text().length() > 50) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " text was over 50 characters.");
|
|
||||||
}
|
|
||||||
if (provider.description().length() > 150) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " description was over 150 characters.");
|
|
||||||
}
|
|
||||||
|
|
||||||
percentageProviderMethods.add(method);
|
|
||||||
}
|
|
||||||
|
|
||||||
return percentageProviderMethods;
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<Method> extractStringMethods() {
|
|
||||||
List<Method> stringProviderMethods = new ArrayList<>();
|
|
||||||
|
|
||||||
Class<? extends DataExtension> extClass = extension.getClass();
|
|
||||||
for (Method method : extClass.getMethods()) {
|
|
||||||
StringProvider provider = method.getAnnotation(StringProvider.class);
|
|
||||||
if (provider == null) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return type check
|
|
||||||
Class<?> returnType = method.getReturnType();
|
|
||||||
if (!double.class.isAssignableFrom(returnType)) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " has invalid return type. was: " + returnType.getName() + ", expected: " + double.class.getName());
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Length restriction checks
|
|
||||||
if (provider.text().length() > 50) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " text was over 50 characters.");
|
|
||||||
}
|
|
||||||
if (provider.description().length() > 150) {
|
|
||||||
errors.add(extClass.getName() + "." + method.getName() + " description was over 150 characters.");
|
|
||||||
}
|
|
||||||
|
|
||||||
stringProviderMethods.add(method);
|
|
||||||
}
|
|
||||||
|
|
||||||
return stringProviderMethods;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,76 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of Player Analytics (Plan).
|
||||||
|
*
|
||||||
|
* Plan is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Lesser General Public License v3 as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* Plan 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 Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public License
|
||||||
|
* along with Plan. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
package com.djrapitops.plan.extension.extractor;
|
||||||
|
|
||||||
|
import java.lang.annotation.Annotation;
|
||||||
|
import java.lang.reflect.Method;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Implementation detail, utility class for handling method annotations.
|
||||||
|
*
|
||||||
|
* @author Rsl1122
|
||||||
|
*/
|
||||||
|
public class MethodAnnotations {
|
||||||
|
|
||||||
|
private final Map<Class, Map<Method, Annotation>> methodAnnotations;
|
||||||
|
|
||||||
|
public MethodAnnotations() {
|
||||||
|
methodAnnotations = new HashMap<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean hasAnyOf(Method method, Class... annotationClasses) {
|
||||||
|
for (Annotation annotation : method.getAnnotations()) {
|
||||||
|
for (Class annotationClass : annotationClasses) {
|
||||||
|
if (annotationClass.isAssignableFrom(annotation.getClass())) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static <T extends Annotation> Optional<T> get(Method from, Class<T> ofClass) {
|
||||||
|
return Optional.ofNullable(from.getAnnotation(ofClass));
|
||||||
|
}
|
||||||
|
|
||||||
|
public <T extends Annotation> void put(Method method, Class<T> annotationClass, T annotation) {
|
||||||
|
Map<Method, Annotation> methods = methodAnnotations.getOrDefault(annotationClass, new HashMap<>());
|
||||||
|
methods.put(method, annotation);
|
||||||
|
methodAnnotations.put(annotationClass, methods);
|
||||||
|
}
|
||||||
|
|
||||||
|
public <T extends Annotation> Map<Method, T> getMethodAnnotations(Class<T> ofType) {
|
||||||
|
return (Map<Method, T>) methodAnnotations.getOrDefault(ofType, new HashMap<>());
|
||||||
|
}
|
||||||
|
|
||||||
|
public <T extends Annotation> Collection<T> getAnnotations(Class<T> ofType) {
|
||||||
|
return getMethodAnnotations(ofType).values();
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isEmpty() {
|
||||||
|
return methodAnnotations.isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "MethodAnnotations{" + methodAnnotations + '}';
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,243 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of Player Analytics (Plan).
|
||||||
|
*
|
||||||
|
* Plan is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Lesser General Public License v3 as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* Plan 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 Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public License
|
||||||
|
* along with Plan. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
package com.djrapitops.plan.extension.extractor;
|
||||||
|
|
||||||
|
import com.djrapitops.plan.extension.DataExtension;
|
||||||
|
import com.djrapitops.plan.extension.annotation.*;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import org.junit.platform.runner.JUnitPlatform;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for different validations of ExtensionExtractor.
|
||||||
|
* <p>
|
||||||
|
* This Test class contains only INVALID implementations of the DataExtension API.
|
||||||
|
*
|
||||||
|
* @author Rsl1122
|
||||||
|
*/
|
||||||
|
@RunWith(JUnitPlatform.class)
|
||||||
|
class ExtensionExtractorTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void pluginInfoIsRequired() {
|
||||||
|
class Extension implements DataExtension {}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Given class had no PluginInfo annotation", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void providerMethodsAreRequired() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Extension class had no methods annotated with a Provider annotation", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void publicProviderMethodsAreRequired() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@BooleanProvider(text = "Banned")
|
||||||
|
private boolean method(UUID playerUUID) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Extension class had no methods annotated with a Provider annotation", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void pluginInfoNameOver50Chars() {
|
||||||
|
@PluginInfo(name = "five five five five five five five five five five -")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@BooleanProvider(text = "Required Provider")
|
||||||
|
public boolean method() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Warnings: [Extension PluginInfo 'name' was over 50 characters.]", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void booleanProviderMustReturnBoolean() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@BooleanProvider(text = "Banned")
|
||||||
|
public String method(UUID playerUUID) {
|
||||||
|
return "false";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Extension.method has invalid return type. was: java.lang.String, expected: boolean", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void booleanProviderMustReturnPrimitiveBoolean() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@BooleanProvider(text = "Banned")
|
||||||
|
public Boolean method(UUID playerUUID) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Extension.method has invalid return type. was: java.lang.Boolean, expected: boolean", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void numberProviderMustReturnPrimitiveLong() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@NumberProvider(text = "Achievements")
|
||||||
|
public Long method(UUID playerUUID) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Extension.method has invalid return type. was: java.lang.Long, expected: long", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void doubleProviderMustReturnPrimitiveDouble() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@DoubleProvider(text = "Money")
|
||||||
|
public Double method(UUID playerUUID) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Extension.method has invalid return type. was: java.lang.Double, expected: double", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void percentageProviderMustReturnPrimitiveDouble() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@PercentageProvider(text = "Achievements awarded")
|
||||||
|
public Double method(UUID playerUUID) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Extension.method has invalid return type. was: java.lang.Double, expected: double", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void stringProviderMustReturnString() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@StringProvider(text = "Town")
|
||||||
|
public Double method(UUID playerUUID) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Extension.method has invalid return type. was: java.lang.Double, expected: java.lang.String", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void booleanProviderCanNotSupplyItsOwnConditional() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@Conditional("hasJoined")
|
||||||
|
@BooleanProvider(text = "Banned", conditionName = "hasJoined")
|
||||||
|
public boolean method(UUID playerUUID) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Warnings: [Extension.method can not be conditional of itself. required condition: hasJoined, provided condition: hasJoined]", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void conditionalMethodRequiresProvider() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@Conditional("hasJoined")
|
||||||
|
public boolean method(UUID playerUUID) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Extension.method did not have any associated Provider for Conditional.", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void conditionalNeedsToBeProvided() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@Conditional("hasJoined")
|
||||||
|
@BooleanProvider(text = "Banned", conditionName = "isBanned")
|
||||||
|
public boolean method(UUID playerUUID) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Warnings: [Extension: 'hasJoined' Condition was not provided by any BooleanProvider.]", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void methodNeedsValidParameters() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@Conditional("hasJoined")
|
||||||
|
@BooleanProvider(text = "Banned", conditionName = "isBanned")
|
||||||
|
public boolean method(Integer invalid) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Extension.method has invalid parameter: 'java.lang.Integer' one of [class java.util.UUID, class java.lang.String, interface com.djrapitops.plan.extension.Group] is required as a parameter.", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void methodHasTooManyParameters() {
|
||||||
|
@PluginInfo(name = "Extension")
|
||||||
|
class Extension implements DataExtension {
|
||||||
|
@Conditional("hasJoined")
|
||||||
|
@BooleanProvider(text = "Banned", conditionName = "isBanned")
|
||||||
|
public boolean method(String playerName, UUID playerUUID) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ExtensionExtractor underTest = new ExtensionExtractor(new Extension());
|
||||||
|
assertEquals("Extension.method has too many parameters, only one of [class java.util.UUID, class java.lang.String, interface com.djrapitops.plan.extension.Group] is required as a parameter.", assertThrows(IllegalArgumentException.class, underTest::validateAnnotations).getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user