mirror of
https://github.com/LuckPerms/LuckPerms.git
synced 2025-01-01 05:57:51 +01:00
Change argument tokenizer to support unicode double quote characters (#1999)
This commit is contained in:
parent
4f63a00bad
commit
c6bda0875c
@ -25,12 +25,7 @@
|
||||
|
||||
package me.lucko.luckperms.common.command.utils;
|
||||
|
||||
import com.google.common.base.Joiner;
|
||||
import com.google.common.base.Splitter;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
import java.util.regex.Pattern;
|
||||
|
||||
/**
|
||||
* Tokenizes command input into distinct "argument" tokens.
|
||||
@ -40,46 +35,22 @@ import java.util.regex.Pattern;
|
||||
public enum ArgumentTokenizer {
|
||||
|
||||
EXECUTE {
|
||||
@Override
|
||||
public List<String> tokenizeInput(String[] args) {
|
||||
return stripQuotes(EXECUTE_ARGUMENT_SPLITTER.split(ARGUMENT_JOINER.join(args)));
|
||||
}
|
||||
|
||||
@Override
|
||||
public List<String> tokenizeInput(String args) {
|
||||
return stripQuotes(EXECUTE_ARGUMENT_SPLITTER.split(args));
|
||||
return new QuotedStringTokenizer(args).tokenize(true);
|
||||
}
|
||||
},
|
||||
TAB_COMPLETE {
|
||||
@Override
|
||||
public List<String> tokenizeInput(String[] args) {
|
||||
return stripQuotes(TAB_COMPLETE_ARGUMENT_SPLITTER.split(ARGUMENT_JOINER.join(args)));
|
||||
}
|
||||
|
||||
@Override
|
||||
public List<String> tokenizeInput(String args) {
|
||||
return stripQuotes(TAB_COMPLETE_ARGUMENT_SPLITTER.split(args));
|
||||
return new QuotedStringTokenizer(args).tokenize(false);
|
||||
}
|
||||
};
|
||||
|
||||
private static final Pattern ARGUMENT_SEPARATOR_PATTERN = Pattern.compile(" (?=([^\\\"]*\\\"[^\\\"]*\\\")*[^\\\"]*$)");
|
||||
private static final Splitter TAB_COMPLETE_ARGUMENT_SPLITTER = Splitter.on(ARGUMENT_SEPARATOR_PATTERN);
|
||||
private static final Splitter EXECUTE_ARGUMENT_SPLITTER = TAB_COMPLETE_ARGUMENT_SPLITTER.omitEmptyStrings();
|
||||
private static final Joiner ARGUMENT_JOINER = Joiner.on(' ');
|
||||
|
||||
public abstract List<String> tokenizeInput(String[] args);
|
||||
public List<String> tokenizeInput(String[] args) {
|
||||
return tokenizeInput(String.join(" ", args));
|
||||
}
|
||||
|
||||
public abstract List<String> tokenizeInput(String args);
|
||||
|
||||
private static List<String> stripQuotes(Iterable<String> input) {
|
||||
List<String> list = new ArrayList<>();
|
||||
for (String argument : input) {
|
||||
if (argument.length() >= 3 && argument.charAt(0) == '"' && argument.charAt(argument.length() - 1) == '"') {
|
||||
list.add(argument.substring(1, argument.length() - 1));
|
||||
} else {
|
||||
list.add(argument);
|
||||
}
|
||||
}
|
||||
return list;
|
||||
}
|
||||
}
|
||||
|
@ -0,0 +1,119 @@
|
||||
/*
|
||||
* This file is part of LuckPerms, licensed under the MIT License.
|
||||
*
|
||||
* Copyright (c) lucko (Luck) <luck@lucko.me>
|
||||
* Copyright (c) contributors
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in all
|
||||
* copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
* SOFTWARE.
|
||||
*/
|
||||
|
||||
package me.lucko.luckperms.common.command.utils;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Tokenizes strings on whitespace, but ignoring whitespace enclosed within quotes.
|
||||
*/
|
||||
public class QuotedStringTokenizer {
|
||||
private final String string;
|
||||
private int cursor;
|
||||
|
||||
public QuotedStringTokenizer(String string) {
|
||||
this.string = string;
|
||||
}
|
||||
|
||||
public List<String> tokenize(boolean omitEmptyStringAtEnd) {
|
||||
List<String> output = new ArrayList<>();
|
||||
while (hasNext()) {
|
||||
output.add(readString());
|
||||
}
|
||||
if (!omitEmptyStringAtEnd && isWhitespace(peek(-1))) {
|
||||
output.add("");
|
||||
}
|
||||
return output;
|
||||
}
|
||||
|
||||
private static boolean isQuotedStringStart(char c) {
|
||||
// return c == '"' || c == '“' || c == '”';
|
||||
return c == '\u0022' || c == '\u201C' || c == '\u201D';
|
||||
}
|
||||
|
||||
private static boolean isWhitespace(char c) {
|
||||
return c == ' ';
|
||||
}
|
||||
|
||||
private String readString() {
|
||||
if (isQuotedStringStart(peek())) {
|
||||
return readQuotedString();
|
||||
} else {
|
||||
return readUnquotedString();
|
||||
}
|
||||
}
|
||||
|
||||
private String readUnquotedString() {
|
||||
final int start = this.cursor;
|
||||
while (hasNext() && !isWhitespace(peek())) {
|
||||
skip();
|
||||
}
|
||||
final int end = this.cursor;
|
||||
|
||||
if (hasNext()) {
|
||||
skip(); // skip whitespace
|
||||
}
|
||||
|
||||
return this.string.substring(start, end);
|
||||
}
|
||||
|
||||
private String readQuotedString() {
|
||||
skip(); // skip start quote
|
||||
|
||||
final int start = this.cursor;
|
||||
while (hasNext() && !isQuotedStringStart(peek())) {
|
||||
skip();
|
||||
}
|
||||
final int end = this.cursor;
|
||||
|
||||
if (hasNext()) {
|
||||
skip(); // skip end quote
|
||||
}
|
||||
if (hasNext() && isWhitespace(peek())) {
|
||||
skip(); // skip whitespace
|
||||
}
|
||||
|
||||
return this.string.substring(start, end);
|
||||
}
|
||||
|
||||
private boolean hasNext() {
|
||||
return this.cursor + 1 <= this.string.length();
|
||||
}
|
||||
|
||||
private char peek() {
|
||||
return this.string.charAt(this.cursor);
|
||||
}
|
||||
|
||||
private char peek(int offset) {
|
||||
return this.string.charAt(this.cursor + offset);
|
||||
}
|
||||
|
||||
private void skip() {
|
||||
this.cursor++;
|
||||
}
|
||||
|
||||
}
|
Loading…
Reference in New Issue
Block a user