Skip to content

Commit 29cb138

Browse files
committed
Merge remote-tracking branch 'origin/main' into env-order
2 parents 391249a + 099d397 commit 29cb138

37 files changed

+462
-129
lines changed

.github/workflows/test.yml

+8-1
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,13 @@ jobs:
6262
sudo apt-get update
6363
sudo apt-get -y install podman
6464
65+
# free some space to prevent reaching GHA disk space limits
66+
- name: Clean docker images
67+
if: runner.os == 'Linux'
68+
run: |
69+
docker system prune -a -f
70+
df -h
71+
6572
- name: Install dependencies
6673
run: |
6774
python -m pip install ".[test]"
@@ -103,7 +110,7 @@ jobs:
103110
104111
- name: Set up QEMU
105112
id: qemu
106-
uses: docker/setup-qemu-action@v2
113+
uses: docker/setup-qemu-action@v3
107114
with:
108115
platforms: all
109116

.pre-commit-config.yaml

+1-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ repos:
1919
- id: black
2020

2121
- repo: https://github.com/astral-sh/ruff-pre-commit
22-
rev: v0.0.288
22+
rev: v0.0.290
2323
hooks:
2424
- id: ruff
2525
args: ["--fix", "--show-fixes"]

README.md

+13-11
Original file line numberDiff line numberDiff line change
@@ -97,7 +97,7 @@ jobs:
9797
- uses: actions/setup-python@v3
9898

9999
- name: Install cibuildwheel
100-
run: python -m pip install cibuildwheel==2.15.0
100+
run: python -m pip install cibuildwheel==2.16.0
101101

102102
- name: Build wheels
103103
run: python -m cibuildwheel --output-dir wheelhouse
@@ -213,6 +213,18 @@ Changelog
213213

214214
<!-- this section was generated by bin/update_readme_changelog.py -- do not edit manually -->
215215

216+
### v2.16.0
217+
218+
_18 September 2023_
219+
220+
- ✨ Add the ability to pass additional flags to a build frontend through the [CIBW_BUILD_FRONTEND](https://cibuildwheel.readthedocs.io/en/stable/options/#build-frontend) option (#1588).
221+
- ✨ The environment variable SOURCE_DATE_EPOCH is now automatically passed through to container Linux builds (useful for [reproducible builds](https://reproducible-builds.org/docs/source-date-epoch/)!) (#1589)
222+
- 🛠 Updates the prerelease CPython 3.12 version to 3.12.0rc2 (#1604)
223+
- 🐛 Fix `requires_python` auto-detection from setup.py when the call to `setup()` is within an `if __name__ == "__main__" block (#1613)
224+
- 🐛 Fix a bug that prevented building Linux wheels in Docker on a Windows host (#1573)
225+
- 🐛 `--only` can now select prerelease-pythons (#1564)
226+
- 📚 Docs & examples updates (#1582, #1593, #1598, #1615)
227+
216228
### v2.15.0
217229

218230
_8 August 2023_
@@ -243,16 +255,6 @@ _10 June 2023_
243255
- 🛠 Updates the prerelease CPython 3.12 version to 3.12.0b2. (#1516)
244256
- 🛠 Adds a moving `v<major>.<minor>` tag for use in GitHub Actions workflow files. If you use this, you'll get the latest patch release within a minor version. Additionally, Dependabot won't send you PRs for patch releases. (#1517)
245257

246-
### v2.13.0
247-
248-
_28 May 2023_
249-
250-
- ✨ Adds CPython 3.12 support, under the prerelease flag [CIBW_PRERELEASE_PYTHONS](https://cibuildwheel.readthedocs.io/en/stable/options/#prerelease-pythons). This version of cibuildwheel uses 3.12.0b1.
251-
252-
While CPython is in beta, the ABI can change, so your wheels might not be compatible with the final release. For this reason, we don't recommend distributing wheels until RC1, at which point 3.12 will be available in cibuildwheel without the flag. (#1507)
253-
254-
- ✨ Adds the ability to pass arguments to the container engine when the container is created, using the [CIBW_CONTAINER_ENGINE](https://cibuildwheel.readthedocs.io/en/stable/options/#container-engine) option. (#1499)
255-
256258
<!-- END bin/update_readme_changelog.py -->
257259

258260
---

cibuildwheel/__init__.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
11
from __future__ import annotations
22

3-
__version__ = "2.15.0"
3+
__version__ = "2.16.0"

cibuildwheel/linux.py

+7-6
Original file line numberDiff line numberDiff line change
@@ -16,9 +16,9 @@
1616
from .typing import PathOrStr
1717
from .util import (
1818
AlreadyBuiltWheelError,
19+
BuildFrontendConfig,
1920
BuildSelector,
2021
NonPlatformWheelError,
21-
build_frontend_or_default,
2222
find_compatible_wheel,
2323
get_build_verbosity_extra_flags,
2424
prepare_command,
@@ -177,7 +177,7 @@ def build_in_container(
177177
for config in platform_configs:
178178
log.build_start(config.identifier)
179179
build_options = options.build_options(config.identifier)
180-
build_frontend = build_frontend_or_default(build_options.build_frontend)
180+
build_frontend = build_options.build_frontend or BuildFrontendConfig("pip")
181181

182182
dependency_constraint_flags: list[PathOrStr] = []
183183

@@ -243,9 +243,10 @@ def build_in_container(
243243
container.call(["rm", "-rf", built_wheel_dir])
244244
container.call(["mkdir", "-p", built_wheel_dir])
245245

246-
extra_flags = split_config_settings(build_options.config_settings, build_frontend)
246+
extra_flags = split_config_settings(build_options.config_settings, build_frontend.name)
247+
extra_flags += build_frontend.args
247248

248-
if build_frontend == "pip":
249+
if build_frontend.name == "pip":
249250
extra_flags += get_build_verbosity_extra_flags(build_options.build_verbosity)
250251
container.call(
251252
[
@@ -260,7 +261,7 @@ def build_in_container(
260261
],
261262
env=env,
262263
)
263-
elif build_frontend == "build":
264+
elif build_frontend.name == "build":
264265
if not 0 <= build_options.build_verbosity < 2:
265266
msg = f"build_verbosity {build_options.build_verbosity} is not supported for build frontend. Ignoring."
266267
log.warning(msg)
@@ -422,7 +423,7 @@ def build(options: Options, tmp_path: Path) -> None: # noqa: ARG001
422423

423424
with OCIContainer(
424425
image=build_step.container_image,
425-
simulate_32_bit=build_step.platform_tag.endswith("i686"),
426+
enforce_32_bit=build_step.platform_tag.endswith("i686"),
426427
cwd=container_project_path,
427428
engine=options.globals.container_engine,
428429
) as container:

cibuildwheel/macos.py

+11-8
Original file line numberDiff line numberDiff line change
@@ -25,10 +25,10 @@
2525
from .util import (
2626
CIBW_CACHE_PATH,
2727
AlreadyBuiltWheelError,
28-
BuildFrontend,
28+
BuildFrontendConfig,
29+
BuildFrontendName,
2930
BuildSelector,
3031
NonPlatformWheelError,
31-
build_frontend_or_default,
3232
call,
3333
detect_ci_provider,
3434
download,
@@ -165,7 +165,7 @@ def setup_python(
165165
python_configuration: PythonConfiguration,
166166
dependency_constraint_flags: Sequence[PathOrStr],
167167
environment: ParsedEnvironment,
168-
build_frontend: BuildFrontend,
168+
build_frontend: BuildFrontendName,
169169
) -> dict[str, str]:
170170
tmp.mkdir()
171171
implementation_id = python_configuration.identifier.split("-")[0]
@@ -334,7 +334,7 @@ def build(options: Options, tmp_path: Path) -> None:
334334

335335
for config in python_configurations:
336336
build_options = options.build_options(config.identifier)
337-
build_frontend = build_frontend_or_default(build_options.build_frontend)
337+
build_frontend = build_options.build_frontend or BuildFrontendConfig("pip")
338338
log.build_start(config.identifier)
339339

340340
identifier_tmp_dir = tmp_path / config.identifier
@@ -357,7 +357,7 @@ def build(options: Options, tmp_path: Path) -> None:
357357
config,
358358
dependency_constraint_flags,
359359
build_options.environment,
360-
build_frontend,
360+
build_frontend.name,
361361
)
362362

363363
compatible_wheel = find_compatible_wheel(built_wheels, config.identifier)
@@ -378,9 +378,12 @@ def build(options: Options, tmp_path: Path) -> None:
378378
log.step("Building wheel...")
379379
built_wheel_dir.mkdir()
380380

381-
extra_flags = split_config_settings(build_options.config_settings, build_frontend)
381+
extra_flags = split_config_settings(
382+
build_options.config_settings, build_frontend.name
383+
)
384+
extra_flags += build_frontend.args
382385

383-
if build_frontend == "pip":
386+
if build_frontend.name == "pip":
384387
extra_flags += get_build_verbosity_extra_flags(build_options.build_verbosity)
385388
# Path.resolve() is needed. Without it pip wheel may try to fetch package from pypi.org
386389
# see https://github.com/pypa/cibuildwheel/pull/369
@@ -395,7 +398,7 @@ def build(options: Options, tmp_path: Path) -> None:
395398
*extra_flags,
396399
env=env,
397400
)
398-
elif build_frontend == "build":
401+
elif build_frontend.name == "build":
399402
if not 0 <= build_options.build_verbosity < 2:
400403
msg = f"build_verbosity {build_options.build_verbosity} is not supported for build frontend. Ignoring."
401404
log.warning(msg)

cibuildwheel/oci_container.py

+18-5
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@
1717
from typing import IO, Dict, Literal
1818

1919
from .typing import PathOrStr, PopenBytes
20-
from .util import CIProvider, detect_ci_provider, parse_key_value_string
20+
from .util import CIProvider, call, detect_ci_provider, parse_key_value_string
2121

2222
ContainerEngineName = Literal["docker", "podman"]
2323

@@ -29,7 +29,9 @@ class OCIContainerEngineConfig:
2929

3030
@staticmethod
3131
def from_config_string(config_string: str) -> OCIContainerEngineConfig:
32-
config_dict = parse_key_value_string(config_string, ["name"])
32+
config_dict = parse_key_value_string(
33+
config_string, ["name"], ["create_args", "create-args"]
34+
)
3335
name = " ".join(config_dict["name"])
3436
if name not in {"docker", "podman"}:
3537
msg = f"unknown container engine {name}"
@@ -83,7 +85,7 @@ def __init__(
8385
self,
8486
*,
8587
image: str,
86-
simulate_32_bit: bool = False,
88+
enforce_32_bit: bool = False,
8789
cwd: PathOrStr | None = None,
8890
engine: OCIContainerEngineConfig = DEFAULT_ENGINE,
8991
):
@@ -92,7 +94,7 @@ def __init__(
9294
raise ValueError(msg)
9395

9496
self.image = image
95-
self.simulate_32_bit = simulate_32_bit
97+
self.enforce_32_bit = enforce_32_bit
9698
self.cwd = cwd
9799
self.name: str | None = None
98100
self.engine = engine
@@ -108,13 +110,24 @@ def __enter__(self) -> OCIContainer:
108110
if detect_ci_provider() == CIProvider.travis_ci and platform.machine() == "ppc64le":
109111
network_args = ["--network=host"]
110112

111-
shell_args = ["linux32", "/bin/bash"] if self.simulate_32_bit else ["/bin/bash"]
113+
simulate_32_bit = False
114+
if self.enforce_32_bit:
115+
# If the architecture running the image is already the right one
116+
# or the image entrypoint takes care of enforcing this, then we don't need to
117+
# simulate this
118+
container_machine = call(
119+
self.engine.name, "run", "--rm", self.image, "uname", "-m", capture_stdout=True
120+
).strip()
121+
simulate_32_bit = container_machine != "i686"
122+
123+
shell_args = ["linux32", "/bin/bash"] if simulate_32_bit else ["/bin/bash"]
112124

113125
subprocess.run(
114126
[
115127
self.engine.name,
116128
"create",
117129
"--env=CIBUILDWHEEL",
130+
"--env=SOURCE_DATE_EPOCH",
118131
f"--name={self.name}",
119132
"--interactive",
120133
"--volume=/:/host", # ignored on CircleCI

cibuildwheel/options.py

+15-13
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@
2727
from .util import (
2828
MANYLINUX_ARCHS,
2929
MUSLLINUX_ARCHS,
30-
BuildFrontend,
30+
BuildFrontendConfig,
3131
BuildSelector,
3232
DependencyConstraints,
3333
TestSelector,
@@ -92,7 +92,7 @@ class BuildOptions:
9292
test_requires: list[str]
9393
test_extras: str
9494
build_verbosity: int
95-
build_frontend: BuildFrontend | Literal["default"]
95+
build_frontend: BuildFrontendConfig | None
9696
config_settings: str
9797

9898
@property
@@ -488,7 +488,6 @@ def build_options(self, identifier: str | None) -> BuildOptions:
488488
with self.reader.identifier(identifier):
489489
before_all = self.reader.get("before-all", sep=" && ")
490490

491-
build_frontend_str = self.reader.get("build-frontend", env_plat=False)
492491
environment_config = self.reader.get(
493492
"environment", table={"item": '{k}="{v}"', "sep": " "}
494493
)
@@ -506,17 +505,20 @@ def build_options(self, identifier: str | None) -> BuildOptions:
506505
test_extras = self.reader.get("test-extras", sep=",")
507506
build_verbosity_str = self.reader.get("build-verbosity")
508507

509-
build_frontend: BuildFrontend | Literal["default"]
510-
if build_frontend_str == "build":
511-
build_frontend = "build"
512-
elif build_frontend_str == "pip":
513-
build_frontend = "pip"
514-
elif build_frontend_str == "default":
515-
build_frontend = "default"
508+
build_frontend_str = self.reader.get(
509+
"build-frontend",
510+
env_plat=False,
511+
table={"item": "{k}:{v}", "sep": "; ", "quote": shlex.quote},
512+
)
513+
build_frontend: BuildFrontendConfig | None
514+
if not build_frontend_str or build_frontend_str == "default":
515+
build_frontend = None
516516
else:
517-
msg = f"cibuildwheel: Unrecognised build frontend {build_frontend_str!r}, only 'pip' and 'build' are supported"
518-
print(msg, file=sys.stderr)
519-
sys.exit(2)
517+
try:
518+
build_frontend = BuildFrontendConfig.from_config_string(build_frontend_str)
519+
except ValueError as e:
520+
print(f"cibuildwheel: {e}", file=sys.stderr)
521+
sys.exit(2)
520522

521523
try:
522524
environment = parse_environment(environment_config)

cibuildwheel/projectfiles.py

+48-2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,43 @@
88
from ._compat import tomllib
99

1010

11+
def get_parent(node: ast.AST | None, depth: int = 1) -> ast.AST | None:
12+
for _ in range(depth):
13+
node = getattr(node, "parent", None)
14+
return node
15+
16+
17+
def is_main(parent: ast.AST | None) -> bool:
18+
if parent is None:
19+
return False
20+
21+
# This would be much nicer with 3.10's pattern matching!
22+
if not isinstance(parent, ast.If):
23+
return False
24+
if not isinstance(parent.test, ast.Compare):
25+
return False
26+
27+
try:
28+
(op,) = parent.test.ops
29+
(comp,) = parent.test.comparators
30+
except ValueError:
31+
return False
32+
33+
if not isinstance(op, ast.Eq):
34+
return False
35+
36+
values = {comp, parent.test.left}
37+
38+
mains = {x for x in values if isinstance(x, ast.Constant) and x.value == "__main__"}
39+
if len(mains) != 1:
40+
return False
41+
consts = {x for x in values if isinstance(x, ast.Name) and x.id == "__name__"}
42+
if len(consts) != 1:
43+
return False
44+
45+
return True
46+
47+
1148
class Analyzer(ast.NodeVisitor):
1249
def __init__(self) -> None:
1350
self.requires_python: str | None = None
@@ -19,13 +56,22 @@ def visit(self, node: ast.AST) -> None:
1956
super().visit(node)
2057

2158
def visit_keyword(self, node: ast.keyword) -> None:
59+
# Must not be nested except for if __name__ == "__main__"
60+
2261
self.generic_visit(node)
23-
# Must not be nested in an if or other structure
2462
# This will be Module -> Expr -> Call -> keyword
63+
parent = get_parent(node, 4)
64+
unnested = parent is None
65+
66+
# This will be Module -> If -> Expr -> Call -> keyword
67+
name_main_unnested = (
68+
parent is not None and get_parent(parent) is None and is_main(get_parent(node, 3))
69+
)
70+
2571
if (
2672
node.arg == "python_requires"
27-
and not hasattr(node.parent.parent.parent, "parent") # type: ignore[attr-defined]
2873
and isinstance(node.value, ast.Constant)
74+
and (unnested or name_main_unnested)
2975
):
3076
self.requires_python = node.value.value
3177

0 commit comments

Comments
 (0)