From 6aa21e9040625d936fdba4362dc4f1e9c908f9c2 Mon Sep 17 00:00:00 2001 From: Kunal Mehta Date: Fri, 20 Dec 2024 15:17:47 -0800 Subject: [PATCH] Improve test_grsecurity_paxtest and update for noble * Remove the unnecessary diffing and templating, just keep what we want to assert against as a plain string. * Avoid shelling out to grep, we can do the filtering in Python. --- molecule/testinfra/common/paxtest_results.j2 | 18 ---- molecule/testinfra/common/test_grsecurity.py | 91 ++++++++++++++------ 2 files changed, 66 insertions(+), 43 deletions(-) delete mode 100644 molecule/testinfra/common/paxtest_results.j2 diff --git a/molecule/testinfra/common/paxtest_results.j2 b/molecule/testinfra/common/paxtest_results.j2 deleted file mode 100644 index e534612895..0000000000 --- a/molecule/testinfra/common/paxtest_results.j2 +++ /dev/null @@ -1,18 +0,0 @@ -Executable anonymous mapping : Killed -Executable bss : Killed -Executable data : Killed -Executable heap : Killed -Executable stack : Killed -Executable shared library bss : Killed -Executable shared library data : Killed -Executable anonymous mapping (mprotect) : Killed -Executable bss (mprotect) : Killed -Executable data (mprotect) : Killed -Executable heap (mprotect) : Killed -Executable stack (mprotect) : Killed -Executable shared library bss (mprotect) : Killed -Executable shared library data (mprotect): Killed -Return to function (strcpy) : paxtest: return address contains a NULL byte. -Return to function (memcpy) : {{ memcpy_result }} -Return to function (strcpy, PIE) : paxtest: return address contains a NULL byte. -Return to function (memcpy, PIE) : {{ memcpy_result }} diff --git a/molecule/testinfra/common/test_grsecurity.py b/molecule/testinfra/common/test_grsecurity.py index 935e386ed9..a35eb816ed 100644 --- a/molecule/testinfra/common/test_grsecurity.py +++ b/molecule/testinfra/common/test_grsecurity.py @@ -1,10 +1,7 @@ -import difflib -import os import warnings import pytest import testutils -from jinja2 import Template sdvars = testutils.securedrop_test_vars testinfra_hosts = [sdvars.app_hostname, sdvars.monitor_hostname] @@ -95,6 +92,52 @@ def test_grsecurity_sysctl_options(host, sysctl_opt): assert host.sysctl(sysctl_opt[0]) == sysctl_opt[1] +# Versions of paxtest newer than 0.9.12 or so will report +# "Vulnerable" on memcpy tests, see details in +# https://github.com/freedomofpress/securedrop/issues/1039 +PAXTEST_FOCAL = """\ +Executable anonymous mapping : Killed +Executable bss : Killed +Executable data : Killed +Executable heap : Killed +Executable stack : Killed +Executable shared library bss : Killed +Executable shared library data : Killed +Executable anonymous mapping (mprotect) : Killed +Executable bss (mprotect) : Killed +Executable data (mprotect) : Killed +Executable heap (mprotect) : Killed +Executable stack (mprotect) : Killed +Executable shared library bss (mprotect) : Killed +Executable shared library data (mprotect): Killed +Return to function (strcpy) : paxtest: return address contains a NULL byte. +Return to function (memcpy) : Vulnerable +Return to function (strcpy, PIE) : paxtest: return address contains a NULL byte. +Return to function (memcpy, PIE) : Vulnerable +""" + +PAXTEST_NOBLE = """\ +Executable anonymous mapping : Killed +Executable bss : Killed +Executable data : Killed +Executable heap : Killed +Executable stack : Killed +Executable shared library bss : Killed +Executable shared library data : Killed +Executable anonymous mapping (mprotect) : Killed +Executable bss (mprotect) : Killed +Executable data (mprotect) : Killed +Executable heap (mprotect) : Killed +Executable stack (mprotect) : Killed +Executable shared library bss (mprotect) : Killed +Executable shared library data (mprotect): Killed +Return to function (strcpy) : paxtest: return address contains a NULL byte. +Return to function (memcpy) : Vulnerable +Return to function (strcpy, PIE) : paxtest: return address contains a NULL byte. +Return to function (memcpy, PIE) : Vulnerable +""" + + def test_grsecurity_paxtest(host): """ Check that paxtest reports the expected mitigations. These are @@ -109,29 +152,27 @@ def test_grsecurity_paxtest(host): try: with host.sudo(): # Log to /tmp to avoid cluttering up /root. - paxtest_cmd = "paxtest blackhat /tmp/paxtest.log" - # Select only predictably formatted lines; omit - # the guesses, since the number of bits can vary - paxtest_cmd += " | grep -P '^(Executable|Return)'" - paxtest_results = host.check_output(paxtest_cmd) - - paxtest_template_path = f"{os.path.dirname(os.path.abspath(__file__))}/paxtest_results.j2" - - memcpy_result = "Killed" - # Versions of paxtest newer than 0.9.12 or so will report - # "Vulnerable" on memcpy tests, see details in - # https://github.com/freedomofpress/securedrop/issues/1039 + paxtest_results = host.check_output("paxtest blackhat /tmp/paxtest.log") + + # Select only predictably formatted lines; omit + # the guesses, since the number of bits can vary + paxtest_results = ( + "\n".join( + line + for line in paxtest_results.split("\n") + if line.startswith(("Executable", "Return")) + ) + + "\n" + ) + print("paxtest results:\n" + paxtest_results) + if host.system_info.codename == "focal": - memcpy_result = "Vulnerable" - with open(paxtest_template_path) as f: - paxtest_template = Template(f.read().rstrip()) - paxtest_expected = paxtest_template.render(memcpy_result=memcpy_result) - - # The stdout prints here will only be displayed if the test fails - for paxtest_diff in difflib.context_diff( - paxtest_expected.split("\n"), paxtest_results.split("\n") - ): - print(paxtest_diff) + paxtest_expected = PAXTEST_FOCAL + elif host.system_info.codename == "noble": + paxtest_expected = PAXTEST_NOBLE + else: + pytest.fail(f"Unexpected codename {host.system_info.codename}") + assert paxtest_results == paxtest_expected finally: with host.sudo():