From b09ba10ba548da211b4715c83373d2eab3f5a8d0 Mon Sep 17 00:00:00 2001 From: Paul King Date: Sun, 18 Feb 2024 21:16:27 +1000 Subject: [PATCH] GROOVY-11314: JsonOutput Pretty Print always escapes characters --- .../src/main/java/groovy/json/JsonOutput.java | 24 ++++++++++++++----- .../groovy/groovy/json/JsonOutputTest.groovy | 16 +++++++++++++ 2 files changed, 34 insertions(+), 6 deletions(-) diff --git a/subprojects/groovy-json/src/main/java/groovy/json/JsonOutput.java b/subprojects/groovy-json/src/main/java/groovy/json/JsonOutput.java index cb55f417c21..3c997e70e77 100644 --- a/subprojects/groovy-json/src/main/java/groovy/json/JsonOutput.java +++ b/subprojects/groovy-json/src/main/java/groovy/json/JsonOutput.java @@ -156,17 +156,29 @@ public static String toJson(Map m) { /** * Pretty print a JSON payload. * - * @param jsonPayload + * @param jsonPayload a JSON payload * @return a pretty representation of JSON payload. */ public static String prettyPrint(String jsonPayload) { + return prettyPrint(jsonPayload, false); + } + + /** + * Pretty print a JSON payload. + * + * @param jsonPayload a JSON payload + * @param disableUnicodeEscaping whether to disable unicode escaping + * @return a pretty representation of JSON payload. + * @since 4.0.19 + */ + public static String prettyPrint(String jsonPayload, boolean disableUnicodeEscaping) { int indentSize = 0; // Just a guess that the pretty view will take 20 percent more than original. final CharBuf output = CharBuf.create((int) (jsonPayload.length() * 1.2)); JsonLexer lexer = new JsonLexer(new StringReader(jsonPayload)); // Will store already created indents. - Map indentCache = new HashMap(); + Map indentCache = new HashMap<>(); while (lexer.hasNext()) { JsonToken token = lexer.next(); switch (token.getType()) { @@ -209,8 +221,8 @@ public static String prettyPrint(String jsonPayload) { case STRING: String textStr = token.getText(); String textWithoutQuotes = textStr.substring(1, textStr.length() - 1); - if (textWithoutQuotes.length() > 0) { - output.addJsonEscapedString(textWithoutQuotes); + if (!textWithoutQuotes.isEmpty()) { + output.addJsonEscapedString(textWithoutQuotes, disableUnicodeEscaping); } else { output.addQuoted(Chr.array()); } @@ -225,7 +237,7 @@ public static String prettyPrint(String jsonPayload) { } /** - * Creates new indent if it not exists in the indent cache. + * Creates a new indent if it doesn't exist in the indent cache. * * @return indent with the specified size. */ @@ -254,7 +266,7 @@ public static JsonUnescaped unescaped(CharSequence text) { * Represents unescaped JSON */ public static class JsonUnescaped { - private CharSequence text; + private final CharSequence text; public JsonUnescaped(CharSequence text) { this.text = text; diff --git a/subprojects/groovy-json/src/test/groovy/groovy/json/JsonOutputTest.groovy b/subprojects/groovy-json/src/test/groovy/groovy/json/JsonOutputTest.groovy index 8fdd2c7408f..772b7ca6dd4 100644 --- a/subprojects/groovy-json/src/test/groovy/groovy/json/JsonOutputTest.groovy +++ b/subprojects/groovy-json/src/test/groovy/groovy/json/JsonOutputTest.groovy @@ -314,6 +314,22 @@ final class JsonOutputTest { }""".stripIndent() } + @Test + void testPrettyPrintUnicodeEscapeVariants() { + def map = [name: "Järry"] + def json = JsonOutput.toJson(map) + // with escaping + assert JsonOutput.prettyPrint(json) == '''\ + { + "name": "J\\u00e4rry" + }'''.stripIndent() + // with escaping disabled + assert JsonOutput.prettyPrint(json, true) == '''\ + { + "name": "Järry" + }'''.stripIndent() + } + @Test void testSerializePogos() { def city = new JsonCity("Paris", [