Skip to content

Commit 316f397

Browse files
authored
Merge pull request #7580 from nulano/cibuildwheel-docker
Build Windows wheels using cibuildwheel
2 parents 63bec07 + e105976 commit 316f397

10 files changed

+196
-73
lines changed

.github/workflows/test-windows.yml

+3-45
Original file line numberDiff line numberDiff line change
@@ -72,10 +72,10 @@ jobs:
7272
- name: Install dependencies
7373
id: install
7474
run: |
75-
7z x winbuild\depends\nasm-2.16.01-win64.zip "-o$env:RUNNER_WORKSPACE\"
76-
echo "$env:RUNNER_WORKSPACE\nasm-2.16.01" >> $env:GITHUB_PATH
75+
choco install nasm --no-progress
76+
echo "C:\Program Files\NASM" >> $env:GITHUB_PATH
7777
78-
choco install ghostscript --version=10.0.0.20230317
78+
choco install ghostscript --version=10.0.0.20230317 --no-progress
7979
echo "C:\Program Files\gs\gs10.00.0\bin" >> $env:GITHUB_PATH
8080
8181
# Install extra test images
@@ -167,7 +167,6 @@ jobs:
167167
- name: Build Pillow
168168
run: |
169169
$FLAGS="-C raqm=vendor -C fribidi=vendor"
170-
if ('${{ github.event_name }}' -ne 'pull_request') { $FLAGS+=" -C imagequant=disable" }
171170
cmd /c "winbuild\build\build_env.cmd && $env:pythonLocation\python.exe -m pip install -v $FLAGS ."
172171
& $env:pythonLocation\python.exe selftest.py --installed
173172
shell: pwsh
@@ -209,47 +208,6 @@ jobs:
209208
flags: GHA_Windows
210209
name: ${{ runner.os }} Python ${{ matrix.python-version }}
211210

212-
- name: Build wheel
213-
id: wheel
214-
if: "github.event_name != 'pull_request'"
215-
run: |
216-
mkdir fribidi
217-
copy winbuild\build\bin\fribidi* fribidi
218-
setlocal EnableDelayedExpansion
219-
for %%f in (winbuild\build\license\*) do (
220-
set x=%%~nf
221-
rem Skip FriBiDi license, it is not included in the wheel.
222-
set fribidi=!x:~0,7!
223-
if NOT !fribidi!==fribidi (
224-
rem Skip imagequant license, it is not included in the wheel.
225-
set libimagequant=!x:~0,13!
226-
if NOT !libimagequant!==libimagequant (
227-
echo. >> LICENSE
228-
echo ===== %%~nf ===== >> LICENSE
229-
echo. >> LICENSE
230-
type %%f >> LICENSE
231-
)
232-
)
233-
)
234-
for /f "tokens=3 delims=/" %%a in ("${{ github.ref }}") do echo dist=dist-%%a >> %GITHUB_OUTPUT%
235-
call winbuild\\build\\build_env.cmd
236-
%pythonLocation%\python.exe -m pip wheel -v -C raqm=vendor -C fribidi=vendor -C imagequant=disable .
237-
shell: cmd
238-
239-
- name: Upload wheel
240-
uses: actions/upload-artifact@v3
241-
if: "github.event_name != 'pull_request'"
242-
with:
243-
name: ${{ steps.wheel.outputs.dist }}
244-
path: "*.whl"
245-
246-
- name: Upload fribidi.dll
247-
if: "github.event_name != 'pull_request' && matrix.python-version == 3.11"
248-
uses: actions/upload-artifact@v3
249-
with:
250-
name: fribidi
251-
path: fribidi\*
252-
253211
success:
254212
permissions:
255213
contents: none

.github/workflows/wheels-test.ps1

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
param ([string]$venv, [string]$pillow="C:\pillow")
2+
$ErrorActionPreference = 'Stop'
3+
$ProgressPreference = 'SilentlyContinue'
4+
Set-PSDebug -Trace 1
5+
if ("$venv" -like "*\cibw-run-*\pp*-win_amd64\*") {
6+
# unlike CPython, PyPy requires Visual C++ Redistributable to be installed
7+
[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12
8+
Invoke-WebRequest -Uri 'https://aka.ms/vs/15/release/vc_redist.x64.exe' -OutFile 'vc_redist.x64.exe'
9+
C:\vc_redist.x64.exe /install /quiet /norestart | Out-Null
10+
}
11+
$env:path += ";$pillow\winbuild\build\bin\"
12+
& "$venv\Scripts\activate.ps1"
13+
& reg add "HKLM\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\python.exe" /v "GlobalFlag" /t REG_SZ /d "0x02000000" /f
14+
cd $pillow
15+
& python -VV
16+
if (!$?) { exit $LASTEXITCODE }
17+
& python selftest.py
18+
if (!$?) { exit $LASTEXITCODE }
19+
& python -m pytest -vx Tests\check_wheel.py
20+
if (!$?) { exit $LASTEXITCODE }
21+
& python -m pytest -vx Tests
22+
if (!$?) { exit $LASTEXITCODE }

.github/workflows/wheels-test.sh

+1-21
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,6 @@
11
#!/bin/bash
22
set -e
33

4-
EXP_CODECS="jpg jpg_2000 libtiff zlib"
5-
EXP_MODULES="freetype2 littlecms2 pil tkinter webp"
6-
EXP_FEATURES="fribidi harfbuzz libjpeg_turbo raqm transp_webp webp_anim webp_mux xcb"
7-
84
if [[ "$OSTYPE" == "darwin"* ]]; then
95
brew install fribidi
106
export PKG_CONFIG_PATH="/usr/local/opt/openblas/lib/pkgconfig"
@@ -25,21 +21,5 @@ fi
2521

2622
# Runs tests
2723
python3 selftest.py
24+
python3 -m pytest Tests/check_wheel.py
2825
python3 -m pytest
29-
30-
# Test against expected codecs, modules and features
31-
codecs=$(python3 -c 'from PIL.features import *; print(" ".join(sorted(get_supported_codecs())))')
32-
if [ "$codecs" != "$EXP_CODECS" ]; then
33-
echo "Codecs should be: '$EXP_CODECS'; but are '$codecs'"
34-
exit 1
35-
fi
36-
modules=$(python3 -c 'from PIL.features import *; print(" ".join(sorted(get_supported_modules())))')
37-
if [ "$modules" != "$EXP_MODULES" ]; then
38-
echo "Modules should be: '$EXP_MODULES'; but are '$modules'"
39-
exit 1
40-
fi
41-
features=$(python3 -c 'from PIL.features import *; print(" ".join(sorted(get_supported_features())))')
42-
if [ "$features" != "$EXP_FEATURES" ]; then
43-
echo "Features should be: '$EXP_FEATURES'; but are '$features'"
44-
exit 1
45-
fi

.github/workflows/wheels.yml

+105-4
Original file line numberDiff line numberDiff line change
@@ -3,14 +3,20 @@ name: Wheels
33
on:
44
push:
55
paths:
6-
- ".github/workflows/wheels*.yml"
6+
- ".ci/requirements-cibw.txt"
7+
- ".github/workflows/wheel*"
78
- "wheels/*"
9+
- "winbuild/build_prepare.py"
10+
- "winbuild/fribidi.cmake"
811
tags:
912
- "*"
1013
pull_request:
1114
paths:
12-
- ".github/workflows/wheels*.yml"
15+
- ".ci/requirements-cibw.txt"
16+
- ".github/workflows/wheel*"
1317
- "wheels/*"
18+
- "winbuild/build_prepare.py"
19+
- "winbuild/fribidi.cmake"
1420
workflow_dispatch:
1521

1622
permissions:
@@ -63,7 +69,6 @@ jobs:
6369
env:
6470
CIBW_ARCHS: ${{ matrix.archs }}
6571
CIBW_BUILD: ${{ matrix.build }}
66-
CIBW_CONFIG_SETTINGS: raqm=enable raqm=vendor fribidi=vendor
6772
CIBW_MANYLINUX_PYPY_X86_64_IMAGE: ${{ matrix.manylinux }}
6873
CIBW_MANYLINUX_X86_64_IMAGE: ${{ matrix.manylinux }}
6974
CIBW_SKIP: pp38-*
@@ -75,6 +80,102 @@ jobs:
7580
name: dist
7681
path: ./wheelhouse/*.whl
7782

83+
windows:
84+
name: Windows ${{ matrix.arch }}
85+
runs-on: windows-latest
86+
strategy:
87+
fail-fast: false
88+
matrix:
89+
include:
90+
- arch: x86
91+
cibw_arch: x86
92+
- arch: x64
93+
cibw_arch: AMD64
94+
- arch: ARM64
95+
cibw_arch: ARM64
96+
steps:
97+
- uses: actions/checkout@v4
98+
99+
- name: Checkout extra test images
100+
uses: actions/checkout@v4
101+
with:
102+
repository: python-pillow/test-images
103+
path: Tests\test-images
104+
105+
- uses: actions/setup-python@v4
106+
with:
107+
python-version: "3.x"
108+
109+
- name: Prepare for build
110+
run: |
111+
choco install nasm --no-progress
112+
echo "C:\Program Files\NASM" >> $env:GITHUB_PATH
113+
114+
# Install extra test images
115+
xcopy /S /Y Tests\test-images\* Tests\images
116+
117+
& python.exe -m pip install -r .ci/requirements-cibw.txt
118+
119+
# Cannot cross-compile FriBiDi (only used for tests)
120+
$FLAGS = ("--no-imagequant", "--architecture=${{ matrix.arch }}")
121+
if ('${{ matrix.arch }}' -eq 'ARM64') { $FLAGS += "--no-fribidi" }
122+
& python.exe winbuild\build_prepare.py -v @FLAGS
123+
shell: pwsh
124+
125+
- name: Build wheels
126+
run: |
127+
setlocal EnableDelayedExpansion
128+
for %%f in (winbuild\build\license\*) do (
129+
set x=%%~nf
130+
rem Skip FriBiDi license, it is not included in the wheel.
131+
set fribidi=!x:~0,7!
132+
if NOT !fribidi!==fribidi (
133+
rem Skip imagequant license, it is not included in the wheel.
134+
set libimagequant=!x:~0,13!
135+
if NOT !libimagequant!==libimagequant (
136+
echo. >> LICENSE
137+
echo ===== %%~nf ===== >> LICENSE
138+
echo. >> LICENSE
139+
type %%f >> LICENSE
140+
)
141+
)
142+
)
143+
call winbuild\\build\\build_env.cmd
144+
%pythonLocation%\python.exe -m cibuildwheel . --output-dir wheelhouse
145+
env:
146+
CIBW_ARCHS: ${{ matrix.cibw_arch }}
147+
CIBW_BEFORE_ALL: "{package}\\winbuild\\build\\build_dep_all.cmd"
148+
CIBW_CACHE_PATH: "C:\\cibw"
149+
CIBW_TEST_SKIP: "*-win_arm64"
150+
CIBW_TEST_COMMAND: 'docker run --rm
151+
-v {project}:C:\pillow
152+
-v C:\cibw:C:\cibw
153+
-v %CD%\..\venv-test:%CD%\..\venv-test
154+
-e CI -e GITHUB_ACTIONS
155+
mcr.microsoft.com/windows/servercore:ltsc2022
156+
powershell C:\pillow\.github\workflows\wheels-test.ps1 %CD%\..\venv-test'
157+
shell: cmd
158+
159+
- name: Upload wheels
160+
uses: actions/upload-artifact@v3
161+
with:
162+
name: dist
163+
path: ./wheelhouse/*.whl
164+
165+
- name: Prepare to upload FriBiDi
166+
if: "matrix.arch != 'ARM64'"
167+
run: |
168+
mkdir fribidi\${{ matrix.arch }}
169+
copy winbuild\build\bin\fribidi* fribidi\${{ matrix.arch }}
170+
shell: cmd
171+
172+
- name: Upload fribidi.dll
173+
if: "matrix.arch != 'ARM64'"
174+
uses: actions/upload-artifact@v3
175+
with:
176+
name: fribidi
177+
path: fribidi\*
178+
78179
sdist:
79180
runs-on: ubuntu-latest
80181
steps:
@@ -97,7 +198,7 @@ jobs:
97198
success:
98199
permissions:
99200
contents: none
100-
needs: [build, sdist]
201+
needs: [build, windows, sdist]
101202
runs-on: ubuntu-latest
102203
name: Wheels Successful
103204
steps:

.travis.yml

-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@ if: tag IS present OR type = api
33
env:
44
global:
55
- CIBW_ARCHS=aarch64
6-
- CIBW_CONFIG_SETTINGS="raqm=enable raqm=vendor fribidi=vendor"
76
- CIBW_SKIP=pp38-*
87

98
language: python

Tests/check_wheel.py

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
import sys
2+
3+
from PIL import features
4+
5+
6+
def test_wheel_modules():
7+
expected_modules = {"pil", "tkinter", "freetype2", "littlecms2", "webp"}
8+
9+
# tkinter is not available in cibuildwheel installed CPython on Windows
10+
try:
11+
import tkinter
12+
13+
assert tkinter
14+
except ImportError:
15+
expected_modules.remove("tkinter")
16+
17+
assert set(features.get_supported_modules()) == expected_modules
18+
19+
20+
def test_wheel_codecs():
21+
expected_codecs = {"jpg", "jpg_2000", "zlib", "libtiff"}
22+
23+
assert set(features.get_supported_codecs()) == expected_codecs
24+
25+
26+
def test_wheel_features():
27+
expected_features = {
28+
"webp_anim",
29+
"webp_mux",
30+
"transp_webp",
31+
"raqm",
32+
"fribidi",
33+
"harfbuzz",
34+
"libjpeg_turbo",
35+
"xcb",
36+
}
37+
38+
if sys.platform == "win32":
39+
expected_features.remove("xcb")
40+
41+
assert set(features.get_supported_features()) == expected_features

Tests/helper.py

+13-2
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
import logging
66
import os
77
import shutil
8+
import subprocess
89
import sys
910
import sysconfig
1011
import tempfile
@@ -258,11 +259,21 @@ def hopper(mode=None, cache={}):
258259

259260

260261
def djpeg_available():
261-
return bool(shutil.which("djpeg"))
262+
if shutil.which("djpeg"):
263+
try:
264+
subprocess.check_call(["djpeg", "-version"])
265+
return True
266+
except subprocess.CalledProcessError: # pragma: no cover
267+
return False
262268

263269

264270
def cjpeg_available():
265-
return bool(shutil.which("cjpeg"))
271+
if shutil.which("cjpeg"):
272+
try:
273+
subprocess.check_call(["cjpeg", "-version"])
274+
return True
275+
except subprocess.CalledProcessError: # pragma: no cover
276+
return False
266277

267278

268279
def netpbm_available():

Tests/test_imagegrab.py

+4
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,10 @@
1111

1212

1313
class TestImageGrab:
14+
@pytest.mark.skipif(
15+
os.environ.get("USERNAME") == "ContainerAdministrator",
16+
reason="can't grab screen when running in Docker",
17+
)
1418
@pytest.mark.skipif(
1519
sys.platform not in ("win32", "darwin"), reason="requires Windows or macOS"
1620
)

pyproject.toml

+2
Original file line numberDiff line numberDiff line change
@@ -88,6 +88,8 @@ version = {attr = "PIL.__version__"}
8888

8989
[tool.cibuildwheel]
9090
before-all = ".github/workflows/wheels-dependencies.sh"
91+
build-verbosity = 1
92+
config-settings = "raqm=enable raqm=vendor fribidi=vendor imagequant=disable"
9193
test-command = "cd {project} && .github/workflows/wheels-test.sh"
9294
test-extras = "tests"
9395

winbuild/build_prepare.py

+5
Original file line numberDiff line numberDiff line change
@@ -586,14 +586,19 @@ def build_dep(name: str) -> str:
586586

587587
def build_dep_all() -> None:
588588
lines = [r'call "{build_dir}\build_env.cmd"']
589+
gha_groups = "GITHUB_ACTIONS" in os.environ
589590
for dep_name in DEPS:
590591
print()
591592
if dep_name in disabled:
592593
print(f"Skipping disabled dependency {dep_name}")
593594
continue
594595
script = build_dep(dep_name)
596+
if gha_groups:
597+
lines.append(f"@echo ::group::Running {script}")
595598
lines.append(rf'cmd.exe /c "{{build_dir}}\{script}"')
596599
lines.append("if errorlevel 1 echo Build failed! && exit /B 1")
600+
if gha_groups:
601+
lines.append("@echo ::endgroup::")
597602
print()
598603
lines.append("@echo All Pillow dependencies built successfully!")
599604
write_script("build_dep_all.cmd", lines)

0 commit comments

Comments
 (0)