mirror of
https://github.com/tahoe-lafs/tahoe-lafs.git
synced 2025-01-02 11:16:41 +00:00
f4f69f4a95
Using pkg_resources is probably better if it works -- zope.interface doesn't have a __version__ attribute that we can query, but pkg_resources knows zope.interface's version number, for one thing. This code falls back to the old way -- looking at the __version__ attributes and __file__ attributes -- if the pkg_resources way doesn't answer. Note that this patch also changes the capitalization of "Nevow", "Twisted", and "pyOpenSSL", and the spelling of "allmydata-tahoe". These changes are not frivolous: they are reflecting the fact that we are naming Python packages (technically called Python "distributions") instead of Python modules (technically and confusingly called Python "packages") here. The package ("distribution") is named "allmydata-tahoe". The module ("package") is named "allmydata".
180 lines
6.6 KiB
Python
180 lines
6.6 KiB
Python
"""
|
|
Decentralized storage grid.
|
|
|
|
maintainer web site: U{http://allmydata.com/}
|
|
|
|
community web site: U{http://allmydata.org/}
|
|
"""
|
|
|
|
__version__ = "unknown"
|
|
try:
|
|
from _version import __version__
|
|
except ImportError:
|
|
# We're running in a tree that hasn't run "./setup.py darcsver", and didn't
|
|
# come with a _version.py, so we don't know what our version is. This should
|
|
# not happen very often.
|
|
pass
|
|
|
|
hush_pyflakes = __version__
|
|
del hush_pyflakes
|
|
|
|
import _auto_deps
|
|
_auto_deps.require_auto_deps()
|
|
|
|
import os, platform, re, subprocess, sys
|
|
_distributor_id_cmdline_re = re.compile("(?:Distributor ID:)\s*(.*)", re.I)
|
|
_release_cmdline_re = re.compile("(?:Release:)\s*(.*)", re.I)
|
|
|
|
_distributor_id_file_re = re.compile("(?:DISTRIB_ID\s*=)\s*(.*)", re.I)
|
|
_release_file_re = re.compile("(?:DISTRIB_RELEASE\s*=)\s*(.*)", re.I)
|
|
|
|
global _distname,_version
|
|
_distname = None
|
|
_version = None
|
|
|
|
def get_linux_distro():
|
|
""" Tries to determine the name of the Linux OS distribution name.
|
|
|
|
First, try to parse a file named "/etc/lsb-release". If it exists, and
|
|
contains the "DISTRIB_ID=" line and the "DISTRIB_RELEASE=" line, then return
|
|
the strings parsed from that file.
|
|
|
|
If that doesn't work, then invoke platform.dist().
|
|
|
|
If that doesn't work, then try to execute "lsb_release", as standardized in
|
|
2001:
|
|
|
|
http://refspecs.freestandards.org/LSB_1.0.0/gLSB/lsbrelease.html
|
|
|
|
The current version of the standard is here:
|
|
|
|
http://refspecs.freestandards.org/LSB_3.2.0/LSB-Core-generic/LSB-Core-generic/lsbrelease.html
|
|
|
|
that lsb_release emitted, as strings.
|
|
|
|
Returns a tuple (distname,version). Distname is what LSB calls a
|
|
"distributor id", e.g. "Ubuntu". Version is what LSB calls a "release",
|
|
e.g. "8.04".
|
|
|
|
A version of this has been submitted to python as a patch for the standard
|
|
library module "platform":
|
|
|
|
http://bugs.python.org/issue3937
|
|
"""
|
|
global _distname,_version
|
|
if _distname and _version:
|
|
return (_distname, _version)
|
|
|
|
try:
|
|
etclsbrel = open("/etc/lsb-release", "rU")
|
|
for line in etclsbrel:
|
|
m = _distributor_id_file_re.search(line)
|
|
if m:
|
|
_distname = m.group(1).strip()
|
|
if _distname and _version:
|
|
return (_distname, _version)
|
|
m = _release_file_re.search(line)
|
|
if m:
|
|
_version = m.group(1).strip()
|
|
if _distname and _version:
|
|
return (_distname, _version)
|
|
except EnvironmentError:
|
|
pass
|
|
|
|
(_distname, _version) = platform.dist()[:2]
|
|
if _distname and _version:
|
|
return (_distname, _version)
|
|
|
|
try:
|
|
p = subprocess.Popen(["lsb_release", "--all"], stdout=subprocess.PIPE, stderr=subprocess.PIPE)
|
|
rc = p.wait()
|
|
if rc == 0:
|
|
for line in p.stdout.readlines():
|
|
m = _distributor_id_cmdline_re.search(line)
|
|
if m:
|
|
_distname = m.group(1).strip()
|
|
if _distname and _version:
|
|
return (_distname, _version)
|
|
|
|
m = _release_cmdline_re.search(p.stdout.read())
|
|
if m:
|
|
_version = m.group(1).strip()
|
|
if _distname and _version:
|
|
return (_distname, _version)
|
|
except EnvironmentError:
|
|
pass
|
|
|
|
if os.path.exists("/etc/arch-release"):
|
|
return ("Arch_Linux", "")
|
|
|
|
return (_distname,_version)
|
|
|
|
def get_platform():
|
|
# Our version of platform.platform(), telling us both less and more than the
|
|
# Python Standard Library's version does.
|
|
# We omit details such as the Linux kernel version number, but we add a
|
|
# more detailed and correct rendition of the Linux distribution and
|
|
# distribution-version.
|
|
if "linux" in platform.system().lower():
|
|
return platform.system()+"-"+"_".join(get_linux_distro())+"-"+platform.machine()+"-"+"_".join([x for x in platform.architecture() if x])
|
|
else:
|
|
return platform.platform()
|
|
|
|
def get_package_versions_and_locations():
|
|
# because there are a few dependencies that are outside setuptools's ken (Python and
|
|
# platform), and because setuptools might fail to find something even though import finds
|
|
# it:
|
|
import OpenSSL, allmydata, foolscap, nevow, platform, pycryptopp, setuptools, simplejson, twisted, zfec, zope.interface
|
|
|
|
d1 = {
|
|
'pyOpenSSL': (OpenSSL.__version__, os.path.dirname(OpenSSL.__file__)),
|
|
'allmydata-tahoe': (allmydata.__version__, os.path.dirname(allmydata.__file__)),
|
|
'foolscap': (foolscap.__version__, os.path.dirname(foolscap.__file__)),
|
|
'Nevow': (nevow.__version__, os.path.dirname(nevow.__file__)),
|
|
'pycryptopp': (pycryptopp.__version__, os.path.dirname(pycryptopp.__file__)),
|
|
'setuptools': (setuptools.__version__, os.path.dirname(setuptools.__file__)),
|
|
'simplejson': (simplejson.__version__, os.path.dirname(simplejson.__file__)),
|
|
'zope.interface': ('unknown', os.path.dirname(zope.interface.__file__)),
|
|
'Twisted': (twisted.__version__, os.path.dirname(twisted.__file__)),
|
|
'zfec': (zfec.__version__, os.path.dirname(zfec.__file__)),
|
|
'python': (platform.python_version(), sys.executable),
|
|
'platform': (get_platform(), None),
|
|
}
|
|
|
|
# But we prefer to get all the dependencies as known by setuptools:
|
|
import pkg_resources
|
|
try:
|
|
d2 = _auto_deps.get_package_versions_from_setuptools()
|
|
except pkg_resources.DistributionNotFound:
|
|
# See docstring in _auto_deps.require_auto_deps() to explain why it makes sense to ignore this exception.
|
|
pass
|
|
else:
|
|
d1.update(d2)
|
|
|
|
return d1
|
|
|
|
def get_package_versions():
|
|
return dict([(k, v) for k, (v, l) in get_package_versions_and_locations().iteritems()])
|
|
|
|
def get_package_locations():
|
|
return dict([(k, l) for k, (v, l) in get_package_versions_and_locations().iteritems()])
|
|
|
|
def get_package_versions_string(show_paths=False):
|
|
vers_and_locs = get_package_versions_and_locations()
|
|
res = []
|
|
for p in ["allmydata-tahoe", "foolscap", "pycryptopp", "zfec", "Twisted", "Nevow", "zope.interface", "python", "platform"]:
|
|
(ver, loc) = vers_and_locs.get(p, ('UNKNOWN', 'UNKNOWN'))
|
|
info = str(p) + ": " + str(ver)
|
|
if show_paths:
|
|
info = info + " (%s)" % str(loc)
|
|
res.append(info)
|
|
if vers_and_locs.has_key(p):
|
|
del vers_and_locs[p]
|
|
|
|
for p, (v, loc) in vers_and_locs.iteritems():
|
|
info = str(p) + ": " + str(v)
|
|
if show_paths:
|
|
info = info + " (%s)" % str(loc)
|
|
res.append(info)
|
|
return ', '.join(res)
|