|
@@
-1,369
+1,381
b''
|
|
1
|
1
|
# encoding: utf-8
|
|
2
|
2
|
"""
|
|
3
|
3
|
This module defines the things that are used in setup.py for building IPython
|
|
4
|
4
|
|
|
5
|
5
|
This includes:
|
|
6
|
6
|
|
|
7
|
7
|
* The basic arguments to setup
|
|
8
|
8
|
* Functions for finding things like packages, package data, etc.
|
|
9
|
9
|
* A function for checking dependencies.
|
|
10
|
10
|
"""
|
|
11
|
11
|
from __future__ import print_function
|
|
12
|
12
|
|
|
13
|
13
|
#-------------------------------------------------------------------------------
|
|
14
|
14
|
# Copyright (C) 2008 The IPython Development Team
|
|
15
|
15
|
#
|
|
16
|
16
|
# Distributed under the terms of the BSD License. The full license is in
|
|
17
|
17
|
# the file COPYING, distributed as part of this software.
|
|
18
|
18
|
#-------------------------------------------------------------------------------
|
|
19
|
19
|
|
|
20
|
20
|
#-------------------------------------------------------------------------------
|
|
21
|
21
|
# Imports
|
|
22
|
22
|
#-------------------------------------------------------------------------------
|
|
23
|
23
|
import os
|
|
24
|
24
|
import sys
|
|
25
|
25
|
|
|
26
|
26
|
from ConfigParser import ConfigParser
|
|
27
|
27
|
from distutils.command.build_py import build_py
|
|
28
|
28
|
from glob import glob
|
|
29
|
29
|
|
|
30
|
30
|
from setupext import install_data_ext
|
|
31
|
31
|
|
|
32
|
32
|
#-------------------------------------------------------------------------------
|
|
33
|
33
|
# Useful globals and utility functions
|
|
34
|
34
|
#-------------------------------------------------------------------------------
|
|
35
|
35
|
|
|
36
|
36
|
# A few handy globals
|
|
37
|
37
|
isfile = os.path.isfile
|
|
38
|
38
|
pjoin = os.path.join
|
|
39
|
39
|
|
|
40
|
40
|
def oscmd(s):
|
|
41
|
41
|
print(">", s)
|
|
42
|
42
|
os.system(s)
|
|
43
|
43
|
|
|
44
|
44
|
# A little utility we'll need below, since glob() does NOT allow you to do
|
|
45
|
45
|
# exclusion on multiple endings!
|
|
46
|
46
|
def file_doesnt_endwith(test,endings):
|
|
47
|
47
|
"""Return true if test is a file and its name does NOT end with any
|
|
48
|
48
|
of the strings listed in endings."""
|
|
49
|
49
|
if not isfile(test):
|
|
50
|
50
|
return False
|
|
51
|
51
|
for e in endings:
|
|
52
|
52
|
if test.endswith(e):
|
|
53
|
53
|
return False
|
|
54
|
54
|
return True
|
|
55
|
55
|
|
|
56
|
56
|
#---------------------------------------------------------------------------
|
|
57
|
57
|
# Basic project information
|
|
58
|
58
|
#---------------------------------------------------------------------------
|
|
59
|
59
|
|
|
60
|
60
|
# release.py contains version, authors, license, url, keywords, etc.
|
|
61
|
61
|
execfile(pjoin('IPython','core','release.py'))
|
|
62
|
62
|
|
|
63
|
63
|
# Create a dict with the basic information
|
|
64
|
64
|
# This dict is eventually passed to setup after additional keys are added.
|
|
65
|
65
|
setup_args = dict(
|
|
66
|
66
|
name = name,
|
|
67
|
67
|
version = version,
|
|
68
|
68
|
description = description,
|
|
69
|
69
|
long_description = long_description,
|
|
70
|
70
|
author = author,
|
|
71
|
71
|
author_email = author_email,
|
|
72
|
72
|
url = url,
|
|
73
|
73
|
download_url = download_url,
|
|
74
|
74
|
license = license,
|
|
75
|
75
|
platforms = platforms,
|
|
76
|
76
|
keywords = keywords,
|
|
77
|
77
|
cmdclass = {'install_data': install_data_ext},
|
|
78
|
78
|
)
|
|
79
|
79
|
|
|
80
|
80
|
|
|
81
|
81
|
#---------------------------------------------------------------------------
|
|
82
|
82
|
# Find packages
|
|
83
|
83
|
#---------------------------------------------------------------------------
|
|
84
|
84
|
|
|
85
|
85
|
def add_package(packages,pname,config=False,tests=False,scripts=False,
|
|
86
|
86
|
others=None):
|
|
87
|
87
|
"""
|
|
88
|
88
|
Add a package to the list of packages, including certain subpackages.
|
|
89
|
89
|
"""
|
|
90
|
90
|
packages.append('.'.join(['IPython',pname]))
|
|
91
|
91
|
if config:
|
|
92
|
92
|
packages.append('.'.join(['IPython',pname,'config']))
|
|
93
|
93
|
if tests:
|
|
94
|
94
|
packages.append('.'.join(['IPython',pname,'tests']))
|
|
95
|
95
|
if scripts:
|
|
96
|
96
|
packages.append('.'.join(['IPython',pname,'scripts']))
|
|
97
|
97
|
if others is not None:
|
|
98
|
98
|
for o in others:
|
|
99
|
99
|
packages.append('.'.join(['IPython',pname,o]))
|
|
100
|
100
|
|
|
101
|
101
|
def find_packages():
|
|
102
|
102
|
"""
|
|
103
|
103
|
Find all of IPython's packages.
|
|
104
|
104
|
"""
|
|
105
|
105
|
packages = ['IPython']
|
|
106
|
106
|
add_package(packages, 'config', tests=True, others=['default','profile'])
|
|
107
|
107
|
add_package(packages, 'core', tests=True)
|
|
108
|
108
|
add_package(packages, 'deathrow', tests=True)
|
|
109
|
109
|
add_package(packages, 'extensions')
|
|
110
|
110
|
add_package(packages, 'external')
|
|
|
111
|
add_package(packages, 'external.argparse')
|
|
|
112
|
add_package(packages, 'external.configobj')
|
|
|
113
|
add_package(packages, 'external.decorator')
|
|
|
114
|
add_package(packages, 'external.decorators')
|
|
|
115
|
add_package(packages, 'external.guid')
|
|
|
116
|
add_package(packages, 'external.Itpl')
|
|
|
117
|
add_package(packages, 'external.mglob')
|
|
|
118
|
add_package(packages, 'external.path')
|
|
|
119
|
add_package(packages, 'external.pretty')
|
|
|
120
|
add_package(packages, 'external.pyparsing')
|
|
|
121
|
add_package(packages, 'external.simplegeneric')
|
|
|
122
|
add_package(packages, 'external.validate')
|
|
111
|
123
|
add_package(packages, 'frontend')
|
|
112
|
124
|
add_package(packages, 'frontend.qt')
|
|
113
|
125
|
add_package(packages, 'frontend.qt.console', tests=True)
|
|
114
|
126
|
add_package(packages, 'frontend.terminal', tests=True)
|
|
115
|
127
|
add_package(packages, 'kernel', config=False, tests=True, scripts=True)
|
|
116
|
128
|
add_package(packages, 'kernel.core', config=False, tests=True)
|
|
117
|
129
|
add_package(packages, 'lib', tests=True)
|
|
118
|
130
|
add_package(packages, 'quarantine', tests=True)
|
|
119
|
131
|
add_package(packages, 'scripts')
|
|
120
|
132
|
add_package(packages, 'testing', tests=True)
|
|
121
|
133
|
add_package(packages, 'testing.plugin', tests=False)
|
|
122
|
134
|
add_package(packages, 'utils', tests=True)
|
|
123
|
135
|
add_package(packages, 'zmq')
|
|
124
|
136
|
add_package(packages, 'zmq.pylab')
|
|
125
|
137
|
return packages
|
|
126
|
138
|
|
|
127
|
139
|
#---------------------------------------------------------------------------
|
|
128
|
140
|
# Find package data
|
|
129
|
141
|
#---------------------------------------------------------------------------
|
|
130
|
142
|
|
|
131
|
143
|
def find_package_data():
|
|
132
|
144
|
"""
|
|
133
|
145
|
Find IPython's package_data.
|
|
134
|
146
|
"""
|
|
135
|
147
|
# This is not enough for these things to appear in an sdist.
|
|
136
|
148
|
# We need to muck with the MANIFEST to get this to work
|
|
137
|
149
|
package_data = {
|
|
138
|
150
|
'IPython.config.userconfig' : ['*'],
|
|
139
|
151
|
'IPython.testing' : ['*.txt']
|
|
140
|
152
|
}
|
|
141
|
153
|
return package_data
|
|
142
|
154
|
|
|
143
|
155
|
|
|
144
|
156
|
#---------------------------------------------------------------------------
|
|
145
|
157
|
# Find data files
|
|
146
|
158
|
#---------------------------------------------------------------------------
|
|
147
|
159
|
|
|
148
|
160
|
def make_dir_struct(tag,base,out_base):
|
|
149
|
161
|
"""Make the directory structure of all files below a starting dir.
|
|
150
|
162
|
|
|
151
|
163
|
This is just a convenience routine to help build a nested directory
|
|
152
|
164
|
hierarchy because distutils is too stupid to do this by itself.
|
|
153
|
165
|
|
|
154
|
166
|
XXX - this needs a proper docstring!
|
|
155
|
167
|
"""
|
|
156
|
168
|
|
|
157
|
169
|
# we'll use these a lot below
|
|
158
|
170
|
lbase = len(base)
|
|
159
|
171
|
pathsep = os.path.sep
|
|
160
|
172
|
lpathsep = len(pathsep)
|
|
161
|
173
|
|
|
162
|
174
|
out = []
|
|
163
|
175
|
for (dirpath,dirnames,filenames) in os.walk(base):
|
|
164
|
176
|
# we need to strip out the dirpath from the base to map it to the
|
|
165
|
177
|
# output (installation) path. This requires possibly stripping the
|
|
166
|
178
|
# path separator, because otherwise pjoin will not work correctly
|
|
167
|
179
|
# (pjoin('foo/','/bar') returns '/bar').
|
|
168
|
180
|
|
|
169
|
181
|
dp_eff = dirpath[lbase:]
|
|
170
|
182
|
if dp_eff.startswith(pathsep):
|
|
171
|
183
|
dp_eff = dp_eff[lpathsep:]
|
|
172
|
184
|
# The output path must be anchored at the out_base marker
|
|
173
|
185
|
out_path = pjoin(out_base,dp_eff)
|
|
174
|
186
|
# Now we can generate the final filenames. Since os.walk only produces
|
|
175
|
187
|
# filenames, we must join back with the dirpath to get full valid file
|
|
176
|
188
|
# paths:
|
|
177
|
189
|
pfiles = [pjoin(dirpath,f) for f in filenames]
|
|
178
|
190
|
# Finally, generate the entry we need, which is a pari of (output
|
|
179
|
191
|
# path, files) for use as a data_files parameter in install_data.
|
|
180
|
192
|
out.append((out_path, pfiles))
|
|
181
|
193
|
|
|
182
|
194
|
return out
|
|
183
|
195
|
|
|
184
|
196
|
|
|
185
|
197
|
def find_data_files():
|
|
186
|
198
|
"""
|
|
187
|
199
|
Find IPython's data_files.
|
|
188
|
200
|
|
|
189
|
201
|
Most of these are docs.
|
|
190
|
202
|
"""
|
|
191
|
203
|
|
|
192
|
204
|
docdirbase = pjoin('share', 'doc', 'ipython')
|
|
193
|
205
|
manpagebase = pjoin('share', 'man', 'man1')
|
|
194
|
206
|
|
|
195
|
207
|
# Simple file lists can be made by hand
|
|
196
|
208
|
manpages = filter(isfile, glob(pjoin('docs','man','*.1.gz')))
|
|
197
|
209
|
igridhelpfiles = filter(isfile,
|
|
198
|
210
|
glob(pjoin('IPython','extensions','igrid_help.*')))
|
|
199
|
211
|
|
|
200
|
212
|
# For nested structures, use the utility above
|
|
201
|
213
|
example_files = make_dir_struct(
|
|
202
|
214
|
'data',
|
|
203
|
215
|
pjoin('docs','examples'),
|
|
204
|
216
|
pjoin(docdirbase,'examples')
|
|
205
|
217
|
)
|
|
206
|
218
|
manual_files = make_dir_struct(
|
|
207
|
219
|
'data',
|
|
208
|
220
|
pjoin('docs','dist'),
|
|
209
|
221
|
pjoin(docdirbase,'manual')
|
|
210
|
222
|
)
|
|
211
|
223
|
|
|
212
|
224
|
# And assemble the entire output list
|
|
213
|
225
|
data_files = [ (manpagebase, manpages),
|
|
214
|
226
|
(pjoin(docdirbase, 'extensions'), igridhelpfiles),
|
|
215
|
227
|
] + manual_files + example_files
|
|
216
|
228
|
|
|
217
|
229
|
return data_files
|
|
218
|
230
|
|
|
219
|
231
|
|
|
220
|
232
|
def make_man_update_target(manpage):
|
|
221
|
233
|
"""Return a target_update-compliant tuple for the given manpage.
|
|
222
|
234
|
|
|
223
|
235
|
Parameters
|
|
224
|
236
|
----------
|
|
225
|
237
|
manpage : string
|
|
226
|
238
|
Name of the manpage, must include the section number (trailing number).
|
|
227
|
239
|
|
|
228
|
240
|
Example
|
|
229
|
241
|
-------
|
|
230
|
242
|
|
|
231
|
243
|
>>> make_man_update_target('ipython.1') #doctest: +NORMALIZE_WHITESPACE
|
|
232
|
244
|
('docs/man/ipython.1.gz',
|
|
233
|
245
|
['docs/man/ipython.1'],
|
|
234
|
246
|
'cd docs/man && gzip -9c ipython.1 > ipython.1.gz')
|
|
235
|
247
|
"""
|
|
236
|
248
|
man_dir = pjoin('docs', 'man')
|
|
237
|
249
|
manpage_gz = manpage + '.gz'
|
|
238
|
250
|
manpath = pjoin(man_dir, manpage)
|
|
239
|
251
|
manpath_gz = pjoin(man_dir, manpage_gz)
|
|
240
|
252
|
gz_cmd = ( "cd %(man_dir)s && gzip -9c %(manpage)s > %(manpage_gz)s" %
|
|
241
|
253
|
locals() )
|
|
242
|
254
|
return (manpath_gz, [manpath], gz_cmd)
|
|
243
|
255
|
|
|
244
|
256
|
#---------------------------------------------------------------------------
|
|
245
|
257
|
# Find scripts
|
|
246
|
258
|
#---------------------------------------------------------------------------
|
|
247
|
259
|
|
|
248
|
260
|
def find_scripts():
|
|
249
|
261
|
"""
|
|
250
|
262
|
Find IPython's scripts.
|
|
251
|
263
|
"""
|
|
252
|
264
|
kernel_scripts = pjoin('IPython','kernel','scripts')
|
|
253
|
265
|
main_scripts = pjoin('IPython','scripts')
|
|
254
|
266
|
scripts = [pjoin(kernel_scripts, 'ipengine'),
|
|
255
|
267
|
pjoin(kernel_scripts, 'ipcontroller'),
|
|
256
|
268
|
pjoin(kernel_scripts, 'ipcluster'),
|
|
257
|
269
|
pjoin(main_scripts, 'ipython'),
|
|
258
|
270
|
pjoin(main_scripts, 'ipython-qtconsole'),
|
|
259
|
271
|
pjoin(main_scripts, 'pycolor'),
|
|
260
|
272
|
pjoin(main_scripts, 'irunner'),
|
|
261
|
273
|
pjoin(main_scripts, 'iptest')
|
|
262
|
274
|
]
|
|
263
|
275
|
|
|
264
|
276
|
# Script to be run by the windows binary installer after the default setup
|
|
265
|
277
|
# routine, to add shortcuts and similar windows-only things. Windows
|
|
266
|
278
|
# post-install scripts MUST reside in the scripts/ dir, otherwise distutils
|
|
267
|
279
|
# doesn't find them.
|
|
268
|
280
|
if 'bdist_wininst' in sys.argv:
|
|
269
|
281
|
if len(sys.argv) > 2 and \
|
|
270
|
282
|
('sdist' in sys.argv or 'bdist_rpm' in sys.argv):
|
|
271
|
283
|
print("ERROR: bdist_wininst must be run alone. Exiting.",
|
|
272
|
284
|
file=sys.stderr)
|
|
273
|
285
|
sys.exit(1)
|
|
274
|
286
|
scripts.append(pjoin('scripts','ipython_win_post_install.py'))
|
|
275
|
287
|
|
|
276
|
288
|
return scripts
|
|
277
|
289
|
|
|
278
|
290
|
#---------------------------------------------------------------------------
|
|
279
|
291
|
# Verify all dependencies
|
|
280
|
292
|
#---------------------------------------------------------------------------
|
|
281
|
293
|
|
|
282
|
294
|
def check_for_dependencies():
|
|
283
|
295
|
"""Check for IPython's dependencies.
|
|
284
|
296
|
|
|
285
|
297
|
This function should NOT be called if running under setuptools!
|
|
286
|
298
|
"""
|
|
287
|
299
|
from setupext.setupext import (
|
|
288
|
300
|
print_line, print_raw, print_status,
|
|
289
|
301
|
check_for_zopeinterface, check_for_twisted,
|
|
290
|
302
|
check_for_foolscap, check_for_pyopenssl,
|
|
291
|
303
|
check_for_sphinx, check_for_pygments,
|
|
292
|
304
|
check_for_nose, check_for_pexpect
|
|
293
|
305
|
)
|
|
294
|
306
|
print_line()
|
|
295
|
307
|
print_raw("BUILDING IPYTHON")
|
|
296
|
308
|
print_status('python', sys.version)
|
|
297
|
309
|
print_status('platform', sys.platform)
|
|
298
|
310
|
if sys.platform == 'win32':
|
|
299
|
311
|
print_status('Windows version', sys.getwindowsversion())
|
|
300
|
312
|
|
|
301
|
313
|
print_raw("")
|
|
302
|
314
|
print_raw("OPTIONAL DEPENDENCIES")
|
|
303
|
315
|
|
|
304
|
316
|
check_for_zopeinterface()
|
|
305
|
317
|
check_for_twisted()
|
|
306
|
318
|
check_for_foolscap()
|
|
307
|
319
|
check_for_pyopenssl()
|
|
308
|
320
|
check_for_sphinx()
|
|
309
|
321
|
check_for_pygments()
|
|
310
|
322
|
check_for_nose()
|
|
311
|
323
|
check_for_pexpect()
|
|
312
|
324
|
|
|
313
|
325
|
|
|
314
|
326
|
def record_commit_info(pkg_dir, build_cmd=build_py):
|
|
315
|
327
|
""" Return extended build command class for recording commit
|
|
316
|
328
|
|
|
317
|
329
|
The extended command tries to run git to find the current commit, getting
|
|
318
|
330
|
the empty string if it fails. It then writes the commit hash into a file
|
|
319
|
331
|
in the `pkg_dir` path, named ``.git_commit_info.ini``.
|
|
320
|
332
|
|
|
321
|
333
|
In due course this information can be used by the package after it is
|
|
322
|
334
|
installed, to tell you what commit it was installed from if known.
|
|
323
|
335
|
|
|
324
|
336
|
To make use of this system, you need a package with a .git_commit_info.ini
|
|
325
|
337
|
file - e.g. ``myproject/.git_commit_info.ini`` - that might well look like
|
|
326
|
338
|
this::
|
|
327
|
339
|
|
|
328
|
340
|
# This is an ini file that may contain information about the code state
|
|
329
|
341
|
[commit hash]
|
|
330
|
342
|
# The line below may contain a valid hash if it has been substituted
|
|
331
|
343
|
# during 'git archive'
|
|
332
|
344
|
archive_subst_hash=$Format:%h$
|
|
333
|
345
|
# This line may be modified by the install process
|
|
334
|
346
|
install_hash=
|
|
335
|
347
|
|
|
336
|
348
|
The .git_commit_info file above is also designed to be used with git
|
|
337
|
349
|
substitution - so you probably also want a ``.gitattributes`` file in the
|
|
338
|
350
|
root directory of your working tree that contains something like this::
|
|
339
|
351
|
|
|
340
|
352
|
myproject/.git_commit_info.ini export-subst
|
|
341
|
353
|
|
|
342
|
354
|
That will cause the ``.git_commit_info.ini`` file to get filled in by ``git
|
|
343
|
355
|
archive`` - useful in case someone makes such an archive - for example with
|
|
344
|
356
|
via the github 'download source' button.
|
|
345
|
357
|
|
|
346
|
358
|
Although all the above will work as is, you might consider having something
|
|
347
|
359
|
like a ``get_info()`` function in your package to display the commit
|
|
348
|
360
|
information at the terminal. See the ``pkg_info.py`` module in the nipy
|
|
349
|
361
|
package for an example.
|
|
350
|
362
|
"""
|
|
351
|
363
|
class MyBuildPy(build_cmd):
|
|
352
|
364
|
''' Subclass to write commit data into installation tree '''
|
|
353
|
365
|
def run(self):
|
|
354
|
366
|
build_py.run(self)
|
|
355
|
367
|
import subprocess
|
|
356
|
368
|
proc = subprocess.Popen('git rev-parse --short HEAD',
|
|
357
|
369
|
stdout=subprocess.PIPE,
|
|
358
|
370
|
stderr=subprocess.PIPE,
|
|
359
|
371
|
shell=True)
|
|
360
|
372
|
repo_commit, _ = proc.communicate()
|
|
361
|
373
|
# We write the installation commit even if it's empty
|
|
362
|
374
|
cfg_parser = ConfigParser()
|
|
363
|
375
|
cfg_parser.read(pjoin(pkg_dir, '.git_commit_info.ini'))
|
|
364
|
376
|
cfg_parser.set('commit hash', 'install_hash', repo_commit)
|
|
365
|
377
|
out_pth = pjoin(self.build_lib, pkg_dir, '.git_commit_info.ini')
|
|
366
|
378
|
out_file = open(out_pth, 'wt')
|
|
367
|
379
|
cfg_parser.write(out_file)
|
|
368
|
380
|
out_file.close()
|
|
369
|
381
|
return MyBuildPy
|