1
1
#! /usr/bin/env python
3
# This is an installation script for bzr. Run it with
4
# './setup.py install', or
5
# './setup.py --help' for more options
3
"""Installation script for bzr.
5
'./setup.py install', or
6
'./setup.py --help' for more options
13
if sys.version_info < (2, 4):
14
sys.stderr.write("[ERROR] Not a supported Python version. Need 2.4+\n")
17
# NOTE: The directory containing setup.py, whether run by 'python setup.py' or
18
# './setup.py' or the equivalent with another path, should always be at the
19
# start of the path, so this should find the right one...
22
def get_long_description():
23
dirname = os.path.dirname(__file__)
24
readme = os.path.join(dirname, 'README')
25
f = open(readme, 'rb')
33
# META INFORMATION FOR SETUP
34
# see http://docs.python.org/dist/meta-data.html
37
'version': bzrlib.__version__,
38
'author': 'Canonical Ltd',
39
'author_email': 'bazaar@lists.canonical.com',
40
'url': 'http://www.bazaar-vcs.org/',
41
'description': 'Friendly distributed version control system',
42
'license': 'GNU GPL v2',
43
'download_url': 'http://bazaar-vcs.org/Download',
44
'long_description': get_long_description(),
46
'Development Status :: 6 - Mature',
47
'Environment :: Console',
48
'Intended Audience :: Developers',
49
'Intended Audience :: System Administrators',
50
'License :: OSI Approved :: GNU General Public License (GPL)',
51
'Operating System :: Microsoft :: Windows',
52
'Operating System :: OS Independent',
53
'Operating System :: POSIX',
54
'Programming Language :: Python',
55
'Programming Language :: C',
56
'Topic :: Software Development :: Version Control',
60
# The list of packages is automatically generated later. Add other things
61
# that are part of BZRLIB here.
64
PKG_DATA = {# install files from selftest suite
65
'package_data': {'bzrlib': ['doc/api/*.txt',
66
'tests/test_patches_data/*',
67
'help_topics/en/*.txt',
72
def get_bzrlib_packages():
73
"""Recurse through the bzrlib directory, and extract the package names"""
76
base_path = os.path.dirname(os.path.abspath(bzrlib.__file__))
77
for root, dirs, files in os.walk(base_path):
78
if '__init__.py' in files:
79
assert root.startswith(base_path)
80
# Get just the path below bzrlib
81
package_path = root[len(base_path):]
82
# Remove leading and trailing slashes
83
package_path = package_path.strip('\\/')
85
package_name = 'bzrlib'
87
package_name = ('bzrlib.' +
88
package_path.replace('/', '.').replace('\\', '.'))
89
packages.append(package_name)
90
return sorted(packages)
93
BZRLIB['packages'] = get_bzrlib_packages()
7
96
from distutils.core import setup
12
author_email='mbp@sourcefrog.net',
13
url='http://www.bazaar-ng.org/',
14
description='Friendly distributed version control system',
97
from distutils.command.install_scripts import install_scripts
98
from distutils.command.build import build
100
###############################
101
# Overridden distutils actions
102
###############################
104
class my_install_scripts(install_scripts):
105
""" Customized install_scripts distutils action.
106
Create bzr.bat for win32.
109
install_scripts.run(self) # standard action
111
if sys.platform == "win32":
113
scripts_dir = os.path.join(sys.prefix, 'Scripts')
114
script_path = self._quoted_path(os.path.join(scripts_dir,
116
python_exe = self._quoted_path(sys.executable)
117
args = self._win_batch_args()
118
batch_str = "@%s %s %s" % (python_exe, script_path, args)
119
batch_path = os.path.join(self.install_dir, "bzr.bat")
120
f = file(batch_path, "w")
123
print "Created:", batch_path
125
print "ERROR: Unable to create %s: %s" % (batch_path, e)
127
def _quoted_path(self, path):
129
return '"' + path + '"'
133
def _win_batch_args(self):
134
from bzrlib.win32utils import winver
135
if winver == 'Windows NT':
138
return '%1 %2 %3 %4 %5 %6 %7 %8 %9'
139
#/class my_install_scripts
142
class bzr_build(build):
143
"""Customized build distutils action.
150
generate_docs.main(argv=["bzr", "man"])
153
########################
155
########################
157
command_classes = {'install_scripts': my_install_scripts,
159
from distutils import log
160
from distutils.errors import CCompilerError, DistutilsPlatformError
161
from distutils.extension import Extension
164
from Pyrex.Distutils import build_ext
167
# try to build the extension from the prior generated source.
169
print ("The python package 'Pyrex' is not available."
170
" If the .c files are available,")
171
print ("they will be built,"
172
" but modifying the .pyx files will not rebuild them.")
174
from distutils.command.build_ext import build_ext
179
class build_ext_if_possible(build_ext):
184
except DistutilsPlatformError, e:
186
log.warn('Extensions cannot be built, '
187
'will use the Python versions instead')
189
def build_extension(self, ext):
191
build_ext.build_extension(self, ext)
192
except CCompilerError:
193
log.warn('Building of "%s" extension failed, '
194
'will use the Python version instead' % (ext.name,))
197
# Override the build_ext if we have Pyrex available
198
command_classes['build_ext'] = build_ext_if_possible
199
unavailable_files = []
202
def add_pyrex_extension(module_name, **kwargs):
203
"""Add a pyrex module to build.
205
This will use Pyrex to auto-generate the .c file if it is available.
206
Otherwise it will fall back on the .c file. If the .c file is not
207
available, it will warn, and not add anything.
209
You can pass any extra options to Extension through kwargs. One example is
212
:param module_name: The python path to the module. This will be used to
213
determine the .pyx and .c files to use.
215
path = module_name.replace('.', '/')
216
pyrex_name = path + '.pyx'
219
ext_modules.append(Extension(module_name, [pyrex_name]))
221
if not os.path.isfile(c_name):
222
unavailable_files.append(c_name)
224
ext_modules.append(Extension(module_name, [c_name]))
227
add_pyrex_extension('bzrlib._dirstate_helpers_c')
228
add_pyrex_extension('bzrlib._knit_load_data_c')
229
ext_modules.append(Extension('bzrlib._patiencediff_c', ['bzrlib/_patiencediff_c.c']))
232
if unavailable_files:
233
print 'C extension(s) not found:'
234
print ' %s' % ('\n '.join(unavailable_files),)
235
print 'The python versions will be used instead.'
239
if 'bdist_wininst' in sys.argv:
242
for root, dirs, files in os.walk('doc'):
245
if (os.path.splitext(f)[1] in ('.html','.css','.png','.pdf')
246
or f == 'quick-start-summary.svg'):
247
r.append(os.path.join(root, f))
251
target = os.path.join('Doc\\Bazaar', relative)
253
target = 'Doc\\Bazaar'
254
docs.append((target, r))
257
# python's distutils-based win32 installer
258
ARGS = {'scripts': ['bzr', 'tools/win32/bzr-win32-bdist-postinstall.py'],
259
'ext_modules': ext_modules,
261
'data_files': find_docs(),
262
# for building pyrex extensions
263
'cmdclass': {'build_ext': build_ext_if_possible},
266
ARGS.update(META_INFO)
268
ARGS.update(PKG_DATA)
272
elif 'py2exe' in sys.argv:
277
# pick real bzr version
281
for i in bzrlib.version_info[:4]:
286
version_number.append(str(i))
287
version_str = '.'.join(version_number)
289
target = py2exe.build_exe.Target(script = "bzr",
291
icon_resources = [(0,'bzr.ico')],
292
name = META_INFO['name'],
293
version = version_str,
294
description = META_INFO['description'],
295
author = META_INFO['author'],
296
copyright = "(c) Canonical Ltd, 2005-2007",
297
company_name = "Canonical Ltd.",
298
comments = META_INFO['description'],
301
packages = BZRLIB['packages']
302
packages.remove('bzrlib')
303
packages = [i for i in packages if not i.startswith('bzrlib.plugins')]
305
for i in glob.glob('bzrlib\\*.py'):
306
module = i[:-3].replace('\\', '.')
307
if module.endswith('__init__'):
308
module = module[:-len('__init__')]
309
includes.append(module)
311
additional_packages = set()
312
if sys.version.startswith('2.4'):
313
# adding elementtree package
314
additional_packages.add('elementtree')
315
elif sys.version.startswith('2.5'):
316
additional_packages.add('xml.etree')
319
warnings.warn('Unknown Python version.\n'
320
'Please check setup.py script for compatibility.')
321
# email package from std python library use lazy import,
322
# so we need to explicitly add all package
323
additional_packages.add('email')
325
# text files for help topis
326
text_topics = glob.glob('bzrlib/help_topics/en/*.txt')
327
topics_files = [('lib/help_topics/en', text_topics)]
331
for root, dirs, files in os.walk('bzrlib/plugins'):
334
if not i.endswith('.py'):
336
if i == '__init__.py' and root == 'bzrlib/plugins':
338
x.append(os.path.join(root, i))
340
target_dir = root[len('bzrlib/'):] # install to 'plugins/...'
341
plugins_files.append((target_dir, x))
342
# find modules for built-in plugins
343
import tools.package_mf
344
mf = tools.package_mf.CustomModuleFinder()
345
mf.run_package('bzrlib/plugins')
346
packs, mods = mf.get_result()
347
additional_packages.update(packs)
349
options_list = {"py2exe": {"packages": packages + list(additional_packages),
350
"includes": includes + mods,
351
"excludes": ["Tkinter", "medusa", "tools"],
352
"dist_dir": "win32_bzr.exe",
355
setup(options=options_list,
357
'tools/win32/bzr_postinstall.py',
359
zipfile='lib/library.zip',
360
data_files=topics_files + plugins_files,
364
# ad-hoc for easy_install
366
if not 'bdist_egg' in sys.argv:
367
# generate and install bzr.1 only with plain install, not easy_install one
368
DATA_FILES = [('man/man1', ['bzr.1'])]
371
ARGS = {'scripts': ['bzr'],
372
'data_files': DATA_FILES,
373
'cmdclass': command_classes,
374
'ext_modules': ext_modules,
377
ARGS.update(META_INFO)
379
ARGS.update(PKG_DATA)