Skip to content

Commit

Permalink
Handle invalid UTF-8 characters in efibootmgr output
Browse files Browse the repository at this point in the history
Modified the code to handle invalid UTF-8 characters in the output of the efibootmgr command by
using `errors="replace"`. This prevents decode exceptions when encountering non-UTF-8 sequences
in the EFI boot manager output [1].

[1] https://bugzilla.redhat.com/show_bug.cgi?id=2292493#c21
Resolves: rhbz#2254801
  • Loading branch information
KKoukiou committed Jan 8, 2025
1 parent 91fe91a commit 6dea007
Show file tree
Hide file tree
Showing 3 changed files with 36 additions and 8 deletions.
22 changes: 16 additions & 6 deletions pyanaconda/core/util.py
Original file line number Diff line number Diff line change
Expand Up @@ -170,6 +170,7 @@ def preexec():


def _run_program(argv, root='/', stdin=None, stdout=None, env_prune=None,
replace_utf_decode_errors=False,
log_output=True, binary_output=False, filter_stderr=False,
do_preexec=True, env_add=None, user=None):
""" Run an external program, log the output and return it to the caller
Expand All @@ -182,6 +183,7 @@ def _run_program(argv, root='/', stdin=None, stdout=None, env_prune=None,
:param stdin: The file object to read stdin from.
:param stdout: Optional file object to write the output to.
:param env_prune: environment variable to remove before execution
:param replace_utf_decode_errors: whether to substitute � for decoding errors.
:param log_output: whether to log the output of command
:param binary_output: whether to treat the output of command as binary data
:param filter_stderr: whether to exclude the contents of stderr from the returned output
Expand All @@ -201,7 +203,10 @@ def _run_program(argv, root='/', stdin=None, stdout=None, env_prune=None,

(output_string, err_string) = proc.communicate()
if not binary_output:
output_string = output_string.decode("utf-8")
output_string = output_string.decode(
"utf-8",
errors="strict" if not replace_utf_decode_errors else "ignore"
)
if output_string and output_string[-1] != "\n":
output_string = output_string + "\n"

Expand Down Expand Up @@ -261,12 +266,14 @@ def execWithRedirect(command, argv, stdin=None, stdout=None, root='/',
:return: The return code of the command
"""
argv = [command] + argv
return _run_program(argv, stdin=stdin, stdout=stdout, root=root, env_prune=env_prune,
env_add=env_add, log_output=log_output, binary_output=binary_output,
return _run_program(argv, stdin=stdin, stdout=stdout, root=root,
env_prune=env_prune, env_add=env_add,
log_output=log_output, binary_output=binary_output,
do_preexec=do_preexec)[0]


def execWithCapture(command, argv, stdin=None, root='/', env_prune=None, env_add=None,
def execWithCapture(command, argv, stdin=None, root='/',
env_prune=None, env_add=None, replace_utf_decode_errors=False,
log_output=True, filter_stderr=False, do_preexec=True):
""" Run an external program and capture standard out and err.
Expand All @@ -276,15 +283,18 @@ def execWithCapture(command, argv, stdin=None, root='/', env_prune=None, env_add
:param root: The directory to chroot to before running command.
:param env_prune: environment variable to remove before execution
:param env_add: environment variables added for the execution
:param replace_utf_decode_errors: whether to ignore decode errors
:param log_output: Whether to log the output of command
:param filter_stderr: Whether stderr should be excluded from the returned output
:param do_preexec: whether to use the preexec function
:return: The output of the command
"""
argv = [command] + argv

return _run_program(argv, stdin=stdin, root=root, log_output=log_output, env_prune=env_prune,
env_add=env_add, filter_stderr=filter_stderr, do_preexec=do_preexec)[1]
return _run_program(argv, stdin=stdin, root=root, log_output=log_output,
env_prune=env_prune, env_add=env_add,
replace_utf_decode_errors=replace_utf_decode_errors,
filter_stderr=filter_stderr, do_preexec=do_preexec)[1]


def execWithCaptureAsLiveUser(command, argv, stdin=None, root='/', log_output=True,
Expand Down
4 changes: 3 additions & 1 deletion pyanaconda/modules/storage/bootloader/efi.py
Original file line number Diff line number Diff line change
Expand Up @@ -111,7 +111,9 @@ def add_efi_boot_target(self):
self._add_single_efi_boot_target(parent)

def remove_efi_boot_target(self):
buf = self.efibootmgr(capture=True)
# FIXME: Stop using replace_utf_decode_errors=True once
# https://github.com/rhboot/efibootmgr/pull/221/ is merged
buf = self.efibootmgr(capture=True, replace_utf_decode_errors=True)
for line in buf.splitlines():
try:
(slot, _product) = line.split(None, 1)
Expand Down
18 changes: 17 additions & 1 deletion tests/unit_tests/pyanaconda_tests/core/test_util.py
Original file line number Diff line number Diff line change
Expand Up @@ -60,10 +60,11 @@ def test_run_program_binary(self):
"""Test _run_program with binary output."""

# Echo something that cannot be decoded as utf-8
# non utf-8 output should be replaced with U+FFFD
retcode, output = util._run_program(['echo', '-en', r'\xa0\xa1\xa2'], binary_output=True)

assert retcode == 0
assert output == b'\xa0\xa1\xa2'
assert output == '���'

def test_exec_with_redirect(self):
"""Test execWithRedirect."""
Expand Down Expand Up @@ -123,6 +124,21 @@ def test_exec_with_capture_as_live_user(self, mock_get_live_user, mock_start_pro
assert mock_start_program.call_args.kwargs["env_add"] == {"TEST": "test"}
assert mock_start_program.call_args.kwargs["env_prune"] == ("TEST_PRUNE",)

def test_exec_with_capture_non_utf8_handling(self):
"""Test execWithCapture with non-utf8 output ignored."""

# Echo something that cannot be decoded as utf-8
output = util.execWithCapture(
'echo', ['-en', r'Hello world! \xa0\xa1\xa2'],
replace_utf_decode_errors=True
)

assert output == 'Hello world! \n'

# If replace_utf_decode_errors is False (default), the non-utf8 output should raise an Exception
with pytest.raises(UnicodeDecodeError):
util.execWithCapture('echo', ['-en', r'\xa0\xa1\xa2'])

def test_exec_readlines(self):
"""Test execReadlines."""

Expand Down

0 comments on commit 6dea007

Please sign in to comment.