Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions changelog/681.bugfix.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Fix tracing output to safely escape surrogate characters in hook arguments and return values.
2 changes: 1 addition & 1 deletion src/pluggy/_manager.py
Original file line number Diff line number Diff line change
Expand Up @@ -503,7 +503,7 @@ def after(
kwargs: Mapping[str, object],
) -> None:
if outcome.exception is None:
hooktrace("finish", hook_name, "-->", outcome.get_result())
hooktrace("finish", hook_name, "-->", repr(outcome.get_result()))
hooktrace.root.indent -= 1

return self.add_hookcall_monitoring(before, after)
Expand Down
2 changes: 1 addition & 1 deletion src/pluggy/_tracing.py
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@ def _format_message(self, tags: Sequence[str], args: Sequence[object]) -> str:
lines = [f"{indent}{content} [{':'.join(tags)}]\n"]

for name, value in extra.items():
lines.append(f"{indent} {name}: {value}\n")
lines.append(f"{indent} {name}: {value!r}\n")

return "".join(lines)

Expand Down
33 changes: 33 additions & 0 deletions testing/test_pluginmanager.py
Original file line number Diff line number Diff line change
Expand Up @@ -695,6 +695,39 @@ def he_method1(self):
undo()


def test_hook_tracing_escapes_surrogate_values(pm: PluginManager) -> None:
class Hooks:
@hookspec(firstresult=True)
def he_method1(self, arg: object) -> object:
raise NotImplementedError()

class Plugin:
@hookimpl
def he_method1(self, arg: object) -> object:
return arg

out: list[str] = []

def write(message: str) -> None:
message.encode()
out.append(message)

pm.add_hookspecs(Hooks)
pm.register(Plugin())
pm.trace.root.setwriter(write)
undo = pm.enable_tracing()
try:
result = pm.hook.he_method1(arg="\ud800")
finally:
undo()

assert result == "\ud800"
assert out == [
" he_method1 [hook]\n arg: '\\ud800'\n",
" finish he_method1 --> '\\ud800' [hook]\n",
]


@pytest.mark.parametrize("historic", [False, True])
def test_register_while_calling(
pm: PluginManager,
Expand Down
6 changes: 6 additions & 0 deletions testing/test_tracer.py
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,12 @@ def test_readable_output_dictargs(rootlogger: TagTracer) -> None:
assert out2 == "test [test]\n a: 1\n"


def test_dictargs_escape_surrogate_values(rootlogger: TagTracer) -> None:
out = rootlogger._format_message(["test"], ["test", {"arg": "\ud800"}])
assert out == "test [test]\n arg: '\\ud800'\n"
out.encode()


def test_setprocessor(rootlogger: TagTracer) -> None:
log = rootlogger.get("1")
log2 = log.get("2")
Expand Down
Loading