]> git.rkrishnan.org Git - tahoe-lafs/tahoe-lafs.git/blob - src/allmydata/__init__.py
Suppress the PowmInsecureWarning from PyCrypto. refs #1586
[tahoe-lafs/tahoe-lafs.git] / src / allmydata / __init__.py
1 """
2 Decentralized storage grid.
3
4 community web site: U{https://tahoe-lafs.org/}
5 """
6
7 class PackagingError(EnvironmentError):
8     """
9     Raised when there is an error in packaging of Tahoe-LAFS or its
10     dependencies which makes it impossible to proceed safely.
11     """
12     pass
13
14 __version__ = "unknown"
15 try:
16     from allmydata._version import __version__
17 except ImportError:
18     # We're running in a tree that hasn't run "./setup.py darcsver", and didn't
19     # come with a _version.py, so we don't know what our version is. This should
20     # not happen very often.
21     pass
22
23 __appname__ = "unknown"
24 try:
25     from allmydata._appname import __appname__
26 except ImportError:
27     # We're running in a tree that hasn't run "./setup.py".  This shouldn't happen.
28     pass
29
30 # __full_version__ is the one that you ought to use when identifying yourself in the
31 # "application" part of the Tahoe versioning scheme:
32 # https://tahoe-lafs.org/trac/tahoe-lafs/wiki/Versioning
33 __full_version__ = __appname__ + '/' + str(__version__)
34
35 import os, platform, re, subprocess, sys, traceback
36 _distributor_id_cmdline_re = re.compile("(?:Distributor ID:)\s*(.*)", re.I)
37 _release_cmdline_re = re.compile("(?:Release:)\s*(.*)", re.I)
38
39 _distributor_id_file_re = re.compile("(?:DISTRIB_ID\s*=)\s*(.*)", re.I)
40 _release_file_re = re.compile("(?:DISTRIB_RELEASE\s*=)\s*(.*)", re.I)
41
42 global _distname,_version
43 _distname = None
44 _version = None
45
46 def get_linux_distro():
47     """ Tries to determine the name of the Linux OS distribution name.
48
49     First, try to parse a file named "/etc/lsb-release".  If it exists, and
50     contains the "DISTRIB_ID=" line and the "DISTRIB_RELEASE=" line, then return
51     the strings parsed from that file.
52
53     If that doesn't work, then invoke platform.dist().
54
55     If that doesn't work, then try to execute "lsb_release", as standardized in
56     2001:
57
58     http://refspecs.freestandards.org/LSB_1.0.0/gLSB/lsbrelease.html
59
60     The current version of the standard is here:
61
62     http://refspecs.freestandards.org/LSB_3.2.0/LSB-Core-generic/LSB-Core-generic/lsbrelease.html
63
64     that lsb_release emitted, as strings.
65
66     Returns a tuple (distname,version). Distname is what LSB calls a
67     "distributor id", e.g. "Ubuntu".  Version is what LSB calls a "release",
68     e.g. "8.04".
69
70     A version of this has been submitted to python as a patch for the standard
71     library module "platform":
72
73     http://bugs.python.org/issue3937
74     """
75     global _distname,_version
76     if _distname and _version:
77         return (_distname, _version)
78
79     try:
80         etclsbrel = open("/etc/lsb-release", "rU")
81         for line in etclsbrel:
82             m = _distributor_id_file_re.search(line)
83             if m:
84                 _distname = m.group(1).strip()
85                 if _distname and _version:
86                     return (_distname, _version)
87             m = _release_file_re.search(line)
88             if m:
89                 _version = m.group(1).strip()
90                 if _distname and _version:
91                     return (_distname, _version)
92     except EnvironmentError:
93         pass
94
95     (_distname, _version) = platform.dist()[:2]
96     if _distname and _version:
97         return (_distname, _version)
98
99     try:
100         p = subprocess.Popen(["lsb_release", "--all"], stdout=subprocess.PIPE, stderr=subprocess.PIPE)
101         rc = p.wait()
102         if rc == 0:
103             for line in p.stdout.readlines():
104                 m = _distributor_id_cmdline_re.search(line)
105                 if m:
106                     _distname = m.group(1).strip()
107                     if _distname and _version:
108                         return (_distname, _version)
109
110                 m = _release_cmdline_re.search(p.stdout.read())
111                 if m:
112                     _version = m.group(1).strip()
113                     if _distname and _version:
114                         return (_distname, _version)
115     except EnvironmentError:
116         pass
117
118     if os.path.exists("/etc/arch-release"):
119         return ("Arch_Linux", "")
120
121     return (_distname,_version)
122
123 def get_platform():
124     # Our version of platform.platform(), telling us both less and more than the
125     # Python Standard Library's version does.
126     # We omit details such as the Linux kernel version number, but we add a
127     # more detailed and correct rendition of the Linux distribution and
128     # distribution-version.
129     if "linux" in platform.system().lower():
130         return platform.system()+"-"+"_".join(get_linux_distro())+"-"+platform.machine()+"-"+"_".join([x for x in platform.architecture() if x])
131     else:
132         return platform.platform()
133
134
135 from allmydata.util import verlib
136 def normalized_version(verstr, what=None):
137     try:
138         return verlib.NormalizedVersion(verlib.suggest_normalized_version(verstr))
139     except (StandardError, verlib.IrrationalVersionError):
140         cls, value, trace = sys.exc_info()
141         raise PackagingError, ("could not parse %s due to %s: %s"
142                                % (what or repr(verstr), cls.__name__, value)), trace
143
144
145 def get_package_versions_and_locations():
146     import warnings
147     from _auto_deps import package_imports, deprecation_messages, \
148         user_warning_messages, runtime_warning_messages, warning_imports
149
150     def package_dir(srcfile):
151         return os.path.dirname(os.path.dirname(os.path.normcase(os.path.realpath(srcfile))))
152
153     # pkg_resources.require returns the distribution that pkg_resources attempted to put
154     # on sys.path, which can differ from the one that we actually import due to #1258,
155     # or any other bug that causes sys.path to be set up incorrectly. Therefore we
156     # must import the packages in order to check their versions and paths.
157
158     # This warning is generated by twisted, PyRex, and possibly other packages,
159     # but can happen at any time, not only when they are imported. See ticket #1129.
160     warnings.filterwarnings("ignore", category=DeprecationWarning,
161         message="BaseException.message has been deprecated as of Python 2.6",
162         append=True)
163
164     # This is to suppress various DeprecationWarnings and UserWarnings that
165     # occur when modules are imported.  See #859, #1435 and
166     # http://divmod.org/trac/ticket/2994 [broken link].
167
168     for msg in deprecation_messages:
169         warnings.filterwarnings("ignore", category=DeprecationWarning, message=msg, append=True)
170     for msg in user_warning_messages:
171         warnings.filterwarnings("ignore", category=UserWarning, message=msg, append=True)
172     for msg in runtime_warning_messages:
173         warnings.filterwarnings("ignore", category=RuntimeWarning, message=msg, append=True)
174     try:
175         for modulename in warning_imports:
176             try:
177                 __import__(modulename)
178             except ImportError:
179                 pass
180     finally:
181         for ign in runtime_warning_messages + user_warning_messages + deprecation_messages:
182             warnings.filters.pop()
183
184     packages = []
185
186     def get_version(module, attr):
187         return str(getattr(module, attr, 'unknown'))
188
189     for pkgname, modulename in [(__appname__, 'allmydata')] + package_imports:
190         if modulename:
191             try:
192                 __import__(modulename)
193                 module = sys.modules[modulename]
194             except ImportError:
195                 etype, emsg, etrace = sys.exc_info()
196                 trace_info = (etype, str(emsg), ([None] + traceback.extract_tb(etrace))[-1])
197                 packages.append( (pkgname, (None, None, trace_info)) )
198             else:
199                 if 'sqlite' in pkgname:
200                     packages.append( (pkgname, (get_version(module, 'version'), package_dir(module.__file__),
201                                                'sqlite %s' % (get_version(module, 'sqlite_version'),))) )
202                 else:
203                     comment = None
204                     if pkgname == 'setuptools' and hasattr(module, '_distribute'):
205                         # distribute does not report its version in any module variables
206                         comment = 'distribute'
207                     packages.append( (pkgname, (get_version(module, '__version__'), package_dir(module.__file__), comment)) )
208         elif pkgname == 'python':
209             packages.append( (pkgname, (platform.python_version(), sys.executable, None)) )
210         elif pkgname == 'platform':
211             packages.append( (pkgname, (get_platform(), None, None)) )
212
213     return packages
214
215
216 def check_requirement(req, vers_and_locs):
217     # TODO: check [] options
218     # We support only disjunctions of <=, >=, and ==
219
220     reqlist = req.split(',')
221     name = reqlist[0].split('<=')[0].split('>=')[0].split('==')[0].strip(' ').split('[')[0]
222     if name not in vers_and_locs:
223         raise PackagingError("no version info for %s" % (name,))
224     if req.strip(' ') == name:
225         return
226     (actual, location, comment) = vers_and_locs[name]
227     if actual is None:
228         # comment is (type, message, (filename, line number, function name, text)) for the original ImportError
229         raise ImportError("for requirement %r: %s" % (req, comment))
230     if actual == 'unknown':
231         return
232     actualver = normalized_version(actual, what="actual version %r of %s from %r" % (actual, name, location))
233
234     for r in reqlist:
235         s = r.split('<=')
236         if len(s) == 2:
237             required = s[1].strip(' ')
238             if actualver <= normalized_version(required, what="required maximum version %r in %r" % (required, req)):
239                 return  # maximum requirement met
240         else:
241             s = r.split('>=')
242             if len(s) == 2:
243                 required = s[1].strip(' ')
244                 if actualver >= normalized_version(required, what="required minimum version %r in %r" % (required, req)):
245                     return  # minimum requirement met
246             else:
247                 s = r.split('==')
248                 if len(s) == 2:
249                     required = s[1].strip(' ')
250                     if actualver == normalized_version(required, what="required exact version %r in %r" % (required, req)):
251                         return  # exact requirement met
252                 else:
253                     raise PackagingError("no version info or could not understand requirement %r" % (req,))
254
255     msg = ("We require %s, but could only find version %s.\n" % (req, actual))
256     if location and location != 'unknown':
257         msg += "The version we found is from %r.\n" % (location,)
258     msg += ("To resolve this problem, uninstall that version, either using your\n"
259             "operating system's package manager or by moving aside the directory.")
260     raise PackagingError(msg)
261
262
263 _vers_and_locs_list = get_package_versions_and_locations()
264
265
266 def cross_check_pkg_resources_versus_import():
267     """This function returns a list of errors due to any failed cross-checks."""
268
269     import pkg_resources
270     from _auto_deps import install_requires
271
272     pkg_resources_vers_and_locs = dict([(p.project_name.lower(), (str(p.version), p.location))
273                                         for p in pkg_resources.require(install_requires)])
274
275     return cross_check(pkg_resources_vers_and_locs, _vers_and_locs_list)
276
277
278 def cross_check(pkg_resources_vers_and_locs, imported_vers_and_locs_list):
279     """This function returns a list of errors due to any failed cross-checks."""
280
281     errors = []
282     not_pkg_resourceable = set(['sqlite3', 'python', 'platform', __appname__.lower()])
283     not_import_versionable = set(['zope.interface', 'mock', 'pyasn1'])
284     ignorable = set(['argparse', 'pyutil', 'zbase32', 'distribute', 'twisted-web', 'twisted-core'])
285
286     for name, (imp_ver, imp_loc, imp_comment) in imported_vers_and_locs_list:
287         name = name.lower()
288         if name not in not_pkg_resourceable:
289             if name not in pkg_resources_vers_and_locs:
290                 if name == "setuptools" and "distribute" in pkg_resources_vers_and_locs:
291                     pr_ver, pr_loc = pkg_resources_vers_and_locs["distribute"]
292                     if not (os.path.normpath(os.path.realpath(pr_loc)) == os.path.normpath(os.path.realpath(imp_loc))
293                             and imp_comment == "distribute"):
294                         errors.append("Warning: dependency 'setuptools' found to be version %r of 'distribute' from %r "
295                                       "by pkg_resources, but 'import setuptools' gave version %r [%s] from %r. "
296                                       "A version mismatch is expected, but a location mismatch is not."
297                                       % (pr_ver, pr_loc, imp_ver, imp_comment or 'probably *not* distribute', imp_loc))
298                 else:
299                     errors.append("Warning: dependency %r (version %r imported from %r) was not found by pkg_resources."
300                                   % (name, imp_ver, imp_loc))
301                 continue
302
303             pr_ver, pr_loc = pkg_resources_vers_and_locs[name]
304             try:
305                 pr_normver = normalized_version(pr_ver)
306             except Exception, e:
307                 errors.append("Warning: version number %r found for dependency %r by pkg_resources could not be parsed. "
308                               "The version found by import was %r from %r. "
309                               "pkg_resources thought it should be found at %r. "
310                               "The exception was %s: %s"
311                               % (pr_ver, name, imp_ver, imp_loc, pr_loc, e.__class__.__name__, e))
312             else:
313                 if imp_ver == 'unknown':
314                     if name not in not_import_versionable:
315                         errors.append("Warning: unexpectedly could not find a version number for dependency %r imported from %r. "
316                                       "pkg_resources thought it should be version %r at %r."
317                                       % (name, imp_loc, pr_ver, pr_loc))
318                 else:
319                     try:
320                         imp_normver = normalized_version(imp_ver)
321                     except Exception, e:
322                         errors.append("Warning: version number %r found for dependency %r (imported from %r) could not be parsed. "
323                                       "pkg_resources thought it should be version %r at %r. "
324                                       "The exception was %s: %s"
325                                       % (imp_ver, name, imp_loc, pr_ver, pr_loc, e.__class__.__name__, e))
326                     else:
327                         if pr_ver == 'unknown' or (pr_normver != imp_normver):
328                             if not os.path.normpath(os.path.realpath(pr_loc)) == os.path.normpath(os.path.realpath(imp_loc)):
329                                 errors.append("Warning: dependency %r found to have version number %r (normalized to %r, from %r) "
330                                               "by pkg_resources, but version %r (normalized to %r, from %r) by import."
331                                               % (name, pr_ver, str(pr_normver), pr_loc, imp_ver, str(imp_normver), imp_loc))
332
333     imported_packages = set([p.lower() for (p, _) in imported_vers_and_locs_list])
334     for pr_name, (pr_ver, pr_loc) in pkg_resources_vers_and_locs.iteritems():
335         if pr_name not in imported_packages and pr_name not in ignorable:
336             errors.append("Warning: dependency %r (version %r) found by pkg_resources not found by import."
337                           % (pr_name, pr_ver))
338
339     return errors
340
341
342 def get_error_string(errors, debug=False):
343     from allmydata._auto_deps import install_requires
344
345     msg = "\n%s\n" % ("\n".join(errors),)
346     if debug:
347         msg += ("\n"
348                 "For debugging purposes, the PYTHONPATH was\n"
349                 "  %r\n"
350                 "install_requires was\n"
351                 "  %r\n"
352                 "sys.path after importing pkg_resources was\n"
353                 "  %s\n"
354                 % (os.environ.get('PYTHONPATH'), install_requires, (os.pathsep+"\n  ").join(sys.path)) )
355     return msg
356
357 def check_all_requirements():
358     """This function returns a list of errors due to any failed checks."""
359
360     from allmydata._auto_deps import install_requires
361
362     errors = []
363
364     # we require 2.4.4 on non-UCS-2, non-Redhat builds to avoid <http://www.python.org/news/security/PSF-2006-001/>
365     # we require 2.4.3 on non-UCS-2 Redhat, because 2.4.3 is common on Redhat-based distros and will have patched the above bug
366     # we require at least 2.4.2 in any case to avoid a bug in the base64 module: <http://bugs.python.org/issue1171487>
367     if sys.maxunicode == 65535:
368         if sys.version_info < (2, 4, 2) or sys.version_info[0] > 2:
369             errors.append("Tahoe-LAFS current requires Python v2.4.2 or greater "
370                           "for a UCS-2 build (but less than v3), not %r" %
371                           (sys.version_info,))
372     elif platform.platform().lower().find('redhat') >= 0:
373         if sys.version_info < (2, 4, 3) or sys.version_info[0] > 2:
374             errors.append("Tahoe-LAFS current requires Python v2.4.3 or greater "
375                           "on Redhat-based distributions (but less than v3), not %r" %
376                           (sys.version_info,))
377     else:
378         if sys.version_info < (2, 4, 4) or sys.version_info[0] > 2:
379             errors.append("Tahoe-LAFS current requires Python v2.4.4 or greater "
380                           "for a non-UCS-2 build (but less than v3), not %r" %
381                           (sys.version_info,))
382
383     vers_and_locs = dict(_vers_and_locs_list)
384     for requirement in install_requires:
385         try:
386             check_requirement(requirement, vers_and_locs)
387         except (ImportError, PackagingError), e:
388             errors.append("%s: %s" % (e.__class__.__name__, e))
389
390     if errors:
391         raise PackagingError(get_error_string(errors, debug=True))
392
393 check_all_requirements()
394
395
396 def get_package_versions():
397     return dict([(k, v) for k, (v, l, c) in _vers_and_locs_list])
398
399 def get_package_locations():
400     return dict([(k, l) for k, (v, l, c) in _vers_and_locs_list])
401
402 def get_package_versions_string(show_paths=False, debug=False):
403     res = []
404     for p, (v, loc, comment) in _vers_and_locs_list:
405         info = str(p) + ": " + str(v)
406         if comment:
407             info = info + " [%s]" % str(comment)
408         if show_paths:
409             info = info + " (%s)" % str(loc)
410         res.append(info)
411
412     output = ",\n".join(res) + "\n"
413
414     if not hasattr(sys, 'frozen'):
415         errors = cross_check_pkg_resources_versus_import()
416         if errors:
417             output += get_error_string(errors, debug=debug)
418
419     return output