mirror of
				https://github.com/python-pillow/Pillow.git
				synced 2025-10-24 20:51:16 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			1119 lines
		
	
	
		
			40 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			1119 lines
		
	
	
		
			40 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # > pyroma .
 | |
| # ------------------------------
 | |
| # Checking .
 | |
| # Found Pillow
 | |
| # ------------------------------
 | |
| # Final rating: 10/10
 | |
| # Your cheese is so fresh most people think it's a cream: Mascarpone
 | |
| # ------------------------------
 | |
| from __future__ import annotations
 | |
| 
 | |
| import os
 | |
| import re
 | |
| import shutil
 | |
| import struct
 | |
| import subprocess
 | |
| import sys
 | |
| import warnings
 | |
| from collections.abc import Iterator
 | |
| 
 | |
| from pybind11.setup_helpers import ParallelCompile
 | |
| from setuptools import Extension, setup
 | |
| from setuptools.command.build_ext import build_ext
 | |
| 
 | |
| configuration: dict[str, list[str]] = {}
 | |
| 
 | |
| # parse configuration from _custom_build/backend.py
 | |
| while sys.argv[-1].startswith("--pillow-configuration="):
 | |
|     _, key, value = sys.argv.pop().split("=", 2)
 | |
|     configuration.setdefault(key, []).append(value)
 | |
| 
 | |
| default = int(configuration.get("parallel", ["0"])[-1])
 | |
| ParallelCompile("MAX_CONCURRENCY", default).install()
 | |
| 
 | |
| 
 | |
| def get_version() -> str:
 | |
|     version_file = "src/PIL/_version.py"
 | |
|     with open(version_file, encoding="utf-8") as f:
 | |
|         return f.read().split('"')[1]
 | |
| 
 | |
| 
 | |
| PILLOW_VERSION = get_version()
 | |
| AVIF_ROOT = None
 | |
| FREETYPE_ROOT = None
 | |
| HARFBUZZ_ROOT = None
 | |
| FRIBIDI_ROOT = None
 | |
| IMAGEQUANT_ROOT = None
 | |
| JPEG2K_ROOT = None
 | |
| JPEG_ROOT = None
 | |
| LCMS_ROOT = None
 | |
| RAQM_ROOT = None
 | |
| TIFF_ROOT = None
 | |
| WEBP_ROOT = None
 | |
| ZLIB_ROOT = None
 | |
| FUZZING_BUILD = "LIB_FUZZING_ENGINE" in os.environ
 | |
| 
 | |
| if sys.platform == "win32" and sys.version_info >= (3, 15):
 | |
|     import atexit
 | |
| 
 | |
|     atexit.register(
 | |
|         lambda: warnings.warn(
 | |
|             f"Pillow {PILLOW_VERSION} does not support Python "
 | |
|             f"{sys.version_info.major}.{sys.version_info.minor} and does not provide "
 | |
|             "prebuilt Windows binaries. We do not recommend building from source on "
 | |
|             "Windows.",
 | |
|             RuntimeWarning,
 | |
|         )
 | |
|     )
 | |
| 
 | |
| 
 | |
| _IMAGING = ("decode", "encode", "map", "display", "outline", "path")
 | |
| 
 | |
| _LIB_IMAGING = (
 | |
|     "Access",
 | |
|     "AlphaComposite",
 | |
|     "Arrow",
 | |
|     "Resample",
 | |
|     "Reduce",
 | |
|     "Bands",
 | |
|     "BcnDecode",
 | |
|     "BcnEncode",
 | |
|     "BitDecode",
 | |
|     "Blend",
 | |
|     "Chops",
 | |
|     "ColorLUT",
 | |
|     "Convert",
 | |
|     "ConvertYCbCr",
 | |
|     "Copy",
 | |
|     "Crop",
 | |
|     "Dib",
 | |
|     "Draw",
 | |
|     "Effects",
 | |
|     "EpsEncode",
 | |
|     "File",
 | |
|     "Fill",
 | |
|     "Filter",
 | |
|     "FliDecode",
 | |
|     "Geometry",
 | |
|     "GetBBox",
 | |
|     "GifDecode",
 | |
|     "GifEncode",
 | |
|     "HexDecode",
 | |
|     "Histo",
 | |
|     "JpegDecode",
 | |
|     "JpegEncode",
 | |
|     "Matrix",
 | |
|     "ModeFilter",
 | |
|     "Negative",
 | |
|     "Offset",
 | |
|     "Pack",
 | |
|     "PackDecode",
 | |
|     "Palette",
 | |
|     "Paste",
 | |
|     "Quant",
 | |
|     "QuantOctree",
 | |
|     "QuantHash",
 | |
|     "QuantHeap",
 | |
|     "PcdDecode",
 | |
|     "PcxDecode",
 | |
|     "PcxEncode",
 | |
|     "Point",
 | |
|     "RankFilter",
 | |
|     "RawDecode",
 | |
|     "RawEncode",
 | |
|     "Storage",
 | |
|     "SgiRleDecode",
 | |
|     "SunRleDecode",
 | |
|     "TgaRleDecode",
 | |
|     "TgaRleEncode",
 | |
|     "Unpack",
 | |
|     "UnpackYCC",
 | |
|     "UnsharpMask",
 | |
|     "XbmDecode",
 | |
|     "XbmEncode",
 | |
|     "ZipDecode",
 | |
|     "ZipEncode",
 | |
|     "TiffDecode",
 | |
|     "Jpeg2KDecode",
 | |
|     "Jpeg2KEncode",
 | |
|     "BoxBlur",
 | |
|     "QuantPngQuant",
 | |
|     "codec_fd",
 | |
| )
 | |
| 
 | |
| DEBUG = False
 | |
| 
 | |
| 
 | |
| class DependencyException(Exception):
 | |
|     pass
 | |
| 
 | |
| 
 | |
| class RequiredDependencyException(Exception):
 | |
|     pass
 | |
| 
 | |
| 
 | |
| PLATFORM_MINGW = os.name == "nt" and "GCC" in sys.version
 | |
| 
 | |
| 
 | |
| def _dbg(s: str, tp: str | tuple[str, ...] | None = None) -> None:
 | |
|     if DEBUG:
 | |
|         if tp:
 | |
|             print(s % tp)
 | |
|             return
 | |
|         print(s)
 | |
| 
 | |
| 
 | |
| def _find_library_dirs_ldconfig() -> list[str]:
 | |
|     # Based on ctypes.util from Python 2
 | |
| 
 | |
|     ldconfig = "ldconfig" if shutil.which("ldconfig") else "/sbin/ldconfig"
 | |
|     args: list[str]
 | |
|     env: dict[str, str]
 | |
|     expr: str
 | |
|     if sys.platform.startswith(("linux", "gnu")):
 | |
|         if struct.calcsize("l") == 4:
 | |
|             machine = os.uname()[4] + "-32"
 | |
|         else:
 | |
|             machine = os.uname()[4] + "-64"
 | |
|         mach_map = {
 | |
|             "x86_64-64": "libc6,x86-64",
 | |
|             "ppc64-64": "libc6,64bit",
 | |
|             "sparc64-64": "libc6,64bit",
 | |
|             "s390x-64": "libc6,64bit",
 | |
|             "ia64-64": "libc6,IA-64",
 | |
|         }
 | |
|         abi_type = mach_map.get(machine, "libc6")
 | |
| 
 | |
|         # Assuming GLIBC's ldconfig (with option -p)
 | |
|         # Alpine Linux uses musl that can't print cache
 | |
|         args = [ldconfig, "-p"]
 | |
|         expr = rf".*\({abi_type}.*\) => (.*)"
 | |
|         env = dict(os.environ)
 | |
|         env["LC_ALL"] = "C"
 | |
|         env["LANG"] = "C"
 | |
| 
 | |
|     elif sys.platform.startswith("freebsd"):
 | |
|         args = [ldconfig, "-r"]
 | |
|         expr = r".* => (.*)"
 | |
|         env = {}
 | |
| 
 | |
|     try:
 | |
|         p = subprocess.Popen(
 | |
|             args, stderr=subprocess.DEVNULL, stdout=subprocess.PIPE, env=env, text=True
 | |
|         )
 | |
|     except OSError:  # E.g. command not found
 | |
|         return []
 | |
|     data = p.communicate()[0]
 | |
| 
 | |
|     dirs = []
 | |
|     for dll in re.findall(expr, data):
 | |
|         dir = os.path.dirname(dll)
 | |
|         if dir not in dirs:
 | |
|             dirs.append(dir)
 | |
|     return dirs
 | |
| 
 | |
| 
 | |
| def _add_directory(
 | |
|     path: list[str], subdir: str | None, where: int | None = None
 | |
| ) -> None:
 | |
|     if subdir is None:
 | |
|         return
 | |
|     subdir = os.path.realpath(subdir)
 | |
|     if os.path.isdir(subdir) and subdir not in path:
 | |
|         if where is None:
 | |
|             _dbg("Appending path %s", subdir)
 | |
|             path.append(subdir)
 | |
|         else:
 | |
|             _dbg("Inserting path %s", subdir)
 | |
|             path.insert(where, subdir)
 | |
|     elif subdir in path and where is not None:
 | |
|         path.remove(subdir)
 | |
|         path.insert(where, subdir)
 | |
| 
 | |
| 
 | |
| def _find_include_file(self: pil_build_ext, include: str) -> str | None:
 | |
|     for directory in self.compiler.include_dirs:
 | |
|         _dbg("Checking for include file %s in %s", (include, directory))
 | |
|         path = os.path.join(directory, include)
 | |
|         if os.path.isfile(path):
 | |
|             _dbg("Found %s", include)
 | |
|             return path
 | |
|     return None
 | |
| 
 | |
| 
 | |
| def _find_library_file(self: pil_build_ext, library: str) -> str | None:
 | |
|     ret = self.compiler.find_library_file(self.compiler.library_dirs, library)
 | |
|     if ret:
 | |
|         _dbg("Found library %s at %s", (library, ret))
 | |
|     else:
 | |
|         _dbg("Couldn't find library %s in %s", (library, self.compiler.library_dirs))
 | |
|     return ret
 | |
| 
 | |
| 
 | |
| def _find_include_dir(self: pil_build_ext, dirname: str, include: str) -> bool | str:
 | |
|     for directory in self.compiler.include_dirs:
 | |
|         _dbg("Checking for include file %s in %s", (include, directory))
 | |
|         if os.path.isfile(os.path.join(directory, include)):
 | |
|             _dbg("Found %s in %s", (include, directory))
 | |
|             return True
 | |
|         subdir = os.path.join(directory, dirname)
 | |
|         _dbg("Checking for include file %s in %s", (include, subdir))
 | |
|         if os.path.isfile(os.path.join(subdir, include)):
 | |
|             _dbg("Found %s in %s", (include, subdir))
 | |
|             return subdir
 | |
|     return False
 | |
| 
 | |
| 
 | |
| def _cmd_exists(cmd: str) -> bool:
 | |
|     if "PATH" not in os.environ:
 | |
|         return False
 | |
|     return any(
 | |
|         os.access(os.path.join(path, cmd), os.X_OK)
 | |
|         for path in os.environ["PATH"].split(os.pathsep)
 | |
|     )
 | |
| 
 | |
| 
 | |
| def _pkg_config(name: str) -> tuple[list[str], list[str]] | None:
 | |
|     command = os.environ.get("PKG_CONFIG", "pkg-config")
 | |
|     for keep_system in (True, False):
 | |
|         try:
 | |
|             command_libs = [command, "--libs-only-L", name]
 | |
|             command_cflags = [command, "--cflags-only-I", name]
 | |
|             stderr = None
 | |
|             if keep_system:
 | |
|                 command_libs.append("--keep-system-libs")
 | |
|                 command_cflags.append("--keep-system-cflags")
 | |
|                 stderr = subprocess.DEVNULL
 | |
|             if not DEBUG:
 | |
|                 command_libs.append("--silence-errors")
 | |
|                 command_cflags.append("--silence-errors")
 | |
|             libs = re.split(
 | |
|                 r"(^|\s+)-L",
 | |
|                 subprocess.check_output(command_libs, stderr=stderr)
 | |
|                 .decode("utf8")
 | |
|                 .strip(),
 | |
|             )[::2][1:]
 | |
|             cflags = re.split(
 | |
|                 r"(^|\s+)-I",
 | |
|                 subprocess.check_output(command_cflags).decode("utf8").strip(),
 | |
|             )[::2][1:]
 | |
|             return libs, cflags
 | |
|         except Exception:
 | |
|             pass
 | |
|     return None
 | |
| 
 | |
| 
 | |
| class pil_build_ext(build_ext):
 | |
|     class ext_feature:
 | |
|         features = [
 | |
|             "zlib",
 | |
|             "jpeg",
 | |
|             "tiff",
 | |
|             "freetype",
 | |
|             "raqm",
 | |
|             "lcms",
 | |
|             "webp",
 | |
|             "jpeg2000",
 | |
|             "imagequant",
 | |
|             "xcb",
 | |
|             "avif",
 | |
|         ]
 | |
| 
 | |
|         required = {"jpeg", "zlib"}
 | |
|         vendor: set[str] = set()
 | |
| 
 | |
|         def __init__(self) -> None:
 | |
|             self._settings: dict[str, str | bool | None] = {}
 | |
|             for f in self.features:
 | |
|                 self.set(f, None)
 | |
| 
 | |
|         def require(self, feat: str) -> bool:
 | |
|             return feat in self.required
 | |
| 
 | |
|         def get(self, feat: str) -> str | bool | None:
 | |
|             return self._settings[feat]
 | |
| 
 | |
|         def set(self, feat: str, value: str | bool | None) -> None:
 | |
|             self._settings[feat] = value
 | |
| 
 | |
|         def want(self, feat: str) -> bool:
 | |
|             return self._settings[feat] is None
 | |
| 
 | |
|         def want_vendor(self, feat: str) -> bool:
 | |
|             return feat in self.vendor
 | |
| 
 | |
|         def __iter__(self) -> Iterator[str]:
 | |
|             yield from self.features
 | |
| 
 | |
|     feature = ext_feature()
 | |
| 
 | |
|     user_options = (
 | |
|         build_ext.user_options
 | |
|         + [(f"disable-{x}", None, f"Disable support for {x}") for x in feature]
 | |
|         + [(f"enable-{x}", None, f"Enable support for {x}") for x in feature]
 | |
|         + [
 | |
|             (f"vendor-{x}", None, f"Use vendored version of {x}")
 | |
|             for x in ("raqm", "fribidi")
 | |
|         ]
 | |
|         + [
 | |
|             ("disable-platform-guessing", None, "Disable platform guessing"),
 | |
|             ("debug", None, "Debug logging"),
 | |
|         ]
 | |
|         + [("add-imaging-libs=", None, "Add libs to _imaging build")]
 | |
|     )
 | |
| 
 | |
|     @staticmethod
 | |
|     def check_configuration(option: str, value: str) -> bool | None:
 | |
|         return True if value in configuration.get(option, []) else None
 | |
| 
 | |
|     def initialize_options(self) -> None:
 | |
|         self.disable_platform_guessing = self.check_configuration(
 | |
|             "platform-guessing", "disable"
 | |
|         )
 | |
|         self.add_imaging_libs = ""
 | |
|         build_ext.initialize_options(self)
 | |
|         for x in self.feature:
 | |
|             setattr(self, f"disable_{x}", self.check_configuration(x, "disable"))
 | |
|             setattr(self, f"enable_{x}", self.check_configuration(x, "enable"))
 | |
|         for x in ("raqm", "fribidi"):
 | |
|             setattr(self, f"vendor_{x}", self.check_configuration(x, "vendor"))
 | |
|         if self.check_configuration("debug", "true"):
 | |
|             self.debug = True
 | |
|         self.parallel = configuration.get("parallel", [None])[-1]
 | |
| 
 | |
|     def finalize_options(self) -> None:
 | |
|         build_ext.finalize_options(self)
 | |
|         if self.debug:
 | |
|             global DEBUG
 | |
|             DEBUG = True
 | |
|         if not self.parallel:
 | |
|             # If --parallel (or -j) wasn't specified, we want to reproduce the same
 | |
|             # behavior as before, that is, auto-detect the number of jobs.
 | |
|             self.parallel = None
 | |
| 
 | |
|             cpu_count = os.cpu_count()
 | |
|             if cpu_count is not None:
 | |
|                 try:
 | |
|                     self.parallel = int(os.environ.get("MAX_CONCURRENCY", cpu_count))
 | |
|                 except TypeError:
 | |
|                     pass
 | |
|         for x in self.feature:
 | |
|             if getattr(self, f"disable_{x}"):
 | |
|                 self.feature.set(x, False)
 | |
|                 self.feature.required.discard(x)
 | |
|                 _dbg("Disabling %s", x)
 | |
|                 if getattr(self, f"enable_{x}"):
 | |
|                     msg = f"Conflicting options: '-C {x}=enable' and '-C {x}=disable'"
 | |
|                     raise ValueError(msg)
 | |
|                 if x == "freetype":
 | |
|                     _dbg("'-C freetype=disable' implies '-C raqm=disable'")
 | |
|                     if getattr(self, "enable_raqm"):
 | |
|                         msg = (
 | |
|                             "Conflicting options: "
 | |
|                             "'-C raqm=enable' and '-C freetype=disable'"
 | |
|                         )
 | |
|                         raise ValueError(msg)
 | |
|                     setattr(self, "disable_raqm", True)
 | |
|             if getattr(self, f"enable_{x}"):
 | |
|                 _dbg("Requiring %s", x)
 | |
|                 self.feature.required.add(x)
 | |
|                 if x == "raqm":
 | |
|                     _dbg("'-C raqm=enable' implies '-C freetype=enable'")
 | |
|                     self.feature.required.add("freetype")
 | |
|         for x in ("raqm", "fribidi"):
 | |
|             if getattr(self, f"vendor_{x}"):
 | |
|                 if getattr(self, "disable_raqm"):
 | |
|                     msg = f"Conflicting options: '-C {x}=vendor' and '-C raqm=disable'"
 | |
|                     raise ValueError(msg)
 | |
|                 if x == "fribidi" and not getattr(self, "vendor_raqm"):
 | |
|                     msg = (
 | |
|                         f"Conflicting options: '-C {x}=vendor' and not '-C raqm=vendor'"
 | |
|                     )
 | |
|                     raise ValueError(msg)
 | |
|                 _dbg("Using vendored version of %s", x)
 | |
|                 self.feature.vendor.add(x)
 | |
| 
 | |
|     def _update_extension(
 | |
|         self,
 | |
|         name: str,
 | |
|         libraries: list[str] | list[str | bool | None],
 | |
|         define_macros: list[tuple[str, str | None]] | None = None,
 | |
|         sources: list[str] | None = None,
 | |
|     ) -> None:
 | |
|         for extension in self.extensions:
 | |
|             if extension.name == name:
 | |
|                 extension.libraries += libraries
 | |
|                 if define_macros is not None:
 | |
|                     extension.define_macros += define_macros
 | |
|                 if sources is not None:
 | |
|                     extension.sources += sources
 | |
|                 if FUZZING_BUILD:
 | |
|                     extension.language = "c++"
 | |
|                     extension.extra_link_args = ["--stdlib=libc++"]
 | |
|                 break
 | |
| 
 | |
|     def _remove_extension(self, name: str) -> None:
 | |
|         for extension in self.extensions:
 | |
|             if extension.name == name:
 | |
|                 self.extensions.remove(extension)
 | |
|                 break
 | |
| 
 | |
|     def get_macos_sdk_path(self) -> str | None:
 | |
|         try:
 | |
|             sdk_path = (
 | |
|                 subprocess.check_output(["xcrun", "--show-sdk-path", "--sdk", "macosx"])
 | |
|                 .strip()
 | |
|                 .decode("latin1")
 | |
|             )
 | |
|         except Exception:
 | |
|             sdk_path = None
 | |
|         if (
 | |
|             not sdk_path
 | |
|             or sdk_path == "/Applications/Xcode.app/Contents/Developer"
 | |
|             "/Platforms/MacOSX.platform/Developer/SDKs/MacOSX.sdk"
 | |
|         ):
 | |
|             commandlinetools_sdk_path = (
 | |
|                 "/Library/Developer/CommandLineTools/SDKs/MacOSX.sdk"
 | |
|             )
 | |
|             if os.path.exists(commandlinetools_sdk_path):
 | |
|                 sdk_path = commandlinetools_sdk_path
 | |
|         return sdk_path
 | |
| 
 | |
|     def get_ios_sdk_path(self) -> str:
 | |
|         try:
 | |
|             sdk = sys.implementation._multiarch.split("-")[-1]
 | |
|             _dbg("Using %s SDK", sdk)
 | |
|             return (
 | |
|                 subprocess.check_output(["xcrun", "--show-sdk-path", "--sdk", sdk])
 | |
|                 .strip()
 | |
|                 .decode("latin1")
 | |
|             )
 | |
|         except Exception:
 | |
|             msg = "Unable to identify location of iOS SDK."
 | |
|             raise ValueError(msg)
 | |
| 
 | |
|     def build_extensions(self) -> None:
 | |
|         library_dirs: list[str] = []
 | |
|         include_dirs: list[str] = []
 | |
| 
 | |
|         pkg_config = None
 | |
|         if _cmd_exists(os.environ.get("PKG_CONFIG", "pkg-config")):
 | |
|             pkg_config = _pkg_config
 | |
| 
 | |
|         #
 | |
|         # add configured kits
 | |
|         for root_name, lib_name in {
 | |
|             "AVIF_ROOT": "avif",
 | |
|             "JPEG_ROOT": "libjpeg",
 | |
|             "JPEG2K_ROOT": "libopenjp2",
 | |
|             "TIFF_ROOT": ("libtiff-5", "libtiff-4"),
 | |
|             "ZLIB_ROOT": "zlib",
 | |
|             "FREETYPE_ROOT": "freetype2",
 | |
|             "HARFBUZZ_ROOT": "harfbuzz",
 | |
|             "FRIBIDI_ROOT": "fribidi",
 | |
|             "RAQM_ROOT": "raqm",
 | |
|             "WEBP_ROOT": "libwebp",
 | |
|             "LCMS_ROOT": "lcms2",
 | |
|             "IMAGEQUANT_ROOT": "libimagequant",
 | |
|         }.items():
 | |
|             root = globals()[root_name]
 | |
| 
 | |
|             if root is None and root_name in os.environ:
 | |
|                 root_prefix = os.environ[root_name]
 | |
|                 root = (
 | |
|                     os.path.join(root_prefix, "lib"),
 | |
|                     os.path.join(root_prefix, "include"),
 | |
|                 )
 | |
| 
 | |
|             if root is None and pkg_config:
 | |
|                 if isinstance(lib_name, str):
 | |
|                     _dbg("Looking for `%s` using pkg-config.", lib_name)
 | |
|                     root = pkg_config(lib_name)
 | |
|                 else:
 | |
|                     for lib_name2 in lib_name:
 | |
|                         _dbg("Looking for `%s` using pkg-config.", lib_name2)
 | |
|                         root = pkg_config(lib_name2)
 | |
|                         if root:
 | |
|                             break
 | |
| 
 | |
|             if isinstance(root, tuple):
 | |
|                 lib_root, include_root = root
 | |
|             else:
 | |
|                 lib_root = include_root = root
 | |
| 
 | |
|             if lib_root is not None:
 | |
|                 if not isinstance(lib_root, (tuple, list)):
 | |
|                     lib_root = (lib_root,)
 | |
|                 for lib_dir in lib_root:
 | |
|                     _add_directory(library_dirs, lib_dir)
 | |
|             if include_root is not None:
 | |
|                 if not isinstance(include_root, (tuple, list)):
 | |
|                     include_root = (include_root,)
 | |
|                 for include_dir in include_root:
 | |
|                     _add_directory(include_dirs, include_dir)
 | |
| 
 | |
|         # respect CFLAGS/CPPFLAGS/LDFLAGS
 | |
|         for k in ("CFLAGS", "CPPFLAGS", "LDFLAGS"):
 | |
|             if k in os.environ:
 | |
|                 for match in re.finditer(r"-I([^\s]+)", os.environ[k]):
 | |
|                     _add_directory(include_dirs, match.group(1))
 | |
|                 for match in re.finditer(r"-L([^\s]+)", os.environ[k]):
 | |
|                     _add_directory(library_dirs, match.group(1))
 | |
| 
 | |
|         # include, rpath, if set as environment variables:
 | |
|         for k in ("C_INCLUDE_PATH", "CPATH", "INCLUDE"):
 | |
|             if k in os.environ:
 | |
|                 for d in os.environ[k].split(os.path.pathsep):
 | |
|                     _add_directory(include_dirs, d)
 | |
| 
 | |
|         for k in ("LD_RUN_PATH", "LIBRARY_PATH", "LIB"):
 | |
|             if k in os.environ:
 | |
|                 for d in os.environ[k].split(os.path.pathsep):
 | |
|                     _add_directory(library_dirs, d)
 | |
| 
 | |
|         _add_directory(library_dirs, os.path.join(sys.prefix, "lib"))
 | |
|         _add_directory(include_dirs, os.path.join(sys.prefix, "include"))
 | |
| 
 | |
|         #
 | |
|         # add platform directories
 | |
| 
 | |
|         if self.disable_platform_guessing:
 | |
|             pass
 | |
| 
 | |
|         elif sys.platform == "cygwin":
 | |
|             # pythonX.Y.dll.a is in the /usr/lib/pythonX.Y/config directory
 | |
|             self.compiler.shared_lib_extension = ".dll.a"
 | |
|             _add_directory(
 | |
|                 library_dirs,
 | |
|                 os.path.join(
 | |
|                     "/usr/lib", "python{}.{}".format(*sys.version_info), "config"
 | |
|                 ),
 | |
|             )
 | |
| 
 | |
|         elif sys.platform == "darwin":
 | |
|             # attempt to make sure we pick freetype2 over other versions
 | |
|             _add_directory(include_dirs, "/sw/include/freetype2")
 | |
|             _add_directory(include_dirs, "/sw/lib/freetype2/include")
 | |
|             # fink installation directories
 | |
|             _add_directory(library_dirs, "/sw/lib")
 | |
|             _add_directory(include_dirs, "/sw/include")
 | |
|             # darwin ports installation directories
 | |
|             _add_directory(library_dirs, "/opt/local/lib")
 | |
|             _add_directory(include_dirs, "/opt/local/include")
 | |
| 
 | |
|             # if Homebrew is installed, use its lib and include directories
 | |
|             try:
 | |
|                 prefix = (
 | |
|                     subprocess.check_output(["brew", "--prefix"])
 | |
|                     .strip()
 | |
|                     .decode("latin1")
 | |
|                 )
 | |
|             except Exception:
 | |
|                 # Homebrew not installed
 | |
|                 prefix = None
 | |
| 
 | |
|             ft_prefix = None
 | |
| 
 | |
|             if prefix:
 | |
|                 # add Homebrew's include and lib directories
 | |
|                 _add_directory(library_dirs, os.path.join(prefix, "lib"))
 | |
|                 _add_directory(include_dirs, os.path.join(prefix, "include"))
 | |
|                 _add_directory(
 | |
|                     include_dirs, os.path.join(prefix, "opt", "zlib", "include")
 | |
|                 )
 | |
|                 ft_prefix = os.path.join(prefix, "opt", "freetype")
 | |
| 
 | |
|             if ft_prefix and os.path.isdir(ft_prefix):
 | |
|                 # freetype might not be linked into Homebrew's prefix
 | |
|                 _add_directory(library_dirs, os.path.join(ft_prefix, "lib"))
 | |
|                 _add_directory(include_dirs, os.path.join(ft_prefix, "include"))
 | |
|             else:
 | |
|                 # fall back to freetype from XQuartz if
 | |
|                 # Homebrew's freetype is missing
 | |
|                 _add_directory(library_dirs, "/usr/X11/lib")
 | |
|                 _add_directory(include_dirs, "/usr/X11/include")
 | |
| 
 | |
|             # Add the macOS SDK path.
 | |
|             sdk_path = self.get_macos_sdk_path()
 | |
|             if sdk_path:
 | |
|                 _add_directory(library_dirs, os.path.join(sdk_path, "usr", "lib"))
 | |
|                 _add_directory(include_dirs, os.path.join(sdk_path, "usr", "include"))
 | |
| 
 | |
|                 for extension in self.extensions:
 | |
|                     extension.extra_compile_args = ["-Wno-nullability-completeness"]
 | |
| 
 | |
|         elif sys.platform == "ios":
 | |
|             # Add the iOS SDK path.
 | |
|             sdk_path = self.get_ios_sdk_path()
 | |
| 
 | |
|             # Add the iOS SDK path.
 | |
|             _add_directory(library_dirs, os.path.join(sdk_path, "usr", "lib"))
 | |
|             _add_directory(include_dirs, os.path.join(sdk_path, "usr", "include"))
 | |
| 
 | |
|             for extension in self.extensions:
 | |
|                 extension.extra_compile_args = ["-Wno-nullability-completeness"]
 | |
| 
 | |
|         elif sys.platform.startswith(("linux", "gnu", "freebsd")):
 | |
|             for dirname in _find_library_dirs_ldconfig():
 | |
|                 _add_directory(library_dirs, dirname)
 | |
|             if sys.platform.startswith("linux") and os.environ.get("ANDROID_ROOT"):
 | |
|                 # termux support for android.
 | |
|                 # system libraries (zlib) are installed in /system/lib
 | |
|                 # headers are at $PREFIX/include
 | |
|                 # user libs are at $PREFIX/lib
 | |
|                 _add_directory(
 | |
|                     library_dirs,
 | |
|                     os.path.join(
 | |
|                         os.environ["ANDROID_ROOT"],
 | |
|                         "lib" if struct.calcsize("l") == 4 else "lib64",
 | |
|                     ),
 | |
|                 )
 | |
| 
 | |
|         elif sys.platform.startswith("netbsd"):
 | |
|             _add_directory(library_dirs, "/usr/pkg/lib")
 | |
|             _add_directory(include_dirs, "/usr/pkg/include")
 | |
| 
 | |
|         elif sys.platform.startswith("sunos5"):
 | |
|             _add_directory(library_dirs, "/opt/local/lib")
 | |
|             _add_directory(include_dirs, "/opt/local/include")
 | |
| 
 | |
|         # FIXME: check /opt/stuff directories here?
 | |
| 
 | |
|         # standard locations
 | |
|         if not self.disable_platform_guessing:
 | |
|             _add_directory(library_dirs, "/usr/local/lib")
 | |
|             _add_directory(include_dirs, "/usr/local/include")
 | |
| 
 | |
|             _add_directory(library_dirs, "/usr/lib")
 | |
|             _add_directory(include_dirs, "/usr/include")
 | |
|             # alpine, at least
 | |
|             _add_directory(library_dirs, "/lib")
 | |
| 
 | |
|         if sys.platform == "win32":
 | |
|             # on Windows, look for the OpenJPEG libraries in the location that
 | |
|             # the official installer puts them
 | |
|             program_files = os.environ.get("ProgramFiles", "")
 | |
|             best_version = (0, 0)
 | |
|             best_path = None
 | |
|             for name in os.listdir(program_files):
 | |
|                 if name.startswith("OpenJPEG "):
 | |
|                     version = tuple(int(x) for x in name[9:].strip().split("."))
 | |
|                     if version > best_version:
 | |
|                         best_version = version
 | |
|                         best_path = os.path.join(program_files, name)
 | |
| 
 | |
|             if best_path:
 | |
|                 _dbg("Adding %s to search list", best_path)
 | |
|                 _add_directory(library_dirs, os.path.join(best_path, "lib"))
 | |
|                 _add_directory(include_dirs, os.path.join(best_path, "include"))
 | |
| 
 | |
|         #
 | |
|         # insert new dirs *before* default libs, to avoid conflicts
 | |
|         # between Python PYD stub libs and real libraries
 | |
| 
 | |
|         self.compiler.library_dirs = library_dirs + self.compiler.library_dirs
 | |
|         self.compiler.include_dirs = include_dirs + self.compiler.include_dirs
 | |
| 
 | |
|         #
 | |
|         # look for available libraries
 | |
| 
 | |
|         feature = self.feature
 | |
| 
 | |
|         if feature.want("zlib"):
 | |
|             _dbg("Looking for zlib")
 | |
|             if _find_include_file(self, "zlib.h"):
 | |
|                 if _find_library_file(self, "z"):
 | |
|                     feature.set("zlib", "z")
 | |
|                 elif sys.platform == "win32" and _find_library_file(self, "zlib"):
 | |
|                     feature.set("zlib", "zlib")  # alternative name
 | |
|                 elif sys.platform == "win32" and _find_library_file(self, "zdll"):
 | |
|                     feature.set("zlib", "zdll")  # dll import library
 | |
| 
 | |
|         if feature.want("jpeg"):
 | |
|             _dbg("Looking for jpeg")
 | |
|             if _find_include_file(self, "jpeglib.h"):
 | |
|                 if _find_library_file(self, "jpeg"):
 | |
|                     feature.set("jpeg", "jpeg")
 | |
|                 elif sys.platform == "win32" and _find_library_file(self, "libjpeg"):
 | |
|                     feature.set("jpeg", "libjpeg")  # alternative name
 | |
| 
 | |
|         feature.set("openjpeg_version", None)
 | |
|         if feature.want("jpeg2000"):
 | |
|             _dbg("Looking for jpeg2000")
 | |
|             best_version: tuple[int, ...] | None = None
 | |
|             best_path = None
 | |
| 
 | |
|             # Find the best version
 | |
|             for directory in self.compiler.include_dirs:
 | |
|                 _dbg("Checking for openjpeg-#.# in %s", directory)
 | |
|                 try:
 | |
|                     listdir = os.listdir(directory)
 | |
|                 except Exception:
 | |
|                     # OSError, FileNotFoundError
 | |
|                     continue
 | |
|                 for name in listdir:
 | |
|                     if name.startswith("openjpeg-") and os.path.isfile(
 | |
|                         os.path.join(directory, name, "openjpeg.h")
 | |
|                     ):
 | |
|                         _dbg("Found openjpeg.h in %s/%s", (directory, name))
 | |
|                         version = tuple(int(x) for x in name[9:].split("."))
 | |
|                         if best_version is None or version > best_version:
 | |
|                             best_version = version
 | |
|                             best_path = os.path.join(directory, name)
 | |
|                             _dbg(
 | |
|                                 "Best openjpeg version %s so far in %s",
 | |
|                                 (str(best_version), best_path),
 | |
|                             )
 | |
| 
 | |
|             if best_version and _find_library_file(self, "openjp2"):
 | |
|                 # Add the directory to the include path so we can include
 | |
|                 # <openjpeg.h> rather than having to cope with the versioned
 | |
|                 # include path
 | |
|                 _add_directory(self.compiler.include_dirs, best_path, 0)
 | |
|                 feature.set("jpeg2000", "openjp2")
 | |
|                 feature.set("openjpeg_version", ".".join(str(x) for x in best_version))
 | |
| 
 | |
|         if feature.want("imagequant"):
 | |
|             _dbg("Looking for imagequant")
 | |
|             if _find_include_file(self, "libimagequant.h"):
 | |
|                 if _find_library_file(self, "imagequant"):
 | |
|                     feature.set("imagequant", "imagequant")
 | |
|                 elif _find_library_file(self, "libimagequant"):
 | |
|                     feature.set("imagequant", "libimagequant")
 | |
| 
 | |
|         if feature.want("tiff"):
 | |
|             _dbg("Looking for tiff")
 | |
|             if _find_include_file(self, "tiff.h"):
 | |
|                 if sys.platform in ["win32", "darwin"] and _find_library_file(
 | |
|                     self, "libtiff"
 | |
|                 ):
 | |
|                     feature.set("tiff", "libtiff")
 | |
|                 elif _find_library_file(self, "tiff"):
 | |
|                     feature.set("tiff", "tiff")
 | |
| 
 | |
|         if feature.want("freetype"):
 | |
|             _dbg("Looking for freetype")
 | |
|             if _find_library_file(self, "freetype"):
 | |
|                 # look for freetype2 include files
 | |
|                 freetype_version = 0
 | |
|                 for subdir in self.compiler.include_dirs:
 | |
|                     _dbg("Checking for include file %s in %s", ("ft2build.h", subdir))
 | |
|                     if os.path.isfile(os.path.join(subdir, "ft2build.h")):
 | |
|                         _dbg("Found %s in %s", ("ft2build.h", subdir))
 | |
|                         freetype_version = 21
 | |
|                         subdir = os.path.join(subdir, "freetype2")
 | |
|                         break
 | |
|                     subdir = os.path.join(subdir, "freetype2")
 | |
|                     _dbg("Checking for include file %s in %s", ("ft2build.h", subdir))
 | |
|                     if os.path.isfile(os.path.join(subdir, "ft2build.h")):
 | |
|                         _dbg("Found %s in %s", ("ft2build.h", subdir))
 | |
|                         freetype_version = 21
 | |
|                         break
 | |
|                 if freetype_version:
 | |
|                     feature.set("freetype", "freetype")
 | |
|                     if subdir:
 | |
|                         _add_directory(self.compiler.include_dirs, subdir, 0)
 | |
| 
 | |
|         if feature.get("freetype") and feature.want("raqm"):
 | |
|             if not feature.want_vendor("raqm"):  # want system Raqm
 | |
|                 _dbg("Looking for Raqm")
 | |
|                 if _find_include_file(self, "raqm.h"):
 | |
|                     if _find_library_file(self, "raqm"):
 | |
|                         feature.set("raqm", "raqm")
 | |
|                     elif _find_library_file(self, "libraqm"):
 | |
|                         feature.set("raqm", "libraqm")
 | |
|             else:  # want to build Raqm from src/thirdparty
 | |
|                 _dbg("Looking for HarfBuzz")
 | |
|                 feature.set("harfbuzz", None)
 | |
|                 hb_dir = _find_include_dir(self, "harfbuzz", "hb.h")
 | |
|                 if hb_dir:
 | |
|                     if isinstance(hb_dir, str):
 | |
|                         _add_directory(self.compiler.include_dirs, hb_dir, 0)
 | |
|                     if _find_library_file(self, "harfbuzz"):
 | |
|                         feature.set("harfbuzz", "harfbuzz")
 | |
|                 if feature.get("harfbuzz"):
 | |
|                     if not feature.want_vendor("fribidi"):  # want system FriBiDi
 | |
|                         _dbg("Looking for FriBiDi")
 | |
|                         feature.set("fribidi", None)
 | |
|                         fribidi_dir = _find_include_dir(self, "fribidi", "fribidi.h")
 | |
|                         if fribidi_dir:
 | |
|                             if isinstance(fribidi_dir, str):
 | |
|                                 _add_directory(
 | |
|                                     self.compiler.include_dirs, fribidi_dir, 0
 | |
|                                 )
 | |
|                             if _find_library_file(self, "fribidi"):
 | |
|                                 feature.set("fribidi", "fribidi")
 | |
|                                 feature.set("raqm", True)
 | |
|                     else:  # want to build FriBiDi shim from src/thirdparty
 | |
|                         feature.set("raqm", True)
 | |
| 
 | |
|         if feature.want("lcms"):
 | |
|             _dbg("Looking for lcms")
 | |
|             if _find_include_file(self, "lcms2.h"):
 | |
|                 if _find_library_file(self, "lcms2"):
 | |
|                     feature.set("lcms", "lcms2")
 | |
|                 elif _find_library_file(self, "lcms2_static"):
 | |
|                     # alternate Windows name.
 | |
|                     feature.set("lcms", "lcms2_static")
 | |
| 
 | |
|         if feature.want("webp"):
 | |
|             _dbg("Looking for webp")
 | |
|             if all(
 | |
|                 _find_include_file(self, "webp/" + include)
 | |
|                 for include in ("encode.h", "decode.h", "mux.h", "demux.h")
 | |
|             ):
 | |
|                 # In Google's precompiled zip it is called "libwebp"
 | |
|                 for prefix in ("", "lib"):
 | |
|                     if all(
 | |
|                         _find_library_file(self, prefix + library)
 | |
|                         for library in ("webp", "webpmux", "webpdemux")
 | |
|                     ):
 | |
|                         feature.set("webp", prefix + "webp")
 | |
|                         break
 | |
| 
 | |
|         if feature.want("xcb"):
 | |
|             _dbg("Looking for xcb")
 | |
|             if _find_include_file(self, "xcb/xcb.h"):
 | |
|                 if _find_library_file(self, "xcb"):
 | |
|                     feature.set("xcb", "xcb")
 | |
| 
 | |
|         if feature.want("avif"):
 | |
|             _dbg("Looking for avif")
 | |
|             if avif_h := _find_include_file(self, "avif/avif.h"):
 | |
|                 with open(avif_h, "rb") as fp:
 | |
|                     major_version = int(
 | |
|                         fp.read().split(b"#define AVIF_VERSION_MAJOR ")[1].split()[0]
 | |
|                     )
 | |
|                     if major_version >= 1 and _find_library_file(self, "avif"):
 | |
|                         feature.set("avif", "avif")
 | |
| 
 | |
|         for f in feature:
 | |
|             if not feature.get(f) and feature.require(f):
 | |
|                 if f in ("jpeg", "zlib"):
 | |
|                     raise RequiredDependencyException(f)
 | |
|                 raise DependencyException(f)
 | |
| 
 | |
|         #
 | |
|         # core library
 | |
| 
 | |
|         libs: list[str | bool | None] = []
 | |
|         libs.extend(self.add_imaging_libs.split())
 | |
|         defs: list[tuple[str, str | None]] = []
 | |
|         if feature.get("tiff"):
 | |
|             libs.append(feature.get("tiff"))
 | |
|             defs.append(("HAVE_LIBTIFF", None))
 | |
|             if sys.platform == "win32":
 | |
|                 # This define needs to be defined if-and-only-if it was defined
 | |
|                 # when compiling LibTIFF. LibTIFF doesn't expose it in `tiffconf.h`,
 | |
|                 # so we have to guess; by default it is defined in all Windows builds.
 | |
|                 # See #4237, #5243, #5359 for more information.
 | |
|                 defs.append(("USE_WIN32_FILEIO", None))
 | |
|             elif sys.platform == "ios":
 | |
|                 # Ensure transitive dependencies are linked.
 | |
|                 libs.append("lzma")
 | |
|         if feature.get("jpeg"):
 | |
|             libs.append(feature.get("jpeg"))
 | |
|             defs.append(("HAVE_LIBJPEG", None))
 | |
|         if feature.get("jpeg2000"):
 | |
|             libs.append(feature.get("jpeg2000"))
 | |
|             defs.append(("HAVE_OPENJPEG", None))
 | |
|             if sys.platform == "win32" and not PLATFORM_MINGW:
 | |
|                 defs.append(("OPJ_STATIC", None))
 | |
|         if feature.get("zlib"):
 | |
|             libs.append(feature.get("zlib"))
 | |
|             defs.append(("HAVE_LIBZ", None))
 | |
|         if feature.get("imagequant"):
 | |
|             libs.append(feature.get("imagequant"))
 | |
|             defs.append(("HAVE_LIBIMAGEQUANT", None))
 | |
|         if feature.get("xcb"):
 | |
|             libs.append(feature.get("xcb"))
 | |
|             if sys.platform == "ios":
 | |
|                 # Ensure transitive dependencies are linked.
 | |
|                 libs.append("Xau")
 | |
|             defs.append(("HAVE_XCB", None))
 | |
|         if sys.platform == "win32":
 | |
|             libs.extend(["kernel32", "user32", "gdi32"])
 | |
|         if struct.unpack("h", b"\0\1")[0] == 1:
 | |
|             defs.append(("WORDS_BIGENDIAN", None))
 | |
| 
 | |
|         defs.append(("PILLOW_VERSION", f'"{PILLOW_VERSION}"'))
 | |
| 
 | |
|         self._update_extension("PIL._imaging", libs, defs)
 | |
| 
 | |
|         #
 | |
|         # additional libraries
 | |
| 
 | |
|         if feature.get("freetype"):
 | |
|             srcs = []
 | |
|             libs = ["freetype"]
 | |
|             defs = []
 | |
|             if feature.get("raqm"):
 | |
|                 if not feature.want_vendor("raqm"):  # using system Raqm
 | |
|                     defs.append(("HAVE_RAQM", None))
 | |
|                     defs.append(("HAVE_RAQM_SYSTEM", None))
 | |
|                     libs.append(feature.get("raqm"))
 | |
|                 else:  # building Raqm from src/thirdparty
 | |
|                     defs.append(("HAVE_RAQM", None))
 | |
|                     srcs.append("src/thirdparty/raqm/raqm.c")
 | |
|                     libs.append(feature.get("harfbuzz"))
 | |
|                     if not feature.want_vendor("fribidi"):  # using system FriBiDi
 | |
|                         defs.append(("HAVE_FRIBIDI_SYSTEM", None))
 | |
|                         libs.append(feature.get("fribidi"))
 | |
|                     else:  # building FriBiDi shim from src/thirdparty
 | |
|                         srcs.append("src/thirdparty/fribidi-shim/fribidi.c")
 | |
| 
 | |
|             if sys.platform == "ios":
 | |
|                 # Ensure transitive dependencies are linked.
 | |
|                 libs.extend(["z", "bz2", "brotlicommon", "brotlidec", "png"])
 | |
| 
 | |
|             self._update_extension("PIL._imagingft", libs, defs, srcs)
 | |
| 
 | |
|         else:
 | |
|             self._remove_extension("PIL._imagingft")
 | |
| 
 | |
|         if feature.get("lcms"):
 | |
|             libs = [feature.get("lcms")]
 | |
|             if sys.platform == "win32":
 | |
|                 libs.extend(["user32", "gdi32"])
 | |
|             self._update_extension("PIL._imagingcms", libs)
 | |
|         else:
 | |
|             self._remove_extension("PIL._imagingcms")
 | |
| 
 | |
|         webp = feature.get("webp")
 | |
|         if isinstance(webp, str):
 | |
|             libs = [webp, webp + "mux", webp + "demux"]
 | |
|             if sys.platform == "ios":
 | |
|                 # Ensure transitive dependencies are linked.
 | |
|                 libs.append("sharpyuv")
 | |
|             self._update_extension("PIL._webp", libs)
 | |
|         else:
 | |
|             self._remove_extension("PIL._webp")
 | |
| 
 | |
|         if feature.get("avif"):
 | |
|             libs = [feature.get("avif")]
 | |
|             if sys.platform == "win32":
 | |
|                 libs.extend(["ntdll", "userenv", "ws2_32", "bcrypt"])
 | |
|             self._update_extension("PIL._avif", libs)
 | |
|         else:
 | |
|             self._remove_extension("PIL._avif")
 | |
| 
 | |
|         tk_libs = ["psapi"] if sys.platform in ("win32", "cygwin") else []
 | |
|         self._update_extension("PIL._imagingtk", tk_libs)
 | |
| 
 | |
|         build_ext.build_extensions(self)
 | |
| 
 | |
|         #
 | |
|         # sanity checks
 | |
| 
 | |
|         self.summary_report(feature)
 | |
| 
 | |
|     def summary_report(self, feature: ext_feature) -> None:
 | |
|         print("-" * 68)
 | |
|         print("PIL SETUP SUMMARY")
 | |
|         print("-" * 68)
 | |
|         print(f"version      Pillow {PILLOW_VERSION}")
 | |
|         version = sys.version.split("[")
 | |
|         print(f"platform     {sys.platform} {version[0].strip()}")
 | |
|         for v in version[1:]:
 | |
|             print(f"             [{v.strip()}")
 | |
|         print("-" * 68)
 | |
| 
 | |
|         raqm_extra_info = ""
 | |
|         if feature.want_vendor("raqm"):
 | |
|             raqm_extra_info += "bundled"
 | |
|             if feature.want_vendor("fribidi"):
 | |
|                 raqm_extra_info += ", FriBiDi shim"
 | |
| 
 | |
|         options = [
 | |
|             (feature.get("jpeg"), "JPEG"),
 | |
|             (
 | |
|                 feature.get("jpeg2000"),
 | |
|                 "OPENJPEG (JPEG2000)",
 | |
|                 feature.get("openjpeg_version"),
 | |
|             ),
 | |
|             (feature.get("zlib"), "ZLIB (PNG/ZIP)"),
 | |
|             (feature.get("imagequant"), "LIBIMAGEQUANT"),
 | |
|             (feature.get("tiff"), "LIBTIFF"),
 | |
|             (feature.get("freetype"), "FREETYPE2"),
 | |
|             (feature.get("raqm"), "RAQM (Text shaping)", raqm_extra_info),
 | |
|             (feature.get("lcms"), "LITTLECMS2"),
 | |
|             (feature.get("webp"), "WEBP"),
 | |
|             (feature.get("xcb"), "XCB (X protocol)"),
 | |
|             (feature.get("avif"), "LIBAVIF"),
 | |
|         ]
 | |
| 
 | |
|         all = 1
 | |
|         for option in options:
 | |
|             if option[0]:
 | |
|                 extra_info = ""
 | |
|                 if len(option) >= 3 and option[2]:
 | |
|                     extra_info = f" ({option[2]})"
 | |
|                 print(f"--- {option[1]} support available{extra_info}")
 | |
|             else:
 | |
|                 print(f"*** {option[1]} support not available")
 | |
|                 all = 0
 | |
| 
 | |
|         print("-" * 68)
 | |
| 
 | |
|         if not all:
 | |
|             print("To add a missing option, make sure you have the required")
 | |
|             print("library and headers.")
 | |
|             print(
 | |
|                 "See https://pillow.readthedocs.io/en/latest/installation."
 | |
|                 "html#building-from-source"
 | |
|             )
 | |
|             print("")
 | |
| 
 | |
|         print("To check the build, run the selftest.py script.")
 | |
|         print("")
 | |
| 
 | |
| 
 | |
| def debug_build() -> bool:
 | |
|     return hasattr(sys, "gettotalrefcount") or FUZZING_BUILD
 | |
| 
 | |
| 
 | |
| files: list[str | os.PathLike[str]] = ["src/_imaging.c"]
 | |
| for src_file in _IMAGING:
 | |
|     files.append("src/" + src_file + ".c")
 | |
| for src_file in _LIB_IMAGING:
 | |
|     files.append(os.path.join("src/libImaging", src_file + ".c"))
 | |
| ext_modules = [
 | |
|     Extension("PIL._imaging", files),
 | |
|     Extension("PIL._imagingft", ["src/_imagingft.c"]),
 | |
|     Extension("PIL._imagingcms", ["src/_imagingcms.c"]),
 | |
|     Extension("PIL._webp", ["src/_webp.c"]),
 | |
|     Extension("PIL._avif", ["src/_avif.c"]),
 | |
|     Extension("PIL._imagingtk", ["src/_imagingtk.c", "src/Tk/tkImaging.c"]),
 | |
|     Extension("PIL._imagingmath", ["src/_imagingmath.c"]),
 | |
|     Extension("PIL._imagingmorph", ["src/_imagingmorph.c"]),
 | |
| ]
 | |
| 
 | |
| 
 | |
| try:
 | |
|     setup(
 | |
|         cmdclass={"build_ext": pil_build_ext},
 | |
|         ext_modules=ext_modules,
 | |
|         zip_safe=not (debug_build() or PLATFORM_MINGW),
 | |
|     )
 | |
| except RequiredDependencyException as err:
 | |
|     msg = f"""
 | |
| 
 | |
| The headers or library files could not be found for {str(err)},
 | |
| a required dependency when compiling Pillow from source.
 | |
| 
 | |
| Please see the install instructions at:
 | |
|    https://pillow.readthedocs.io/en/latest/installation/basic-installation.html
 | |
| 
 | |
| """
 | |
|     sys.stderr.write(msg)
 | |
|     raise RequiredDependencyException(msg)
 | |
| except DependencyException as err:
 | |
|     msg = f"""
 | |
| 
 | |
| The headers or library files could not be found for {str(err)},
 | |
| which was requested by the option flag '-C {str(err)}=enable'
 | |
| 
 | |
| """
 | |
|     sys.stderr.write(msg)
 | |
|     raise DependencyException(msg)
 |