mirror of
https://github.com/python-pillow/Pillow.git
synced 2025-01-28 02:04:36 +03:00
Isolate macOS build from Homebrew.
This commit is contained in:
parent
96fb0e63ea
commit
0fe55d6115
96
.github/workflows/wheels-dependencies.sh
vendored
96
.github/workflows/wheels-dependencies.sh
vendored
|
@ -23,6 +23,8 @@ OPENJPEG_VERSION=2.5.2
|
||||||
XZ_VERSION=5.6.3
|
XZ_VERSION=5.6.3
|
||||||
TIFF_VERSION=4.6.0
|
TIFF_VERSION=4.6.0
|
||||||
LCMS2_VERSION=2.16
|
LCMS2_VERSION=2.16
|
||||||
|
RAQM_VERSION=0.7.1
|
||||||
|
FRIBIDI_VERSION=1.0.16
|
||||||
if [[ -n "$IS_MACOS" ]]; then
|
if [[ -n "$IS_MACOS" ]]; then
|
||||||
GIFLIB_VERSION=5.2.2
|
GIFLIB_VERSION=5.2.2
|
||||||
else
|
else
|
||||||
|
@ -38,7 +40,23 @@ BZIP2_VERSION=1.0.8
|
||||||
LIBXCB_VERSION=1.17.0
|
LIBXCB_VERSION=1.17.0
|
||||||
BROTLI_VERSION=1.1.0
|
BROTLI_VERSION=1.1.0
|
||||||
|
|
||||||
|
function build_pkg_config {
|
||||||
|
if [ -e pkg-config-stamp ]; then return; fi
|
||||||
|
# This essentially duplicates the Homebrew recipe:
|
||||||
|
# https://github.com/Homebrew/homebrew-core/blob/master/Formula/p/pkg-config.rb
|
||||||
|
ORIGINAL_CFLAGS=$CFLAGS
|
||||||
|
CFLAGS="$CFLAGS -Wno-int-conversion"
|
||||||
|
build_simple pkg-config 0.29.2 https://pkg-config.freedesktop.org/releases tar.gz \
|
||||||
|
--disable-debug --disable-host-tool --with-internal-glib \
|
||||||
|
--with-pc-path=$BUILD_PREFIX/share/pkgconfig:$BUILD_PREFIX/lib/pkgconfig \
|
||||||
|
--with-system-include-path=$(xcrun --show-sdk-path --sdk macosx)/usr/include
|
||||||
|
CFLAGS=$ORIGINAL_CFLAGS
|
||||||
|
export PKG_CONFIG=$BUILD_PREFIX/bin/pkg-config
|
||||||
|
touch pkg-config-stamp
|
||||||
|
}
|
||||||
|
|
||||||
function build_brotli {
|
function build_brotli {
|
||||||
|
if [ -e brotli-stamp ]; then return; fi
|
||||||
local cmake=$(get_modern_cmake)
|
local cmake=$(get_modern_cmake)
|
||||||
local out_dir=$(fetch_unpack https://github.com/google/brotli/archive/v$BROTLI_VERSION.tar.gz brotli-$BROTLI_VERSION.tar.gz)
|
local out_dir=$(fetch_unpack https://github.com/google/brotli/archive/v$BROTLI_VERSION.tar.gz brotli-$BROTLI_VERSION.tar.gz)
|
||||||
(cd $out_dir \
|
(cd $out_dir \
|
||||||
|
@ -48,25 +66,25 @@ function build_brotli {
|
||||||
cp /usr/local/lib64/libbrotli* /usr/local/lib
|
cp /usr/local/lib64/libbrotli* /usr/local/lib
|
||||||
cp /usr/local/lib64/pkgconfig/libbrotli* /usr/local/lib/pkgconfig
|
cp /usr/local/lib64/pkgconfig/libbrotli* /usr/local/lib/pkgconfig
|
||||||
fi
|
fi
|
||||||
|
touch brotli-stamp
|
||||||
}
|
}
|
||||||
|
|
||||||
function build_harfbuzz {
|
function build_harfbuzz {
|
||||||
python3 -m pip install meson ninja
|
if [ -e harfbuzz-stamp ]; then return; fi
|
||||||
|
python -m pip install meson ninja
|
||||||
|
|
||||||
local out_dir=$(fetch_unpack https://github.com/harfbuzz/harfbuzz/releases/download/$HARFBUZZ_VERSION/$HARFBUZZ_VERSION.tar.xz harfbuzz-$HARFBUZZ_VERSION.tar.xz)
|
local out_dir=$(fetch_unpack https://github.com/harfbuzz/harfbuzz/releases/download/$HARFBUZZ_VERSION/$HARFBUZZ_VERSION.tar.xz harfbuzz-$HARFBUZZ_VERSION.tar.xz)
|
||||||
(cd $out_dir \
|
(cd $out_dir \
|
||||||
&& meson setup build --buildtype=release -Dfreetype=enabled -Dglib=disabled)
|
&& meson setup build --prefix=$BUILD_PREFIX --buildtype=release -Dfreetype=enabled -Dglib=disabled)
|
||||||
(cd $out_dir/build \
|
(cd $out_dir/build \
|
||||||
&& meson install)
|
&& meson install)
|
||||||
if [[ "$MB_ML_LIBC" == "manylinux" ]]; then
|
if [[ "$MB_ML_LIBC" == "manylinux" ]]; then
|
||||||
cp /usr/local/lib64/libharfbuzz* /usr/local/lib
|
cp /usr/local/lib64/libharfbuzz* /usr/local/lib
|
||||||
fi
|
fi
|
||||||
|
touch harfbuzz-stamp
|
||||||
}
|
}
|
||||||
|
|
||||||
function build {
|
function build {
|
||||||
if [[ -n "$IS_MACOS" ]] && [[ "$CIBW_ARCHS" == "arm64" ]]; then
|
|
||||||
sudo chown -R runner /usr/local
|
|
||||||
fi
|
|
||||||
build_xz
|
build_xz
|
||||||
if [ -z "$IS_ALPINE" ] && [ -z "$IS_MACOS" ]; then
|
if [ -z "$IS_ALPINE" ] && [ -z "$IS_MACOS" ]; then
|
||||||
yum remove -y zlib-devel
|
yum remove -y zlib-devel
|
||||||
|
@ -77,17 +95,23 @@ function build {
|
||||||
if [ -n "$IS_MACOS" ]; then
|
if [ -n "$IS_MACOS" ]; then
|
||||||
build_simple xorgproto 2024.1 https://www.x.org/pub/individual/proto
|
build_simple xorgproto 2024.1 https://www.x.org/pub/individual/proto
|
||||||
build_simple libXau 1.0.11 https://www.x.org/pub/individual/lib
|
build_simple libXau 1.0.11 https://www.x.org/pub/individual/lib
|
||||||
|
build_simple libXdmcp 1.1.5 https://www.x.org/pub/individual/lib
|
||||||
build_simple libpthread-stubs 0.5 https://xcb.freedesktop.org/dist
|
build_simple libpthread-stubs 0.5 https://xcb.freedesktop.org/dist
|
||||||
if [[ "$CIBW_ARCHS" == "arm64" ]]; then
|
|
||||||
cp /usr/local/share/pkgconfig/xcb-proto.pc /usr/local/lib/pkgconfig
|
|
||||||
fi
|
|
||||||
else
|
else
|
||||||
sed s/\${pc_sysrootdir\}// /usr/local/share/pkgconfig/xcb-proto.pc > /usr/local/lib/pkgconfig/xcb-proto.pc
|
sed s/\${pc_sysrootdir\}// /usr/local/share/pkgconfig/xcb-proto.pc > /usr/local/lib/pkgconfig/xcb-proto.pc
|
||||||
fi
|
fi
|
||||||
build_simple libxcb $LIBXCB_VERSION https://www.x.org/releases/individual/lib
|
build_simple libxcb $LIBXCB_VERSION https://www.x.org/releases/individual/lib
|
||||||
|
|
||||||
build_libjpeg_turbo
|
build_libjpeg_turbo
|
||||||
build_tiff
|
if [ -n "$IS_MACOS" ]; then
|
||||||
|
# Custom tiff build to include jpeg; by default, configure won't include
|
||||||
|
# headers/libs in the custom macOS prefix
|
||||||
|
build_simple tiff $TIFF_VERSION https://download.osgeo.org/libtiff tar.gz \
|
||||||
|
--with-jpeg-include-dir=$BUILD_PREFIX/include --with-jpeg-lib-dir=$BUILD_PREFIX/lib
|
||||||
|
else
|
||||||
|
build_tiff
|
||||||
|
fi
|
||||||
|
|
||||||
build_libpng
|
build_libpng
|
||||||
build_lcms2
|
build_lcms2
|
||||||
build_openjpeg
|
build_openjpeg
|
||||||
|
@ -113,33 +137,55 @@ function build {
|
||||||
fi
|
fi
|
||||||
|
|
||||||
build_harfbuzz
|
build_harfbuzz
|
||||||
|
|
||||||
|
if [ -n "$IS_MACOS" ]; then
|
||||||
|
build_simple fribidi $FRIBIDI_VERSION https://github.com/fribidi/fribidi/releases/download/v$FRIBIDI_VERSION tar.xz --enable-shared
|
||||||
|
build_simple raqm $RAQM_VERSION https://github.com/Host_Oman/libraqm/releases/download/v$RAQM_VERSION tar.gz --enable-shared
|
||||||
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Perform all dependency builds in the build subfolder.
|
||||||
|
mkdir -p build
|
||||||
|
pushd build > /dev/null
|
||||||
|
|
||||||
# Any stuff that you need to do before you start building the wheels
|
# Any stuff that you need to do before you start building the wheels
|
||||||
# Runs in the root directory of this repository.
|
# Runs in the root directory of this repository.
|
||||||
curl -fsSL -o pillow-depends-main.zip https://github.com/python-pillow/pillow-depends/archive/main.zip
|
if [[ ! -d pillow-depends-main ]]; then
|
||||||
untar pillow-depends-main.zip
|
if [[ ! -f pillow-depends-main.zip ]]; then
|
||||||
|
echo "Download pillow dependency sources..."
|
||||||
|
curl -fSL -o pillow-depends-main.zip https://github.com/python-pillow/pillow-depends/archive/main.zip
|
||||||
|
fi
|
||||||
|
untar pillow-depends-main.zip
|
||||||
|
fi
|
||||||
|
|
||||||
if [[ -n "$IS_MACOS" ]]; then
|
if [[ -n "$IS_MACOS" ]]; then
|
||||||
# libtiff and libxcb cause a conflict with building libtiff and libxcb
|
# Build and install into the `deps` folder.
|
||||||
# libxau and libxdmcp cause an issue on macOS < 11
|
BUILD_PREFIX=$(pwd)/deps
|
||||||
# remove cairo to fix building harfbuzz on arm64
|
|
||||||
# remove lcms2 and libpng to fix building openjpeg on arm64
|
|
||||||
# remove jpeg-turbo to avoid inclusion on arm64
|
|
||||||
# remove webp and zstd to avoid inclusion on x86_64
|
|
||||||
# curl from brew requires zstd, use system curl
|
|
||||||
brew remove --ignore-dependencies libpng libtiff libxcb libxau libxdmcp curl cairo lcms2 zstd
|
|
||||||
if [[ "$CIBW_ARCHS" == "arm64" ]]; then
|
|
||||||
brew remove --ignore-dependencies jpeg-turbo
|
|
||||||
else
|
|
||||||
brew remove --ignore-dependencies webp
|
|
||||||
fi
|
|
||||||
|
|
||||||
brew install pkg-config
|
# Homebrew (or similar packaging environments) install can contain some of
|
||||||
|
# the libraries that we're going to build. However, they may be compiled
|
||||||
|
# with a MACOSX_DEPLOYMENT_TARGET that doesn't match what we want to use,
|
||||||
|
# and they may bring in other dependencies that we don't want. The same will
|
||||||
|
# be true of any other locations on the path. To avoid conflicts, strip the
|
||||||
|
# path down to the bare mimimum (which, on macOS, won't include any
|
||||||
|
# development dependencies).
|
||||||
|
export PATH="$BUILD_PREFIX/bin:/usr/bin:/bin:/usr/sbin:/sbin:/Library/Apple/usr/bin:$(dirname $(which python))"
|
||||||
|
export CMAKE_PREFIX_PATH=$BUILD_PREFIX
|
||||||
|
|
||||||
|
# Link the brew command into our isolated build directory.
|
||||||
|
mkdir -p "$BUILD_PREFIX/bin"
|
||||||
|
mkdir -p "$BUILD_PREFIX/lib"
|
||||||
|
|
||||||
|
# Ensure pkg-confg and cmake are available
|
||||||
|
build_pkg_config
|
||||||
|
python -m pip install cmake
|
||||||
fi
|
fi
|
||||||
|
|
||||||
wrap_wheel_builder build
|
wrap_wheel_builder build
|
||||||
|
|
||||||
|
# Return to the project root to finish the build
|
||||||
|
popd > /dev/null
|
||||||
|
|
||||||
# Append licenses
|
# Append licenses
|
||||||
for filename in wheels/dependency_licenses/*; do
|
for filename in wheels/dependency_licenses/*; do
|
||||||
echo -e "\n\n----\n\n$(basename $filename | cut -f 1 -d '.')\n" | cat >> LICENSE
|
echo -e "\n\n----\n\n$(basename $filename | cut -f 1 -d '.')\n" | cat >> LICENSE
|
||||||
|
|
15
.github/workflows/wheels-test.sh
vendored
15
.github/workflows/wheels-test.sh
vendored
|
@ -1,16 +1,13 @@
|
||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
set -e
|
set -e
|
||||||
|
|
||||||
if [[ "$OSTYPE" == "darwin"* ]]; then
|
# For Unix, Ensure fibidi is installed by the system.
|
||||||
brew install fribidi
|
if [[ "$OSTYPE" != "darwin"* ]]; then
|
||||||
export PKG_CONFIG_PATH="/usr/local/opt/openblas/lib/pkgconfig"
|
if [ "${AUDITWHEEL_POLICY::9}" == "musllinux" ]; then
|
||||||
if [ -f /opt/homebrew/lib/libfribidi.dylib ]; then
|
apk add curl fribidi
|
||||||
sudo cp /opt/homebrew/lib/libfribidi.dylib /usr/local/lib
|
else
|
||||||
|
yum install -y fribidi
|
||||||
fi
|
fi
|
||||||
elif [ "${AUDITWHEEL_POLICY::9}" == "musllinux" ]; then
|
|
||||||
apk add curl fribidi
|
|
||||||
else
|
|
||||||
yum install -y fribidi
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
python3 -m pip install numpy
|
python3 -m pip install numpy
|
||||||
|
|
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -19,6 +19,7 @@ lib64/
|
||||||
parts/
|
parts/
|
||||||
sdist/
|
sdist/
|
||||||
var/
|
var/
|
||||||
|
wheelhouse/
|
||||||
*.egg-info/
|
*.egg-info/
|
||||||
.installed.cfg
|
.installed.cfg
|
||||||
*.egg
|
*.egg
|
||||||
|
|
|
@ -93,10 +93,17 @@ version = { attr = "PIL.__version__" }
|
||||||
[tool.cibuildwheel]
|
[tool.cibuildwheel]
|
||||||
before-all = ".github/workflows/wheels-dependencies.sh"
|
before-all = ".github/workflows/wheels-dependencies.sh"
|
||||||
build-verbosity = 1
|
build-verbosity = 1
|
||||||
|
|
||||||
config-settings = "raqm=enable raqm=vendor fribidi=vendor imagequant=disable"
|
config-settings = "raqm=enable raqm=vendor fribidi=vendor imagequant=disable"
|
||||||
|
# Add an explicit dependencies prefix for macOS, and don't request vendored libraries.
|
||||||
|
macos.config-settings = "raqm=enable imagequant=disable dependencies-prefix=./build/deps"
|
||||||
|
|
||||||
test-command = "cd {project} && .github/workflows/wheels-test.sh"
|
test-command = "cd {project} && .github/workflows/wheels-test.sh"
|
||||||
test-extras = "tests"
|
test-extras = "tests"
|
||||||
|
|
||||||
|
[tool.cibuildwheel.macos.environment]
|
||||||
|
PATH = "$(pwd)/build/deps/bin:/usr/bin:/bin:/usr/sbin:/sbin:/Library/Apple/usr/bin:$(dirname $(which python))"
|
||||||
|
|
||||||
[tool.black]
|
[tool.black]
|
||||||
exclude = "wheels/multibuild"
|
exclude = "wheels/multibuild"
|
||||||
|
|
||||||
|
|
94
setup.py
94
setup.py
|
@ -344,6 +344,11 @@ class pil_build_ext(build_ext):
|
||||||
for x in ("raqm", "fribidi")
|
for x in ("raqm", "fribidi")
|
||||||
]
|
]
|
||||||
+ [
|
+ [
|
||||||
|
(
|
||||||
|
"dependencies-prefix",
|
||||||
|
None,
|
||||||
|
"The prefix where build dependencies are located.",
|
||||||
|
),
|
||||||
("disable-platform-guessing", None, "Disable platform guessing on Linux"),
|
("disable-platform-guessing", None, "Disable platform guessing on Linux"),
|
||||||
("debug", None, "Debug logging"),
|
("debug", None, "Debug logging"),
|
||||||
]
|
]
|
||||||
|
@ -355,6 +360,7 @@ class pil_build_ext(build_ext):
|
||||||
return True if value in configuration.get(option, []) else None
|
return True if value in configuration.get(option, []) else None
|
||||||
|
|
||||||
def initialize_options(self) -> None:
|
def initialize_options(self) -> None:
|
||||||
|
self.dependencies_prefix = configuration.get("dependencies-prefix", [])
|
||||||
self.disable_platform_guessing = self.check_configuration(
|
self.disable_platform_guessing = self.check_configuration(
|
||||||
"platform-guessing", "disable"
|
"platform-guessing", "disable"
|
||||||
)
|
)
|
||||||
|
@ -564,48 +570,56 @@ class pil_build_ext(build_ext):
|
||||||
)
|
)
|
||||||
|
|
||||||
elif sys.platform == "darwin":
|
elif sys.platform == "darwin":
|
||||||
# attempt to make sure we pick freetype2 over other versions
|
if self.dependencies_prefix:
|
||||||
_add_directory(include_dirs, "/sw/include/freetype2")
|
# Use the explicitly provided prefixes for dependencies.
|
||||||
_add_directory(include_dirs, "/sw/lib/freetype2/include")
|
for prefix in self.dependencies_prefix:
|
||||||
# fink installation directories
|
_add_directory(library_dirs, os.path.join(prefix, "lib"))
|
||||||
_add_directory(library_dirs, "/sw/lib")
|
_add_directory(include_dirs, os.path.join(prefix, "include"))
|
||||||
_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:
|
else:
|
||||||
# fall back to freetype from XQuartz if
|
# Guess the dependency locations based on homebrew/fink/macports
|
||||||
# Homebrew's freetype is missing
|
# attempt to make sure we pick freetype2 over other versions
|
||||||
_add_directory(library_dirs, "/usr/X11/lib")
|
_add_directory(include_dirs, "/sw/include/freetype2")
|
||||||
_add_directory(include_dirs, "/usr/X11/include")
|
_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()
|
sdk_path = self.get_macos_sdk_path()
|
||||||
if sdk_path:
|
if sdk_path:
|
||||||
_add_directory(library_dirs, os.path.join(sdk_path, "usr", "lib"))
|
_add_directory(library_dirs, os.path.join(sdk_path, "usr", "lib"))
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
Subproject commit 452dd2d1705f6b2375369a6570c415beb3163f70
|
Subproject commit 8a9333e307fa3f24db2089a66f25e38a8272c261
|
Loading…
Reference in New Issue
Block a user