1
1
from __future__ import annotations
2
2
3
3
import dataclasses
4
+ import hashlib
5
+ import os
4
6
import re
5
7
import sys
6
8
import sysconfig
9
+ import tarfile
10
+ import tempfile
7
11
from collections .abc import Iterable , Mapping , Sequence
8
12
from pathlib import Path
13
+ from typing import BinaryIO
9
14
10
15
from packaging .version import Version
11
16
22
27
get_python_library ,
23
28
get_soabi ,
24
29
)
30
+ from .wheel_tag import WheelTag
25
31
26
32
__all__ : list [str ] = ["Builder" , "get_archs" , "archs_to_tags" ]
27
33
@@ -64,6 +70,83 @@ def archs_to_tags(archs: list[str]) -> list[str]:
64
70
return archs
65
71
66
72
73
+ @dataclasses .dataclass (init = False )
74
+ class BuildEnvArchive :
75
+ _archive_file : BinaryIO
76
+ hash : hashlib ._Hash
77
+
78
+ def __init__ (self , env_dir : Path ) -> None :
79
+ self ._archive_file = tempfile .TemporaryFile (prefix = "build-env-archive-" , suffix = ".tar" ) # type: ignore[assignment]
80
+
81
+ # Rewrite environment path to be relative to root
82
+ # Example:
83
+ # /tmp/pip-build-env-pklovjqz/overlay/lib/python3.11/site-packages
84
+ # is rewritten into
85
+ # tmp/pip-build-env-pklovjqz/overlay/lib/python3.11/site-packages
86
+ prefix = Path (env_dir )
87
+ prefix = prefix .relative_to (prefix .root )
88
+
89
+ def ext_filter (ti : tarfile .TarInfo ) -> tarfile .TarInfo | None :
90
+ pname = Path (ti .name )
91
+
92
+ if ti .type is tarfile .LNKTYPE :
93
+ logger .warning (
94
+ "Unexpected link inside build environment archive (path={})" , pname
95
+ )
96
+ elif (
97
+ ti .type is not tarfile .REGTYPE
98
+ and ti .type is not tarfile .AREGTYPE
99
+ and ti .type is not tarfile .DIRTYPE
100
+ ):
101
+ logger .warning (
102
+ "Unexpected file type inside build environment archive (path={})" ,
103
+ pname ,
104
+ )
105
+
106
+ # Rewrite name to be relative to site-packages inside the build environment
107
+ ti .name = str (pname .relative_to (prefix ))
108
+
109
+ # FIXME: __pycache__ files don't have consistent hashes - why?
110
+ if "__pycache__" in ti .name :
111
+ return None
112
+
113
+ # Reset mtime to zero
114
+ # This is safe (regarding build tool out-of-date detection)
115
+ # since the resulting archive is content-addressed through its hash
116
+ ti .mtime = 0
117
+
118
+ return ti
119
+
120
+ with tarfile .open (
121
+ fileobj = self ._archive_file , mode = "x" , dereference = True
122
+ ) as dir_tar :
123
+ dir_tar .add (env_dir , filter = ext_filter )
124
+
125
+ self ._archive_file .flush ()
126
+
127
+ archive_len = self ._archive_file .tell ()
128
+ self ._archive_file .seek (0 )
129
+
130
+ self .hash = hashlib .file_digest (self ._archive_file , hashlib .sha256 ) # type: ignore[attr-defined]
131
+ self ._archive_file .seek (0 )
132
+
133
+ logger .debug (
134
+ "created build env archive len={} sha256={}" ,
135
+ archive_len ,
136
+ self .hash .hexdigest (),
137
+ )
138
+
139
+ def extract (self , destination : Path ) -> None :
140
+ self ._archive_file .seek (0 )
141
+ with tarfile .open (fileobj = self ._archive_file , mode = "r" ) as dir_tar :
142
+ dir_tar .extractall (path = destination )
143
+
144
+ # Reset atime/mtime of the destination directory
145
+ # Otherwise CMake would consider the directory out of date
146
+ # FIXME: Apparently not necessary?
147
+ # os.utime(destination, times=(0,0))
148
+
149
+
67
150
@dataclasses .dataclass
68
151
class Builder :
69
152
settings : ScikitBuildSettings
@@ -79,6 +162,31 @@ def get_cmake_args(self) -> list[str]:
79
162
80
163
return [* self .settings .cmake .args , * env_cmake_args ]
81
164
165
+ # FIXME: Proper setting for build env dir
166
+ def _build_dir (self ) -> Path :
167
+ tags = WheelTag .compute_best (
168
+ archs_to_tags (get_archs (os .environ )),
169
+ self .settings .wheel .py_api ,
170
+ expand_macos = self .settings .wheel .expand_macos_universal_tags ,
171
+ )
172
+
173
+ assert self .settings .build_dir is not None
174
+ # A build dir can be specified, otherwise use a temporary directory
175
+ build_dir = Path (
176
+ self .settings .build_dir .format (
177
+ cache_tag = sys .implementation .cache_tag ,
178
+ wheel_tag = str (tags ),
179
+ )
180
+ )
181
+ logger .info ("Build directory: {}" , build_dir .resolve ())
182
+
183
+ return build_dir .resolve ()
184
+
185
+ def _build_env_cache_dir (self , hash : hashlib ._Hash ) -> Path :
186
+ base_dir = self ._build_dir ()
187
+ base_dir = base_dir .with_name (base_dir .name + "-build-env-cache" )
188
+ return base_dir / hash .hexdigest ()
189
+
82
190
def configure (
83
191
self ,
84
192
* ,
@@ -103,9 +211,20 @@ def configure(
103
211
site_packages = Path (sysconfig .get_path ("purelib" ))
104
212
self .config .prefix_dirs .append (site_packages )
105
213
logger .debug ("SITE_PACKAGES: {}" , site_packages )
106
- if site_packages != DIR .parent .parent :
214
+
215
+ if self .settings .cache_build_env :
216
+ if not self .settings .experimental :
217
+ msg = "Experimental features must be enabled to use build environment caching"
218
+ raise AssertionError (msg )
219
+
220
+ archive = BuildEnvArchive (DIR .parent .parent )
221
+ targettree = self ._build_env_cache_dir (archive .hash )
222
+ archive .extract (targettree )
223
+ self .config .prefix_dirs .append (targettree )
224
+
225
+ elif site_packages != DIR .parent .parent :
107
226
self .config .prefix_dirs .append (DIR .parent .parent )
108
- logger .debug ("Extra SITE_PACKAGES: {}" , site_packages )
227
+ logger .debug ("Extra SITE_PACKAGES: {}" , DIR . parent . parent )
109
228
110
229
# Add the FindPython backport if needed
111
230
fp_backport = self .settings .backport .find_python
0 commit comments