##// END OF EJS Templates
setup: cleanup coding style
Martin Geisler -
r10000:16f49d67 default
parent child Browse files
Show More
@@ -1,286 +1,286 b''
1 #!/usr/bin/env python
1 #!/usr/bin/env python
2 #
2 #
3 # This is the mercurial setup script.
3 # This is the mercurial setup script.
4 #
4 #
5 # 'python setup.py install', or
5 # 'python setup.py install', or
6 # 'python setup.py --help' for more options
6 # 'python setup.py --help' for more options
7
7
8 import sys
8 import sys
9 if not hasattr(sys, 'version_info') or sys.version_info < (2, 4, 0, 'final'):
9 if not hasattr(sys, 'version_info') or sys.version_info < (2, 4, 0, 'final'):
10 raise SystemExit("Mercurial requires Python 2.4 or later.")
10 raise SystemExit("Mercurial requires Python 2.4 or later.")
11
11
12 # Solaris Python packaging brain damage
12 # Solaris Python packaging brain damage
13 try:
13 try:
14 import hashlib
14 import hashlib
15 sha = hashlib.sha1()
15 sha = hashlib.sha1()
16 except:
16 except:
17 try:
17 try:
18 import sha
18 import sha
19 except:
19 except:
20 raise SystemExit(
20 raise SystemExit(
21 "Couldn't import standard hashlib (incomplete Python install).")
21 "Couldn't import standard hashlib (incomplete Python install).")
22
22
23 try:
23 try:
24 import zlib
24 import zlib
25 except:
25 except:
26 raise SystemExit(
26 raise SystemExit(
27 "Couldn't import standard zlib (incomplete Python install).")
27 "Couldn't import standard zlib (incomplete Python install).")
28
28
29 import os, subprocess, time
29 import os, subprocess, time
30 import shutil
30 import shutil
31 import tempfile
31 import tempfile
32 from distutils.core import setup, Extension
32 from distutils.core import setup, Extension
33 from distutils.dist import Distribution
33 from distutils.dist import Distribution
34 from distutils.command.install_data import install_data
34 from distutils.command.install_data import install_data
35 from distutils.command.build import build
35 from distutils.command.build import build
36 from distutils.command.build_py import build_py
36 from distutils.command.build_py import build_py
37 from distutils.spawn import spawn, find_executable
37 from distutils.spawn import spawn, find_executable
38 from distutils.ccompiler import new_compiler
38 from distutils.ccompiler import new_compiler
39
39
40 extra = {}
40 extra = {}
41 scripts = ['hg']
41 scripts = ['hg']
42 if os.name == 'nt':
42 if os.name == 'nt':
43 scripts.append('contrib/win32/hg.bat')
43 scripts.append('contrib/win32/hg.bat')
44
44
45 # simplified version of distutils.ccompiler.CCompiler.has_function
45 # simplified version of distutils.ccompiler.CCompiler.has_function
46 # that actually removes its temporary files.
46 # that actually removes its temporary files.
47 def has_function(cc, funcname):
47 def hasfunction(cc, funcname):
48 tmpdir = tempfile.mkdtemp(prefix='hg-install-')
48 tmpdir = tempfile.mkdtemp(prefix='hg-install-')
49 devnull = oldstderr = None
49 devnull = oldstderr = None
50 try:
50 try:
51 try:
51 try:
52 fname = os.path.join(tmpdir, 'funcname.c')
52 fname = os.path.join(tmpdir, 'funcname.c')
53 f = open(fname, 'w')
53 f = open(fname, 'w')
54 f.write('int main(void) {\n')
54 f.write('int main(void) {\n')
55 f.write(' %s();\n' % funcname)
55 f.write(' %s();\n' % funcname)
56 f.write('}\n')
56 f.write('}\n')
57 f.close()
57 f.close()
58 # Redirect stderr to /dev/null to hide any error messages
58 # Redirect stderr to /dev/null to hide any error messages
59 # from the compiler.
59 # from the compiler.
60 # This will have to be changed if we ever have to check
60 # This will have to be changed if we ever have to check
61 # for a function on Windows.
61 # for a function on Windows.
62 devnull = open('/dev/null', 'w')
62 devnull = open('/dev/null', 'w')
63 oldstderr = os.dup(sys.stderr.fileno())
63 oldstderr = os.dup(sys.stderr.fileno())
64 os.dup2(devnull.fileno(), sys.stderr.fileno())
64 os.dup2(devnull.fileno(), sys.stderr.fileno())
65 objects = cc.compile([fname], output_dir=tmpdir)
65 objects = cc.compile([fname], output_dir=tmpdir)
66 cc.link_executable(objects, os.path.join(tmpdir, "a.out"))
66 cc.link_executable(objects, os.path.join(tmpdir, "a.out"))
67 except:
67 except:
68 return False
68 return False
69 return True
69 return True
70 finally:
70 finally:
71 if oldstderr is not None:
71 if oldstderr is not None:
72 os.dup2(oldstderr, sys.stderr.fileno())
72 os.dup2(oldstderr, sys.stderr.fileno())
73 if devnull is not None:
73 if devnull is not None:
74 devnull.close()
74 devnull.close()
75 shutil.rmtree(tmpdir)
75 shutil.rmtree(tmpdir)
76
76
77 # py2exe needs to be installed to work
77 # py2exe needs to be installed to work
78 try:
78 try:
79 import py2exe
79 import py2exe
80
80
81 # Help py2exe to find win32com.shell
81 # Help py2exe to find win32com.shell
82 try:
82 try:
83 import modulefinder
83 import modulefinder
84 import win32com
84 import win32com
85 for p in win32com.__path__[1:]: # Take the path to win32comext
85 for p in win32com.__path__[1:]: # Take the path to win32comext
86 modulefinder.AddPackagePath("win32com", p)
86 modulefinder.AddPackagePath("win32com", p)
87 pn = "win32com.shell"
87 pn = "win32com.shell"
88 __import__(pn)
88 __import__(pn)
89 m = sys.modules[pn]
89 m = sys.modules[pn]
90 for p in m.__path__[1:]:
90 for p in m.__path__[1:]:
91 modulefinder.AddPackagePath(pn, p)
91 modulefinder.AddPackagePath(pn, p)
92 except ImportError:
92 except ImportError:
93 pass
93 pass
94
94
95 extra['console'] = ['hg']
95 extra['console'] = ['hg']
96
96
97 except ImportError:
97 except ImportError:
98 pass
98 pass
99
99
100 def runcmd(cmd, env):
100 def runcmd(cmd, env):
101 p = subprocess.Popen(cmd, stdout=subprocess.PIPE,
101 p = subprocess.Popen(cmd, stdout=subprocess.PIPE,
102 stderr=subprocess.PIPE, env=env)
102 stderr=subprocess.PIPE, env=env)
103 out, err = p.communicate()
103 out, err = p.communicate()
104 # If root is executing setup.py, but the repository is owned by
104 # If root is executing setup.py, but the repository is owned by
105 # another user (as in "sudo python setup.py install") we will get
105 # another user (as in "sudo python setup.py install") we will get
106 # trust warnings since the .hg/hgrc file is untrusted. That is
106 # trust warnings since the .hg/hgrc file is untrusted. That is
107 # fine, we don't want to load it anyway.
107 # fine, we don't want to load it anyway.
108 err = [e for e in err.splitlines()
108 err = [e for e in err.splitlines()
109 if not e.startswith('Not trusting file')]
109 if not e.startswith('Not trusting file')]
110 if err:
110 if err:
111 return ''
111 return ''
112 return out
112 return out
113
113
114 version = ''
114 version = ''
115
115
116 if os.path.isdir('.hg'):
116 if os.path.isdir('.hg'):
117 # Execute hg out of this directory with a custom environment which
117 # Execute hg out of this directory with a custom environment which
118 # includes the pure Python modules in mercurial/pure. We also take
118 # includes the pure Python modules in mercurial/pure. We also take
119 # care to not use any hgrc files and do no localization.
119 # care to not use any hgrc files and do no localization.
120 pypath = ['mercurial', os.path.join('mercurial', 'pure')]
120 pypath = ['mercurial', os.path.join('mercurial', 'pure')]
121 env = {'PYTHONPATH': os.pathsep.join(pypath),
121 env = {'PYTHONPATH': os.pathsep.join(pypath),
122 'HGRCPATH': '',
122 'HGRCPATH': '',
123 'LANGUAGE': 'C'}
123 'LANGUAGE': 'C'}
124 if 'LD_LIBRARY_PATH' in os.environ:
124 if 'LD_LIBRARY_PATH' in os.environ:
125 env['LD_LIBRARY_PATH'] = os.environ['LD_LIBRARY_PATH']
125 env['LD_LIBRARY_PATH'] = os.environ['LD_LIBRARY_PATH']
126 if 'SystemRoot' in os.environ:
126 if 'SystemRoot' in os.environ:
127 # Copy SystemRoot into the custom environment for Python 2.6
127 # Copy SystemRoot into the custom environment for Python 2.6
128 # under Windows. Otherwise, the subprocess will fail with
128 # under Windows. Otherwise, the subprocess will fail with
129 # error 0xc0150004. See: http://bugs.python.org/issue3440
129 # error 0xc0150004. See: http://bugs.python.org/issue3440
130 env['SystemRoot'] = os.environ['SystemRoot']
130 env['SystemRoot'] = os.environ['SystemRoot']
131 cmd = [sys.executable, 'hg', 'id', '-i', '-t']
131 cmd = [sys.executable, 'hg', 'id', '-i', '-t']
132 l = runcmd(cmd, env).split()
132 l = runcmd(cmd, env).split()
133 while len(l) > 1 and l[-1][0].isalpha(): # remove non-numbered tags
133 while len(l) > 1 and l[-1][0].isalpha(): # remove non-numbered tags
134 l.pop()
134 l.pop()
135 if len(l) > 1: # tag found
135 if len(l) > 1: # tag found
136 version = l[-1]
136 version = l[-1]
137 if l[0].endswith('+'): # propagate the dirty status to the tag
137 if l[0].endswith('+'): # propagate the dirty status to the tag
138 version += '+'
138 version += '+'
139 elif len(l) == 1: # no tag found
139 elif len(l) == 1: # no tag found
140 cmd = [sys.executable, 'hg', 'parents', '--template',
140 cmd = [sys.executable, 'hg', 'parents', '--template',
141 '{latesttag}+{latesttagdistance}-']
141 '{latesttag}+{latesttagdistance}-']
142 version = runcmd(cmd, env) + l[0]
142 version = runcmd(cmd, env) + l[0]
143 if version.endswith('+'):
143 if version.endswith('+'):
144 version += time.strftime('%Y%m%d')
144 version += time.strftime('%Y%m%d')
145 elif os.path.exists('.hg_archival.txt'):
145 elif os.path.exists('.hg_archival.txt'):
146 kw = dict([t.strip() for t in l.split(':', 1)]
146 kw = dict([t.strip() for t in l.split(':', 1)]
147 for l in open('.hg_archival.txt'))
147 for l in open('.hg_archival.txt'))
148 if 'tag' in kw:
148 if 'tag' in kw:
149 version = kw['tag']
149 version = kw['tag']
150 elif 'latesttag' in kw:
150 elif 'latesttag' in kw:
151 version = '%(latesttag)s+%(latesttagdistance)s-%(node).12s' % kw
151 version = '%(latesttag)s+%(latesttagdistance)s-%(node).12s' % kw
152 else:
152 else:
153 version = kw.get('node', '')[:12]
153 version = kw.get('node', '')[:12]
154
154
155 if version:
155 if version:
156 f = open("mercurial/__version__.py", "w")
156 f = open("mercurial/__version__.py", "w")
157 f.write('# this file is autogenerated by setup.py\n')
157 f.write('# this file is autogenerated by setup.py\n')
158 f.write('version = "%s"\n' % version)
158 f.write('version = "%s"\n' % version)
159 f.close()
159 f.close()
160
160
161
161
162 try:
162 try:
163 from mercurial import __version__
163 from mercurial import __version__
164 version = __version__.version
164 version = __version__.version
165 except ImportError:
165 except ImportError:
166 version = 'unknown'
166 version = 'unknown'
167
167
168 class build_mo(build):
168 class hgbuildmo(build):
169
169
170 description = "build translations (.mo files)"
170 description = "build translations (.mo files)"
171
171
172 def run(self):
172 def run(self):
173 if not find_executable('msgfmt'):
173 if not find_executable('msgfmt'):
174 self.warn("could not find msgfmt executable, no translations "
174 self.warn("could not find msgfmt executable, no translations "
175 "will be built")
175 "will be built")
176 return
176 return
177
177
178 podir = 'i18n'
178 podir = 'i18n'
179 if not os.path.isdir(podir):
179 if not os.path.isdir(podir):
180 self.warn("could not find %s/ directory" % podir)
180 self.warn("could not find %s/ directory" % podir)
181 return
181 return
182
182
183 join = os.path.join
183 join = os.path.join
184 for po in os.listdir(podir):
184 for po in os.listdir(podir):
185 if not po.endswith('.po'):
185 if not po.endswith('.po'):
186 continue
186 continue
187 pofile = join(podir, po)
187 pofile = join(podir, po)
188 modir = join('locale', po[:-3], 'LC_MESSAGES')
188 modir = join('locale', po[:-3], 'LC_MESSAGES')
189 mofile = join(modir, 'hg.mo')
189 mofile = join(modir, 'hg.mo')
190 mobuildfile = join('mercurial', mofile)
190 mobuildfile = join('mercurial', mofile)
191 cmd = ['msgfmt', '-v', '-o', mobuildfile, pofile]
191 cmd = ['msgfmt', '-v', '-o', mobuildfile, pofile]
192 if sys.platform != 'sunos5':
192 if sys.platform != 'sunos5':
193 # msgfmt on Solaris does not know about -c
193 # msgfmt on Solaris does not know about -c
194 cmd.append('-c')
194 cmd.append('-c')
195 self.mkpath(join('mercurial', modir))
195 self.mkpath(join('mercurial', modir))
196 self.make_file([pofile], mobuildfile, spawn, (cmd,))
196 self.make_file([pofile], mobuildfile, spawn, (cmd,))
197
197
198 # Insert build_mo first so that files in mercurial/locale/ are found
198 # Insert hgbuildmo first so that files in mercurial/locale/ are found
199 # when build_py is run next.
199 # when build_py is run next.
200 build.sub_commands.insert(0, ('build_mo', None))
200 build.sub_commands.insert(0, ('build_mo', None))
201
201
202 Distribution.pure = 0
202 Distribution.pure = 0
203 Distribution.global_options.append(('pure', None, "use pure (slow) Python "
203 Distribution.global_options.append(('pure', None, "use pure (slow) Python "
204 "code instead of C extensions"))
204 "code instead of C extensions"))
205
205
206 class hg_build_py(build_py):
206 class hgbuildpy(build_py):
207
207
208 def finalize_options(self):
208 def finalize_options(self):
209 build_py.finalize_options(self)
209 build_py.finalize_options(self)
210
210
211 if self.distribution.pure:
211 if self.distribution.pure:
212 if self.py_modules is None:
212 if self.py_modules is None:
213 self.py_modules = []
213 self.py_modules = []
214 for ext in self.distribution.ext_modules:
214 for ext in self.distribution.ext_modules:
215 if ext.name.startswith("mercurial."):
215 if ext.name.startswith("mercurial."):
216 self.py_modules.append("mercurial.pure.%s" % ext.name[10:])
216 self.py_modules.append("mercurial.pure.%s" % ext.name[10:])
217 self.distribution.ext_modules = []
217 self.distribution.ext_modules = []
218
218
219 def find_modules(self):
219 def find_modules(self):
220 modules = build_py.find_modules(self)
220 modules = build_py.find_modules(self)
221 for module in modules:
221 for module in modules:
222 if module[0] == "mercurial.pure":
222 if module[0] == "mercurial.pure":
223 if module[1] != "__init__":
223 if module[1] != "__init__":
224 yield ("mercurial", module[1], module[2])
224 yield ("mercurial", module[1], module[2])
225 else:
225 else:
226 yield module
226 yield module
227
227
228 cmdclass = {'build_mo': build_mo,
228 cmdclass = {'build_mo': hgbuildmo,
229 'build_py': hg_build_py}
229 'build_py': hgbuildpy}
230
230
231 ext_modules=[
231 extmodules = [
232 Extension('mercurial.base85', ['mercurial/base85.c']),
232 Extension('mercurial.base85', ['mercurial/base85.c']),
233 Extension('mercurial.bdiff', ['mercurial/bdiff.c']),
233 Extension('mercurial.bdiff', ['mercurial/bdiff.c']),
234 Extension('mercurial.diffhelpers', ['mercurial/diffhelpers.c']),
234 Extension('mercurial.diffhelpers', ['mercurial/diffhelpers.c']),
235 Extension('mercurial.mpatch', ['mercurial/mpatch.c']),
235 Extension('mercurial.mpatch', ['mercurial/mpatch.c']),
236 Extension('mercurial.parsers', ['mercurial/parsers.c']),
236 Extension('mercurial.parsers', ['mercurial/parsers.c']),
237 Extension('mercurial.osutil', ['mercurial/osutil.c']),
237 Extension('mercurial.osutil', ['mercurial/osutil.c']),
238 ]
238 ]
239
239
240 packages = ['mercurial', 'mercurial.hgweb', 'hgext', 'hgext.convert',
240 packages = ['mercurial', 'mercurial.hgweb', 'hgext', 'hgext.convert',
241 'hgext.highlight', 'hgext.zeroconf', ]
241 'hgext.highlight', 'hgext.zeroconf', ]
242
242
243 if sys.platform == 'linux2' and os.uname()[2] > '2.6':
243 if sys.platform == 'linux2' and os.uname()[2] > '2.6':
244 # The inotify extension is only usable with Linux 2.6 kernels.
244 # The inotify extension is only usable with Linux 2.6 kernels.
245 # You also need a reasonably recent C library.
245 # You also need a reasonably recent C library.
246 cc = new_compiler()
246 cc = new_compiler()
247 if has_function(cc, 'inotify_add_watch'):
247 if hasfunction(cc, 'inotify_add_watch'):
248 ext_modules.append(Extension('hgext.inotify.linux._inotify',
248 extmodules.append(Extension('hgext.inotify.linux._inotify',
249 ['hgext/inotify/linux/_inotify.c']))
249 ['hgext/inotify/linux/_inotify.c']))
250 packages.extend(['hgext.inotify', 'hgext.inotify.linux'])
250 packages.extend(['hgext.inotify', 'hgext.inotify.linux'])
251
251
252 packagedata = {'mercurial': ['locale/*/LC_MESSAGES/hg.mo',
252 packagedata = {'mercurial': ['locale/*/LC_MESSAGES/hg.mo',
253 'help/*.txt']}
253 'help/*.txt']}
254
254
255 def ordinarypath(p):
255 def ordinarypath(p):
256 return p and p[0] != '.' and p[-1] != '~'
256 return p and p[0] != '.' and p[-1] != '~'
257
257
258 for root in ('templates', ):
258 for root in ('templates', ):
259 for curdir, dirs, files in os.walk(os.path.join('mercurial', root)):
259 for curdir, dirs, files in os.walk(os.path.join('mercurial', root)):
260 curdir = curdir.split(os.sep, 1)[1]
260 curdir = curdir.split(os.sep, 1)[1]
261 dirs[:] = filter(ordinarypath, dirs)
261 dirs[:] = filter(ordinarypath, dirs)
262 for f in filter(ordinarypath, files):
262 for f in filter(ordinarypath, files):
263 f = os.path.join(curdir, f)
263 f = os.path.join(curdir, f)
264 packagedata['mercurial'].append(f)
264 packagedata['mercurial'].append(f)
265
265
266 datafiles = []
266 datafiles = []
267
267
268 setup(name='mercurial',
268 setup(name='mercurial',
269 version=version,
269 version=version,
270 author='Matt Mackall',
270 author='Matt Mackall',
271 author_email='mpm@selenic.com',
271 author_email='mpm@selenic.com',
272 url='http://mercurial.selenic.com/',
272 url='http://mercurial.selenic.com/',
273 description='Scalable distributed SCM',
273 description='Scalable distributed SCM',
274 license='GNU GPL',
274 license='GNU GPL',
275 scripts=scripts,
275 scripts=scripts,
276 packages=packages,
276 packages=packages,
277 ext_modules=ext_modules,
277 ext_modules=extmodules,
278 data_files=datafiles,
278 data_files=datafiles,
279 package_data=packagedata,
279 package_data=packagedata,
280 cmdclass=cmdclass,
280 cmdclass=cmdclass,
281 options=dict(py2exe=dict(packages=['hgext', 'email']),
281 options=dict(py2exe=dict(packages=['hgext', 'email']),
282 bdist_mpkg=dict(zipdist=True,
282 bdist_mpkg=dict(zipdist=True,
283 license='COPYING',
283 license='COPYING',
284 readme='contrib/macosx/Readme.html',
284 readme='contrib/macosx/Readme.html',
285 welcome='contrib/macosx/Welcome.html')),
285 welcome='contrib/macosx/Welcome.html')),
286 **extra)
286 **extra)
General Comments 0
You need to be logged in to leave comments. Login now