From 2461d2edb574825cf394afb4d5970335cc8d1f50 Mon Sep 17 00:00:00 2001 From: Kevin Deldycke Date: Tue, 30 May 2023 14:56:51 +0400 Subject: [PATCH] Keep track of and mix in CliRunner results. Closes #2522 --- CHANGES.rst | 7 ++- src/click/testing.py | 138 ++++++++++++++++++++++++++++++------------ tests/test_termui.py | 2 +- tests/test_testing.py | 31 ++++------ 4 files changed, 116 insertions(+), 62 deletions(-) diff --git a/CHANGES.rst b/CHANGES.rst index a43a5531f..6fb338b41 100644 --- a/CHANGES.rst +++ b/CHANGES.rst @@ -1,4 +1,5 @@ -.. currentmodule:: click +rom 0) +Unpacking objects: 100% (9/9), 2.32 KiB | 2.32 MiB/s, done... currentmodule:: click Version 8.2.0 ------------- @@ -39,6 +40,10 @@ Unreleased - Add ``get_help_extra`` method on ``Option`` to fetch the generated extra items used in ``get_help_record`` to render help text. :issue:`2516` :pr:`2517` +- Keep `` and `` streams independent in `CliRunner`. Always + collect `` output and never raise an exception. Add a new + `` stream to simulate what the user sees in its terminal. Removes + the ``mix_stderr`` parameter in ``CliRunner``. :issue:`2522` :pr:`2523` Version 8.1.8 diff --git a/src/click/testing.py b/src/click/testing.py index b91170e91..2d0bb0cf0 100644 --- a/src/click/testing.py +++ b/src/click/testing.py @@ -18,6 +18,8 @@ from ._compat import _find_binary_reader if t.TYPE_CHECKING: + from _typeshed import ReadableBuffer + from .core import Command @@ -65,6 +67,39 @@ def _pause_echo(stream: EchoingStdin | None) -> cabc.Iterator[None]: stream._paused = False +class BytesIOCopy(io.BytesIO): + """Patch ``io.BytesIO`` to let the written stream be copied to another. + + .. versionadded:: 8.2 + """ + + def __init__(self, copy_to: io.BytesIO) -> None: + super().__init__() + self.copy_to = copy_to + + def flush(self) -> None: + super().flush() + self.copy_to.flush() + + def write(self, b: ReadableBuffer) -> int: + self.copy_to.write(b) + return super().write(b) + + +class StreamMixer: + """Mixes `` and `` streams. + + The result is available in the ``output`` attribute. + + .. versionadded:: 8.2 + """ + + def __init__(self) -> None: + self.output: io.BytesIO = io.BytesIO() + self.stdout: io.BytesIO = BytesIOCopy(copy_to=self.output) + self.stderr: io.BytesIO = BytesIOCopy(copy_to=self.output) + + class _NamedTextIOWrapper(io.TextIOWrapper): def __init__( self, buffer: t.BinaryIO, name: str, mode: str, **kwargs: t.Any @@ -109,7 +144,8 @@ def __init__( self, runner: CliRunner, stdout_bytes: bytes, - stderr_bytes: bytes | None, + stderr_bytes: bytes, + output_bytes: bytes, return_value: t.Any, exit_code: int, exception: BaseException | None, @@ -120,8 +156,16 @@ def __init__( self.runner = runner #: The standard output as bytes. self.stdout_bytes = stdout_bytes - #: The standard error as bytes, or None if not available + #: The standard error as bytes. + #: + #: .. versionchanged:: 8.2 + #: No longer optional. self.stderr_bytes = stderr_bytes + #: A mix of `stdout_bytes` and `stderr_bytes``, as the user would see + # it in its terminal. + #: + #: .. versionadded:: 8.2 + self.output_bytes = output_bytes #: The value returned from the invoked command. #: #: .. versionadded:: 8.0 @@ -135,8 +179,15 @@ def __init__( @property def output(self) -> str: - """The (standard) output as unicode string.""" - return self.stdout + """The terminal output as unicode string, as the user would see it. + + .. versionchanged:: 8.2 + No longer a proxy for ``self.stdout``. Now has its own independent stream + that is mixing `` and ``, in the order they were written. + """ + return self.output_bytes.decode(self.runner.charset, "replace").replace( + "\r\n", "\n" + ) @property def stdout(self) -> str: @@ -147,9 +198,11 @@ def stdout(self) -> str: @property def stderr(self) -> str: - """The standard error as unicode string.""" - if self.stderr_bytes is None: - raise ValueError("stderr not separately captured") + """The standard error as unicode string. + + .. versionchanged:: 8.2 + No longer raise an exception, always returns the `` string. + """ return self.stderr_bytes.decode(self.runner.charset, "replace").replace( "\r\n", "\n" ) @@ -167,15 +220,13 @@ class CliRunner: :param charset: the character set for the input and output data. :param env: a dictionary with environment variables for overriding. - :param echo_stdin: if this is set to `True`, then reading from stdin writes - to stdout. This is useful for showing examples in + :param echo_stdin: if this is set to `True`, then reading from `` writes + to ``. This is useful for showing examples in some circumstances. Note that regular prompts will automatically echo the input. - :param mix_stderr: if this is set to `False`, then stdout and stderr are - preserved as independent streams. This is useful for - Unix-philosophy apps that have predictable stdout and - noisy stderr, such that each may be measured - independently + + .. versionchanged:: 8.2 + ``mix_stderr`` parameter has been removed. """ def __init__( @@ -183,12 +234,10 @@ def __init__( charset: str = "utf-8", env: cabc.Mapping[str, str | None] | None = None, echo_stdin: bool = False, - mix_stderr: bool = True, ) -> None: self.charset = charset self.env: cabc.Mapping[str, str | None] = env or {} self.echo_stdin = echo_stdin - self.mix_stderr = mix_stderr def get_default_prog_name(self, cli: Command) -> str: """Given a command object it will return the default program name @@ -212,22 +261,29 @@ def isolation( input: str | bytes | t.IO[t.Any] | None = None, env: cabc.Mapping[str, str | None] | None = None, color: bool = False, - ) -> cabc.Iterator[tuple[io.BytesIO, io.BytesIO | None]]: + ) -> cabc.Iterator[tuple[io.BytesIO, io.BytesIO, io.BytesIO]]: """A context manager that sets up the isolation for invoking of a - command line tool. This sets up stdin with the given input data + command line tool. This sets up `` with the given input data and `os.environ` with the overrides from the given dictionary. This also rebinds some internals in Click to be mocked (like the prompt functionality). This is automatically done in the :meth:`invoke` method. - :param input: the input stream to put into sys.stdin. + :param input: the input stream to put into `sys.stdin`. :param env: the environment overrides as dictionary. :param color: whether the output should contain color codes. The application can still override this explicitly. + .. versionadded:: 8.2 + An additional output stream is returned, which is a mix of + `` and `` streams. + + .. versionchanged:: 8.2 + Always returns the `` stream. + .. versionchanged:: 8.0 - ``stderr`` is opened with ``errors="backslashreplace"`` + `` is opened with ``errors="backslashreplace"`` instead of the default ``"strict"``. .. versionchanged:: 4.0 @@ -244,11 +300,11 @@ def isolation( env = self.make_env(env) - bytes_output = io.BytesIO() + stream_mixer = StreamMixer() if self.echo_stdin: bytes_input = echo_input = t.cast( - t.BinaryIO, EchoingStdin(bytes_input, bytes_output) + t.BinaryIO, EchoingStdin(bytes_input, stream_mixer.stdout) ) sys.stdin = text_input = _NamedTextIOWrapper( @@ -261,21 +317,16 @@ def isolation( text_input._CHUNK_SIZE = 1 # type: ignore sys.stdout = _NamedTextIOWrapper( - bytes_output, encoding=self.charset, name="", mode="w" + stream_mixer.stdout, encoding=self.charset, name="", mode="w" ) - bytes_error = None - if self.mix_stderr: - sys.stderr = sys.stdout - else: - bytes_error = io.BytesIO() - sys.stderr = _NamedTextIOWrapper( - bytes_error, - encoding=self.charset, - name="", - mode="w", - errors="backslashreplace", - ) + sys.stderr = _NamedTextIOWrapper( + stream_mixer.stderr, + encoding=self.charset, + name="", + mode="w", + errors="backslashreplace", + ) @_pause_echo(echo_input) # type: ignore def visible_input(prompt: str | None = None) -> str: @@ -331,7 +382,7 @@ def should_strip_ansi( pass else: os.environ[key] = value - yield (bytes_output, bytes_error) + yield (stream_mixer.stdout, stream_mixer.stderr, stream_mixer.output) finally: for key, value in old_env.items(): if value is None: @@ -380,6 +431,14 @@ def invoke( :param color: whether the output should contain color codes. The application can still override this explicitly. + .. versionadded:: 8.2 + The result object has the ``output_bytes`` attribute with + the mix of ``stdout_bytes`` and ``stderr_bytes``, as the user would + see it in its terminal. + + .. versionchanged:: 8.2 + The result object always returns the ``stderr_bytes`` stream. + .. versionchanged:: 8.0 The result object has the ``return_value`` attribute with the value returned from the invoked command. @@ -436,15 +495,14 @@ def invoke( finally: sys.stdout.flush() stdout = outstreams[0].getvalue() - if self.mix_stderr: - stderr = None - else: - stderr = outstreams[1].getvalue() # type: ignore + stderr = outstreams[1].getvalue() + output = outstreams[2].getvalue() return Result( runner=self, stdout_bytes=stdout, stderr_bytes=stderr, + output_bytes=output, return_value=return_value, exit_code=exit_code, exception=exception, diff --git a/tests/test_termui.py b/tests/test_termui.py index 2af7ea18f..0dcd2330c 100644 --- a/tests/test_termui.py +++ b/tests/test_termui.py @@ -246,7 +246,7 @@ def test_secho(runner): ("value", "expect"), [(123, b"\x1b[45m123\x1b[0m"), (b"test", b"test")] ) def test_secho_non_text(runner, value, expect): - with runner.isolation() as (out, _): + with runner.isolation() as (out, _, _): click.secho(value, nl=False, color=True, bg="magenta") result = out.getvalue() assert result == expect diff --git a/tests/test_testing.py b/tests/test_testing.py index e67b3a911..afb5ef9a2 100644 --- a/tests/test_testing.py +++ b/tests/test_testing.py @@ -301,32 +301,23 @@ def cli_env(): def test_stderr(): @click.command() def cli_stderr(): - click.echo("stdout") - click.echo("stderr", err=True) - - runner = CliRunner(mix_stderr=False) - - result = runner.invoke(cli_stderr) - - assert result.output == "stdout\n" - assert result.stdout == "stdout\n" - assert result.stderr == "stderr\n" + click.echo("1 - stdout") + click.echo("2 - stderr", err=True) + click.echo("3 - stdout") + click.echo("4 - stderr", err=True) - runner_mix = CliRunner(mix_stderr=True) + runner_mix = CliRunner() result_mix = runner_mix.invoke(cli_stderr) - assert result_mix.output == "stdout\nstderr\n" - assert result_mix.stdout == "stdout\nstderr\n" - - with pytest.raises(ValueError): - assert result_mix.stderr # noqa B018 + assert result_mix.output == "1 - stdout\n2 - stderr\n3 - stdout\n4 - stderr\n" + assert result_mix.stdout == "1 - stdout\n3 - stdout\n" + assert result_mix.stderr == "2 - stderr\n4 - stderr\n" @click.command() def cli_empty_stderr(): click.echo("stdout") - runner = CliRunner(mix_stderr=False) - + runner = CliRunner() result = runner.invoke(cli_empty_stderr) assert result.output == "stdout\n" @@ -410,9 +401,9 @@ def test_isolation_stderr_errors(): """Writing to stderr should escape invalid characters instead of raising a UnicodeEncodeError. """ - runner = CliRunner(mix_stderr=False) + runner = CliRunner() - with runner.isolation() as (_, err): + with runner.isolation() as (_, err, _): click.echo("\udce2", err=True, nl=False) assert err.getvalue() == b"\\udce2"