-
Notifications
You must be signed in to change notification settings - Fork 0
/
cythexts.py
292 lines (258 loc) · 10.9 KB
/
cythexts.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
import os
from os.path import splitext, sep as filesep, join as pjoin, relpath
from hashlib import sha1
from setuptools.command.build_ext import build_ext
from setuptools.command.sdist import sdist
from packaging.version import Version
def derror_maker(klass, msg):
""" Decorate setuptools class to make run method raise error """
class K(klass):
def run(self):
raise RuntimeError(msg)
return K
def stamped_pyx_ok(exts, hash_stamp_fname):
""" Check for match of recorded hashes for pyx, corresponding c files
Parameters
----------
exts : sequence of ``Extension``
setuptools ``Extension`` instances, in fact only need to contain a
``sources`` sequence field.
hash_stamp_fname : str
filename of text file containing hash stamps
Returns
-------
tf : bool
True if there is a corresponding c file for each pyx or py file in
`exts` sources, and the hash for both the (pyx, py) file *and* the c
file match those recorded in the file named in `hash_stamp_fname`.
"""
# Calculate hashes for pyx and c files. Check for presence of c files.
stamps = {}
for mod in exts:
for source in mod.sources:
base, ext = splitext(source)
if ext not in ('.pyx', '.py'):
continue
source_hash = sha1(open(source, 'rb').read()).hexdigest()
c_fname = base + '.c'
try:
c_file = open(c_fname, 'rb')
except IOError:
return False
c_hash = sha1(c_file.read()).hexdigest()
stamps[source_hash] = source
stamps[c_hash] = c_fname
# Read stamps from hash_stamp_fname; check in stamps dictionary
try:
stamp_file = open(hash_stamp_fname, 'rt')
except IOError:
return False
for line in stamp_file:
if line.startswith('#'):
continue
fname, hash = [e.strip() for e in line.split(',')]
if hash not in stamps:
return False
# Compare path made canonical for \/
fname = fname.replace(filesep, '/')
if not stamps[hash].replace(filesep, '/') == fname:
return False
stamps.pop(hash)
# All good if we found all hashes we need
return len(stamps) == 0
def cyproc_exts(exts, cython_min_version,
hash_stamps_fname='pyx-stamps',
build_ext=build_ext):
""" Process sequence of `exts` to check if we need Cython. Return builder
Parameters
----------
exts : sequence of Setuptools ``Extension``
If we already have good c files for any pyx or py sources, we replace
the pyx or py files with their compiled up c versions inplace.
cython_min_version : str
Minimum cython version needed for compile
hash_stamps_fname : str, optional
filename with hashes for pyx/py and c files known to be in sync. Default
is 'pyx-stamps'
build_ext : Setuptools command
default build_ext to return if not cythonizing. Default is setuptools
``build_ext`` class
Returns
-------
builder : ``setuptools`` ``build_ext`` class or similar
Can be ``build_ext`` input (if we have good c files) or cython
``build_ext`` if we have a good cython, or a class raising an informative
error on ``run()``
need_cython : bool
True if we need Cython to build extensions, False otherwise.
"""
if stamped_pyx_ok(exts, hash_stamps_fname):
# Replace pyx with c files, use standard builder
for mod in exts:
sources = []
for source in mod.sources:
base, ext = splitext(source)
if ext in ('.pyx', '.py'):
sources.append(base + '.c')
else:
sources.append(source)
mod.sources = sources
return build_ext, False
# We need cython
try:
from Cython.Compiler.Version import version as cyversion
except ImportError:
return derror_maker(build_ext,
'Need cython>={0} to build extensions '
'but cannot import "Cython"'.format(
cython_min_version)), True
if Version(cyversion) >= Version(cython_min_version):
from Cython.Distutils import build_ext as extbuilder
return extbuilder, True
return derror_maker(build_ext,
'Need cython>={0} to build extensions'
'but found cython version {1}'.format(
cython_min_version, cyversion)), True
def build_stamp(pyxes, include_dirs=()):
""" Cythonize files in `pyxes`, return pyx, C filenames, hashes
Parameters
----------
pyxes : sequence
sequence of filenames of files on which to run Cython
include_dirs : sequence
Any extra include directories in which to find Cython files.
Returns
-------
pyx_defs : dict
dict has key, value pairs of <pyx_filename>, <pyx_info>, where
<pyx_info> is a dict with key, value pairs of "pyx_hash", <pyx file
SHA1 hash>; "c_filename", <c filemane>; "c_hash", <c file SHA1 hash>.
"""
pyx_defs = {}
from Cython.Compiler.Main import compile
from Cython.Compiler.CmdLine import parse_command_line
includes = sum([['--include-dir', d] for d in include_dirs], [])
for source in pyxes:
base, ext = splitext(source)
pyx_hash = sha1((open(source, 'rt').read().encode())).hexdigest()
c_filename = base + '.c'
options, sources = parse_command_line(['-3'] + includes + [source])
result = compile(sources, options)
if result.num_errors > 0:
raise RuntimeError('Cython failed to compile ' + source)
c_hash = sha1(open(c_filename, 'rt').read().encode()).hexdigest()
pyx_defs[source] = dict(pyx_hash=pyx_hash,
c_filename=c_filename,
c_hash=c_hash)
return pyx_defs
def write_stamps(pyx_defs, stamp_fname='pyx-stamps'):
""" Write stamp information in `pyx_defs` to filename `stamp_fname`
Parameters
----------
pyx_defs : dict
dict has key, value pairs of <pyx_filename>, <pyx_info>, where
<pyx_info> is a dict with key, value pairs of "pyx_hash", <pyx file
SHA1 hash>; "c_filename", <c filemane>; "c_hash", <c file SHA1 hash>.
stamp_fname : str
filename to which to write stamp information
"""
with open(stamp_fname, 'wt') as stamp_file:
stamp_file.write('# SHA1 hashes for pyx files and generated c files\n')
stamp_file.write('# Auto-generated file, do not edit\n')
for pyx_fname, pyx_info in pyx_defs.items():
stamp_file.write('%s, %s\n' % (pyx_fname,
pyx_info['pyx_hash']))
stamp_file.write('%s, %s\n' % (pyx_info['c_filename'],
pyx_info['c_hash']))
def find_pyx(root_dir):
""" Recursively find files with extension '.pyx' starting at `root_dir`
Parameters
----------
root_dir : str
Directory from which to search for pyx files.
Returns
-------
pyxes : list
list of filenames relative to `root_dir`
"""
pyxes = []
for dirpath, dirnames, filenames in os.walk(root_dir):
for filename in filenames:
if not filename.endswith('.pyx'):
continue
base = relpath(dirpath, root_dir)
pyxes.append(pjoin(base, filename))
return pyxes
def get_pyx_sdist(sdist_like=sdist, hash_stamps_fname='pyx-stamps',
include_dirs=()):
""" Add pyx->c conversion, hash recording to sdist command `sdist_like`
Parameters
----------
sdist_like : sdist command class, optional
command that will do work of ``setuptools.command.sdist.sdist``. By
default we use the setuptools version
hash_stamps_fname : str, optional
filename to which to write hashes of pyx / py and c files. Default is
``pyx-stamps``
include_dirs : sequence
Any extra include directories in which to find Cython files.
Returns
-------
modified_sdist : sdist-like command class
decorated `sdist_like` class, for compiling pyx / py files to c,
putting the .c files in the the source archive, and writing hashes
for these into the file named from `hash_stamps_fname`
"""
class PyxSDist(sdist_like):
"""Custom setuptools sdist command to generate .c files from pyx files.
Running the command object ``obj.run()`` will compile the pyx-py
files in any extensions, into c files, and add them to the list of
files to put into the source archive, as well as the usual behavior of
distutils ``sdist``. It will also take the sha1 hashes of the pyx-py
and c files, and store them in a file ``pyx-stamps``, and put this
file in the release tree. This allows someone who has the archive
to know that the pyx and c files that they have are the ones packed
into the archive, and therefore they may not need Cython at
install time.
See Also
--------
``cython_process_exts`` for the build-time command.
"""
def make_distribution(self):
""" Compile pyx to c files, add to sources, stamp sha1s """
pyxes = []
for mod in self.distribution.ext_modules:
for source in mod.sources:
base, ext = splitext(source)
if ext in ('.pyx', '.py'):
pyxes.append(source)
self.pyx_defs = build_stamp(pyxes, include_dirs)
for pyx_fname, pyx_info in self.pyx_defs.items():
self.filelist.append(pyx_info['c_filename'])
sdist_like.make_distribution(self)
def make_release_tree(self, base_dir, files):
""" Put pyx stamps file into release tree """
sdist_like.make_release_tree(self, base_dir, files)
stamp_fname = pjoin(base_dir, hash_stamps_fname)
write_stamps(self.pyx_defs, stamp_fname)
return PyxSDist
def build_stamp_source(root_dir=None, stamp_fname='pyx-stamps',
include_dirs=None):
""" Build cython c files, make stamp file in source tree `root_dir`
Parameters
----------
root_dir : None or str, optional
Directory from which to find ``.pyx`` files. If None, use current
working directory.
stamp_fname : str, optional
Filename for stamp file we will write
include_dirs : None or sequence
Any extra Cython include directories
"""
if root_dir is None:
root_dir = os.getcwd()
if include_dirs is None:
include_dirs = [pjoin(root_dir, 'src')]
pyxes = find_pyx(root_dir)
pyx_defs = build_stamp(pyxes, include_dirs=include_dirs)
write_stamps(pyx_defs, stamp_fname)