forked from Garyfallidis/dipy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
cythexts.py
186 lines (167 loc) · 7.18 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
from os.path import splitext, sep as filesep, join as pjoin
from hashlib import sha1
from subprocess import check_call
from distutils.command.build_ext import build_ext
from distutils.command.sdist import sdist
from distutils.version import LooseVersion
def derror_maker(klass, msg):
""" Decorate distutils 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``
distutils ``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 not ext in ('.pyx', '.py'):
continue
source_hash = sha1(open(source, 'rt').read()).hexdigest()
c_fname = base + '.c'
try:
c_file = open(c_fname, 'rt')
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 not hash 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 distutils ``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 neede 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 : distutils command
default build_ext to return if not cythonizing. Default is distutils
``build_ext`` class
Returns
-------
builder : ``distutils`` ``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()``
"""
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
# We need cython
try:
from Cython.Compiler.Version import version as cyversion
except ImportError:
cython_ok = False
else:
cython_ok = LooseVersion(cyversion) >= cython_min_version
if cython_ok:
from Cython.Distutils import build_ext as extbuilder
return extbuilder
return derror_maker(build_ext,
'Need cython>=%s to build extensions'
% cython_min_version)
def get_pyx_sdist(sdist_like=sdist, hash_stamps_fname='pyx-stamps'):
""" Add pyx->c conversion, hash recording to sdist command `sdist_like`
Parameters
----------
sdist_like : sdist command class, optional
command that will do work of ``distutils.command.sdist.sdist``. By
default we use the distutils version
hash_stamps_fname : str, optional
filename to which to write hashes of pyx / py and c files. Default is
``pyx-stamps``
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 distutils 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
``cython_process_exts`` for the build-time command.
"""
def make_distribution(self):
""" Compile pyx to c files, add to sources, stamp sha1s """
stamps = []
for mod in self.distribution.ext_modules:
for source in mod.sources:
base, ext = splitext(source)
if not ext in ('.pyx', '.py'):
continue
source_hash = sha1(open(source, 'rt').read()).hexdigest()
stamps.append('%s, %s\n' % (source, source_hash))
c_fname = base + '.c'
check_call('cython ' + source, shell=True)
c_hash = sha1(open(c_fname, 'rt').read()).hexdigest()
stamps.append('%s, %s\n' % (c_fname, c_hash))
self.filelist.append(c_fname)
self.stamps = stamps
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)
stamp_file = open(stamp_fname, 'wt')
stamp_file.write('# SHA1 hashes for pyx files and generated c files\n')
stamp_file.write('# Auto-generated file, do not edit\n')
stamp_file.writelines(self.stamps)
stamp_file.close()
return PyxSDist