diff --git a/.github/workflows/wheels.yml b/.github/workflows/wheels.yml index 727852e..66d2a05 100644 --- a/.github/workflows/wheels.yml +++ b/.github/workflows/wheels.yml @@ -9,7 +9,7 @@ env: MACOSX_DEPLOYMENT_TARGET: "10.10" WHEEL_SDIR: wheelhouse CONFIG_PATH: pillow-avif-plugin/wheelbuild/config.sh - LIBAVIF_VERSION: 0.10.1 + LIBAVIF_VERSION: ee29bec775ab8e6d555f602775301c14302b96e7 jobs: build: @@ -159,7 +159,7 @@ jobs: strategy: fail-fast: false matrix: - python-version: ["3.7", "3.8", "3.9", "3.10", "3.11"] + python-version: ["3.7", "3.8", "3.9", "3.10", "3.11", "3.12"] architecture: ["x64"] include: - architecture: "x64" @@ -202,12 +202,12 @@ jobs: - name: Install dependencies id: install run: | - 7z x winbuild\depends\nasm-2.14.02-win64.zip "-o$env:RUNNER_WORKSPACE\" - echo "$env:RUNNER_WORKSPACE\nasm-2.14.02" >> $env:GITHUB_PATH + 7z x winbuild\depends\nasm-2.16.01-win64.zip "-o$env:RUNNER_WORKSPACE\" + echo "$env:RUNNER_WORKSPACE\nasm-2.16.01" >> $env:GITHUB_PATH - winbuild\depends\gs9533w32.exe /S - echo "C:\Program Files (x86)\gs\gs9.53.3\bin" >> $env:GITHUB_PATH + python -m pip install meson + # make cache key depend on VS version & "C:\Program Files (x86)\Microsoft Visual Studio\Installer\vswhere.exe" ` | find """catalog_buildVersion""" ` | ForEach-Object { $a = $_.split(" ")[1]; echo "vs=$a" >> $env:GITHUB_OUTPUT } @@ -224,7 +224,7 @@ jobs: - name: Prepare build if: steps.build-cache.outputs.cache-hit != 'true' run: | - & python.exe winbuild\build_prepare.py -v --python=$env:pythonLocation --srcdir + & python.exe winbuild\build_prepare.py -v shell: pwsh - name: Build dependencies / libjpeg-turbo @@ -248,21 +248,28 @@ jobs: run: python -m pip install meson shell: cmd + - name: Build dependencies / rav1e + if: steps.build-cache.outputs.cache-hit != 'true' + run: "& winbuild\\build\\build_dep_rav1e.cmd" + - name: Build dependencies / libavif if: steps.build-cache.outputs.cache-hit != 'true' run: "& winbuild\\build\\build_dep_libavif.cmd" - - name: Install dependencies / Pillow - run: "& winbuild\\build\\install_pillow.cmd" - # trim ~150MB x 9 - name: Optimize build cache if: steps.build-cache.outputs.cache-hit != 'true' run: rmdir /S /Q winbuild\build\src shell: cmd + - name: Install dependencies / Pillow + run: | + cmd /c "winbuild\build\build_env.cmd && $env:pythonLocation\python.exe -m pip install -v Pillow" + shell: pwsh + - name: Build pillow-avif-plugin - run: "& winbuild\\build\\build_pillow_avif_plugin.cmd install" + run: | + cmd /c "winbuild\build\build_env.cmd && $env:pythonLocation\python.exe -m pip install -v ." shell: pwsh # failing with PyPy3 @@ -305,15 +312,14 @@ jobs: id: wheel if: "github.event_name == 'push'" run: | - for /f "tokens=3 delims=/" %%a in ("${{ github.ref }}") do echo dist=dist-%%a >> %GITHUB_OUTPUT% - winbuild\\build\\build_pillow_avif_plugin.cmd bdist_wheel - shell: cmd + cmd /c "winbuild\build\build_env.cmd && $env:pythonLocation\python.exe -m pip wheel -v ." + shell: pwsh - uses: actions/upload-artifact@v3 if: "github.event_name == 'push'" with: name: wheels - path: dist\*.whl + path: "*.whl" success: needs: [build, windows] diff --git a/depends/install_libavif.sh b/depends/install_libavif.sh index 831c1d5..709e71f 100755 --- a/depends/install_libavif.sh +++ b/depends/install_libavif.sh @@ -20,10 +20,15 @@ PKGCONFIG=${PKGCONFIG:-pkg-config} export CFLAGS="-fPIC -O3 $CFLAGS" export CXXFLAGS="-fPIC -O3 $CXXFLAGS" +ARCHIVE="${LIBAVIF_VERSION}.tar.gz" +if [[ "$LIBAVIF_VERSION" == *"."* ]]; then + ARCHIVE="v${ARCHIVE}" +fi + echo "::group::Fetching libavif" mkdir -p libavif-$LIBAVIF_VERSION curl -sLo - \ - https://github.com/AOMediaCodec/libavif/archive/v$LIBAVIF_VERSION.tar.gz \ + https://github.com/AOMediaCodec/libavif/archive/$ARCHIVE \ | tar --strip-components=1 -C libavif-$LIBAVIF_VERSION -zxf - pushd libavif-$LIBAVIF_VERSION echo "::endgroup::" diff --git a/wheelbuild/config.sh b/wheelbuild/config.sh index 6571b70..18bcc31 100644 --- a/wheelbuild/config.sh +++ b/wheelbuild/config.sh @@ -4,11 +4,11 @@ set -eo pipefail CONFIG_DIR=$(abspath $(dirname "${BASH_SOURCE[0]}")) ARCHIVE_SDIR=pillow-avif-plugin-depends -LIBAVIF_VERSION=1.0.1 +LIBAVIF_VERSION=ee29bec775ab8e6d555f602775301c14302b96e7 AOM_VERSION=3.7.0 DAV1D_VERSION=1.2.1 SVT_AV1_VERSION=1.7.0 -RAV1E_VERSION=p20230911 +RAV1E_VERSION=p20231003 LIBWEBP_SHA=e2c85878f6a33f29948b43d3492d9cdaf801aa54 LIBYUV_SHA=464c51a0 CCACHE_VERSION=4.7.1 @@ -23,6 +23,11 @@ alias trace_off='{ set +x; } 2>/dev/null' alias trace_suppress='{ [[ $- =~ .*x.* ]] && trace_enabled=1 || trace_enabled=0; set +x; } 2>/dev/null' alias trace_restore='{ [ $trace_enabled -eq 1 ] && trace_on || trace_off; } 2>/dev/null' +if [ -n "$IS_MACOS" ] && [ -n "$MACOSX_DEPLOYMENT_TARGET" ]; then + CFLAGS="${CFLAGS} -mmacosx-version-min=$MACOSX_DEPLOYMENT_TARGET" + LDFLAGS="${LDFLAGS} -mmacosx-version-min=$MACOSX_DEPLOYMENT_TARGET" +fi + call_and_restore_trace() { local rc local force_trace @@ -222,6 +227,7 @@ function build_aom { (cd libaom-$AOM_VERSION/build/work \ && cmake \ -DCMAKE_BUILD_TYPE=Release \ + -DCONFIG_PIC=1 \ -DCMAKE_INSTALL_PREFIX="${BUILD_PREFIX}" \ -DCMAKE_INSTALL_LIBDIR=lib \ -DBUILD_SHARED_LIBS=0 \ @@ -345,8 +351,8 @@ function build_rav1e { fi curl -sLo - \ - https://github.com/fdintino/rav1e/releases/download/$RAV1E_VERSION/$librav1e_tgz \ - | tar -C $BUILD_PREFIX -zxf - + https://github.com/xiph/rav1e/releases/download/$RAV1E_VERSION/$librav1e_tgz \ + | tar -C $BUILD_PREFIX --exclude LICENSE -zxf - if [ ! -n "$IS_MACOS" ]; then sed -i 's/-lgcc_s/-lgcc_eh/g' "${BUILD_PREFIX}/lib/pkgconfig/rav1e.pc" @@ -467,14 +473,11 @@ function build_libavif { group_start "Download libavif source" fetch_unpack \ - "https://github.com/AOMediaCodec/libavif/archive/v$LIBAVIF_VERSION.tar.gz" \ + "https://github.com/AOMediaCodec/libavif/archive/$LIBAVIF_VERSION.tar.gz" \ "libavif-$LIBAVIF_VERSION.tar.gz" group_end - (cd libavif-$LIBAVIF_VERSION \ - && patch -p1 -i $CONFIG_DIR/libavif-1.0.1-local-static.patch) - build_libsharpyuv mv libwebp-$LIBWEBP_SHA libavif-$LIBAVIF_VERSION/ext/libwebp LIBAVIF_CMAKE_FLAGS+=(-DAVIF_LOCAL_LIBSHARPYUV=ON) @@ -489,11 +492,12 @@ function build_libavif { (cd libavif-$LIBAVIF_VERSION/build \ && cmake .. \ + -G "Ninja" \ -DCMAKE_INSTALL_PREFIX=$BUILD_PREFIX \ -DCMAKE_BUILD_TYPE=Release \ - -DCMAKE_VERBOSE_MAKEFILE:BOOL=ON \ + -DBUILD_SHARED_LIBS=OFF \ "${LIBAVIF_CMAKE_FLAGS[@]}" \ - && make install) + && ninja -v install/strip) group_end } diff --git a/winbuild/build_prepare.py b/winbuild/build_prepare.py index 5bbbe21..957a8c5 100644 --- a/winbuild/build_prepare.py +++ b/winbuild/build_prepare.py @@ -1,119 +1,137 @@ +from __future__ import annotations + +import argparse import os +import platform +import re import shutil import struct import subprocess -import sys -def cmd_cd(path): - return "cd /D {path}".format(path=path) +def cmd_cd(path: str) -> str: + return f"cd /D {path}" + + +def cmd_set(name: str, value: str) -> str: + return f"set {name}={value}" -def cmd_set(name, value): - return "set {name}={value}".format(name=name, value=value) +def cmd_append(name: str, value: str) -> str: + op = "path " if name == "PATH" else f"set {name}=" + return op + f"%{name}%;{value}" -def cmd_append(name, value): - op = "path " if name == "PATH" else "set {name}=".format(name=name) - return op + "%{name}%;{value}".format(name=name, value=value) +def cmd_copy(src: str, tgt: str) -> str: + return f'copy /Y /B "{src}" "{tgt}"' -def cmd_copy(src, tgt): - return 'copy /Y /B "{src}" "{tgt}"'.format(src=src, tgt=tgt) +def cmd_xcopy(src: str, tgt: str) -> str: + return f'xcopy /Y /E "{src}" "{tgt}"' -def cmd_mkdir(path): - return 'mkdir "{path}"'.format(path=path) +def cmd_mkdir(path: str) -> str: + return f'mkdir "{path}"' -def cmd_rmdir(path): - return 'rmdir /S /Q "{path}"'.format(path=path) +def cmd_rmdir(path: str) -> str: + return f'rmdir /S /Q "{path}"' -def cmd_lib_combine(outfile, *libfiles): +def cmd_lib_combine(outfile: str, *libfiles) -> str: params = " ".join(['"%s"' % f for f in libfiles]) return "LIB.EXE /OUT:{outfile} {params}".format(outfile=outfile, params=params) -def cmd_nmake(makefile=None, target="", params=None): - if params is None: - params = "" - elif isinstance(params, list) or isinstance(params, tuple): - params = " ".join(params) - else: - params = str(params) - - if makefile is not None: - makefile_arg = '-f "{makefile}"'.format(makefile=makefile) - else: - makefile_arg = "" +def cmd_nmake( + makefile: str | None = None, + target: str = "", + params: list[str] | None = None, +) -> str: + params = "" if params is None else " ".join(params) return " ".join( [ "{nmake}", "-nologo", - makefile_arg, - "{params}".format(params=params), - '"{target}"'.format(target=target), + f'-f "{makefile}"' if makefile is not None else "", + f"{params}", + f'"{target}"', ] ) -def cmd_cmake(params=None, file="."): - if params is None: - params = "" - elif isinstance(params, list) or isinstance(params, tuple): - params = " ".join(params) - else: - params = str(params) +def cmds_cmake(target: str | tuple[str, ...] | list[str], *params) -> list[str]: + if not isinstance(target, str): + target = " ".join(target) + + return [ + " ".join( + [ + "{cmake}", + "-DCMAKE_BUILD_TYPE=Release", + "-DCMAKE_VERBOSE_MAKEFILE=ON", + "-DCMAKE_RULE_MESSAGES:BOOL=OFF", # for NMake + "-DCMAKE_C_COMPILER=cl.exe", # for Ninja + "-DCMAKE_CXX_COMPILER=cl.exe", # for Ninja + "-DCMAKE_C_FLAGS=-nologo", + "-DCMAKE_CXX_FLAGS=-nologo", + *params, + '-G "{cmake_generator}"', + ".", + ] + ), + f"{{cmake}} --build . --clean-first --parallel --target {target}", + ] + + +def cmd_msbuild( + file: str, + configuration: str = "Release", + target: str = "Build", + platform: str = "{msbuild_arch}", +) -> str: return " ".join( [ - "{cmake}", - "-DCMAKE_VERBOSE_MAKEFILE=ON", - "-DCMAKE_RULE_MESSAGES:BOOL=OFF", - "-DCMAKE_BUILD_TYPE=Release", - "{params}".format(params=params), - '-G "NMake Makefiles"', - '"{file}"'.format(file=file), + "{msbuild}", + f"{file}", + f'/t:"{target}"', + f'/p:Configuration="{configuration}"', + f"/p:Platform={platform}", + "/m", ] ) SF_PROJECTS = "https://sourceforge.net/projects" -architectures = { +ARCHITECTURES = { "x86": {"vcvars_arch": "x86", "msbuild_arch": "Win32"}, "x64": {"vcvars_arch": "x86_amd64", "msbuild_arch": "x64"}, + "ARM64": {"vcvars_arch": "x86_arm64", "msbuild_arch": "ARM64"}, } -header = [ - cmd_set("INCLUDE", "{inc_dir}"), - cmd_set("INCLIB", "{lib_dir}"), - cmd_set("LIB", "{lib_dir}"), - cmd_append("PATH", "{bin_dir}"), -] - # dependencies, listed in order of compilation -deps = { +DEPS = { "libjpeg": { "url": SF_PROJECTS - + "/libjpeg-turbo/files/2.1.3/libjpeg-turbo-2.1.3.tar.gz/download", - "filename": "libjpeg-turbo-2.1.3.tar.gz", - "dir": "libjpeg-turbo-2.1.3", + + "/libjpeg-turbo/files/3.0.0/libjpeg-turbo-3.0.0.tar.gz/download", + "filename": "libjpeg-turbo-3.0.0.tar.gz", + "dir": "libjpeg-turbo-3.0.0", + "license": ["README.ijg", "LICENSE.md"], + "license_pattern": ( + "(LEGAL ISSUES\n============\n\n.+?)\n\nREFERENCES\n==========" + ".+(libjpeg-turbo Licenses\n======================\n\n.+)$" + ), "build": [ - cmd_cmake( - [ - "-DENABLE_SHARED:BOOL=FALSE", - "-DWITH_JPEG8:BOOL=TRUE", - "-DWITH_CRT_DLL:BOOL=TRUE", - ] + *cmds_cmake( + ("jpeg-static", "cjpeg-static", "djpeg-static"), + "-DENABLE_SHARED:BOOL=FALSE", + "-DWITH_JPEG8:BOOL=TRUE", + "-DWITH_CRT_DLL:BOOL=TRUE", ), - cmd_nmake(target="clean"), - cmd_nmake(target="jpeg-static"), cmd_copy("jpeg-static.lib", "libjpeg.lib"), - cmd_nmake(target="cjpeg-static"), cmd_copy("cjpeg-static.exe", "cjpeg.exe"), - cmd_nmake(target="djpeg-static"), cmd_copy("djpeg-static.exe", "djpeg.exe"), ], "headers": ["j*.h"], @@ -121,9 +139,11 @@ def cmd_cmake(params=None, file="."): "bins": ["cjpeg.exe", "djpeg.exe"], }, "zlib": { - "url": "http://zlib.net/zlib1213.zip", - "filename": "zlib1213.zip", - "dir": "zlib-1.2.13", + "url": "https://zlib.net/zlib13.zip", + "filename": "zlib13.zip", + "dir": "zlib-1.3", + "license": "README", + "license_pattern": "Copyright notice:\n\n(.+)$", "build": [ cmd_nmake(r"win32\Makefile.msc", "clean"), cmd_nmake(r"win32\Makefile.msc", "zlib.lib"), @@ -133,30 +153,42 @@ def cmd_cmake(params=None, file="."): "libs": [r"*.lib"], }, "libpng": { - "url": SF_PROJECTS + "/libpng/files/libpng16/1.6.37/lpng1637.zip/download", - "filename": "lpng1637.zip", - "dir": "lpng1637", + "url": SF_PROJECTS + "/libpng/files/libpng16/1.6.39/lpng1639.zip/download", + "filename": "lpng1639.zip", + "dir": "lpng1639", + "license": "LICENSE", "build": [ - # lint: do not inline - cmd_cmake(("-DPNG_SHARED:BOOL=OFF", "-DPNG_TESTS:BOOL=OFF")), - cmd_nmake(target="clean"), - cmd_nmake(), + *cmds_cmake("png_static", "-DPNG_SHARED:BOOL=OFF", "-DPNG_TESTS:BOOL=OFF"), cmd_copy("libpng16_static.lib", "libpng16.lib"), ], "headers": [r"png*.h"], "libs": [r"libpng16.lib"], }, + "rav1e": { + "url": ( + "https://github.com/xiph/rav1e/releases/download/p20231003/" + "rav1e-windows-msvc-generic.zip" + ), + "filename": "rav1e-windows-msvc-generic.zip", + "dir": "rav1e-windows-msvc-sdk", + "license": "LICENSE", + "build": [ + cmd_xcopy("include", "{inc_dir}"), + ], + "bins": [r"bin\*.dll"], + "libs": [r"lib\*.*"], + }, "libavif": { - "url": "https://github.com/AOMediaCodec/libavif/archive/v1.0.1.tar.gz", - "filename": "libavif-1.0.1.tar.gz", - "dir": "libavif-1.0.1", - "patch": { - "src/codec_aom.c": { - "if (aomCpuUsed >= 7)": "if (0)", - }, - }, + "url": ( + "https://github.com/AOMediaCodec/libavif/archive/" + "ee29bec775ab8e6d555f602775301c14302b96e7.zip" + ), + "filename": "libavif-ee29bec775ab8e6d555f602775301c14302b96e7.zip", + "dir": "libavif-ee29bec775ab8e6d555f602775301c14302b96e7", + "license": "LICENSE", "build": [ - cmd_append("PATH", r"{program_files}\Meson"), + cmd_mkdir(r"ext\rav1e\build.libavif\usr"), + cmd_xcopy(r"..\rav1e-windows-msvc-sdk", r"ext\rav1e\build.libavif\usr"), cmd_cd("ext"), "@echo ::group::Building SVT-AV1", cmd_rmdir("SVT-AV1"), @@ -170,43 +202,38 @@ def cmd_cmake(params=None, file="."): cmd_rmdir("dav1d"), 'cmd.exe /c "dav1d.cmd"', "@echo ::endgroup::", + "@echo ::group::Building libyuv", + cmd_rmdir("libyuv"), + 'cmd.exe /c "libyuv.cmd"', + "@echo ::endgroup::", + "@echo ::group::Building libsharpyuv", + cmd_rmdir("libsharpyuv"), + 'cmd.exe /c "libsharpyuv.cmd"', + "@echo ::endgroup::", "@echo ::group::Building libavif", cmd_cd(".."), - cmd_rmdir("build"), - cmd_mkdir("build"), - cmd_cd("build"), - cmd_cmake( - [ - "-DBUILD_SHARED_LIBS=OFF", - "-DAVIF_CODEC_AOM=ON", - "-DAVIF_LOCAL_AOM=ON", - "-DAVIF_CODEC_DAV1D=ON", - "-DAVIF_LOCAL_DAV1D=ON", - "-DAVIF_CODEC_SVT=ON", - "-DAVIF_LOCAL_SVT=ON", - ], - "..", - ), - cmd_nmake(), - cmd_cd(".."), - cmd_lib_combine( - r"avif.lib", - r"build\avif.lib", - r"ext\aom\build.libavif\aom.lib", - r"ext\dav1d\build\src\libdav1d.a", - r"ext\SVT-AV1\Bin\Release\SvtAv1Enc.lib", + *cmds_cmake( + "avif", + "-DBUILD_SHARED_LIBS=OFF", + "-DAVIF_CODEC_AOM=ON", + "-DAVIF_LOCAL_AOM=ON", + "-DAVIF_LOCAL_LIBYUV=ON", + "-DAVIF_LOCAL_LIBSHARPYUV=ON", + "-DAVIF_CODEC_RAV1E=ON", + "-DAVIF_CODEC_DAV1D=ON", + "-DAVIF_LOCAL_DAV1D=ON", + "-DAVIF_CODEC_SVT=ON", + "-DAVIF_LOCAL_SVT=ON", ), - cmd_mkdir(r"{inc_dir}\avif"), - cmd_copy(r"include\avif\avif.h", r"{inc_dir}\avif"), - "@echo ::endgroup::", + cmd_xcopy("include", "{inc_dir}"), ], - "libs": [r"*.lib"], + "libs": [r"avif.lib"], }, } # based on distutils._msvccompiler from CPython 3.7.4 -def find_msvs(): +def find_msvs() -> dict[str, str] | None: root = os.environ.get("ProgramFiles(x86)") or os.environ.get("ProgramFiles") if not root: print("Program Files not found") @@ -240,23 +267,12 @@ def find_msvs(): print("Visual Studio seems to be missing C compiler") return None - vs = { - "header": [], - # nmake selected by vcvarsall - "nmake": "nmake.exe", - "vs_dir": vspath, - } - # vs2017 msbuild = os.path.join(vspath, "MSBuild", "15.0", "Bin", "MSBuild.exe") - if os.path.isfile(msbuild): - vs["msbuild"] = '"{msbuild}"'.format(msbuild=msbuild) - else: + if not os.path.isfile(msbuild): # vs2019 msbuild = os.path.join(vspath, "MSBuild", "Current", "Bin", "MSBuild.exe") - if os.path.isfile(msbuild): - vs["msbuild"] = '"{msbuild}"'.format(msbuild=msbuild) - else: + if not os.path.isfile(msbuild): print("Visual Studio MSBuild not found") return None @@ -264,65 +280,87 @@ def find_msvs(): if not os.path.isfile(vcvarsall): print("Visual Studio vcvarsall not found") return None - vs["header"].append( - 'call "{vcvarsall}" {{vcvars_arch}}'.format(vcvarsall=vcvarsall) - ) - return vs + return { + "vs_dir": vspath, + "msbuild": f'"{msbuild}"', + "vcvarsall": f'"{vcvarsall}"', + "nmake": "nmake.exe", # nmake selected by vcvarsall + } + +def download_dep(url: str, file: str) -> None: + import urllib.request -def fetch(url, file): - try: - from urllib.request import urlopen - from urllib.error import URLError - except ImportError: - from urllib2 import urlopen - from urllib2 import URLError + ex = None + for i in range(3): + try: + print(f"Fetching {url} (attempt {i + 1})...") + content = urllib.request.urlopen(url).read() + with open(file, "wb") as f: + f.write(content) + break + except urllib.error.URLError as e: + ex = e + else: + raise RuntimeError(ex) - if not os.path.exists(file): - ex = None - for i in range(3): - try: - print("Fetching %s (attempt %d)..." % (url, i + 1)) - content = urlopen(url).read() - with open(file, "wb") as f: - f.write(content) - break - except URLError as e: - ex = e - else: - raise RuntimeError(ex) - - -def extract_dep(url, filename): + +def extract_dep(url: str, filename: str) -> None: import tarfile import zipfile - file = os.path.join(depends_dir, filename) - fetch(url, file) + file = os.path.join(args.depends_dir, filename) + if not os.path.exists(file): + # First try our mirror + mirror_url = ( + f"https://raw.githubusercontent.com/" + f"python-pillow/pillow-depends/main/{filename}" + ) + try: + download_dep(mirror_url, file) + except RuntimeError as exc: + # Otherwise try upstream + print(exc) + download_dep(url, file) + print("Extracting " + filename) + sources_dir_abs = os.path.abspath(sources_dir) if filename.endswith(".zip"): with zipfile.ZipFile(file) as zf: + for member in zf.namelist(): + member_abspath = os.path.abspath(os.path.join(sources_dir, member)) + member_prefix = os.path.commonpath([sources_dir_abs, member_abspath]) + if sources_dir_abs != member_prefix: + msg = "Attempted Path Traversal in Zip File" + raise RuntimeError(msg) zf.extractall(sources_dir) elif filename.endswith(".tar.gz") or filename.endswith(".tgz"): with tarfile.open(file, "r:gz") as tgz: + for member in tgz.getnames(): + member_abspath = os.path.abspath(os.path.join(sources_dir, member)) + member_prefix = os.path.commonpath([sources_dir_abs, member_abspath]) + if sources_dir_abs != member_prefix: + msg = "Attempted Path Traversal in Tar File" + raise RuntimeError(msg) tgz.extractall(sources_dir) else: - raise RuntimeError("Unknown archive type: " + filename) + msg = "Unknown archive type: " + filename + raise RuntimeError(msg) -def write_script(name, lines): - name = os.path.join(build_dir, name) +def write_script(name: str, lines: list[str]) -> None: + name = os.path.join(args.build_dir, name) lines = [line.format(**prefs) for line in lines] print("Writing " + name) - with open(name, "w") as f: - f.write("\n\r".join(lines)) - if verbose: + with open(name, "w", newline="") as f: + f.write(os.linesep.join(lines)) + if args.verbose: for line in lines: print(" " + line) -def get_footer(dep): +def get_footer(dep: dict) -> list[str]: lines = [] for out in dep.get("headers", []): lines.append(cmd_copy(out, "{inc_dir}")) @@ -333,13 +371,45 @@ def get_footer(dep): return lines -def build_dep(name): - dep = deps[name] +def build_env() -> None: + lines = [ + "if defined DISTUTILS_USE_SDK goto end", + cmd_set("INCLUDE", "{inc_dir}"), + cmd_set("INCLIB", "{lib_dir}"), + cmd_set("LIB", "{lib_dir}"), + cmd_append("PATH", "{bin_dir}"), + "call {vcvarsall} {vcvars_arch}", + cmd_set("DISTUTILS_USE_SDK", "1"), # use same compiler to build Pillow + cmd_set("py_vcruntime_redist", "true"), # always use /MD, never /MT + ":end", + "@echo on", + ] + write_script("build_env.cmd", lines) + + +def build_dep(name: str) -> str: + dep = DEPS[name] dir = dep["dir"] - file = "build_dep_{name}.cmd".format(name=name) + file = f"build_dep_{name}.cmd" extract_dep(dep["url"], dep["filename"]) + licenses = dep["license"] + if isinstance(licenses, str): + licenses = [licenses] + license_text = "" + for license_file in licenses: + with open(os.path.join(sources_dir, dir, license_file)) as f: + license_text += f.read() + if "license_pattern" in dep: + match = re.search(dep["license_pattern"], license_text, re.DOTALL) + license_text = "\n".join(match.groups()) + if licenses: + assert len(license_text) > 50 + with open(os.path.join(license_dir, f"{dir}.txt"), "w") as f: + print(f"Writing license {dir}.txt") + f.write(license_text) + for patch_file, patch_list in dep.get("patch", {}).items(): patch_file = os.path.join(sources_dir, dir, patch_file.format(**prefs)) with open(patch_file) as f: @@ -350,53 +420,43 @@ def build_dep(name): assert patch_from in text text = text.replace(patch_from, patch_to) with open(patch_file, "w") as f: + print(f"Patching {patch_file}") f.write(text) - banner = "Building {name} ({dir})".format(name=name, dir=dir) + banner = f"Building {name} ({dir})" lines = [ + r'call "{build_dir}\build_env.cmd"', "@echo " + ("=" * 70), - "@echo ==== {banner:<60} ====".format(banner=banner), + f"@echo ==== {banner:<60} ====", "@echo " + ("=" * 70), - "cd /D %s" % os.path.join(sources_dir, dir), + cmd_cd(os.path.join(sources_dir, dir)), + *dep.get("build", []), + *get_footer(dep), ] - lines += prefs["header"] - lines += dep.get("build", []) - lines += get_footer(dep) + write_script(file, lines) return file -def build_dep_all(): - lines = ["@echo on"] - for dep_name in deps: +def build_dep_all() -> None: + lines = [r'call "{build_dir}\build_env.cmd"'] + for dep_name in DEPS: + print() if dep_name in disabled: + print(f"Skipping disabled dependency {dep_name}") continue script = build_dep(dep_name) - lines.append( - r'cmd.exe /c "{{build_dir}}\{script}"'.format( # noqa: F522 - build_dir=build_dir, - script=script, - ) - ) + lines.append(rf'cmd.exe /c "{{build_dir}}\{script}"') lines.append("if errorlevel 1 echo Build failed! && exit /B 1") - lines.append("@echo All pillow-avif-plugin dependencies built successfully!") + print() + lines.append("@echo All Pillow dependencies built successfully!") write_script("build_dep_all.cmd", lines) -def install_pillow(): - lines = [ - "@echo on", - "@echo ---- Installing pillow ----", - r'"{python_dir}\{python_exe}" -m pip install Pillow', - "@echo Pillow installed successfully", - ] - write_script("install_pillow.cmd", lines) - - def install_meson(): msi_url = "https://github.com/mesonbuild/meson/releases/download/0.56.2/meson-0.56.2-64.msi" # noqa: E501 - msi_file = os.path.join(depends_dir, "meson-0.56.2-64.msi") - fetch(msi_url, msi_file) + msi_file = os.path.join(args.depends_dir, "meson-0.56.2-64.msi") + download_dep(msi_url, msi_file) lines = [ "@echo on", @@ -407,126 +467,121 @@ def install_meson(): write_script("install_meson.cmd", lines) -def build_pillow_avif_plugin(): - lines = ( - [ - "@echo ---- Building pillow-avif-plugin (build_ext %*) ----", - cmd_cd("{pillow_avif_plugin_dir}"), - ] - + prefs["header"] - + [ - cmd_set("DISTUTILS_USE_SDK", "1"), # use same compiler to build pillow-avif - cmd_set("MSSdk", "1"), # for PyPy3.6 - cmd_set("py_vcruntime_redist", "true"), # use /MD, not /MT - r'"{python_dir}\{python_exe}" setup.py build_ext %*', - ] - ) - - write_script("build_pillow_avif_plugin.cmd", lines) - - if __name__ == "__main__": - # winbuild directory winbuild_dir = os.path.dirname(os.path.realpath(__file__)) - - verbose = False - disabled = [] - depends_dir = os.environ.get("PILLOW_DEPS", os.path.join(winbuild_dir, "depends")) - python_dir = os.environ.get("PYTHON") - python_exe = os.environ.get("EXECUTABLE", "python.exe") - architecture = os.environ.get( - "ARCHITECTURE", "x86" if struct.calcsize("P") == 4 else "x64" + pillow_dir = os.path.realpath(os.path.join(winbuild_dir, "..")) + + parser = argparse.ArgumentParser( + prog="winbuild\\build_prepare.py", + description="Download and generate build scripts for Pillow dependencies.", + epilog="""Arguments can also be supplied using the environment variables + PILLOW_BUILD, PILLOW_DEPS, ARCHITECTURE. See winbuild\\build.rst + for more information.""", ) - build_dir = os.environ.get("PILLOW_BUILD", os.path.join(winbuild_dir, "build")) - sources_dir = "" - for arg in sys.argv[1:]: - if arg == "-v": - verbose = True - elif arg.startswith("--depends="): - depends_dir = arg[10:] - elif arg.startswith("--python="): - python_dir = arg[9:] - elif arg.startswith("--executable="): - python_exe = arg[13:] - elif arg.startswith("--architecture="): - architecture = arg[15:] - elif arg.startswith("--dir="): - build_dir = arg[6:] - elif arg == "--srcdir": - sources_dir = os.path.sep + "src" - else: - raise ValueError("Unknown parameter: " + arg) - - # dependency cache directory - if not os.path.exists(depends_dir): - os.makedirs(depends_dir) - print("Caching dependencies in:", depends_dir) - - if python_dir is None: - python_dir = os.path.dirname(os.path.realpath(sys.executable)) - python_exe = os.path.basename(sys.executable) - print("Target Python:", os.path.join(python_dir, python_exe)) + parser.add_argument( + "-v", "--verbose", action="store_true", help="print generated scripts" + ) + parser.add_argument( + "-d", + "--dir", + "--build-dir", + dest="build_dir", + metavar="PILLOW_BUILD", + default=os.environ.get("PILLOW_BUILD", os.path.join(winbuild_dir, "build")), + help="build directory (default: 'winbuild\\build')", + ) + parser.add_argument( + "--depends", + dest="depends_dir", + metavar="PILLOW_DEPS", + default=os.environ.get("PILLOW_DEPS", os.path.join(winbuild_dir, "depends")), + help="directory used to store cached dependencies " + "(default: 'winbuild\\depends')", + ) + parser.add_argument( + "--architecture", + choices=ARCHITECTURES, + default=os.environ.get( + "ARCHITECTURE", + ( + "ARM64" + if platform.machine() == "ARM64" + else ("x86" if struct.calcsize("P") == 4 else "x64") + ), + ), + help="build architecture (default: same as host Python)", + ) + parser.add_argument( + "--nmake", + dest="cmake_generator", + action="store_const", + const="NMake Makefiles", + default="Ninja", + help="build dependencies using NMake instead of Ninja", + ) + args = parser.parse_args() - arch_prefs = architectures[architecture] - print("Target Architecture:", architecture) + arch_prefs = ARCHITECTURES[args.architecture] + print("Target architecture:", args.architecture) msvs = find_msvs() if msvs is None: - raise RuntimeError( - "Visual Studio not found. Please install Visual Studio 2017 or newer." - ) + msg = "Visual Studio not found. Please install Visual Studio 2017 or newer." + raise RuntimeError(msg) print("Found Visual Studio at:", msvs["vs_dir"]) - print("Using output directory:", build_dir) + # dependency cache directory + args.depends_dir = os.path.abspath(args.depends_dir) + os.makedirs(args.depends_dir, exist_ok=True) + print("Caching dependencies in:", args.depends_dir) + + args.build_dir = os.path.abspath(args.build_dir) + print("Using output directory:", args.build_dir) # build directory for *.h files - inc_dir = os.path.join(build_dir, "inc") + inc_dir = os.path.join(args.build_dir, "inc") # build directory for *.lib files - lib_dir = os.path.join(build_dir, "lib") + lib_dir = os.path.join(args.build_dir, "lib") # build directory for *.bin files - bin_dir = os.path.join(build_dir, "bin") + bin_dir = os.path.join(args.build_dir, "bin") # directory for storing project files - sources_dir = build_dir + sources_dir + sources_dir = os.path.join(args.build_dir, "src") + # copy dependency licenses to this directory + license_dir = os.path.join(args.build_dir, "license") - shutil.rmtree(build_dir, ignore_errors=True) - if not os.path.exists(build_dir): - os.makedirs(build_dir) - for path in [inc_dir, lib_dir, bin_dir, sources_dir]: - if not os.path.exists(path): - os.makedirs(path) + shutil.rmtree(args.build_dir, ignore_errors=True) + os.makedirs(args.build_dir, exist_ok=False) + for path in [inc_dir, lib_dir, bin_dir, sources_dir, license_dir]: + os.makedirs(path, exist_ok=True) + + disabled = [] prefs = { - # Python paths / preferences - "python_dir": python_dir, - "python_exe": python_exe, - "architecture": architecture, + "architecture": args.architecture, + **arch_prefs, # Pillow paths - "pillow_avif_plugin_dir": os.path.realpath(os.path.join(winbuild_dir, "..")), + "pillow_dir": pillow_dir, "winbuild_dir": winbuild_dir, # Build paths - "build_dir": build_dir, + "build_dir": args.build_dir, "inc_dir": inc_dir, "lib_dir": lib_dir, "bin_dir": bin_dir, "src_dir": sources_dir, - "program_files": os.environ["ProgramFiles"], + "license_dir": license_dir, # Compilers / Tools + **msvs, "cmake": "cmake.exe", # TODO find CMAKE automatically + "cmake_generator": args.cmake_generator, # TODO find NASM automatically } - prefs.update(arch_prefs) - prefs.update(msvs) - - # script header - prefs["header"] = sum([header, msvs["header"], ["@echo on"]], []) - for k, v in deps.items(): - prefs["dir_%s" % k] = os.path.join(sources_dir, v["dir"]) + for k, v in DEPS.items(): + prefs[f"dir_{k}"] = os.path.join(sources_dir, v["dir"]) print() write_script(".gitignore", ["*"]) - build_dep_all() - install_pillow() + build_env() install_meson() - build_pillow_avif_plugin() + build_dep_all()