SPIGOT-6303: Interpret text that Gson considers to be an empty JSON document as legacy text.

For example, this affects empty Strings, and Strings that consist only
of whitespace. In lenient mode this would also affect text that contains
comments, but that is not the case currently.
Minecraft's component parser (i.e. Gson) produces null for these texts
instead of throwing a parse exception. By interpreting this text as
legacy text we correctly create text components for them that contain
their contents.

By: blablubbabc <lukas@wirsindwir.de>
This commit is contained in:
CraftBukkit/Spigot 2021-01-02 14:46:59 +11:00
parent 904d0ddd37
commit 4b503e57c6

View File

@ -192,14 +192,14 @@ public final class CraftChatMessage {
public static IChatBaseComponent fromJSON(String jsonMessage) throws JsonParseException { public static IChatBaseComponent fromJSON(String jsonMessage) throws JsonParseException {
// Note: This also parses plain Strings to text components. // Note: This also parses plain Strings to text components.
// Note: An empty message (empty, or only consisting of whitespace) results in null rather than a parse exception.
return IChatBaseComponent.ChatSerializer.a(jsonMessage); return IChatBaseComponent.ChatSerializer.a(jsonMessage);
} }
public static IChatBaseComponent fromJSONOrNull(String jsonMessage) { public static IChatBaseComponent fromJSONOrNull(String jsonMessage) {
// Note: An empty message is parsed to an empty text component instead of null.
if (jsonMessage == null) return null; if (jsonMessage == null) return null;
try { try {
return fromJSON(jsonMessage); return fromJSON(jsonMessage); // Can return null
} catch (JsonParseException ex) { } catch (JsonParseException ex) {
return null; return null;
} }
@ -216,13 +216,13 @@ public final class CraftChatMessage {
private static IChatBaseComponent fromJSONOrString(String message, boolean nullable, boolean keepNewlines) { private static IChatBaseComponent fromJSONOrString(String message, boolean nullable, boolean keepNewlines) {
if (message == null) message = ""; if (message == null) message = "";
if (nullable && message.isEmpty()) return null; if (nullable && message.isEmpty()) return null;
// If the message contains color codes, we convert it ourselves: if (isLegacy(message)) {
if (containsColorCodes(message)) {
return fromString(message, keepNewlines)[0]; return fromString(message, keepNewlines)[0];
} else { } else {
try { IChatBaseComponent component = fromJSONOrNull(message);
return fromJSON(message); if (component != null) {
} catch (JsonParseException ex) { return component;
} else {
return fromString(message, keepNewlines)[0]; return fromString(message, keepNewlines)[0];
} }
} }
@ -247,14 +247,13 @@ public final class CraftChatMessage {
public static String fromJSONOrStringToJSON(String message, boolean nullable, boolean keepNewlines, int maxLength, boolean checkJsonContentLength) { public static String fromJSONOrStringToJSON(String message, boolean nullable, boolean keepNewlines, int maxLength, boolean checkJsonContentLength) {
if (message == null) message = ""; if (message == null) message = "";
if (nullable && message.isEmpty()) return null; if (nullable && message.isEmpty()) return null;
// If the message contains color codes, we convert it ourselves: if (isLegacy(message)) {
if (containsColorCodes(message)) {
message = trimMessage(message, maxLength); message = trimMessage(message, maxLength);
return fromStringToJSON(message, keepNewlines); return fromStringToJSON(message, keepNewlines);
} else { } else {
try { // If the input can be parsed as JSON, we use that:
// If the input can be parsed as JSON, we use that: IChatBaseComponent component = fromJSONOrNull(message);
IChatBaseComponent component = fromJSON(message); if (component != null) {
if (checkJsonContentLength) { if (checkJsonContentLength) {
String content = fromComponent(component); String content = fromComponent(component);
String trimmedContent = trimMessage(content, maxLength); String trimmedContent = trimMessage(content, maxLength);
@ -264,8 +263,8 @@ public final class CraftChatMessage {
} }
} }
return message; return message;
} catch (JsonParseException ex) { } else {
// Else we convert the input: // Else we interpret the input as legacy text:
message = trimMessage(message, maxLength); message = trimMessage(message, maxLength);
return fromStringToJSON(message, keepNewlines); return fromStringToJSON(message, keepNewlines);
} }
@ -280,8 +279,12 @@ public final class CraftChatMessage {
} }
} }
public static boolean containsColorCodes(String message) { // Heuristic detection of legacy (plain) text.
return message != null && message.contains(COLOR_CHAR_STRING); // May produce false-negatives: I.e. a return value of false does not imply that the text represents modern (JSON-based) text.
// We also consider empty Strings as legacy. The component deserializer cannot parse them (produces null).
private static boolean isLegacy(String message) {
// assert: message != null
return message.trim().isEmpty() || message.contains(COLOR_CHAR_STRING);
} }
public static String fromStringToJSON(String message) { public static String fromStringToJSON(String message) {