From fd788dae465a920fa6a92cd9d017afc4dc786b5b80a39301c2b92fbc53071036 Mon Sep 17 00:00:00 2001 From: Matej Cepl Date: Wed, 28 Aug 2024 16:41:06 +0000 Subject: [PATCH 1/2] Add back qemu_user_space_build condition OBS-URL: https://build.opensuse.org/package/show/devel:languages:python:Factory/python38?expand=0&rev=166 --- python38.spec | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/python38.spec b/python38.spec index 9698e2d..5b65516 100644 --- a/python38.spec +++ b/python38.spec @@ -36,7 +36,7 @@ %bcond_without general %endif -%if 0%{?do_profiling} +%if 0%{?do_profiling} && !0%{?qemu_user_space_build} %bcond_without profileopt %else %bcond_with profileopt From 17d5df370fae65216e72e7285dc3e1a22759cc6f0209d7f43a2d5a7c6d9a7588 Mon Sep 17 00:00:00 2001 From: Matej Cepl Date: Wed, 28 Aug 2024 16:55:40 +0000 Subject: [PATCH 2/2] - Add CVE-2024-8088-inf-loop-zipfile_Path.patch to prevent malformed payload to cause infinite loops in zipfile.Path (bsc#1229704, CVE-2024-8088). OBS-URL: https://build.opensuse.org/package/show/devel:languages:python:Factory/python38?expand=0&rev=167 --- CVE-2024-8088-inf-loop-zipfile_Path.patch | 142 ++++++++++++++++++++++ python38.changes | 7 ++ python38.spec | 4 + 3 files changed, 153 insertions(+) create mode 100644 CVE-2024-8088-inf-loop-zipfile_Path.patch diff --git a/CVE-2024-8088-inf-loop-zipfile_Path.patch b/CVE-2024-8088-inf-loop-zipfile_Path.patch new file mode 100644 index 0000000..7fdc649 --- /dev/null +++ b/CVE-2024-8088-inf-loop-zipfile_Path.patch @@ -0,0 +1,142 @@ +From dcb320a0c85713c5dfe89a83d6eb295ad1511be8 Mon Sep 17 00:00:00 2001 +From: "Jason R. Coombs" +Date: Tue, 27 Aug 2024 17:10:30 -0400 +Subject: [PATCH] [3.8] [3.9] [3.11] gh-123270: Replaced SanitizedNames with a + more surgical fix. (GH-123354) + +Applies changes from zipp 3.20.1 and jaraco/zippGH-124 +(cherry picked from commit 2231286d78d328c2f575e0b05b16fe447d1656d6) +(cherry picked from commit 17b77bb41409259bad1cd6c74761c18b6ab1e860) +(cherry picked from commit 66d3383) + +Co-authored-by: Jason R. Coombs +--- + Lib/test/test_zipfile.py | 75 ++++++++++ + Lib/zipfile.py | 9 - + Misc/NEWS.d/next/Library/2024-08-26-13-45-20.gh-issue-123270.gXHvNJ.rst | 3 + 3 files changed, 85 insertions(+), 2 deletions(-) + create mode 100644 Misc/NEWS.d/next/Library/2024-08-26-13-45-20.gh-issue-123270.gXHvNJ.rst + +Index: Python-3.8.19/Lib/test/test_zipfile.py +=================================================================== +--- Python-3.8.19.orig/Lib/test/test_zipfile.py ++++ Python-3.8.19/Lib/test/test_zipfile.py +@@ -3007,6 +3007,81 @@ class TestPath(unittest.TestCase): + data = ['/'.join(string.ascii_lowercase + str(n)) for n in range(10000)] + zipfile.CompleteDirs._implied_dirs(data) + ++ def test_malformed_paths(self): ++ """ ++ Path should handle malformed paths gracefully. ++ ++ Paths with leading slashes are not visible. ++ ++ Paths with dots are treated like regular files. ++ """ ++ data = io.BytesIO() ++ zf = zipfile.ZipFile(data, "w") ++ zf.writestr("../parent.txt", b"content") ++ zf.filename = '' ++ root = zipfile.Path(zf) ++ assert list(map(str, root.iterdir())) == ['../'] ++ assert root.joinpath('..').joinpath('parent.txt').read_bytes() == b'content' ++ ++ def test_unsupported_names(self): ++ """ ++ Path segments with special characters are readable. ++ ++ On some platforms or file systems, characters like ++ ``:`` and ``?`` are not allowed, but they are valid ++ in the zip file. ++ """ ++ data = io.BytesIO() ++ zf = zipfile.ZipFile(data, "w") ++ zf.writestr("path?", b"content") ++ zf.writestr("V: NMS.flac", b"fLaC...") ++ zf.filename = '' ++ root = zipfile.Path(zf) ++ contents = root.iterdir() ++ assert next(contents).name == 'path?' ++ assert next(contents).name == 'V: NMS.flac' ++ assert root.joinpath('V: NMS.flac').read_bytes() == b"fLaC..." ++ ++ def test_backslash_not_separator(self): ++ """ ++ In a zip file, backslashes are not separators. ++ """ ++ data = io.BytesIO() ++ zf = zipfile.ZipFile(data, "w") ++ zf.writestr(DirtyZipInfo.for_name("foo\\bar", zf), b"content") ++ zf.filename = '' ++ root = zipfile.Path(zf) ++ (first,) = root.iterdir() ++ assert not first.is_dir() ++ assert first.name == 'foo\\bar' ++ ++ ++class DirtyZipInfo(zipfile.ZipInfo): ++ """ ++ Bypass name sanitization. ++ """ ++ ++ def __init__(self, filename, *args, **kwargs): ++ super().__init__(filename, *args, **kwargs) ++ self.filename = filename ++ ++ @classmethod ++ def for_name(cls, name, archive): ++ """ ++ Construct the same way that ZipFile.writestr does. ++ ++ TODO: extract this functionality and re-use ++ """ ++ self = cls(filename=name, date_time=time.localtime(time.time())[:6]) ++ self.compress_type = archive.compression ++ self.compress_level = archive.compresslevel ++ if self.filename.endswith('/'): # pragma: no cover ++ self.external_attr = 0o40775 << 16 # drwxrwxr-x ++ self.external_attr |= 0x10 # MS-DOS directory flag ++ else: ++ self.external_attr = 0o600 << 16 # ?rw------- ++ return self ++ + + if __name__ == "__main__": + unittest.main() +Index: Python-3.8.19/Lib/zipfile.py +=================================================================== +--- Python-3.8.19.orig/Lib/zipfile.py ++++ Python-3.8.19/Lib/zipfile.py +@@ -2161,7 +2161,7 @@ def _parents(path): + def _ancestry(path): + """ + Given a path with elements separated by +- posixpath.sep, generate all elements of that path ++ posixpath.sep, generate all elements of that path. + + >>> list(_ancestry('b/d')) + ['b/d', 'b'] +@@ -2173,9 +2173,14 @@ def _ancestry(path): + ['b'] + >>> list(_ancestry('')) + [] ++ ++ Multiple separators are treated like a single. ++ ++ >>> list(_ancestry('//b//d///f//')) ++ ['//b//d///f', '//b//d', '//b'] + """ + path = path.rstrip(posixpath.sep) +- while path and path != posixpath.sep: ++ while path.rstrip(posixpath.sep): + yield path + path, tail = posixpath.split(path) + +Index: Python-3.8.19/Misc/NEWS.d/next/Library/2024-08-26-13-45-20.gh-issue-123270.gXHvNJ.rst +=================================================================== +--- /dev/null ++++ Python-3.8.19/Misc/NEWS.d/next/Library/2024-08-26-13-45-20.gh-issue-123270.gXHvNJ.rst +@@ -0,0 +1,3 @@ ++Applied a more surgical fix for malformed payloads in :class:`zipfile.Path` ++causing infinite loops (gh-122905) without breaking contents using ++legitimate characters. diff --git a/python38.changes b/python38.changes index 9a0d410..2472785 100644 --- a/python38.changes +++ b/python38.changes @@ -1,3 +1,10 @@ +------------------------------------------------------------------- +Wed Aug 28 16:54:34 UTC 2024 - Matej Cepl + +- Add CVE-2024-8088-inf-loop-zipfile_Path.patch to prevent + malformed payload to cause infinite loops in zipfile.Path + (bsc#1229704, CVE-2024-8088). + ------------------------------------------------------------------- Thu Aug 8 19:30:36 UTC 2024 - Matej Cepl diff --git a/python38.spec b/python38.spec index 5b65516..6e7c05b 100644 --- a/python38.spec +++ b/python38.spec @@ -207,6 +207,9 @@ Patch47: CVE-2024-6923-email-hdr-inject.patch # PATCH-FIX-UPSTREAM CVE-2024-5642-OpenSSL-API-buf-overread-NPN.patch bsc#1227233 mcepl@suse.com # Remove for support for anything but OpenSSL 1.1.1 or newer Patch48: CVE-2024-5642-OpenSSL-API-buf-overread-NPN.patch +# PATCH-FIX-UPSTREAM CVE-2024-8088-inf-loop-zipfile_Path.patch bsc#1229704 mcepl@suse.com +# avoid denial of service in zipfile +Patch49: CVE-2024-8088-inf-loop-zipfile_Path.patch BuildRequires: autoconf-archive BuildRequires: automake BuildRequires: fdupes @@ -484,6 +487,7 @@ other applications. %patch -p1 -P 46 %patch -p1 -P 47 %patch -p1 -P 48 +%patch -p1 -P 49 # drop Autoconf version requirement sed -i 's/^AC_PREREQ/dnl AC_PREREQ/' configure.ac