From 9da5d2156fc933c41b7ae3cb3ad6784fcd2a1e0a Mon Sep 17 00:00:00 2001 From: Philippe Charles Date: Wed, 3 Feb 2021 11:51:53 +0100 Subject: [PATCH] Fix bad encoding when creating default PrintWriter --- src/main/java/picocli/CommandLine.java | 27 +++++-- src/test/java/picocli/ExecuteTest.java | 15 +--- src/test/java/picocli/HelpSubCommandTest.java | 8 +- src/test/java/picocli/Issue1320.java | 80 +++++++++++++++++++ 4 files changed, 105 insertions(+), 25 deletions(-) create mode 100644 src/test/java/picocli/Issue1320.java diff --git a/src/main/java/picocli/CommandLine.java b/src/main/java/picocli/CommandLine.java index 9d92ed694..226bceaa2 100644 --- a/src/main/java/picocli/CommandLine.java +++ b/src/main/java/picocli/CommandLine.java @@ -1206,7 +1206,7 @@ public CommandLine setColorScheme(Help.ColorScheme colorScheme) { * help with a {@code --help} or similar option, the usage help message is printed to the standard output stream so that it can be easily searched and paged.

* @since 4.0 */ public PrintWriter getOut() { - if (out == null) { setOut(new PrintWriter(System.out, true)); } + if (out == null) { setOut(newPrintWriter(System.out, getStdoutEncoding())); } return out; } @@ -1233,7 +1233,7 @@ public CommandLine setOut(PrintWriter out) { * should use this writer to print error messages (which may include a usage help message) when an unexpected error occurs.

* @since 4.0 */ public PrintWriter getErr() { - if (err == null) { setErr(new PrintWriter(System.err, true)); } + if (err == null) { setErr(newPrintWriter(System.err, getStderrEncoding())); } return err; } @@ -1807,7 +1807,7 @@ protected R throwOrExit(ExecutionException ex) { * @since 2.0 */ @Deprecated public static class DefaultExceptionHandler extends AbstractHandler> implements IExceptionHandler, IExceptionHandler2 { public List handleException(ParameterException ex, PrintStream out, Help.Ansi ansi, String... args) { - internalHandleParseException(ex, new PrintWriter(out, true), Help.defaultColorScheme(ansi)); return Collections.emptyList(); } + internalHandleParseException(ex, newPrintWriter(out, getStdoutEncoding()), Help.defaultColorScheme(ansi)); return Collections.emptyList(); } /** Prints the message of the specified exception, followed by the usage message for the command or subcommand * whose input was invalid, to the stream returned by {@link #err()}. @@ -1817,7 +1817,7 @@ public List handleException(ParameterException ex, PrintStream out, Help * @return the empty list * @since 3.0 */ public R handleParseException(ParameterException ex, String[] args) { - internalHandleParseException(ex, new PrintWriter(err(), true), colorScheme()); return returnResultOrExit(null); } + internalHandleParseException(ex, newPrintWriter(err(), getStderrEncoding()), colorScheme()); return returnResultOrExit(null); } static void internalHandleParseException(ParameterException ex, PrintWriter writer, Help.ColorScheme colorScheme) { writer.println(colorScheme.errorText(ex.getMessage())); @@ -1878,7 +1878,7 @@ public static boolean printHelpIfRequested(ParseResult parseResult) { * @since 3.6 */ @Deprecated public static boolean printHelpIfRequested(List parsedCommands, PrintStream out, PrintStream err, Help.ColorScheme colorScheme) { // for backwards compatibility - for (CommandLine cmd : parsedCommands) { cmd.setOut(new PrintWriter(out, true)).setErr(new PrintWriter(err, true)).setColorScheme(colorScheme); } + for (CommandLine cmd : parsedCommands) { cmd.setOut(newPrintWriter(out, getStdoutEncoding())).setErr(newPrintWriter(err, getStderrEncoding())).setColorScheme(colorScheme); } return executeHelpRequest(parsedCommands) != null; } @@ -2134,8 +2134,8 @@ private T enrichForBackwardsCompatibility(T obj) { // and the application called #useOut, #useErr or #useAnsi on it if (obj instanceof AbstractHandler) { AbstractHandler handler = (AbstractHandler) obj; - if (handler.out() != System.out) { setOut(new PrintWriter(handler.out(), true)); } - if (handler.err() != System.err) { setErr(new PrintWriter(handler.err(), true)); } + if (handler.out() != System.out) { setOut(newPrintWriter(handler.out(), getStdoutEncoding())); } + if (handler.err() != System.err) { setErr(newPrintWriter(handler.err(), getStderrEncoding())); } if (handler.ansi() != Help.Ansi.AUTO) { setColorScheme(handler.colorScheme()); } } return obj; @@ -14476,6 +14476,17 @@ static void close(Closeable closeable) { new Tracer().warn("Could not close " + closeable + ": " + ex.toString()); } } + static Charset getStdoutEncoding() { + String encoding = System.getProperty("sun.stdout.encoding"); + return encoding != null ? Charset.forName(encoding) : Charset.defaultCharset(); + } + static Charset getStderrEncoding() { + String encoding = System.getProperty("sun.stderr.encoding"); + return encoding != null ? Charset.forName(encoding) : Charset.defaultCharset(); + } + static PrintWriter newPrintWriter(OutputStream stream, Charset charset) { + return new PrintWriter(new BufferedWriter(new OutputStreamWriter(stream, charset)), true); + } static class PositionalParametersSorter implements Comparator { private static final Range OPTION_INDEX = new Range(0, 0, false, true, "0"); public int compare(ArgSpec p1, ArgSpec p2) { @@ -18015,7 +18026,7 @@ static List stripErrorMessage(List unmatched) { public boolean isUnknownOption() { return isUnknownOption(unmatched, getCommandLine()); } /** Returns {@code true} and prints suggested solutions to the specified stream if such solutions exist, otherwise returns {@code false}. * @since 3.3.0 */ - public boolean printSuggestions(PrintStream out) { return printSuggestions(new PrintWriter(out, true)); } + public boolean printSuggestions(PrintStream out) { return printSuggestions(newPrintWriter(out, getStdoutEncoding())); } /** Returns {@code true} and prints suggested solutions to the specified stream if such solutions exist, otherwise returns {@code false}. * @since 4.0 */ public boolean printSuggestions(PrintWriter writer) { diff --git a/src/test/java/picocli/ExecuteTest.java b/src/test/java/picocli/ExecuteTest.java index 4eb180179..26df3106b 100644 --- a/src/test/java/picocli/ExecuteTest.java +++ b/src/test/java/picocli/ExecuteTest.java @@ -51,19 +51,8 @@ import static org.hamcrest.CoreMatchers.containsString; import static org.hamcrest.CoreMatchers.startsWith; import static org.junit.Assert.*; -import static picocli.CommandLine.Command; -import static picocli.CommandLine.ExecutionException; -import static picocli.CommandLine.Help; -import static picocli.CommandLine.IExecutionStrategy; +import static picocli.CommandLine.*; import static picocli.CommandLine.Model.UsageMessageSpec.keyValuesMap; -import static picocli.CommandLine.Option; -import static picocli.CommandLine.ParameterException; -import static picocli.CommandLine.Parameters; -import static picocli.CommandLine.ParseResult; -import static picocli.CommandLine.RunAll; -import static picocli.CommandLine.RunFirst; -import static picocli.CommandLine.RunLast; -import static picocli.CommandLine.Spec; public class ExecuteTest { @Rule @@ -415,7 +404,7 @@ public void testExecuteWithInvalidInput_Ansi_ON() { @Test public void testExecuteWithInvalidInput_Ansi_ON_CustomErr() { new CommandLine(new MyCallable()) - .setErr(new PrintWriter(System.out, true)) + .setErr(CommandLine.newPrintWriter(System.out, getStdoutEncoding())) .setColorScheme(Help.defaultColorScheme(Help.Ansi.ON)).execute("invalid input"); assertEquals("", systemErrRule.getLog()); assertEquals(INVALID_INPUT_ANSI + MYCALLABLE_USAGE_ANSI, systemOutRule.getLog()); diff --git a/src/test/java/picocli/HelpSubCommandTest.java b/src/test/java/picocli/HelpSubCommandTest.java index 1e1f439d8..2468cee53 100644 --- a/src/test/java/picocli/HelpSubCommandTest.java +++ b/src/test/java/picocli/HelpSubCommandTest.java @@ -168,8 +168,8 @@ public void run() { } int exitCode = new CommandLine(new App()) .setExecutionStrategy(new RunLast()) - .setOut(new PrintWriter(System.out, true)) - .setErr(new PrintWriter(System.err, true)) + .setOut(CommandLine.newPrintWriter(System.out, getStdoutEncoding())) + .setErr(CommandLine.newPrintWriter(System.err, getStderrEncoding())) .setColorScheme(Help.defaultColorScheme(Help.Ansi.OFF)) .execute("customHelp"); @@ -206,8 +206,8 @@ public void run() { } int exitCode = new CommandLine(new App()) .setExecutionStrategy(new RunLast()) - .setOut(new PrintWriter(System.out, true)) - .setErr(new PrintWriter(System.err, true)) + .setOut(CommandLine.newPrintWriter(System.out, getStdoutEncoding())) + .setErr(CommandLine.newPrintWriter(System.err, getStderrEncoding())) .setColorScheme(Help.defaultColorScheme(Help.Ansi.OFF)) .execute("newCustomHelp"); diff --git a/src/test/java/picocli/Issue1320.java b/src/test/java/picocli/Issue1320.java new file mode 100644 index 000000000..be807f6b4 --- /dev/null +++ b/src/test/java/picocli/Issue1320.java @@ -0,0 +1,80 @@ +package picocli; + +import org.junit.Rule; +import org.junit.Test; +import org.junit.contrib.java.lang.system.RestoreSystemProperties; +import org.junit.contrib.java.lang.system.SystemErrRule; +import org.junit.contrib.java.lang.system.SystemOutRule; +import org.junit.rules.TestRule; +import picocli.CommandLine.Command; +import picocli.CommandLine.Model.CommandSpec; +import picocli.CommandLine.Parameters; +import picocli.CommandLine.Spec; + +import java.nio.charset.Charset; + +import static org.junit.Assert.assertEquals; + +public class Issue1320 { + + @Rule + public final TestRule restoreSystemProperties = new RestoreSystemProperties(); + + @Rule + public final SystemOutRule systemOutRule = new SystemOutRule().enableLog(); + + @Rule + public final SystemErrRule systemErrRule = new SystemErrRule().enableLog(); + + @Command(name = "test") + static class TestCommand implements Runnable { + + @Parameters + String text; + + @Spec + CommandSpec spec; + + @Override + public void run() { + spec.commandLine().getOut().print(text); + spec.commandLine().getOut().flush(); + spec.commandLine().getErr().print(text); + spec.commandLine().getErr().flush(); + } + } + + @Test + public void testIssue1320() { + String unmappable = "[abcµ]"; + + resetLogs(); + System.clearProperty(SUN_STDOUT_ENCODING); + System.clearProperty(SUN_STDERR_ENCODING); + fixLogPrintStream(Charset.defaultCharset().name()); + assertEquals(CommandLine.ExitCode.OK, new CommandLine(new TestCommand()).execute(unmappable)); + assertEquals(unmappable, systemOutRule.getLog()); + assertEquals(unmappable, systemErrRule.getLog()); + + resetLogs(); + System.setProperty(SUN_STDOUT_ENCODING, CP_437); + System.setProperty(SUN_STDERR_ENCODING, CP_437); + fixLogPrintStream(CP_437); + assertEquals(CommandLine.ExitCode.OK, new CommandLine(new TestCommand()).execute(unmappable)); + assertEquals(unmappable, systemOutRule.getLog()); + assertEquals(unmappable, systemErrRule.getLog()); + } + + private void resetLogs() { + systemOutRule.clearLog(); + systemErrRule.clearLog(); + } + + private void fixLogPrintStream(String encoding) { + System.setProperty("file.encoding", encoding); + } + + private static final String CP_437 = "cp437"; + private static final String SUN_STDOUT_ENCODING = "sun.stdout.encoding"; + private static final String SUN_STDERR_ENCODING = "sun.stderr.encoding"; +}