Add CLI modules for standalone apps used in Pandacube (web api, launcher)
This commit is contained in:
parent
3dd2af7814
commit
3bf4184fd2
27
CLI/.classpath
Normal file
27
CLI/.classpath
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<classpath>
|
||||||
|
<classpathentry kind="src" output="target/classes" path="src/main/java">
|
||||||
|
<attributes>
|
||||||
|
<attribute name="optional" value="true"/>
|
||||||
|
<attribute name="maven.pomderived" value="true"/>
|
||||||
|
</attributes>
|
||||||
|
</classpathentry>
|
||||||
|
<classpathentry kind="src" output="target/test-classes" path="src/test/java">
|
||||||
|
<attributes>
|
||||||
|
<attribute name="optional" value="true"/>
|
||||||
|
<attribute name="maven.pomderived" value="true"/>
|
||||||
|
<attribute name="test" value="true"/>
|
||||||
|
</attributes>
|
||||||
|
</classpathentry>
|
||||||
|
<classpathentry kind="con" path="org.eclipse.jdt.launching.JRE_CONTAINER/org.eclipse.jdt.internal.debug.ui.launcher.StandardVMType/JavaSE-16">
|
||||||
|
<attributes>
|
||||||
|
<attribute name="maven.pomderived" value="true"/>
|
||||||
|
</attributes>
|
||||||
|
</classpathentry>
|
||||||
|
<classpathentry kind="con" path="org.eclipse.m2e.MAVEN2_CLASSPATH_CONTAINER">
|
||||||
|
<attributes>
|
||||||
|
<attribute name="maven.pomderived" value="true"/>
|
||||||
|
</attributes>
|
||||||
|
</classpathentry>
|
||||||
|
<classpathentry kind="output" path="target/classes"/>
|
||||||
|
</classpath>
|
1
CLI/.gitignore
vendored
Normal file
1
CLI/.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
/target/
|
23
CLI/.project
Normal file
23
CLI/.project
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<projectDescription>
|
||||||
|
<name>pandalib-cli</name>
|
||||||
|
<comment></comment>
|
||||||
|
<projects>
|
||||||
|
</projects>
|
||||||
|
<buildSpec>
|
||||||
|
<buildCommand>
|
||||||
|
<name>org.eclipse.jdt.core.javabuilder</name>
|
||||||
|
<arguments>
|
||||||
|
</arguments>
|
||||||
|
</buildCommand>
|
||||||
|
<buildCommand>
|
||||||
|
<name>org.eclipse.m2e.core.maven2Builder</name>
|
||||||
|
<arguments>
|
||||||
|
</arguments>
|
||||||
|
</buildCommand>
|
||||||
|
</buildSpec>
|
||||||
|
<natures>
|
||||||
|
<nature>org.eclipse.jdt.core.javanature</nature>
|
||||||
|
<nature>org.eclipse.m2e.core.maven2Nature</nature>
|
||||||
|
</natures>
|
||||||
|
</projectDescription>
|
3
CLI/.settings/org.eclipse.core.resources.prefs
Normal file
3
CLI/.settings/org.eclipse.core.resources.prefs
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
eclipse.preferences.version=1
|
||||||
|
encoding//src/main/java=UTF-8
|
||||||
|
encoding/<project>=UTF-8
|
8
CLI/.settings/org.eclipse.jdt.core.prefs
Normal file
8
CLI/.settings/org.eclipse.jdt.core.prefs
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
eclipse.preferences.version=1
|
||||||
|
org.eclipse.jdt.core.compiler.codegen.targetPlatform=16
|
||||||
|
org.eclipse.jdt.core.compiler.compliance=16
|
||||||
|
org.eclipse.jdt.core.compiler.problem.enablePreviewFeatures=disabled
|
||||||
|
org.eclipse.jdt.core.compiler.problem.forbiddenReference=warning
|
||||||
|
org.eclipse.jdt.core.compiler.problem.reportPreviewFeatures=ignore
|
||||||
|
org.eclipse.jdt.core.compiler.release=disabled
|
||||||
|
org.eclipse.jdt.core.compiler.source=16
|
4
CLI/.settings/org.eclipse.m2e.core.prefs
Normal file
4
CLI/.settings/org.eclipse.m2e.core.prefs
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
activeProfiles=
|
||||||
|
eclipse.preferences.version=1
|
||||||
|
resolveWorkspaceProjects=true
|
||||||
|
version=1
|
51
CLI/pom.xml
Normal file
51
CLI/pom.xml
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||||
|
<modelVersion>4.0.0</modelVersion>
|
||||||
|
|
||||||
|
<parent>
|
||||||
|
<groupId>fr.pandacube.lib</groupId>
|
||||||
|
<artifactId>pandalib-parent</artifactId>
|
||||||
|
<version>1.0-SNAPSHOT</version>
|
||||||
|
</parent>
|
||||||
|
|
||||||
|
<artifactId>pandalib-cli</artifactId>
|
||||||
|
|
||||||
|
<name>PandaLib-CLI</name>
|
||||||
|
|
||||||
|
<repositories>
|
||||||
|
<repository>
|
||||||
|
<id>minecraft-libraries</id>
|
||||||
|
<name>Minecraft Libraries</name>
|
||||||
|
<url>https://libraries.minecraft.net</url>
|
||||||
|
</repository>
|
||||||
|
</repositories>
|
||||||
|
|
||||||
|
<dependencies>
|
||||||
|
<dependency>
|
||||||
|
<groupId>fr.pandacube.lib</groupId>
|
||||||
|
<artifactId>pandalib-core</artifactId>
|
||||||
|
<version>${project.version}</version>
|
||||||
|
<scope>compile</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>fr.pandacube.bungeecord</groupId>
|
||||||
|
<artifactId>bungeecord-log</artifactId>
|
||||||
|
<version>${bungeecord.version}</version>
|
||||||
|
<scope>compile</scope>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>fr.pandacube.bungeecord</groupId>
|
||||||
|
<artifactId>bungeecord-config</artifactId>
|
||||||
|
<version>${bungeecord.version}</version>
|
||||||
|
<scope>compile</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>com.mojang</groupId>
|
||||||
|
<artifactId>brigadier</artifactId>
|
||||||
|
<version>1.0.17</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
|
</dependencies>
|
||||||
|
|
||||||
|
</project>
|
144
CLI/src/main/java/fr/pandacube/lib/cli/BrigadierCommand.java
Normal file
144
CLI/src/main/java/fr/pandacube/lib/cli/BrigadierCommand.java
Normal file
@ -0,0 +1,144 @@
|
|||||||
|
package fr.pandacube.lib.cli;
|
||||||
|
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.CompletableFuture;
|
||||||
|
|
||||||
|
import com.mojang.brigadier.arguments.ArgumentType;
|
||||||
|
import com.mojang.brigadier.builder.LiteralArgumentBuilder;
|
||||||
|
import com.mojang.brigadier.builder.RequiredArgumentBuilder;
|
||||||
|
import com.mojang.brigadier.context.CommandContext;
|
||||||
|
import com.mojang.brigadier.context.ParsedCommandNode;
|
||||||
|
import com.mojang.brigadier.suggestion.Suggestion;
|
||||||
|
import com.mojang.brigadier.suggestion.SuggestionProvider;
|
||||||
|
import com.mojang.brigadier.suggestion.Suggestions;
|
||||||
|
import com.mojang.brigadier.suggestion.SuggestionsBuilder;
|
||||||
|
import com.mojang.brigadier.tree.LiteralCommandNode;
|
||||||
|
|
||||||
|
import fr.pandacube.lib.core.chat.ChatStatic;
|
||||||
|
import fr.pandacube.lib.core.commands.SuggestionsSupplier;
|
||||||
|
import fr.pandacube.lib.core.util.Log;
|
||||||
|
import fr.pandacube.lib.core.util.ReflexionUtil;
|
||||||
|
|
||||||
|
public abstract class BrigadierCommand extends ChatStatic {
|
||||||
|
|
||||||
|
private LiteralCommandNode<Object> commandNode;
|
||||||
|
|
||||||
|
public BrigadierCommand() {
|
||||||
|
LiteralArgumentBuilder<Object> builder = buildCommand();
|
||||||
|
String[] aliases = getAliases();
|
||||||
|
if (aliases == null)
|
||||||
|
aliases = new String[0];
|
||||||
|
|
||||||
|
commandNode = BrigadierDispatcher.instance.register(builder);
|
||||||
|
|
||||||
|
|
||||||
|
for (String alias : aliases) {
|
||||||
|
BrigadierDispatcher.instance.register(literal(alias)
|
||||||
|
.requires(commandNode.getRequirement())
|
||||||
|
.executes(commandNode.getCommand())
|
||||||
|
.redirect(commandNode)
|
||||||
|
);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
protected abstract LiteralArgumentBuilder<Object> buildCommand();
|
||||||
|
|
||||||
|
protected String[] getAliases() {
|
||||||
|
return new String[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public static LiteralArgumentBuilder<Object> literal(String name) {
|
||||||
|
return LiteralArgumentBuilder.literal(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static <T> RequiredArgumentBuilder<Object, T> argument(String name, ArgumentType<T> type) {
|
||||||
|
return RequiredArgumentBuilder.argument(name, type);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static boolean isLiteralParsed(CommandContext<Object> context, String literal) {
|
||||||
|
for (ParsedCommandNode<Object> node : context.getNodes()) {
|
||||||
|
if (!(node.getNode() instanceof LiteralCommandNode))
|
||||||
|
continue;
|
||||||
|
if (((LiteralCommandNode<Object>)node.getNode()).getLiteral().equals(literal))
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static <T> T tryGetArgument(CommandContext<Object> context, String argument, Class<T> type) {
|
||||||
|
return tryGetArgument(context, argument, type, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static <T> T tryGetArgument(CommandContext<Object> context, String argument, Class<T> type, T deflt) {
|
||||||
|
try {
|
||||||
|
return context.getArgument(argument, type);
|
||||||
|
} catch (IllegalArgumentException e) {
|
||||||
|
return deflt;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
protected static SuggestionProvider<Object> wrapSuggestions(SuggestionsSupplier<Object> suggestions) {
|
||||||
|
return (context, builder) -> {
|
||||||
|
Object sender = context.getSource();
|
||||||
|
String message = builder.getInput();
|
||||||
|
try {
|
||||||
|
int tokenStartPos = builder.getStart();
|
||||||
|
|
||||||
|
List<String> results = Collections.emptyList();
|
||||||
|
|
||||||
|
int firstSpacePos = message.indexOf(" ");
|
||||||
|
String[] args = (firstSpacePos + 1 > tokenStartPos - 1) ? new String[0]
|
||||||
|
: message.substring(firstSpacePos + 1, tokenStartPos - 1).split(" ", -1);
|
||||||
|
args = Arrays.copyOf(args, args.length + 1);
|
||||||
|
args[args.length - 1] = message.substring(tokenStartPos);
|
||||||
|
|
||||||
|
results = suggestions.getSuggestions(sender, args.length - 1, args[args.length - 1], args);
|
||||||
|
|
||||||
|
for (String s : results) {
|
||||||
|
if (s != null)
|
||||||
|
builder.suggest(s);
|
||||||
|
}
|
||||||
|
} catch (Throwable e) {
|
||||||
|
Log.severe("Error while tab-completing '" + message/* + "' for " + sender.getName()*/, e);
|
||||||
|
}
|
||||||
|
return completableFutureSuggestionsKeepsOriginalOrdering(builder);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public static CompletableFuture<Suggestions> completableFutureSuggestionsKeepsOriginalOrdering(SuggestionsBuilder builder) {
|
||||||
|
return CompletableFuture.completedFuture(
|
||||||
|
BrigadierDispatcher.createSuggestionsOriginalOrdering(builder.getInput(), getSuggestionsFromSuggestionsBuilder(builder))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
private static List<Suggestion> getSuggestionsFromSuggestionsBuilder(SuggestionsBuilder builder) {
|
||||||
|
try {
|
||||||
|
return (List<Suggestion>) ReflexionUtil.getDeclaredFieldValue(SuggestionsBuilder.class, builder, "result");
|
||||||
|
} catch (ReflectiveOperationException e) {
|
||||||
|
throw new RuntimeException(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
156
CLI/src/main/java/fr/pandacube/lib/cli/BrigadierDispatcher.java
Normal file
156
CLI/src/main/java/fr/pandacube/lib/cli/BrigadierDispatcher.java
Normal file
@ -0,0 +1,156 @@
|
|||||||
|
package fr.pandacube.lib.cli;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.CompletableFuture;
|
||||||
|
|
||||||
|
import com.mojang.brigadier.CommandDispatcher;
|
||||||
|
import com.mojang.brigadier.ParseResults;
|
||||||
|
import com.mojang.brigadier.builder.LiteralArgumentBuilder;
|
||||||
|
import com.mojang.brigadier.context.CommandContextBuilder;
|
||||||
|
import com.mojang.brigadier.context.StringRange;
|
||||||
|
import com.mojang.brigadier.context.SuggestionContext;
|
||||||
|
import com.mojang.brigadier.exceptions.CommandSyntaxException;
|
||||||
|
import com.mojang.brigadier.suggestion.Suggestion;
|
||||||
|
import com.mojang.brigadier.suggestion.Suggestions;
|
||||||
|
import com.mojang.brigadier.suggestion.SuggestionsBuilder;
|
||||||
|
import com.mojang.brigadier.tree.CommandNode;
|
||||||
|
import com.mojang.brigadier.tree.LiteralCommandNode;
|
||||||
|
|
||||||
|
import fr.pandacube.lib.core.util.Log;
|
||||||
|
import jline.console.completer.Completer;
|
||||||
|
|
||||||
|
public class BrigadierDispatcher implements Completer {
|
||||||
|
|
||||||
|
public static final BrigadierDispatcher instance = new BrigadierDispatcher();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
private CommandDispatcher<Object> dispatcher;
|
||||||
|
|
||||||
|
private Object sender = new Object();
|
||||||
|
|
||||||
|
public BrigadierDispatcher() {
|
||||||
|
dispatcher = new CommandDispatcher<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/* package */ LiteralCommandNode<Object> register(LiteralArgumentBuilder<Object> node) {
|
||||||
|
return dispatcher.register(node);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public int execute(String commandWithoutSlash) {
|
||||||
|
ParseResults<Object> parsed = dispatcher.parse(commandWithoutSlash, sender);
|
||||||
|
|
||||||
|
try {
|
||||||
|
return dispatcher.execute(parsed);
|
||||||
|
} catch (CommandSyntaxException e) {
|
||||||
|
Log.severe("Erreur d’utilisation de la commande : " + e.getMessage());
|
||||||
|
return 0;
|
||||||
|
} catch (Throwable e) {
|
||||||
|
Log.severe("Erreur lors de l’exécution de la commande : ", e);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int complete(String buffer, int cursor, List<CharSequence> candidates) {
|
||||||
|
|
||||||
|
String bufferBeforeCursor = buffer.substring(0, cursor);
|
||||||
|
|
||||||
|
Suggestions completeResult = getSuggestions(bufferBeforeCursor);
|
||||||
|
|
||||||
|
completeResult.getList().stream().map(s -> s.getText()).forEach(candidates::add);
|
||||||
|
|
||||||
|
return completeResult.getRange().getStart();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* package */ Suggestions getSuggestions(String buffer) {
|
||||||
|
ParseResults<Object> parsed = dispatcher.parse(buffer, sender);
|
||||||
|
try {
|
||||||
|
CompletableFuture<Suggestions> futureSuggestions = buildSuggestionBrigadier(parsed);
|
||||||
|
return futureSuggestions.join();
|
||||||
|
} catch (Throwable e) {
|
||||||
|
Log.severe("Erreur d’exécution des suggestions :\n" + e.getMessage(), e);
|
||||||
|
return Suggestions.empty().join();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
CompletableFuture<Suggestions> buildSuggestionBrigadier(ParseResults<Object> parsed) {
|
||||||
|
int cursor = parsed.getReader().getTotalLength();
|
||||||
|
final CommandContextBuilder<Object> context = parsed.getContext();
|
||||||
|
|
||||||
|
final SuggestionContext<Object> nodeBeforeCursor = context.findSuggestionContext(cursor);
|
||||||
|
final CommandNode<Object> parent = nodeBeforeCursor.parent;
|
||||||
|
final int start = Math.min(nodeBeforeCursor.startPos, cursor);
|
||||||
|
|
||||||
|
final String fullInput = parsed.getReader().getString();
|
||||||
|
final String truncatedInput = fullInput.substring(0, cursor);
|
||||||
|
@SuppressWarnings("unchecked") final CompletableFuture<Suggestions>[] futures = new CompletableFuture[parent.getChildren().size()];
|
||||||
|
int i = 0;
|
||||||
|
for (final CommandNode<Object> node : parent.getChildren()) {
|
||||||
|
CompletableFuture<Suggestions> future = Suggestions.empty();
|
||||||
|
try {
|
||||||
|
future = node.listSuggestions(context.build(truncatedInput), new SuggestionsBuilder(truncatedInput, start));
|
||||||
|
} catch (final CommandSyntaxException ignored) {
|
||||||
|
}
|
||||||
|
futures[i++] = future;
|
||||||
|
}
|
||||||
|
|
||||||
|
final CompletableFuture<Suggestions> result = new CompletableFuture<>();
|
||||||
|
CompletableFuture.allOf(futures).thenRun(() -> {
|
||||||
|
final List<Suggestions> suggestions = new ArrayList<>();
|
||||||
|
for (final CompletableFuture<Suggestions> future : futures) {
|
||||||
|
suggestions.add(future.join());
|
||||||
|
}
|
||||||
|
result.complete(mergeSuggestionsOriginalOrdering(fullInput, suggestions));
|
||||||
|
});
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
// inspired from com.mojang.brigadier.suggestion.Suggestions#merge, but without the sorting part
|
||||||
|
public static Suggestions mergeSuggestionsOriginalOrdering(String input, Collection<Suggestions> suggestions) {
|
||||||
|
if (suggestions.isEmpty()) {
|
||||||
|
return new Suggestions(StringRange.at(0), new ArrayList<>(0));
|
||||||
|
} else if (suggestions.size() == 1) {
|
||||||
|
return suggestions.iterator().next();
|
||||||
|
}
|
||||||
|
|
||||||
|
final List<Suggestion> texts = new ArrayList<>();
|
||||||
|
for (final Suggestions suggestions1 : suggestions) {
|
||||||
|
texts.addAll(suggestions1.getList());
|
||||||
|
}
|
||||||
|
return createSuggestionsOriginalOrdering(input, texts);
|
||||||
|
}
|
||||||
|
|
||||||
|
// inspired from com.mojang.brigadier.suggestion.Suggestions#create, but without the sorting part
|
||||||
|
public static Suggestions createSuggestionsOriginalOrdering(String command, Collection<Suggestion> suggestions) {
|
||||||
|
if (suggestions.isEmpty()) {
|
||||||
|
return new Suggestions(StringRange.at(0), new ArrayList<>(0));
|
||||||
|
}
|
||||||
|
int start = Integer.MAX_VALUE;
|
||||||
|
int end = Integer.MIN_VALUE;
|
||||||
|
for (final Suggestion suggestion : suggestions) {
|
||||||
|
start = Math.min(suggestion.getRange().getStart(), start);
|
||||||
|
end = Math.max(suggestion.getRange().getEnd(), end);
|
||||||
|
}
|
||||||
|
final StringRange range = new StringRange(start, end);
|
||||||
|
final List<Suggestion> texts = new ArrayList<>(suggestions.size());
|
||||||
|
for (final Suggestion suggestion : suggestions) {
|
||||||
|
texts.add(suggestion.expand(command, range));
|
||||||
|
}
|
||||||
|
return new Suggestions(range, texts);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
143
CLI/src/main/java/fr/pandacube/lib/cli/ConsoleInterface.java
Normal file
143
CLI/src/main/java/fr/pandacube/lib/cli/ConsoleInterface.java
Normal file
@ -0,0 +1,143 @@
|
|||||||
|
package fr.pandacube.lib.cli;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.PrintWriter;
|
||||||
|
import java.util.logging.ErrorManager;
|
||||||
|
import java.util.logging.Handler;
|
||||||
|
import java.util.logging.LogRecord;
|
||||||
|
|
||||||
|
import fr.pandacube.lib.core.util.Log;
|
||||||
|
import jline.console.ConsoleReader;
|
||||||
|
import net.md_5.bungee.log.ConciseFormatter;
|
||||||
|
|
||||||
|
public class ConsoleInterface extends Handler {
|
||||||
|
|
||||||
|
|
||||||
|
public static final String ANSI_RESET = "\u001B[0m";
|
||||||
|
|
||||||
|
public static final String ANSI_BLACK = "\u001B[30m";
|
||||||
|
public static final String ANSI_DARK_RED = "\u001B[31m";
|
||||||
|
public static final String ANSI_DARK_GREEN = "\u001B[32m";
|
||||||
|
public static final String ANSI_GOLD = "\u001B[33m";
|
||||||
|
public static final String ANSI_DARK_BLUE = "\u001B[34m";
|
||||||
|
public static final String ANSI_DARK_PURPLE = "\u001B[35m";
|
||||||
|
public static final String ANSI_DARK_AQUA = "\u001B[36m";
|
||||||
|
public static final String ANSI_GRAY = "\u001B[37m";
|
||||||
|
|
||||||
|
public static final String ANSI_DARK_GRAY = "\u001B[30;1m";
|
||||||
|
public static final String ANSI_RED = "\u001B[31;1m";
|
||||||
|
public static final String ANSI_GREEN = "\u001B[32;1m";
|
||||||
|
public static final String ANSI_YELLOW = "\u001B[33;1m";
|
||||||
|
public static final String ANSI_BLUE = "\u001B[34;1m";
|
||||||
|
public static final String ANSI_LIGHT_PURPLE = "\u001B[35;1m";
|
||||||
|
public static final String ANSI_AQUA = "\u001B[36;1m";
|
||||||
|
public static final String ANSI_WHITE = "\u001B[37;1m";
|
||||||
|
|
||||||
|
public static final String ANSI_BOLD = "\u001B[1m";
|
||||||
|
|
||||||
|
public static final String ANSI_CLEAR_SCREEN = "\u001B[2J\u001B[1;1H";
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
private ConsoleReader reader;
|
||||||
|
private PrintWriter out;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public ConsoleInterface() throws IOException {
|
||||||
|
reader = new ConsoleReader();
|
||||||
|
reader.setBellEnabled(false);
|
||||||
|
reader.setPrompt("\r"+ANSI_LIGHT_PURPLE+">");
|
||||||
|
out = new PrintWriter(reader.getOutput());
|
||||||
|
reader.addCompleter(BrigadierDispatcher.instance);
|
||||||
|
|
||||||
|
// configuration du formatteur pour le logger
|
||||||
|
System.setProperty("net.md_5.bungee.log-date-format", "yyyy-MM-dd HH:mm:ss");
|
||||||
|
setFormatter(new ConciseFormatter(true));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public ConsoleReader getConsoleReader() {
|
||||||
|
return reader;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public void loop() {
|
||||||
|
|
||||||
|
int i = 0;
|
||||||
|
String line;
|
||||||
|
try {
|
||||||
|
while((line = reader.readLine()) != null) {
|
||||||
|
if (line.trim().equals(""))
|
||||||
|
continue;
|
||||||
|
String cmdLine = line;
|
||||||
|
new Thread(() -> {
|
||||||
|
BrigadierDispatcher.instance.execute(cmdLine);
|
||||||
|
}, "CLICmdThread #"+(i++)).start();
|
||||||
|
|
||||||
|
}
|
||||||
|
} catch (IOException e) {
|
||||||
|
Log.severe(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
private synchronized void println(String str) {
|
||||||
|
try {
|
||||||
|
out.println('\r'+ANSI_RESET+str);
|
||||||
|
out.flush();
|
||||||
|
reader.drawLine();
|
||||||
|
reader.flush();
|
||||||
|
} catch (IOException e) {
|
||||||
|
Log.severe(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void close() throws SecurityException { }
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void flush() { }
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void publish(LogRecord record) {
|
||||||
|
if (!isLoggable(record))
|
||||||
|
return;
|
||||||
|
|
||||||
|
String formattedMessage;
|
||||||
|
|
||||||
|
try {
|
||||||
|
formattedMessage = getFormatter().format(record);
|
||||||
|
} catch (Exception ex) {
|
||||||
|
reportError(null, ex, ErrorManager.FORMAT_FAILURE);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
println(formattedMessage.trim());
|
||||||
|
} catch (Exception ex) {
|
||||||
|
reportError(null, ex, ErrorManager.WRITE_FAILURE);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
31
CLI/src/main/java/fr/pandacube/lib/cli/CoreLogger.java
Normal file
31
CLI/src/main/java/fr/pandacube/lib/cli/CoreLogger.java
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
package fr.pandacube.lib.cli;
|
||||||
|
|
||||||
|
import java.io.PrintStream;
|
||||||
|
import java.util.logging.Level;
|
||||||
|
import java.util.logging.LogRecord;
|
||||||
|
import java.util.logging.Logger;
|
||||||
|
|
||||||
|
import net.md_5.bungee.log.LoggingOutputStream;
|
||||||
|
|
||||||
|
public class CoreLogger extends Logger {
|
||||||
|
|
||||||
|
|
||||||
|
public CoreLogger(ConsoleInterface cli) {
|
||||||
|
super("CoreLogger", null);
|
||||||
|
setLevel(Level.ALL);
|
||||||
|
setUseParentHandlers(false);
|
||||||
|
addHandler(cli);
|
||||||
|
System.setErr(new PrintStream(new LoggingOutputStream(this, Level.SEVERE), true));
|
||||||
|
System.setOut(new PrintStream(new LoggingOutputStream(this, Level.INFO), true));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void log(LogRecord record) {
|
||||||
|
record.setLongThreadID(Thread.currentThread().getId());
|
||||||
|
|
||||||
|
super.log(record);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user